
instead of a closed set of languages. I also removed the offsets: I simply use the current region to determine whether the preprocessing directie starts at the beginning of a line. I also removed scanning line indicators, to make the lexer simpler. LexToken.mll: Moved the function [check_right_context] that checks stylistic constraints from Lexer.mll to LexToken.mll. While this triplicates code (as CameLIGO, PascaLIGO and ReasonLIGO share the same constraints), the benefit is that Lexer.mll becomes more generic and the signature for the TOKEN module is simpler (no more exporting predicates, except for EOF). In accordance with the change of the preprocessor, the lexers and parsers for LIGO now depend on the kind of comments, not a fixed set of syntaxes. This gives more versatility when adding a new language: only the kinds of its comments are needed, although Lexer.mll and Preproc.mll may have to be modified if they do not already know the comment delimiters, for example line comments starting with #. **************************************************************** BUG: The exceptions coming from LexToken.mll when a stylistic constraint is broken in [LexToken.check_right_context] are not caught yet. **************************************************************** Lexer.mll: I moved out as much as I could from the header into a new module LexerLib. The aim is to make it easy to reuse as much as possible of the lexer machinerie, when it cannot be used as is.
181 lines
5.7 KiB
OCaml
181 lines
5.7 KiB
OCaml
open Trace
|
|
|
|
module AST = Parser_cameligo.AST
|
|
module LexToken = Parser_reasonligo.LexToken
|
|
module Lexer = Lexer.Make (LexToken)
|
|
module Scoping = Parser_cameligo.Scoping
|
|
module Region = Simple_utils.Region
|
|
module ParErr = Parser_reasonligo.ParErr
|
|
module SyntaxError = Parser_reasonligo.SyntaxError
|
|
module SSet = Set.Make (String)
|
|
|
|
(* Mock IOs TODO: Fill them with CLI options *)
|
|
|
|
module SubIO =
|
|
struct
|
|
type options = <
|
|
libs : string list;
|
|
verbose : SSet.t;
|
|
offsets : bool;
|
|
block : EvalOpt.block_comment option;
|
|
line : EvalOpt.line_comment option;
|
|
ext : string; (* ".religo" *)
|
|
mode : [`Byte | `Point];
|
|
cmd : EvalOpt.command;
|
|
mono : bool
|
|
>
|
|
|
|
let options : options =
|
|
let block = EvalOpt.mk_block ~opening:"/*" ~closing:"*/"
|
|
in object
|
|
method libs = []
|
|
method verbose = SSet.empty
|
|
method offsets = true
|
|
method block = Some block
|
|
method line = Some "//"
|
|
method ext = ".religo"
|
|
method mode = `Point
|
|
method cmd = EvalOpt.Quiet
|
|
method mono = false
|
|
end
|
|
|
|
let make =
|
|
EvalOpt.make ~libs:options#libs
|
|
~verbose:options#verbose
|
|
~offsets:options#offsets
|
|
?block:options#block
|
|
?line:options#line
|
|
~ext:options#ext
|
|
~mode:options#mode
|
|
~cmd:options#cmd
|
|
~mono:options#mono
|
|
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 Unit =
|
|
ParserUnit.Make (Lexer)(AST)(Parser)(ParErr)(ParserLog)(SubIO)
|
|
|
|
module Errors =
|
|
struct
|
|
let generic message =
|
|
let title () = ""
|
|
and message () = message.Region.value
|
|
in Trace.error ~data:[] title message
|
|
|
|
let wrong_function_arguments (expr: AST.expr) =
|
|
let title () = "" in
|
|
let message () =
|
|
"It looks like you are defining a function, \
|
|
however we do not\n\
|
|
understand the parameters declaration.\n\
|
|
Examples of valid functions:\n\
|
|
let x = (a: string, b: int) : int => 3;\n\
|
|
let tuple = ((a, b): (int, int)) => a + b; \n\
|
|
let x = (a: string) : string => \"Hello, \" ++ a;\n" 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
|
|
|
|
let invalid_wild (expr: AST.expr) =
|
|
let title () = "" in
|
|
let message () =
|
|
"It looks like you are using a wild pattern where it cannot be used."
|
|
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 apply parser =
|
|
let local_fail error =
|
|
Trace.fail
|
|
@@ Errors.generic
|
|
@@ Unit.format_error ~offsets:SubIO.options#offsets
|
|
SubIO.options#mode error in
|
|
match parser () with
|
|
Stdlib.Ok semantic_value -> Trace.ok semantic_value
|
|
|
|
(* Lexing and parsing errors *)
|
|
|
|
| Stdlib.Error error -> Trace.fail @@ Errors.generic error
|
|
(* Scoping errors *)
|
|
|
|
| exception Scoping.Error (Scoping.Reserved_name name) ->
|
|
let token =
|
|
Lexer.Token.mk_ident name.Region.value name.Region.region in
|
|
(match token with
|
|
Stdlib.Error LexToken.Reserved_name ->
|
|
Trace.fail @@ Errors.generic @@ Region.wrap_ghost "Reserved name."
|
|
| Ok invalid ->
|
|
local_fail
|
|
("Reserved name.\nHint: Change the name.\n", None, invalid))
|
|
|
|
| exception Scoping.Error (Scoping.Duplicate_variant name) ->
|
|
let token =
|
|
Lexer.Token.mk_constr name.Region.value name.Region.region
|
|
in local_fail
|
|
("Duplicate constructor in this sum type declaration.\n\
|
|
Hint: Change the constructor.\n", None, token)
|
|
|
|
| exception Scoping.Error (Scoping.Non_linear_pattern var) ->
|
|
let token =
|
|
Lexer.Token.mk_ident var.Region.value var.Region.region in
|
|
(match token with
|
|
Stdlib.Error LexToken.Reserved_name ->
|
|
Trace.fail @@ Errors.generic @@ Region.wrap_ghost "Reserved name."
|
|
| Ok invalid ->
|
|
local_fail ("Repeated variable in this pattern.\n\
|
|
Hint: Change the name.\n",
|
|
None, invalid))
|
|
|
|
| exception Scoping.Error (Scoping.Duplicate_field name) ->
|
|
let token =
|
|
Lexer.Token.mk_ident name.Region.value name.Region.region in
|
|
(match token with
|
|
Stdlib.Error LexToken.Reserved_name ->
|
|
Trace.fail @@ Errors.generic @@ Region.wrap_ghost "Reserved name."
|
|
| Ok invalid ->
|
|
local_fail
|
|
("Duplicate field name in this record declaration.\n\
|
|
Hint: Change the name.\n",
|
|
None, invalid))
|
|
|
|
| exception SyntaxError.Error (SyntaxError.WrongFunctionArguments expr) ->
|
|
Trace.fail @@ Errors.wrong_function_arguments expr
|
|
| exception SyntaxError.Error (SyntaxError.InvalidWild expr) ->
|
|
Trace.fail @@ Errors.invalid_wild expr
|
|
|
|
(* Parsing a contract in a file *)
|
|
|
|
let parse_file source = apply (fun () -> Unit.contract_in_file source)
|
|
|
|
(* Parsing a contract in a string *)
|
|
|
|
let parse_string source = apply (fun () -> Unit.contract_in_string source)
|
|
|
|
(* Parsing an expression in a string *)
|
|
|
|
let parse_expression source = apply (fun () -> Unit.expr_in_string source)
|
|
|
|
(* Preprocessing a contract in a file *)
|
|
|
|
let preprocess source = apply (fun () -> Unit.preprocess source)
|