2019-12-10 17:47:31 +04:00
|
|
|
open Trace
|
|
|
|
|
2019-12-12 17:35:07 +04:00
|
|
|
module AST = Parser_cameligo.AST
|
2019-12-10 17:47:31 +04:00
|
|
|
module LexToken = Parser_reasonligo.LexToken
|
|
|
|
module Lexer = Lexer.Make(LexToken)
|
2020-01-14 04:27:35 +04:00
|
|
|
module Scoping = Parser_cameligo.Scoping
|
2020-01-23 21:28:04 +04:00
|
|
|
module Region = Simple_utils.Region
|
|
|
|
module ParErr = Parser_reasonligo.ParErr
|
|
|
|
module SyntaxError = Parser_reasonligo.SyntaxError
|
|
|
|
|
|
|
|
(* Mock IOs TODO: Fill them with CLI options *)
|
|
|
|
|
|
|
|
module type IO =
|
|
|
|
sig
|
|
|
|
val ext : string
|
|
|
|
val options : EvalOpt.options
|
|
|
|
end
|
|
|
|
|
|
|
|
module PreIO =
|
|
|
|
struct
|
|
|
|
let ext = ".ligo"
|
|
|
|
let pre_options =
|
|
|
|
EvalOpt.make ~input:None
|
|
|
|
~libs:[]
|
|
|
|
~verbose:Utils.String.Set.empty
|
|
|
|
~offsets:true
|
|
|
|
~mode:`Point
|
|
|
|
~cmd:EvalOpt.Quiet
|
|
|
|
~mono:true
|
|
|
|
end
|
|
|
|
|
|
|
|
module Parser =
|
|
|
|
struct
|
|
|
|
type ast = AST.t
|
|
|
|
type expr = AST.expr
|
|
|
|
include Parser_reasonligo.Parser
|
|
|
|
end
|
|
|
|
|
|
|
|
module ParserLog =
|
|
|
|
struct
|
|
|
|
type ast = AST.t
|
|
|
|
type expr = AST.expr
|
|
|
|
include Parser_cameligo.ParserLog
|
|
|
|
end
|
|
|
|
|
|
|
|
module PreUnit =
|
|
|
|
ParserUnit.Make (Lexer)(AST)(Parser)(ParErr)(ParserLog)
|
2019-12-19 17:50:57 +04:00
|
|
|
|
2020-01-14 04:27:35 +04:00
|
|
|
module Errors =
|
|
|
|
struct
|
2020-01-23 21:28:04 +04:00
|
|
|
let reserved_name Region.{value; region} =
|
|
|
|
let title () = Printf.sprintf "\nReserved name \"%s\"" value in
|
|
|
|
let message () = "" in
|
2020-01-14 04:27:35 +04:00
|
|
|
let data = [
|
2020-01-23 21:28:04 +04:00
|
|
|
("location",
|
|
|
|
fun () -> Format.asprintf "%a" Location.pp_lift @@ region)]
|
2020-01-14 04:27:35 +04:00
|
|
|
in error ~data title message
|
2019-12-19 17:50:57 +04:00
|
|
|
|
2020-01-23 21:28:04 +04:00
|
|
|
let duplicate_variant Region.{value; region} =
|
|
|
|
let title () =
|
|
|
|
Printf.sprintf "\nDuplicate variant \"%s\" in this \
|
|
|
|
type declaration" value in
|
|
|
|
let message () = "" in
|
|
|
|
let data = [
|
|
|
|
("location",
|
|
|
|
fun () -> Format.asprintf "%a" Location.pp_lift @@ region)]
|
|
|
|
in error ~data title message
|
|
|
|
|
|
|
|
let non_linear_pattern Region.{value; region} =
|
|
|
|
let title () =
|
|
|
|
Printf.sprintf "\nRepeated variable \"%s\" in this pattern" value in
|
2020-01-14 04:27:35 +04:00
|
|
|
let message () = "" in
|
|
|
|
let data = [
|
2020-01-23 21:28:04 +04:00
|
|
|
("location",
|
|
|
|
fun () -> Format.asprintf "%a" Location.pp_lift @@ region)]
|
2020-01-14 04:27:35 +04:00
|
|
|
in error ~data title message
|
2020-01-04 12:24:16 +04:00
|
|
|
|
2020-01-23 21:28:04 +04:00
|
|
|
let duplicate_field Region.{value; region} =
|
|
|
|
let title () =
|
|
|
|
Printf.sprintf "\nDuplicate field name \"%s\" \
|
|
|
|
in this record declaration" value in
|
|
|
|
let message () = "" in
|
|
|
|
let data = [
|
|
|
|
("location",
|
|
|
|
fun () -> Format.asprintf "%a" Location.pp_lift @@ region)]
|
|
|
|
in error ~data title message
|
|
|
|
|
|
|
|
let parser_error Region.{value; region} =
|
|
|
|
let title () = ""
|
|
|
|
and message () = value
|
|
|
|
and loc = region in
|
2020-01-14 04:27:35 +04:00
|
|
|
let data =
|
|
|
|
[("parser_loc",
|
|
|
|
fun () -> Format.asprintf "%a" Location.pp_lift @@ loc)]
|
|
|
|
in error ~data title message
|
2019-12-19 17:50:57 +04:00
|
|
|
|
2020-01-23 21:28:04 +04:00
|
|
|
let lexer_error (e: Lexer.error AST.reg) =
|
|
|
|
let title () = "\nLexer error" in
|
|
|
|
let message () = Lexer.error_to_string e.value in
|
2020-01-14 04:27:35 +04:00
|
|
|
let data = [
|
2020-01-23 21:28:04 +04:00
|
|
|
("parser_loc",
|
|
|
|
fun () -> Format.asprintf "%a" Location.pp_lift @@ e.region)]
|
2020-01-14 04:27:35 +04:00
|
|
|
in error ~data title message
|
2019-12-19 17:50:57 +04:00
|
|
|
|
2020-01-23 21:28:04 +04:00
|
|
|
let wrong_function_arguments (expr: AST.expr) =
|
|
|
|
let title () = "\nWrong function arguments" in
|
|
|
|
let message () = "" in
|
|
|
|
let expression_loc = AST.expr_to_region expr in
|
|
|
|
let data = [
|
|
|
|
("location",
|
|
|
|
fun () -> Format.asprintf "%a" Location.pp_lift @@ expression_loc)]
|
|
|
|
in error ~data title message
|
|
|
|
end
|
|
|
|
|
|
|
|
let parse (module IO : IO) parser =
|
|
|
|
let module Unit = PreUnit (IO) in
|
|
|
|
let mk_error error =
|
|
|
|
Unit.format_error ~offsets:IO.options#offsets
|
|
|
|
IO.options#mode error in
|
|
|
|
match parser () with
|
|
|
|
(* Scoping errors *)
|
|
|
|
|
|
|
|
Stdlib.Ok semantic_value -> ok semantic_value
|
|
|
|
| Stdlib.Error error -> fail @@ Errors.parser_error error
|
|
|
|
| exception Lexer.Error e -> fail @@ Errors.lexer_error e
|
|
|
|
|
|
|
|
| exception SyntaxError.Error (SyntaxError.WrongFunctionArguments expr) ->
|
|
|
|
fail @@ Errors.wrong_function_arguments expr
|
|
|
|
| exception Scoping.Error (Scoping.Reserved_name name) ->
|
|
|
|
let token =
|
|
|
|
Lexer.Token.mk_ident name.Region.value name.Region.region in
|
|
|
|
(match token with
|
|
|
|
(* Cannot fail because [name] is a not a
|
|
|
|
reserved name for the lexer. *)
|
|
|
|
Stdlib.Error _ -> assert false
|
|
|
|
| Ok invalid ->
|
|
|
|
let point =
|
|
|
|
"Reserved name.\nHint: Change the name.\n", None, invalid
|
|
|
|
in fail @@ Errors.reserved_name @@ mk_error point)
|
|
|
|
|
|
|
|
| exception Scoping.Error (Scoping.Duplicate_variant name) ->
|
|
|
|
let token =
|
|
|
|
Lexer.Token.mk_constr name.Region.value name.Region.region in
|
|
|
|
let point =
|
|
|
|
"Duplicate constructor in this sum type declaration.\n\
|
|
|
|
Hint: Change the constructor.\n",
|
|
|
|
None, token
|
|
|
|
in fail @@ Errors.duplicate_variant @@ mk_error point
|
|
|
|
|
|
|
|
| exception Scoping.Error (Scoping.Non_linear_pattern var) ->
|
|
|
|
let token =
|
|
|
|
Lexer.Token.mk_ident var.Region.value var.Region.region in
|
|
|
|
(match token with
|
|
|
|
(* Cannot fail because [var] is a not a
|
|
|
|
reserved name for the lexer. *)
|
|
|
|
Stdlib.Error _ -> assert false
|
|
|
|
| Ok invalid ->
|
|
|
|
let point =
|
|
|
|
"Repeated variable in this pattern.\n\
|
|
|
|
Hint: Change the name.\n",
|
|
|
|
None, invalid
|
|
|
|
in fail @@ Errors.non_linear_pattern @@ mk_error point)
|
|
|
|
|
|
|
|
| exception Scoping.Error (Scoping.Duplicate_field name) ->
|
|
|
|
let token =
|
|
|
|
Lexer.Token.mk_ident name.Region.value name.Region.region in
|
|
|
|
(match token with
|
|
|
|
(* Cannot fail because [name] is a not a
|
|
|
|
reserved name for the lexer. *)
|
|
|
|
Stdlib.Error _ -> assert false
|
|
|
|
| Ok invalid ->
|
|
|
|
let point =
|
|
|
|
"Duplicate field name in this record declaration.\n\
|
|
|
|
Hint: Change the name.\n",
|
|
|
|
None, invalid
|
|
|
|
in fail @@ Errors.duplicate_field @@ mk_error point)
|
|
|
|
|
|
|
|
let parse_file (source: string) =
|
|
|
|
let module IO =
|
|
|
|
struct
|
|
|
|
let ext = PreIO.ext
|
|
|
|
let options = PreIO.pre_options ~expr:false
|
|
|
|
end in
|
2019-12-10 17:47:31 +04:00
|
|
|
let pp_input =
|
|
|
|
let prefix = Filename.(source |> basename |> remove_extension)
|
2020-01-23 21:28:04 +04:00
|
|
|
and suffix = ".pp.ligo"
|
2019-12-10 17:47:31 +04:00
|
|
|
in prefix ^ suffix in
|
|
|
|
let cpp_cmd = Printf.sprintf "cpp -traditional-cpp %s > %s"
|
|
|
|
source pp_input in
|
|
|
|
let%bind () = sys_command cpp_cmd in
|
|
|
|
let%bind channel =
|
2020-01-23 21:28:04 +04:00
|
|
|
generic_try (simple_error "Error when opening file") @@
|
2019-12-10 17:47:31 +04:00
|
|
|
(fun () -> open_in pp_input) in
|
2020-01-23 21:28:04 +04:00
|
|
|
let module Unit = PreUnit (IO) in
|
|
|
|
let instance =
|
|
|
|
match Lexer.open_token_stream (Lexer.Channel channel) with
|
|
|
|
Ok instance -> instance
|
|
|
|
| Stdlib.Error _ -> assert false (* No file opening *) in
|
|
|
|
let thunk () = Unit.apply instance Unit.parse_contract in
|
|
|
|
parse (module IO) thunk
|
2019-12-10 17:47:31 +04:00
|
|
|
|
2020-01-23 21:28:04 +04:00
|
|
|
let parse_string (s: string) =
|
|
|
|
let module IO =
|
|
|
|
struct
|
|
|
|
let ext = PreIO.ext
|
|
|
|
let options = PreIO.pre_options ~expr:false
|
|
|
|
end in
|
|
|
|
let module Unit = PreUnit (IO) in
|
|
|
|
let instance =
|
|
|
|
match Lexer.open_token_stream (Lexer.String s) with
|
|
|
|
Ok instance -> instance
|
|
|
|
| Stdlib.Error _ -> assert false (* No file opening *) in
|
|
|
|
let thunk () = Unit.apply instance Unit.parse_contract in
|
|
|
|
parse (module IO) thunk
|
2019-12-10 17:47:31 +04:00
|
|
|
|
2020-01-23 21:28:04 +04:00
|
|
|
let parse_expression (s: string) =
|
|
|
|
let module IO =
|
|
|
|
struct
|
|
|
|
let ext = PreIO.ext
|
|
|
|
let options = PreIO.pre_options ~expr:true
|
|
|
|
end in
|
|
|
|
let module Unit = PreUnit (IO) in
|
|
|
|
let instance =
|
|
|
|
match Lexer.open_token_stream (Lexer.String s) with
|
|
|
|
Ok instance -> instance
|
|
|
|
| Stdlib.Error _ -> assert false (* No file opening *) in
|
|
|
|
let thunk () = Unit.apply instance Unit.parse_expr in
|
|
|
|
parse (module IO) thunk
|