package yocaml

  1. Overview
  2. Docs

Source file diagnostic.ml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
(* YOCaml a static blog generator.
   Copyright (C) 2024 The Funkyworkers and The YOCaml's developers

   This program is free software: you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation, either version 3 of the License, or
   (at your option) any later version.

   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.

   You should have received a copy of the GNU General Public License
   along with this program.  If not, see <https://www.gnu.org/licenses/>. *)

let comma_sep ppf () = Format.fprintf ppf ";@,"

let rec pp_validation_error custom_error ppf = function
  | Data.Validation.Invalid_shape { expected; given } ->
      Format.fprintf ppf
        "Fail with Invalid shape: @[<2>{ @[<1>expected =@ `%s`@];@,\
         @[<1>given =@ `%a`@];@,\
         }@]"
        expected Data.pp given
  | Data.Validation.With_message { message; given } ->
      Format.fprintf ppf
        "Fail with message: @[<2>{ @[<1>message =@ `%s`@];@,\
         @[<1>given =@ `%s`@];@,\
         }@]"
        message given
  | Data.Validation.Custom custom ->
      Format.fprintf ppf "Fail with Custom error: @[<2>%a@]" custom_error custom
  | Data.Validation.Invalid_list { errors; given } ->
      Format.fprintf ppf
        "Fail with Invalid list @[<2>{ @[<1>errors =@ `%a`@];@,\
         @[<1>given =@ `%a`@];@,\
         }@]"
        (Format.pp_print_list ~pp_sep:comma_sep (fun ppf (i, err) ->
             Format.fprintf ppf "@[<1>%d =@ `%a`@]" i
               (pp_validation_error custom_error)
               err))
        (Nel.to_list errors)
        (Format.pp_print_list ~pp_sep:comma_sep Data.pp)
        given
  | Data.Validation.Invalid_record { errors; given } ->
      Format.fprintf ppf
        "Fail with Invalid record: @[<2>{@[<1>errors =@ `%a`@];@,\
         @[<1>given =@ `%a`@];@,\
         }]"
        (Format.pp_print_list ~pp_sep:comma_sep (pp_record_error custom_error))
        (Nel.to_list errors)
        (Format.pp_print_list ~pp_sep:comma_sep (fun ppf (k, v) ->
             Format.fprintf ppf "@[<1>%s =@ `%a`@]" k Data.pp v))
        given

and pp_record_error custom_error ppf = function
  | Data.Validation.Missing_field { field } ->
      Format.fprintf ppf "Missing field =@ `%s`" field
  | Data.Validation.Invalid_field { given; field; error } ->
      Format.fprintf ppf
        "Invalid field =@ `%s` @[<2>{@[<2>{@[<1>error =@ `%a`@];@,\
         @[<1>given =@ `%a`@];@,\
         }@]"
        field
        (pp_validation_error custom_error)
        error Data.pp given

let pp_provider_error custom_error ppf = function
  | Required.Parsing_error { given; message } ->
      Format.fprintf ppf "Parsing error: @[given: @[`%s`@]\nmessage:@[`%s`@]@]"
        given message
  | Required.Required_metadata { entity } ->
      Format.fprintf ppf "Required metadata: `%s`" entity
  | Required.Validation_error { entity; error } ->
      Format.fprintf ppf "Validation error: `%s`\n @[%a@]" entity
        (pp_validation_error custom_error)
        error

let glob_pp p v backtrace ppf =
  Format.fprintf ppf "--- %a ---\n%a\n---\n%s" Lexicon.there_is_an_error () p v
    backtrace

let exception_to_diagnostic
    ?(custom_error = fun ppf _ -> Format.fprintf ppf "Custom Validation Error")
    ?(in_exception_handler = true) ppf exn =
  let backtrace =
    if in_exception_handler then Printexc.get_backtrace ()
    else Lexicon.backtrace_not_available
  in
  let glob_pp p v = glob_pp p v backtrace ppf in
  match exn with
  | Eff.File_not_exists (source, path) ->
      glob_pp (Lexicon.file_not_exists source path) ()
  | Eff.Invalid_path (source, path) ->
      glob_pp (Lexicon.invalid_path source path) ()
  | Eff.File_is_a_directory (source, path) ->
      glob_pp (Lexicon.file_is_a_directory source path) ()
  | Eff.Directory_not_exists (source, path) ->
      glob_pp (Lexicon.directory_not_exists source path) ()
  | Eff.Directory_is_a_file (source, path) ->
      glob_pp (Lexicon.directory_is_a_file source path) ()
  | Eff.Provider_error error -> glob_pp (pp_provider_error custom_error) error
  | exn -> glob_pp Lexicon.unknown_error exn

let runtime_error_to_diagnostic ppf message =
  let backtrace = Lexicon.backtrace_not_available in
  glob_pp Format.pp_print_string message backtrace ppf