2020-04-16 18:47:35 +03:00
|
|
|
(* This file is part of the Catala compiler, a specification language for tax
|
2020-04-15 16:33:21 +03:00
|
|
|
and social benefits computation rules. Copyright (C) 2020 Inria, contributor:
|
2020-03-12 20:04:27 +03:00
|
|
|
Denis Merigoux <denis.merigoux@inria.fr>
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License"); you may not
|
|
|
|
use this file except in compliance with the License. You may obtain a copy of
|
|
|
|
the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
License for the specific language governing permissions and limitations under
|
|
|
|
the License. *)
|
|
|
|
|
2022-02-18 17:54:40 +03:00
|
|
|
type t = { code_pos : Lexing.position * Lexing.position; law_pos : string list }
|
2021-01-20 17:37:20 +03:00
|
|
|
|
|
|
|
let from_lpos (p : Lexing.position * Lexing.position) : t =
|
|
|
|
{ code_pos = p; law_pos = [] }
|
2020-03-12 20:04:27 +03:00
|
|
|
|
2020-05-25 18:48:02 +03:00
|
|
|
let from_info
|
|
|
|
(file : string)
|
|
|
|
(sline : int)
|
|
|
|
(scol : int)
|
|
|
|
(eline : int)
|
|
|
|
(ecol : int) : t =
|
|
|
|
let spos =
|
|
|
|
{
|
|
|
|
Lexing.pos_fname = file;
|
|
|
|
Lexing.pos_lnum = sline;
|
|
|
|
Lexing.pos_cnum = scol;
|
|
|
|
Lexing.pos_bol = 1;
|
|
|
|
}
|
|
|
|
in
|
|
|
|
let epos =
|
|
|
|
{
|
|
|
|
Lexing.pos_fname = file;
|
|
|
|
Lexing.pos_lnum = eline;
|
|
|
|
Lexing.pos_cnum = ecol;
|
|
|
|
Lexing.pos_bol = 1;
|
|
|
|
}
|
|
|
|
in
|
2021-01-20 17:37:20 +03:00
|
|
|
{ code_pos = spos, epos; law_pos = [] }
|
2020-03-12 20:04:27 +03:00
|
|
|
|
2021-01-20 21:19:17 +03:00
|
|
|
let overwrite_law_info (pos : t) (law_pos : string list) : t =
|
|
|
|
{ pos with law_pos }
|
|
|
|
|
2021-04-03 18:58:31 +03:00
|
|
|
let get_law_info (pos : t) : string list = pos.law_pos
|
|
|
|
|
2020-04-10 13:14:16 +03:00
|
|
|
let get_start_line (pos : t) : int =
|
2021-01-20 17:37:20 +03:00
|
|
|
let s, _ = pos.code_pos in
|
2020-04-10 13:14:16 +03:00
|
|
|
s.Lexing.pos_lnum
|
|
|
|
|
2020-04-26 19:32:03 +03:00
|
|
|
let get_start_column (pos : t) : int =
|
2021-01-20 17:37:20 +03:00
|
|
|
let s, _ = pos.code_pos in
|
2020-04-26 19:32:03 +03:00
|
|
|
s.Lexing.pos_cnum - s.Lexing.pos_bol + 1
|
|
|
|
|
|
|
|
let get_end_line (pos : t) : int =
|
2021-01-20 17:37:20 +03:00
|
|
|
let _, e = pos.code_pos in
|
2020-04-26 19:32:03 +03:00
|
|
|
e.Lexing.pos_lnum
|
|
|
|
|
|
|
|
let get_end_column (pos : t) : int =
|
2021-01-20 17:37:20 +03:00
|
|
|
let _, e = pos.code_pos in
|
2020-04-26 19:32:03 +03:00
|
|
|
e.Lexing.pos_cnum - e.Lexing.pos_bol + 1
|
|
|
|
|
2021-01-20 17:37:20 +03:00
|
|
|
let get_file (pos : t) : string = (fst pos.code_pos).Lexing.pos_fname
|
2020-04-19 20:04:11 +03:00
|
|
|
|
2020-12-26 19:37:41 +03:00
|
|
|
type input_file = FileName of string | Contents of string
|
|
|
|
|
2020-05-25 18:48:02 +03:00
|
|
|
let to_string (pos : t) : string =
|
2021-01-20 17:37:20 +03:00
|
|
|
let s, e = pos.code_pos in
|
2020-05-25 18:48:02 +03:00
|
|
|
Printf.sprintf "in file %s, from %d:%d to %d:%d" s.Lexing.pos_fname
|
|
|
|
s.Lexing.pos_lnum
|
2022-10-27 13:10:23 +03:00
|
|
|
(s.Lexing.pos_cnum - s.Lexing.pos_bol)
|
2020-05-25 18:48:02 +03:00
|
|
|
e.Lexing.pos_lnum
|
2022-10-27 13:10:23 +03:00
|
|
|
(e.Lexing.pos_cnum - e.Lexing.pos_bol)
|
2020-05-25 18:48:02 +03:00
|
|
|
|
|
|
|
let to_string_short (pos : t) : string =
|
2021-01-20 17:37:20 +03:00
|
|
|
let s, e = pos.code_pos in
|
2022-10-27 13:10:23 +03:00
|
|
|
if e.Lexing.pos_lnum = s.Lexing.pos_lnum then
|
2022-11-24 20:00:45 +03:00
|
|
|
Printf.sprintf "%s:%d.%d-%d:" s.Lexing.pos_fname s.Lexing.pos_lnum
|
2022-10-27 13:10:23 +03:00
|
|
|
(s.Lexing.pos_cnum - s.Lexing.pos_bol)
|
|
|
|
(e.Lexing.pos_cnum - e.Lexing.pos_bol)
|
|
|
|
else
|
2022-11-24 20:00:45 +03:00
|
|
|
Printf.sprintf "%s:%d.%d-%d.%d:" s.Lexing.pos_fname s.Lexing.pos_lnum
|
2022-10-27 13:10:23 +03:00
|
|
|
(s.Lexing.pos_cnum - s.Lexing.pos_bol)
|
|
|
|
e.Lexing.pos_lnum
|
|
|
|
(e.Lexing.pos_cnum - e.Lexing.pos_bol)
|
2020-05-25 18:48:02 +03:00
|
|
|
|
|
|
|
let indent_number (s : string) : int =
|
|
|
|
try
|
|
|
|
let rec aux (i : int) = if s.[i] = ' ' then aux (i + 1) else i in
|
|
|
|
aux 0
|
|
|
|
with Invalid_argument _ -> String.length s
|
|
|
|
|
2022-10-27 13:18:00 +03:00
|
|
|
let string_repeat n s =
|
|
|
|
let slen = String.length s in
|
|
|
|
let buf = Bytes.create (n * slen) in
|
|
|
|
for i = 0 to n - 1 do
|
|
|
|
Bytes.blit_string s 0 buf (i * slen) slen
|
|
|
|
done;
|
|
|
|
Bytes.to_string buf
|
|
|
|
|
2022-11-25 13:36:53 +03:00
|
|
|
(* Note: this should do, but remains incorrect for combined unicode characters
|
|
|
|
that display as one (e.g. `e` + postfix `'`). We should switch to Uuseg at
|
|
|
|
some poing *)
|
|
|
|
let string_columns s =
|
|
|
|
let len = String.length s in
|
|
|
|
let rec aux ncols i =
|
|
|
|
if i >= len then ncols
|
|
|
|
else if s.[i] = '\t' then aux (ncols + 8) (i + 1)
|
|
|
|
else
|
|
|
|
aux (ncols + 1) (i + Uchar.utf_decode_length (String.get_utf_8_uchar s i))
|
|
|
|
in
|
|
|
|
aux 0 0
|
|
|
|
|
|
|
|
let utf8_byte_index s ui0 =
|
|
|
|
let rec aux bi ui =
|
|
|
|
if ui >= ui0 then bi
|
|
|
|
else
|
|
|
|
aux (bi + Uchar.utf_decode_length (String.get_utf_8_uchar s bi)) (ui + 1)
|
|
|
|
in
|
|
|
|
aux 0 0
|
|
|
|
|
2020-05-25 18:48:02 +03:00
|
|
|
let retrieve_loc_text (pos : t) : string =
|
2020-12-26 19:37:41 +03:00
|
|
|
try
|
|
|
|
let filename = get_file pos in
|
|
|
|
let blue_style = [ANSITerminal.Bold; ANSITerminal.blue] in
|
|
|
|
if filename = "" then "No position information"
|
|
|
|
else
|
|
|
|
let sline = get_start_line pos in
|
|
|
|
let eline = get_end_line pos in
|
|
|
|
let oc, input_line_opt =
|
|
|
|
if filename = "stdin" then
|
|
|
|
let line_index = ref 0 in
|
|
|
|
let lines = String.split_on_char '\n' !Cli.contents in
|
|
|
|
let input_line_opt () : string option =
|
|
|
|
match List.nth_opt lines !line_index with
|
|
|
|
| Some l ->
|
|
|
|
line_index := !line_index + 1;
|
|
|
|
Some l
|
|
|
|
| None -> None
|
|
|
|
in
|
|
|
|
None, input_line_opt
|
|
|
|
else
|
|
|
|
let oc = open_in filename in
|
|
|
|
let input_line_opt () : string option =
|
|
|
|
try Some (input_line oc) with End_of_file -> None
|
|
|
|
in
|
|
|
|
Some oc, input_line_opt
|
|
|
|
in
|
|
|
|
let print_matched_line (line : string) (line_no : int) : string =
|
|
|
|
let line_indent = indent_number line in
|
|
|
|
let error_indicator_style = [ANSITerminal.red; ANSITerminal.Bold] in
|
2022-11-25 13:36:53 +03:00
|
|
|
let match_start_index =
|
|
|
|
utf8_byte_index line
|
|
|
|
(if line_no = sline then get_start_column pos - 1 else line_indent)
|
|
|
|
in
|
|
|
|
let match_end_index =
|
|
|
|
if line_no = eline then utf8_byte_index line (get_end_column pos - 1)
|
|
|
|
else String.length line
|
|
|
|
in
|
|
|
|
let unmatched_prefix = String.sub line 0 match_start_index in
|
|
|
|
let matched_substring =
|
|
|
|
String.sub line match_start_index
|
|
|
|
(max 0 (match_end_index - match_start_index))
|
|
|
|
in
|
|
|
|
let match_start_col = string_columns unmatched_prefix in
|
|
|
|
let match_num_cols = string_columns matched_substring in
|
|
|
|
String.concat ""
|
|
|
|
(line
|
|
|
|
:: "\n"
|
|
|
|
::
|
|
|
|
(if line_no >= sline && line_no <= eline then
|
|
|
|
[
|
|
|
|
string_repeat match_start_col " ";
|
|
|
|
Cli.with_style error_indicator_style "%s"
|
|
|
|
(string_repeat match_num_cols "‾");
|
|
|
|
]
|
|
|
|
else []))
|
2020-12-26 19:37:41 +03:00
|
|
|
in
|
|
|
|
let include_extra_count = 0 in
|
|
|
|
let rec get_lines (n : int) : string list =
|
|
|
|
match input_line_opt () with
|
|
|
|
| Some line ->
|
|
|
|
if n < sline - include_extra_count then get_lines (n + 1)
|
|
|
|
else if
|
|
|
|
n >= sline - include_extra_count && n <= eline + include_extra_count
|
|
|
|
then print_matched_line line n :: get_lines (n + 1)
|
|
|
|
else []
|
|
|
|
| None -> []
|
|
|
|
in
|
|
|
|
let pos_lines = get_lines 1 in
|
|
|
|
let spaces = int_of_float (log10 (float_of_int eline)) + 1 in
|
2021-01-20 21:58:48 +03:00
|
|
|
let legal_pos_lines =
|
|
|
|
List.rev
|
|
|
|
(List.map
|
|
|
|
(fun s ->
|
|
|
|
Re.Pcre.substitute ~rex:(Re.Pcre.regexp "\n\\s*")
|
|
|
|
~subst:(fun _ -> " ")
|
|
|
|
s)
|
|
|
|
pos.law_pos)
|
|
|
|
in
|
2020-12-26 19:37:41 +03:00
|
|
|
(match oc with None -> () | Some oc -> close_in oc);
|
2022-10-27 13:18:00 +03:00
|
|
|
let buf = Buffer.create 73 in
|
|
|
|
Buffer.add_string buf
|
|
|
|
(Cli.with_style blue_style "┌─⯈ %s" (to_string_short pos));
|
|
|
|
Buffer.add_char buf '\n';
|
|
|
|
(* should be outside of [Cli.with_style] *)
|
|
|
|
Buffer.add_string buf
|
|
|
|
(Cli.with_style blue_style "└%s┐" (string_repeat spaces "─"));
|
|
|
|
Buffer.add_char buf '\n';
|
|
|
|
Buffer.add_string buf
|
2022-11-24 20:00:45 +03:00
|
|
|
(Cli.add_prefix_to_each_line (String.concat "\n" pos_lines) (fun i ->
|
|
|
|
let cur_line = sline - include_extra_count + i in
|
2020-12-26 19:37:41 +03:00
|
|
|
if
|
|
|
|
cur_line >= sline
|
|
|
|
&& cur_line <= sline + (2 * (eline - sline))
|
|
|
|
&& cur_line mod 2 = sline mod 2
|
2022-03-08 15:04:27 +03:00
|
|
|
then
|
2022-10-27 13:18:00 +03:00
|
|
|
Cli.with_style blue_style "%*d │" spaces
|
2022-03-08 15:04:27 +03:00
|
|
|
(sline + ((cur_line - sline) / 2))
|
2020-12-26 19:37:41 +03:00
|
|
|
else if cur_line >= sline - include_extra_count && cur_line < sline
|
2022-10-27 13:18:00 +03:00
|
|
|
then Cli.with_style blue_style "%*d │" spaces (cur_line + 1)
|
2020-12-26 19:37:41 +03:00
|
|
|
else if
|
|
|
|
cur_line
|
|
|
|
<= sline + (2 * (eline - sline)) + 1 + include_extra_count
|
|
|
|
&& cur_line > sline + (2 * (eline - sline)) + 1
|
2022-03-08 15:04:27 +03:00
|
|
|
then
|
2022-10-27 13:18:00 +03:00
|
|
|
Cli.with_style blue_style "%*d │" spaces
|
2022-03-08 15:04:27 +03:00
|
|
|
(cur_line - (eline - sline + 1))
|
2022-10-27 13:18:00 +03:00
|
|
|
else Cli.with_style blue_style "%*s │" spaces ""));
|
|
|
|
Buffer.add_char buf '\n';
|
|
|
|
let () =
|
|
|
|
match legal_pos_lines with
|
|
|
|
| [] -> ()
|
|
|
|
| _ ->
|
|
|
|
let last = List.length legal_pos_lines - 1 in
|
|
|
|
Buffer.add_string buf
|
|
|
|
(Cli.add_prefix_to_each_line
|
|
|
|
(String.concat "\n"
|
|
|
|
(List.map
|
|
|
|
(fun l -> Cli.with_style blue_style "%s" l)
|
|
|
|
legal_pos_lines))
|
|
|
|
(fun i ->
|
|
|
|
if i = last then
|
|
|
|
Cli.with_style blue_style "%*s└─" (spaces + i + 1) ""
|
|
|
|
else Cli.with_style blue_style "%*s└┬" (spaces + i + 1) ""))
|
|
|
|
in
|
|
|
|
Buffer.contents buf
|
2020-12-26 19:37:41 +03:00
|
|
|
with Sys_error _ -> "Location:" ^ to_string pos
|
2020-05-25 18:48:02 +03:00
|
|
|
|
2020-03-12 20:04:27 +03:00
|
|
|
let no_pos : t =
|
|
|
|
let zero_pos =
|
|
|
|
{
|
|
|
|
Lexing.pos_fname = "";
|
|
|
|
Lexing.pos_lnum = 0;
|
|
|
|
Lexing.pos_cnum = 0;
|
|
|
|
Lexing.pos_bol = 0;
|
|
|
|
}
|
|
|
|
in
|
2021-01-20 17:37:20 +03:00
|
|
|
{ code_pos = zero_pos, zero_pos; law_pos = [] }
|