package dune

  1. Overview
  2. Docs
Legend:
Page
Library
Module
Module type
Parameter
Class
Class type
Source

Source file user_message.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
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
module Style = struct
  type t =
    | Loc
    | Error
    | Warning
    | Kwd
    | Id
    | Prompt
    | Details
    | Ok
    | Debug
    | Success
    | Ansi_styles of Ansi_color.Style.t list
end

module Print_config = struct
  type t = Style.t -> Ansi_color.Style.t list

  open Ansi_color.Style

  let default : Style.t -> _ = function
    | Loc     -> [bold]
    | Error   -> [bold; fg_red]
    | Warning -> [bold; fg_magenta]
    | Kwd     -> [bold; fg_blue]
    | Id      -> [bold; fg_yellow]
    | Prompt  -> [bold; fg_green]
    | Details -> [dim; fg_white]
    | Ok      -> [dim; fg_green]
    | Debug   -> [underlined; fg_bright_cyan]
    | Success -> [bold; fg_green]
    | Ansi_styles l -> l
end

type t =
  { loc : Loc0.t option
  ; paragraphs : Style.t Pp.t list
  ; hints : Style.t Pp.t list
  }

let make ?loc ?prefix ?(hints=[]) paragraphs =
  let paragraphs =
    match prefix, paragraphs with
    | None, l -> l
    | Some p, [] -> [p]
    | Some p, x :: l ->
      Pp.concat ~sep:Pp.space [p; x] :: l
  in
  { loc; hints; paragraphs }

let pp { loc; paragraphs; hints } =
  let paragraphs =
    match hints with
    | [] -> paragraphs
    | _ ->
      List.append
        paragraphs
        (List.map hints ~f:(fun hint ->
           Pp.concat ~sep:Pp.space [Pp.verbatim "Hint:"; hint]))
  in
  let paragraphs = List.map paragraphs ~f:Pp.box in
  let paragraphs =
    match loc with
    | None -> paragraphs
    | Some { Loc0.start; stop } ->
      let start_c = start.pos_cnum - start.pos_bol in
      let stop_c  = stop.pos_cnum  - start.pos_bol in
      Pp.tag ~tag:Style.Loc
        (Pp.textf "File %S, line %d, characters %d-%d:"
           start.pos_fname start.pos_lnum start_c stop_c)
      :: paragraphs
  in
  Pp.vbox (Pp.concat_map paragraphs ~sep:Pp.nop
             ~f:(fun pp -> Pp.seq pp Pp.cut))

let print ?(config=Print_config.default) t =
  Ansi_color.print (Pp.map_tags (pp t) ~f:config)

let prerr ?(config=Print_config.default) t =
  Ansi_color.prerr (Pp.map_tags (pp t) ~f:config)

(* As found here http://rosettacode.org/wiki/Levenshtein_distance#OCaml *)
let levenshtein_distance s t =
  let m = String.length s
  and n = String.length t in
  (* for all i and j, d.(i).(j) will hold the Levenshtein distance between
     the first i characters of s and the first j characters of t *)
  let d = Array.make_matrix ~dimx:(m+1) ~dimy:(n+1) 0 in

  for i = 0 to m do
    (* the distance of any first string to an empty second string *)
    d.(i).(0) <- i
  done;
  for j = 0 to n do
    (* the distance of any second string to an empty first string *)
    d.(0).(j) <- j
  done;

  for j = 1 to n do
    for i = 1 to m do

      if s.[i-1] = t.[j-1] then
        d.(i).(j) <- d.(i-1).(j-1)  (* no operation required *)
      else
        d.(i).(j) <- min
                       (d.(i-1).(j) + 1) (* a deletion *)
                       (min
                          (d.(i).(j-1) + 1) (* an insertion *)
                          (d.(i-1).(j-1) + 1) (* a substitution *)
                       )
    done;
  done;

  d.(m).(n)

let did_you_mean s ~candidates =
  let candidates =
    List.filter candidates ~f:(fun candidate ->
      levenshtein_distance s candidate < 3)
  in
  match candidates with
  | [] -> []
  | l -> [Pp.textf "did you mean %s?" (String.enumerate_or l)]
OCaml

Innovation. Community. Security.