From fd175eb431343d0113a8b765773d7ebe6c9009c7 Mon Sep 17 00:00:00 2001 From: Antal Spector-Zabusky Date: Thu, 10 Aug 2023 16:42:56 -0400 Subject: [PATCH 01/26] Expose migration etc. for all AST types needed by `Pprintast` Signed-off-by: Antal Spector-Zabusky --- CHANGES.md | 2 + ast/cinaps/ast_cinaps_helpers.ml | 28 +---- ast/cinaps/dune | 2 +- ast/import.ml | 120 ++++++++++++++++-- ast/versions.ml | 162 +++++++++++++++++++++++++ ast/versions.mli | 36 ++++++ astlib/cinaps/astlib_cinaps_helpers.ml | 34 ++++++ astlib/migrate_414_500.ml | 106 +++++++++++----- astlib/migrate_500_414.ml | 106 +++++++++++----- 9 files changed, 505 insertions(+), 91 deletions(-) diff --git a/CHANGES.md b/CHANGES.md index e2d1dab2a..d30021973 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -7,6 +7,8 @@ unreleased - Restore the "path_arg" functionality in the V3 API (#431, @ELLIOTTCABLE) +- Expose migration/copying/etc. functions for all AST types needed by `Pprintast` (#454, @antalsz) + 0.30.0 (20/06/2023) ------------------- diff --git a/ast/cinaps/ast_cinaps_helpers.ml b/ast/cinaps/ast_cinaps_helpers.ml index 237c572af..f1d9a0882 100644 --- a/ast/cinaps/ast_cinaps_helpers.ml +++ b/ast/cinaps/ast_cinaps_helpers.ml @@ -3,33 +3,13 @@ include StdLabels include Printf -let nl () = printf "\n" - -let qualified_types = - [ - ( "Parsetree", - [ - "structure"; - "signature"; - "toplevel_phrase"; - "core_type"; - "expression"; - "pattern"; - "case"; - "type_declaration"; - "type_extension"; - "extension_constructor"; - ] ); - ] +let nl = Astlib_cinaps_helpers.nl +let qualified_types = Astlib_cinaps_helpers.qualified_types +let foreach_module = Astlib_cinaps_helpers.foreach_module +let foreach_type = Astlib_cinaps_helpers.foreach_type let all_types = List.concat (List.map ~f:snd qualified_types) -let foreach_module f = - nl (); - List.iter qualified_types ~f:(fun (m, types) -> f m types) - -let foreach_type f = foreach_module (fun m -> List.iter ~f:(f m)) - let foreach_version f = nl (); List.iter Supported_version.all ~f:(fun v -> diff --git a/ast/cinaps/dune b/ast/cinaps/dune index 0b5e453a8..b40d6d415 100644 --- a/ast/cinaps/dune +++ b/ast/cinaps/dune @@ -1,3 +1,3 @@ (library (name ast_cinaps_helpers) - (libraries supported_version)) + (libraries supported_version astlib_cinaps_helpers)) diff --git a/ast/import.ml b/ast/import.ml index 7ed293b81..915535809 100644 --- a/ast/import.ml +++ b/ast/import.ml @@ -4,6 +4,8 @@ It must be opened in all modules, especially the ones coming from the compiler. *) +(*$ open Ast_cinaps_helpers $*) + module Js = Versions.OCaml_500 module Ocaml = Versions.OCaml_current @@ -12,18 +14,43 @@ module Select_ast (Ocaml : Versions.OCaml_version) = struct module Type = struct type ('js, 'ocaml) t = - | Signature - : (Js.Ast.Parsetree.signature, Ocaml.Ast.Parsetree.signature) t + (*$ foreach_type (fun _ s -> + printf + " | %s\n\ + \ : ( Js.Ast.Parsetree.%s,\n\ + \ Ocaml.Ast.Parsetree.%s )\n\ + \ t\n" + (String.capitalize_ascii s) s s + ) + *) | Structure - : (Js.Ast.Parsetree.structure, Ocaml.Ast.Parsetree.structure) t + : ( Js.Ast.Parsetree.structure, + Ocaml.Ast.Parsetree.structure ) + t + | Signature + : ( Js.Ast.Parsetree.signature, + Ocaml.Ast.Parsetree.signature ) + t | Toplevel_phrase : ( Js.Ast.Parsetree.toplevel_phrase, Ocaml.Ast.Parsetree.toplevel_phrase ) t - | Expression - : (Js.Ast.Parsetree.expression, Ocaml.Ast.Parsetree.expression) t | Core_type - : (Js.Ast.Parsetree.core_type, Ocaml.Ast.Parsetree.core_type) t + : ( Js.Ast.Parsetree.core_type, + Ocaml.Ast.Parsetree.core_type ) + t + | Expression + : ( Js.Ast.Parsetree.expression, + Ocaml.Ast.Parsetree.expression ) + t + | Pattern + : ( Js.Ast.Parsetree.pattern, + Ocaml.Ast.Parsetree.pattern ) + t + | Case + : ( Js.Ast.Parsetree.case, + Ocaml.Ast.Parsetree.case ) + t | Type_declaration : ( Js.Ast.Parsetree.type_declaration, Ocaml.Ast.Parsetree.type_declaration ) @@ -36,6 +63,43 @@ module Select_ast (Ocaml : Versions.OCaml_version) = struct : ( Js.Ast.Parsetree.extension_constructor, Ocaml.Ast.Parsetree.extension_constructor ) t + | Class_expr + : ( Js.Ast.Parsetree.class_expr, + Ocaml.Ast.Parsetree.class_expr ) + t + | Class_field + : ( Js.Ast.Parsetree.class_field, + Ocaml.Ast.Parsetree.class_field ) + t + | Class_type + : ( Js.Ast.Parsetree.class_type, + Ocaml.Ast.Parsetree.class_type ) + t + | Class_signature + : ( Js.Ast.Parsetree.class_signature, + Ocaml.Ast.Parsetree.class_signature ) + t + | Class_type_field + : ( Js.Ast.Parsetree.class_type_field, + Ocaml.Ast.Parsetree.class_type_field ) + t + | Module_expr + : ( Js.Ast.Parsetree.module_expr, + Ocaml.Ast.Parsetree.module_expr ) + t + | Module_type + : ( Js.Ast.Parsetree.module_type, + Ocaml.Ast.Parsetree.module_type ) + t + | Signature_item + : ( Js.Ast.Parsetree.signature_item, + Ocaml.Ast.Parsetree.signature_item ) + t + | Structure_item + : ( Js.Ast.Parsetree.structure_item, + Ocaml.Ast.Parsetree.structure_item ) + t +(*$*) | List : ('a, 'b) t -> ('a list, 'b list) t | Pair : ('a, 'b) t * ('c, 'd) t -> ('a * 'c, 'b * 'd) t end @@ -48,14 +112,32 @@ module Select_ast (Ocaml : Versions.OCaml_version) = struct let open Of_ocaml in fun node -> match node with - | Signature -> copy_signature + (*$ foreach_type (fun _ s -> + printf + " | %s -> copy_%s\n" + (String.capitalize_ascii s) s + ) + *) | Structure -> copy_structure + | Signature -> copy_signature | Toplevel_phrase -> copy_toplevel_phrase - | Expression -> copy_expression | Core_type -> copy_core_type + | Expression -> copy_expression + | Pattern -> copy_pattern + | Case -> copy_case | Type_declaration -> copy_type_declaration | Type_extension -> copy_type_extension | Extension_constructor -> copy_extension_constructor + | Class_expr -> copy_class_expr + | Class_field -> copy_class_field + | Class_type -> copy_class_type + | Class_signature -> copy_class_signature + | Class_type_field -> copy_class_type_field + | Module_expr -> copy_module_expr + | Module_type -> copy_module_type + | Signature_item -> copy_signature_item + | Structure_item -> copy_structure_item +(*$*) | List t -> List.map (of_ocaml t) | Pair (a, b) -> let f = of_ocaml a in @@ -66,14 +148,32 @@ module Select_ast (Ocaml : Versions.OCaml_version) = struct let open To_ocaml in fun node -> match node with - | Signature -> copy_signature + (*$ foreach_type (fun _ s -> + printf + " | %s -> copy_%s\n" + (String.capitalize_ascii s) s + ) + *) | Structure -> copy_structure + | Signature -> copy_signature | Toplevel_phrase -> copy_toplevel_phrase - | Expression -> copy_expression | Core_type -> copy_core_type + | Expression -> copy_expression + | Pattern -> copy_pattern + | Case -> copy_case | Type_declaration -> copy_type_declaration | Type_extension -> copy_type_extension | Extension_constructor -> copy_extension_constructor + | Class_expr -> copy_class_expr + | Class_field -> copy_class_field + | Class_type -> copy_class_type + | Class_signature -> copy_class_signature + | Class_type_field -> copy_class_type_field + | Module_expr -> copy_module_expr + | Module_type -> copy_module_type + | Signature_item -> copy_signature_item + | Structure_item -> copy_structure_item +(*$*) | List t -> List.map (to_ocaml t) | Pair (a, b) -> let f = to_ocaml a in diff --git a/ast/versions.ml b/ast/versions.ml index cf7bb2081..58598030b 100644 --- a/ast/versions.ml +++ b/ast/versions.ml @@ -46,6 +46,15 @@ module type Ast = sig type type_declaration type type_extension type extension_constructor + type class_expr + type class_field + type class_type + type class_signature + type class_type_field + type module_expr + type module_type + type signature_item + type structure_item end (*$*) module Config : sig @@ -69,6 +78,15 @@ type 'a _types = 'a constraint 'a type_declaration : _; type_extension : _; extension_constructor : _; + class_expr : _; + class_field : _; + class_type : _; + class_signature : _; + class_type_field : _; + module_expr : _; + module_type : _; + signature_item : _; + structure_item : _; (*$*) > ;; @@ -97,6 +115,24 @@ type 'a get_type_extension = 'x constraint 'a _types = < type_extension : 'x; .. > type 'a get_extension_constructor = 'x constraint 'a _types = < extension_constructor : 'x; .. > +type 'a get_class_expr = + 'x constraint 'a _types = < class_expr : 'x; .. > +type 'a get_class_field = + 'x constraint 'a _types = < class_field : 'x; .. > +type 'a get_class_type = + 'x constraint 'a _types = < class_type : 'x; .. > +type 'a get_class_signature = + 'x constraint 'a _types = < class_signature : 'x; .. > +type 'a get_class_type_field = + 'x constraint 'a _types = < class_type_field : 'x; .. > +type 'a get_module_expr = + 'x constraint 'a _types = < module_expr : 'x; .. > +type 'a get_module_type = + 'x constraint 'a _types = < module_type : 'x; .. > +type 'a get_signature_item = + 'x constraint 'a _types = < signature_item : 'x; .. > +type 'a get_structure_item = + 'x constraint 'a _types = < structure_item : 'x; .. > (*$*) module type OCaml_version = sig @@ -115,6 +151,15 @@ module type OCaml_version = sig type_declaration : Ast.Parsetree.type_declaration; type_extension : Ast.Parsetree.type_extension; extension_constructor : Ast.Parsetree.extension_constructor; + class_expr : Ast.Parsetree.class_expr; + class_field : Ast.Parsetree.class_field; + class_type : Ast.Parsetree.class_type; + class_signature : Ast.Parsetree.class_signature; + class_type_field : Ast.Parsetree.class_type_field; + module_expr : Ast.Parsetree.module_expr; + module_type : Ast.Parsetree.module_type; + signature_item : Ast.Parsetree.signature_item; + structure_item : Ast.Parsetree.structure_item; (*$*) > _types type _ witnesses += Version : types witnesses @@ -135,6 +180,15 @@ struct type_declaration : Ast.Parsetree.type_declaration; type_extension : Ast.Parsetree.type_extension; extension_constructor : Ast.Parsetree.extension_constructor; + class_expr : Ast.Parsetree.class_expr; + class_field : Ast.Parsetree.class_field; + class_type : Ast.Parsetree.class_type; + class_signature : Ast.Parsetree.class_signature; + class_type_field : Ast.Parsetree.class_type_field; + module_expr : Ast.Parsetree.module_expr; + module_type : Ast.Parsetree.module_type; + signature_item : Ast.Parsetree.signature_item; + structure_item : Ast.Parsetree.structure_item; (*$*) > _types type _ witnesses += Version : types witnesses @@ -157,6 +211,15 @@ type 'types ocaml_version = and type Ast.Parsetree.type_declaration = 'types get_type_declaration and type Ast.Parsetree.type_extension = 'types get_type_extension and type Ast.Parsetree.extension_constructor = 'types get_extension_constructor + and type Ast.Parsetree.class_expr = 'types get_class_expr + and type Ast.Parsetree.class_field = 'types get_class_field + and type Ast.Parsetree.class_type = 'types get_class_type + and type Ast.Parsetree.class_signature = 'types get_class_signature + and type Ast.Parsetree.class_type_field = 'types get_class_type_field + and type Ast.Parsetree.module_expr = 'types get_module_expr + and type Ast.Parsetree.module_type = 'types get_module_type + and type Ast.Parsetree.signature_item = 'types get_signature_item + and type Ast.Parsetree.structure_item = 'types get_structure_item (*$*) ) @@ -173,6 +236,15 @@ type ('from, 'to_) migration_functions = { copy_type_declaration: 'from get_type_declaration -> 'to_ get_type_declaration; copy_type_extension: 'from get_type_extension -> 'to_ get_type_extension; copy_extension_constructor: 'from get_extension_constructor -> 'to_ get_extension_constructor; + copy_class_expr: 'from get_class_expr -> 'to_ get_class_expr; + copy_class_field: 'from get_class_field -> 'to_ get_class_field; + copy_class_type: 'from get_class_type -> 'to_ get_class_type; + copy_class_signature: 'from get_class_signature -> 'to_ get_class_signature; + copy_class_type_field: 'from get_class_type_field -> 'to_ get_class_type_field; + copy_module_expr: 'from get_module_expr -> 'to_ get_module_expr; + copy_module_type: 'from get_module_type -> 'to_ get_module_type; + copy_signature_item: 'from get_signature_item -> 'to_ get_signature_item; + copy_structure_item: 'from get_structure_item -> 'to_ get_structure_item; (*$*) } @@ -189,6 +261,15 @@ let migration_identity : ('a, 'a) migration_functions = { copy_type_declaration = id; copy_type_extension = id; copy_extension_constructor = id; + copy_class_expr = id; + copy_class_field = id; + copy_class_type = id; + copy_class_signature = id; + copy_class_type_field = id; + copy_module_expr = id; + copy_module_type = id; + copy_signature_item = id; + copy_structure_item = id; (*$*) } @@ -206,6 +287,15 @@ let migration_compose (ab : ('a, 'b) migration_functions) (bc : ('b, 'c) migrati copy_type_declaration = compose bc.copy_type_declaration ab.copy_type_declaration; copy_type_extension = compose bc.copy_type_extension ab.copy_type_extension; copy_extension_constructor = compose bc.copy_extension_constructor ab.copy_extension_constructor; + copy_class_expr = compose bc.copy_class_expr ab.copy_class_expr; + copy_class_field = compose bc.copy_class_field ab.copy_class_field; + copy_class_type = compose bc.copy_class_type ab.copy_class_type; + copy_class_signature = compose bc.copy_class_signature ab.copy_class_signature; + copy_class_type_field = compose bc.copy_class_type_field ab.copy_class_type_field; + copy_module_expr = compose bc.copy_module_expr ab.copy_module_expr; + copy_module_type = compose bc.copy_module_type ab.copy_module_type; + copy_signature_item = compose bc.copy_signature_item ab.copy_signature_item; + copy_structure_item = compose bc.copy_structure_item ab.copy_structure_item; (*$*) } @@ -226,6 +316,15 @@ module type Migrate_module = sig val copy_type_declaration : From.Parsetree.type_declaration -> To.Parsetree.type_declaration val copy_type_extension : From.Parsetree.type_extension -> To.Parsetree.type_extension val copy_extension_constructor: From.Parsetree.extension_constructor -> To.Parsetree.extension_constructor + val copy_class_expr : From.Parsetree.class_expr -> To.Parsetree.class_expr + val copy_class_field : From.Parsetree.class_field -> To.Parsetree.class_field + val copy_class_type : From.Parsetree.class_type -> To.Parsetree.class_type + val copy_class_signature : From.Parsetree.class_signature -> To.Parsetree.class_signature + val copy_class_type_field : From.Parsetree.class_type_field -> To.Parsetree.class_type_field + val copy_module_expr : From.Parsetree.module_expr -> To.Parsetree.module_expr + val copy_module_type : From.Parsetree.module_type -> To.Parsetree.module_type + val copy_signature_item : From.Parsetree.signature_item -> To.Parsetree.signature_item + val copy_structure_item : From.Parsetree.structure_item -> To.Parsetree.structure_item (*$*) end @@ -248,6 +347,15 @@ struct copy_type_declaration; copy_type_extension; copy_extension_constructor; + copy_class_expr; + copy_class_field; + copy_class_type; + copy_class_signature; + copy_class_type_field; + copy_module_expr; + copy_module_type; + copy_signature_item; + copy_structure_item; (*$*) } end @@ -292,6 +400,15 @@ let immediate_migration (type type_declaration) (type type_extension) (type extension_constructor) + (type class_expr) + (type class_field) + (type class_type) + (type class_signature) + (type class_type_field) + (type module_expr) + (type module_type) + (type signature_item) + (type structure_item) (*$*) ((module A) : < (*$ foreach_type (fun _ s -> printf " %-21s : %s;\n" s s) *) @@ -305,6 +422,15 @@ let immediate_migration type_declaration : type_declaration; type_extension : type_extension; extension_constructor : extension_constructor; + class_expr : class_expr; + class_field : class_field; + class_type : class_type; + class_signature : class_signature; + class_type_field : class_type_field; + module_expr : module_expr; + module_type : module_type; + signature_item : signature_item; + structure_item : structure_item; (*$*) > ocaml_version) direction @@ -330,6 +456,15 @@ let migrate (type type_declaration1) (type type_declaration2) (type type_extension1) (type type_extension2) (type extension_constructor1) (type extension_constructor2) + (type class_expr1) (type class_expr2) + (type class_field1) (type class_field2) + (type class_type1) (type class_type2) + (type class_signature1) (type class_signature2) + (type class_type_field1) (type class_type_field2) + (type module_expr1) (type module_expr2) + (type module_type1) (type module_type2) + (type signature_item1) (type signature_item2) + (type structure_item1) (type structure_item2) (*$*) ((module A) : < (*$ foreach_type (fun _ s -> printf " %-21s : %s1;\n" s s) *) @@ -343,6 +478,15 @@ let migrate type_declaration : type_declaration1; type_extension : type_extension1; extension_constructor : extension_constructor1; + class_expr : class_expr1; + class_field : class_field1; + class_type : class_type1; + class_signature : class_signature1; + class_type_field : class_type_field1; + module_expr : module_expr1; + module_type : module_type1; + signature_item : signature_item1; + structure_item : structure_item1; (*$*) > ocaml_version) ((module B) : < @@ -357,6 +501,15 @@ let migrate type_declaration : type_declaration2; type_extension : type_extension2; extension_constructor : extension_constructor2; + class_expr : class_expr2; + class_field : class_field2; + class_type : class_type2; + class_signature : class_signature2; + class_type_field : class_type_field2; + module_expr : module_expr2; + module_type : module_type2; + signature_item : signature_item2; + structure_item : structure_item2; (*$*) > ocaml_version) : (A.types, B.types) migration_functions @@ -392,6 +545,15 @@ module Convert (A : OCaml_version) (B : OCaml_version) = struct copy_type_declaration; copy_type_extension; copy_extension_constructor; + copy_class_expr; + copy_class_field; + copy_class_type; + copy_class_signature; + copy_class_type_field; + copy_module_expr; + copy_module_type; + copy_signature_item; + copy_structure_item; (*$*) } : (A.types, B.types) migration_functions = migrate (module A) (module B) diff --git a/ast/versions.mli b/ast/versions.mli index 83b2ea78f..782c0a856 100644 --- a/ast/versions.mli +++ b/ast/versions.mli @@ -37,6 +37,15 @@ module type Ast = sig type type_declaration type type_extension type extension_constructor + type class_expr + type class_field + type class_type + type class_signature + type class_type_field + type module_expr + type module_type + type signature_item + type structure_item end (*$*) module Config : sig @@ -60,6 +69,15 @@ type 'a _types = 'a constraint 'a type_declaration : _; type_extension : _; extension_constructor : _; + class_expr : _; + class_field : _; + class_type : _; + class_signature : _; + class_type_field : _; + module_expr : _; + module_type : _; + signature_item : _; + structure_item : _; (*$*) > ;; @@ -98,6 +116,15 @@ module type OCaml_version = sig type_declaration : Ast.Parsetree.type_declaration; type_extension : Ast.Parsetree.type_extension; extension_constructor : Ast.Parsetree.extension_constructor; + class_expr : Ast.Parsetree.class_expr; + class_field : Ast.Parsetree.class_field; + class_type : Ast.Parsetree.class_type; + class_signature : Ast.Parsetree.class_signature; + class_type_field : Ast.Parsetree.class_type_field; + module_expr : Ast.Parsetree.module_expr; + module_type : Ast.Parsetree.module_type; + signature_item : Ast.Parsetree.signature_item; + structure_item : Ast.Parsetree.structure_item; (*$*) > _types @@ -155,6 +182,15 @@ module Convert (A : OCaml_version) (B : OCaml_version) : sig val copy_type_declaration : A.Ast.Parsetree.type_declaration -> B.Ast.Parsetree.type_declaration val copy_type_extension : A.Ast.Parsetree.type_extension -> B.Ast.Parsetree.type_extension val copy_extension_constructor : A.Ast.Parsetree.extension_constructor -> B.Ast.Parsetree.extension_constructor + val copy_class_expr : A.Ast.Parsetree.class_expr -> B.Ast.Parsetree.class_expr + val copy_class_field : A.Ast.Parsetree.class_field -> B.Ast.Parsetree.class_field + val copy_class_type : A.Ast.Parsetree.class_type -> B.Ast.Parsetree.class_type + val copy_class_signature : A.Ast.Parsetree.class_signature -> B.Ast.Parsetree.class_signature + val copy_class_type_field : A.Ast.Parsetree.class_type_field -> B.Ast.Parsetree.class_type_field + val copy_module_expr : A.Ast.Parsetree.module_expr -> B.Ast.Parsetree.module_expr + val copy_module_type : A.Ast.Parsetree.module_type -> B.Ast.Parsetree.module_type + val copy_signature_item : A.Ast.Parsetree.signature_item -> B.Ast.Parsetree.signature_item + val copy_structure_item : A.Ast.Parsetree.structure_item -> B.Ast.Parsetree.structure_item (*$*) end diff --git a/astlib/cinaps/astlib_cinaps_helpers.ml b/astlib/cinaps/astlib_cinaps_helpers.ml index 550c84021..7760b4fd8 100644 --- a/astlib/cinaps/astlib_cinaps_helpers.ml +++ b/astlib/cinaps/astlib_cinaps_helpers.ml @@ -37,3 +37,37 @@ let foreach_version_pair f = | [ _ ] | [] -> () in aux supported_versions + +(* Just for 4.14 <-> 5.00, mostly used by [ast_cinaps_helpers] *) + +let qualified_types = + [ + ( "Parsetree", + [ + "structure"; + "signature"; + "toplevel_phrase"; + "core_type"; + "expression"; + "pattern"; + "case"; + "type_declaration"; + "type_extension"; + "extension_constructor"; + "class_expr"; + "class_field"; + "class_type"; + "class_signature"; + "class_type_field"; + "module_expr"; + "module_type"; + "signature_item"; + "structure_item"; + ] ); + ] + +let foreach_module f = + nl (); + List.iter qualified_types ~f:(fun (m, types) -> f m types) + +let foreach_type f = foreach_module (fun m -> List.iter ~f:(f m)) diff --git a/astlib/migrate_414_500.ml b/astlib/migrate_414_500.ml index 6a81b0736..084692807 100644 --- a/astlib/migrate_414_500.ml +++ b/astlib/migrate_414_500.ml @@ -1,40 +1,90 @@ module From = Ast_414 module To = Ast_500 -let copy_structure : Ast_414.Parsetree.structure -> Ast_500.Parsetree.structure - = - fun x -> x +(*$ open Astlib_cinaps_helpers $*) -let copy_signature : Ast_414.Parsetree.signature -> Ast_500.Parsetree.signature - = - fun x -> x +(*$ foreach_type (fun _ s -> + Printf.printf + "let copy_%s\n\ + \ : Ast_414.Parsetree.%s -> Ast_500.Parsetree.%s\n\ + \ = fun x -> x\n\n" + s s s + ) +*) +let copy_structure + : Ast_414.Parsetree.structure -> Ast_500.Parsetree.structure + = fun x -> x -let copy_toplevel_phrase : - Ast_414.Parsetree.toplevel_phrase -> Ast_500.Parsetree.toplevel_phrase = - fun x -> x +let copy_signature + : Ast_414.Parsetree.signature -> Ast_500.Parsetree.signature + = fun x -> x -let copy_core_type : Ast_414.Parsetree.core_type -> Ast_500.Parsetree.core_type - = - fun x -> x +let copy_toplevel_phrase + : Ast_414.Parsetree.toplevel_phrase -> Ast_500.Parsetree.toplevel_phrase + = fun x -> x -let copy_expression : - Ast_414.Parsetree.expression -> Ast_500.Parsetree.expression = - fun x -> x +let copy_core_type + : Ast_414.Parsetree.core_type -> Ast_500.Parsetree.core_type + = fun x -> x -let copy_pattern : Ast_414.Parsetree.pattern -> Ast_500.Parsetree.pattern = - fun x -> x +let copy_expression + : Ast_414.Parsetree.expression -> Ast_500.Parsetree.expression + = fun x -> x -let copy_case : Ast_414.Parsetree.case -> Ast_500.Parsetree.case = fun x -> x +let copy_pattern + : Ast_414.Parsetree.pattern -> Ast_500.Parsetree.pattern + = fun x -> x -let copy_type_declaration : - Ast_414.Parsetree.type_declaration -> Ast_500.Parsetree.type_declaration = - fun x -> x +let copy_case + : Ast_414.Parsetree.case -> Ast_500.Parsetree.case + = fun x -> x -let copy_type_extension : - Ast_414.Parsetree.type_extension -> Ast_500.Parsetree.type_extension = - fun x -> x +let copy_type_declaration + : Ast_414.Parsetree.type_declaration -> Ast_500.Parsetree.type_declaration + = fun x -> x -let copy_extension_constructor : - Ast_414.Parsetree.extension_constructor -> - Ast_500.Parsetree.extension_constructor = - fun x -> x +let copy_type_extension + : Ast_414.Parsetree.type_extension -> Ast_500.Parsetree.type_extension + = fun x -> x + +let copy_extension_constructor + : Ast_414.Parsetree.extension_constructor -> Ast_500.Parsetree.extension_constructor + = fun x -> x + +let copy_class_expr + : Ast_414.Parsetree.class_expr -> Ast_500.Parsetree.class_expr + = fun x -> x + +let copy_class_field + : Ast_414.Parsetree.class_field -> Ast_500.Parsetree.class_field + = fun x -> x + +let copy_class_type + : Ast_414.Parsetree.class_type -> Ast_500.Parsetree.class_type + = fun x -> x + +let copy_class_signature + : Ast_414.Parsetree.class_signature -> Ast_500.Parsetree.class_signature + = fun x -> x + +let copy_class_type_field + : Ast_414.Parsetree.class_type_field -> Ast_500.Parsetree.class_type_field + = fun x -> x + +let copy_module_expr + : Ast_414.Parsetree.module_expr -> Ast_500.Parsetree.module_expr + = fun x -> x + +let copy_module_type + : Ast_414.Parsetree.module_type -> Ast_500.Parsetree.module_type + = fun x -> x + +let copy_signature_item + : Ast_414.Parsetree.signature_item -> Ast_500.Parsetree.signature_item + = fun x -> x + +let copy_structure_item + : Ast_414.Parsetree.structure_item -> Ast_500.Parsetree.structure_item + = fun x -> x + +(*$*) diff --git a/astlib/migrate_500_414.ml b/astlib/migrate_500_414.ml index ac0f27bdc..6b625e209 100644 --- a/astlib/migrate_500_414.ml +++ b/astlib/migrate_500_414.ml @@ -1,40 +1,90 @@ module From = Ast_500 module To = Ast_414 -let copy_structure : Ast_500.Parsetree.structure -> Ast_414.Parsetree.structure - = - fun x -> x +(*$ open Astlib_cinaps_helpers $*) -let copy_signature : Ast_500.Parsetree.signature -> Ast_414.Parsetree.signature - = - fun x -> x +(*$ foreach_type (fun _ s -> + Printf.printf + "let copy_%s\n\ + \ : Ast_500.Parsetree.%s -> Ast_414.Parsetree.%s\n\ + \ = fun x -> x\n\n" + s s s + ) +*) +let copy_structure + : Ast_500.Parsetree.structure -> Ast_414.Parsetree.structure + = fun x -> x -let copy_toplevel_phrase : - Ast_500.Parsetree.toplevel_phrase -> Ast_414.Parsetree.toplevel_phrase = - fun x -> x +let copy_signature + : Ast_500.Parsetree.signature -> Ast_414.Parsetree.signature + = fun x -> x -let copy_core_type : Ast_500.Parsetree.core_type -> Ast_414.Parsetree.core_type - = - fun x -> x +let copy_toplevel_phrase + : Ast_500.Parsetree.toplevel_phrase -> Ast_414.Parsetree.toplevel_phrase + = fun x -> x -let copy_expression : - Ast_500.Parsetree.expression -> Ast_414.Parsetree.expression = - fun x -> x +let copy_core_type + : Ast_500.Parsetree.core_type -> Ast_414.Parsetree.core_type + = fun x -> x -let copy_pattern : Ast_500.Parsetree.pattern -> Ast_414.Parsetree.pattern = - fun x -> x +let copy_expression + : Ast_500.Parsetree.expression -> Ast_414.Parsetree.expression + = fun x -> x -let copy_case : Ast_500.Parsetree.case -> Ast_414.Parsetree.case = fun x -> x +let copy_pattern + : Ast_500.Parsetree.pattern -> Ast_414.Parsetree.pattern + = fun x -> x -let copy_type_declaration : - Ast_500.Parsetree.type_declaration -> Ast_414.Parsetree.type_declaration = - fun x -> x +let copy_case + : Ast_500.Parsetree.case -> Ast_414.Parsetree.case + = fun x -> x -let copy_type_extension : - Ast_500.Parsetree.type_extension -> Ast_414.Parsetree.type_extension = - fun x -> x +let copy_type_declaration + : Ast_500.Parsetree.type_declaration -> Ast_414.Parsetree.type_declaration + = fun x -> x -let copy_extension_constructor : - Ast_500.Parsetree.extension_constructor -> - Ast_414.Parsetree.extension_constructor = - fun x -> x +let copy_type_extension + : Ast_500.Parsetree.type_extension -> Ast_414.Parsetree.type_extension + = fun x -> x + +let copy_extension_constructor + : Ast_500.Parsetree.extension_constructor -> Ast_414.Parsetree.extension_constructor + = fun x -> x + +let copy_class_expr + : Ast_500.Parsetree.class_expr -> Ast_414.Parsetree.class_expr + = fun x -> x + +let copy_class_field + : Ast_500.Parsetree.class_field -> Ast_414.Parsetree.class_field + = fun x -> x + +let copy_class_type + : Ast_500.Parsetree.class_type -> Ast_414.Parsetree.class_type + = fun x -> x + +let copy_class_signature + : Ast_500.Parsetree.class_signature -> Ast_414.Parsetree.class_signature + = fun x -> x + +let copy_class_type_field + : Ast_500.Parsetree.class_type_field -> Ast_414.Parsetree.class_type_field + = fun x -> x + +let copy_module_expr + : Ast_500.Parsetree.module_expr -> Ast_414.Parsetree.module_expr + = fun x -> x + +let copy_module_type + : Ast_500.Parsetree.module_type -> Ast_414.Parsetree.module_type + = fun x -> x + +let copy_signature_item + : Ast_500.Parsetree.signature_item -> Ast_414.Parsetree.signature_item + = fun x -> x + +let copy_structure_item + : Ast_500.Parsetree.structure_item -> Ast_414.Parsetree.structure_item + = fun x -> x + +(*$*) From e748f504db1ed849771d7736d991444e99688412 Mon Sep 17 00:00:00 2001 From: Antal Spector-Zabusky Date: Thu, 10 Aug 2023 17:28:09 -0400 Subject: [PATCH 02/26] Formatting Signed-off-by: Antal Spector-Zabusky --- .ocamlformat-ignore | 5 ++++- ast/cinaps/ast_cinaps_helpers.ml | 1 - 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/.ocamlformat-ignore b/.ocamlformat-ignore index 64cd1004a..1e7f6979a 100644 --- a/.ocamlformat-ignore +++ b/.ocamlformat-ignore @@ -18,10 +18,13 @@ astlib/ast_413.ml astlib/ast_414.ml astlib/ast_501.ml -# Files that use cinaps to generate bode blocks from other code blocks work well, +# Files that use cinaps to generate code blocks from other code blocks work well, # but files that inject freely formatted code via cinaps must be excluded ast/versions.ml ast/versions.mli +ast/import.ml +astlib/migrate_414_500.ml +astlib/migrate_500_414.ml # Currently our expect-test lexer is too strict for our expect tests to # work well with ocamlformat diff --git a/ast/cinaps/ast_cinaps_helpers.ml b/ast/cinaps/ast_cinaps_helpers.ml index f1d9a0882..daad29607 100644 --- a/ast/cinaps/ast_cinaps_helpers.ml +++ b/ast/cinaps/ast_cinaps_helpers.ml @@ -7,7 +7,6 @@ let nl = Astlib_cinaps_helpers.nl let qualified_types = Astlib_cinaps_helpers.qualified_types let foreach_module = Astlib_cinaps_helpers.foreach_module let foreach_type = Astlib_cinaps_helpers.foreach_type - let all_types = List.concat (List.map ~f:snd qualified_types) let foreach_version f = From 95baecb29a89478d3b48b8ea22cdcf032f45be17 Mon Sep 17 00:00:00 2001 From: Antal Spector-Zabusky Date: Thu, 10 Aug 2023 17:30:28 -0400 Subject: [PATCH 03/26] Support versions below 4.05 Signed-off-by: Antal Spector-Zabusky --- ast/cinaps/ast_cinaps_helpers.ml | 3 +++ ast/cinaps/dune | 2 +- ast/import.ml | 6 +++--- 3 files changed, 7 insertions(+), 4 deletions(-) diff --git a/ast/cinaps/ast_cinaps_helpers.ml b/ast/cinaps/ast_cinaps_helpers.ml index daad29607..ce120a969 100644 --- a/ast/cinaps/ast_cinaps_helpers.ml +++ b/ast/cinaps/ast_cinaps_helpers.ml @@ -3,6 +3,9 @@ include StdLabels include Printf +let capitalize_ascii = Stdppx.String.capitalize_ascii + +(* Reexports from [Astlib_cinaps_helpers] *) let nl = Astlib_cinaps_helpers.nl let qualified_types = Astlib_cinaps_helpers.qualified_types let foreach_module = Astlib_cinaps_helpers.foreach_module diff --git a/ast/cinaps/dune b/ast/cinaps/dune index b40d6d415..4ece19d35 100644 --- a/ast/cinaps/dune +++ b/ast/cinaps/dune @@ -1,3 +1,3 @@ (library (name ast_cinaps_helpers) - (libraries supported_version astlib_cinaps_helpers)) + (libraries stdppx supported_version astlib_cinaps_helpers)) diff --git a/ast/import.ml b/ast/import.ml index 915535809..57168b7fd 100644 --- a/ast/import.ml +++ b/ast/import.ml @@ -20,7 +20,7 @@ module Select_ast (Ocaml : Versions.OCaml_version) = struct \ : ( Js.Ast.Parsetree.%s,\n\ \ Ocaml.Ast.Parsetree.%s )\n\ \ t\n" - (String.capitalize_ascii s) s s + (capitalize_ascii s) s s ) *) | Structure @@ -115,7 +115,7 @@ module Select_ast (Ocaml : Versions.OCaml_version) = struct (*$ foreach_type (fun _ s -> printf " | %s -> copy_%s\n" - (String.capitalize_ascii s) s + (capitalize_ascii s) s ) *) | Structure -> copy_structure @@ -151,7 +151,7 @@ module Select_ast (Ocaml : Versions.OCaml_version) = struct (*$ foreach_type (fun _ s -> printf " | %s -> copy_%s\n" - (String.capitalize_ascii s) s + (capitalize_ascii s) s ) *) | Structure -> copy_structure From d1442b0be9f31f922e3aced222006fa96c13d0ff Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Fri, 30 Jun 2023 16:31:10 -0400 Subject: [PATCH 04/26] Preserve quoted attributes on antiquoted constructs Signed-off-by: Nick Roberts --- metaquot/dune | 2 +- metaquot/ppxlib_metaquot.ml | 134 ++++++++++++++++++-- test/metaquot/test.ml | 243 ++++++++++++++++++++++++++++++++++++ test/metaquot/test_510.ml | 243 ++++++++++++++++++++++++++++++++++++ 4 files changed, 608 insertions(+), 14 deletions(-) diff --git a/metaquot/dune b/metaquot/dune index a85cfb2db..dc1169ce6 100644 --- a/metaquot/dune +++ b/metaquot/dune @@ -4,5 +4,5 @@ (kind ppx_rewriter) (flags (:standard -safe-string)) - (libraries ppxlib ppxlib_traverse_builtins ppxlib_metaquot_lifters) + (libraries astlib ppxlib ppxlib_traverse_builtins ppxlib_metaquot_lifters) (ppx_runtime_libraries ppxlib_ast)) diff --git a/metaquot/ppxlib_metaquot.ml b/metaquot/ppxlib_metaquot.ml index dac00dd60..de0a395cf 100644 --- a/metaquot/ppxlib_metaquot.ml +++ b/metaquot/ppxlib_metaquot.ml @@ -3,11 +3,26 @@ open Ast_builder.Default module E = Extension module A = Ast_pattern +type quoted_attributes = + { quoted_attributes : attributes + (* The attributes that appear quoted, e.g. [@foo] in [%expr [%e e] [@foo]] *) + ; field_name : string + (* The field name where attributes are stored for the kind of ASt the quoted attributes + are placed on, e.g. pexp_attributes. *) + } + module Make (M : sig type result val annotate : result -> core_type -> result - val cast : extension -> result + + val cast + : < attributes : attributes -> result; .. > + (* The instance of the [std_lifters] class being used. *) + -> extension + -> quoted_attributes option + -> result + val location : location -> result val location_stack : (location -> result) option val attributes : (location -> result) option @@ -48,49 +63,75 @@ struct method! expression e = match e.pexp_desc with | Pexp_extension (({ txt = "e"; _ }, _) as ext) -> - self#typed (M.cast ext) "expression" + let attributes = + { quoted_attributes = e.pexp_attributes + ; field_name = "pexp_attributes" + } + in + self#typed (M.cast self ext (Some attributes)) "expression" | _ -> super#expression e method! pattern p = match p.ppat_desc with | Ppat_extension (({ txt = "p"; _ }, _) as ext) -> - self#typed (M.cast ext) "pattern" + let attributes = + { quoted_attributes = p.ppat_attributes + ; field_name = "ppat_attributes" + } + in + self#typed (M.cast self ext (Some attributes)) "pattern" | _ -> super#pattern p method! core_type t = match t.ptyp_desc with | Ptyp_extension (({ txt = "t"; _ }, _) as ext) -> - self#typed (M.cast ext) "core_type" + let attributes = + { quoted_attributes = t.ptyp_attributes + ; field_name = "ptyp_attributes" + } + in + self#typed (M.cast self ext (Some attributes)) "core_type" | _ -> super#core_type t method! module_expr m = match m.pmod_desc with | Pmod_extension (({ txt = "m"; _ }, _) as ext) -> - self#typed (M.cast ext) "module_expr" + let attributes = + { quoted_attributes = m.pmod_attributes + ; field_name = "pmod_attributes" + } + in + self#typed (M.cast self ext (Some attributes)) "module_expr" | _ -> super#module_expr m method! module_type m = match m.pmty_desc with | Pmty_extension (({ txt = "m"; _ }, _) as ext) -> - self#typed (M.cast ext) "module_type" + let attributes = + { quoted_attributes = m.pmty_attributes + ; field_name = "pmty_attributes" + } + in + self#typed (M.cast self ext (Some attributes)) "module_type" | _ -> super#module_type m method! structure_item i = match i.pstr_desc with | Pstr_extension ((({ txt = "i"; _ }, _) as ext), attrs) -> assert_no_attributes attrs; - self#typed (M.cast ext) "structure_item" + self#typed (M.cast self ext None) "structure_item" | _ -> super#structure_item i method! signature_item i = match i.psig_desc with | Psig_extension ((({ txt = "i"; _ }, _) as ext), attrs) -> assert_no_attributes attrs; - self#typed (M.cast ext) "signature_item" + self#typed (M.cast self ext None) "signature_item" | _ -> super#signature_item i end end + module Expr = Make (struct type result = expression @@ -102,11 +143,66 @@ module Expr = Make (struct let annotate e core_type = pexp_constraint ~loc:core_type.ptyp_loc e core_type - let cast ext = + let fresh_name = + let counter = ref 0 in + fun () -> + let var = "_ppx_metaquot_helper_var%d" ^ Int.to_string !counter in + incr counter; + var + + (* Append the quoted attributes to the attributes present on the + antiquoted construct. Take this as example: + + [%expr [%e e] [@attr]] + + Suppose e has pexp_attributes = [attr1]. Then the resulting attributes + are [ attr1; [@attr] ]. The decision to put outer attributes (here, + [@attr]) at the end of the list is consistent with other parts of ppxlib + that accumulate attributes. + *) + let add_quoted_attributes self e { quoted_attributes; field_name } ~loc = + match quoted_attributes with + | [] -> e + | _ :: _ -> + let open Ppxlib_ast.Ast_helper in + let loc = { loc with loc_ghost = true } in + let mkloc x = Located.mk x ~loc in + let var = fresh_name () in + let var_expr = Exp.ident (mkloc (Lident var)) in + let field_name = mkloc (Lident field_name) in + let reified_attrs = self#attributes quoted_attributes in + (* append arg1 arg2 = [%expr Stdlib.List.append [%e arg1] [%e arg2]] *) + let append arg1 arg2 = + Exp.apply + (Exp.ident + (mkloc (Ldot (Ldot (Lident "Stdlib", "List"), "append")))) + [ Nolabel, arg1; Nolabel, arg2 ] + in + (* [%expr + let [%p var] = [%e e] in + { [%e var] with field_name = + [%e append [%expr [%e var].field_name] reified_attrs] + ] + + This comment lies a little bit: field_name is actually some other + literal string. + *) + Exp.let_ + Nonrecursive + [ Vb.mk (Pat.var (mkloc var)) e ] + (Exp.record + [ field_name, append (Exp.field var_expr field_name) reified_attrs ] + (Some var_expr)) + + let cast self ext attrs = match snd ext with - | PStr [ { pstr_desc = Pstr_eval (e, attrs); _ } ] -> - assert_no_attributes attrs; - e + | PStr [ { pstr_desc = Pstr_eval (e, inner_attrs); _ } ] -> + assert_no_attributes inner_attrs; + (match attrs with + | None -> e + | Some quoted_attrs -> + add_quoted_attributes self e quoted_attrs + ~loc:(loc_of_extension ext)) | _ -> Ast_builder.Default.( pexp_extension ~loc:(loc_of_extension ext) @@ -128,7 +224,19 @@ module Patt = Make (struct let annotate p core_type = ppat_constraint ~loc:core_type.ptyp_loc p core_type - let cast ext = + let cast _ ext attrs = + begin + match attrs with + | None -> () + | Some { quoted_attributes; field_name = _ } -> + (* In theory, we could create a pattern where [quoted_attributes] + is consed to the front of [p.ppat_attributes]. But this is + inconsistent with [Expression.add_quoted_attributes], which appends + quoted attributes to the end -- and this wouldn't be a legal + pattern. + *) + assert_no_attributes quoted_attributes + end; match snd ext with | PPat (p, None) -> p | PPat (_, Some e) -> diff --git a/test/metaquot/test.ml b/test/metaquot/test.ml index 148ab4ade..b56d98e5d 100644 --- a/test/metaquot/test.ml +++ b/test/metaquot/test.ml @@ -292,6 +292,243 @@ let _ = [%sig: include S] loc_ghost = true}}] |}] +(* attributes *) + +let _ = + let e = [%expr (() [@attr1])] in + [%expr [%e e] [@attr2]].pexp_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + +let _ = + let p = [%pat? (() [@attr1])] in + [%pat? [%p p] [@attr2]].ppat_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + +let _ = + let t = [%type: (unit [@attr1])] in + [%type: [%t t] [@attr2]].ptyp_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + +let _ = + let extract_module_M m = + match m with + | [%stri module M = [%m? m]] -> m + | _ -> assert false + in + let m = extract_module_M [%stri module M = (struct end [@attr1])] in + (extract_module_M [%stri module M = [%m m] [@attr2]]).pmod_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + +let _ = + let extract_module_ty_S s = + match s with + | [%stri module type S = [%m? s]] -> s + | _ -> assert false + in + let s = extract_module_ty_S [%stri module type S = (sig end [@attr1])] in + (extract_module_ty_S [%stri module type S = [%m s] [@attr2]]).pmty_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + (* mistyped escapes (not producing ASTs at all) *) let _ = [%expr [%e ()]] @@ -328,3 +565,9 @@ Line _, characters 21-23: Error: This expression should not be a unit literal, the expected type is Ppxlib_ast.Ast.signature_item |}] + +let _ = [%expr [%e ()] [@attr]] +[%%expect{| +Line _: +Error: This expression has type unit which is not a record type. +|}] diff --git a/test/metaquot/test_510.ml b/test/metaquot/test_510.ml index a4cae3339..949876c41 100644 --- a/test/metaquot/test_510.ml +++ b/test/metaquot/test_510.ml @@ -292,6 +292,243 @@ let _ = [%sig: include S] loc_ghost = true}}] |}] +(* attributes *) + +let _ = + let e = [%expr (() [@attr1])] in + [%expr [%e e] [@attr2]].pexp_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + +let _ = + let p = [%pat? (() [@attr1])] in + [%pat? [%p p] [@attr2]].ppat_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + +let _ = + let t = [%type: (unit [@attr1])] in + [%type: [%t t] [@attr2]].ptyp_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + +let _ = + let extract_module_M m = + match m with + | [%stri module M = [%m? m]] -> m + | _ -> assert false + in + let m = extract_module_M [%stri module M = (struct end [@attr1])] in + (extract_module_M [%stri module M = [%m m] [@attr2]]).pmod_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + +let _ = + let extract_module_ty_S s = + match s with + | [%stri module type S = [%m? s]] -> s + | _ -> assert false + in + let s = extract_module_ty_S [%stri module type S = (sig end [@attr1])] in + (extract_module_ty_S [%stri module type S = [%m s] [@attr2]]).pmty_attributes +[%%expect{| +- : Ppxlib_ast.Ast.attributes = +[{Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr1"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + {Ppxlib_ast.Ast.attr_name = + {Ppxlib_ast.Ast.txt = "attr2"; + loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}; + attr_payload = Ppxlib_ast.Ast.PStr []; + attr_loc = + {Ppxlib_ast.Ast.loc_start = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_end = + {Ppxlib_ast.Ast.pos_fname = "_none_"; pos_lnum = 1; pos_bol = 0; + pos_cnum = -1}; + loc_ghost = true}}] +|}] + (* mistyped escapes (not producing ASTs at all) *) let _ = [%expr [%e ()]] @@ -332,3 +569,9 @@ Line _, characters 21-23: Error: This expression should not be a unit literal, the expected type is Ppxlib_ast.Ast.signature_item |}] + +let _ = [%expr [%e ()] [@attr]] +[%%expect{| +Line _: +Error: This expression has type unit which is not a record type. +|}] From 19b61b269c089e79ab81d48f1a1c1cc35ccff639 Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Fri, 30 Jun 2023 17:59:40 -0400 Subject: [PATCH 05/26] Use more idiomatic Ast_builder, fix formatting, annotate types better Signed-off-by: Nick Roberts --- metaquot/ppxlib_metaquot.ml | 184 +++++++++++++++++++----------------- test/metaquot/test.ml | 35 ++++++- test/metaquot/test_510.ml | 37 ++++++++ 3 files changed, 169 insertions(+), 87 deletions(-) diff --git a/metaquot/ppxlib_metaquot.ml b/metaquot/ppxlib_metaquot.ml index de0a395cf..3c3e17a94 100644 --- a/metaquot/ppxlib_metaquot.ml +++ b/metaquot/ppxlib_metaquot.ml @@ -3,25 +3,32 @@ open Ast_builder.Default module E = Extension module A = Ast_pattern -type quoted_attributes = - { quoted_attributes : attributes - (* The attributes that appear quoted, e.g. [@foo] in [%expr [%e e] [@foo]] *) - ; field_name : string - (* The field name where attributes are stored for the kind of ASt the quoted attributes - are placed on, e.g. pexp_attributes. *) - } +type quoted_attributes = { + quoted_attributes : attributes; + (* The attributes that appear quoted, e.g. [@foo] in [%expr [%e e] [@foo]] *) + field_name : string; + (* The field name where attributes are stored for the kind of AST the quoted + attributes are placed on, e.g. pexp_attributes. *) +} module Make (M : sig type result val annotate : result -> core_type -> result - val cast - : < attributes : attributes -> result; .. > - (* The instance of the [std_lifters] class being used. *) - -> extension - -> quoted_attributes option - -> result + val cast : + (* The instance of the [std_lifters] class being used. *) + < attributes : attributes -> result + ; typed : result -> string -> result + ; .. > -> + extension -> + quoted_attributes option -> + (* e.g. [expression]; the callee is responsible for calling + [self#typed ast type_name] on the AST to add a type annotation + that constrains its type. + *) + type_name:string -> + result val location : location -> result val location_stack : (location -> result) option @@ -64,74 +71,78 @@ struct match e.pexp_desc with | Pexp_extension (({ txt = "e"; _ }, _) as ext) -> let attributes = - { quoted_attributes = e.pexp_attributes - ; field_name = "pexp_attributes" + { + quoted_attributes = e.pexp_attributes; + field_name = "pexp_attributes"; } in - self#typed (M.cast self ext (Some attributes)) "expression" + M.cast self ext (Some attributes) ~type_name:"expression" | _ -> super#expression e method! pattern p = match p.ppat_desc with | Ppat_extension (({ txt = "p"; _ }, _) as ext) -> let attributes = - { quoted_attributes = p.ppat_attributes - ; field_name = "ppat_attributes" + { + quoted_attributes = p.ppat_attributes; + field_name = "ppat_attributes"; } in - self#typed (M.cast self ext (Some attributes)) "pattern" + M.cast self ext (Some attributes) ~type_name:"pattern" | _ -> super#pattern p method! core_type t = match t.ptyp_desc with | Ptyp_extension (({ txt = "t"; _ }, _) as ext) -> let attributes = - { quoted_attributes = t.ptyp_attributes - ; field_name = "ptyp_attributes" + { + quoted_attributes = t.ptyp_attributes; + field_name = "ptyp_attributes"; } in - self#typed (M.cast self ext (Some attributes)) "core_type" + M.cast self ext (Some attributes) ~type_name:"core_type" | _ -> super#core_type t method! module_expr m = match m.pmod_desc with | Pmod_extension (({ txt = "m"; _ }, _) as ext) -> let attributes = - { quoted_attributes = m.pmod_attributes - ; field_name = "pmod_attributes" + { + quoted_attributes = m.pmod_attributes; + field_name = "pmod_attributes"; } in - self#typed (M.cast self ext (Some attributes)) "module_expr" + M.cast self ext (Some attributes) ~type_name:"module_expr" | _ -> super#module_expr m method! module_type m = match m.pmty_desc with | Pmty_extension (({ txt = "m"; _ }, _) as ext) -> let attributes = - { quoted_attributes = m.pmty_attributes - ; field_name = "pmty_attributes" + { + quoted_attributes = m.pmty_attributes; + field_name = "pmty_attributes"; } in - self#typed (M.cast self ext (Some attributes)) "module_type" + M.cast self ext (Some attributes) ~type_name:"module_type" | _ -> super#module_type m method! structure_item i = match i.pstr_desc with | Pstr_extension ((({ txt = "i"; _ }, _) as ext), attrs) -> assert_no_attributes attrs; - self#typed (M.cast self ext None) "structure_item" + M.cast self ext None ~type_name:"structure_item" | _ -> super#structure_item i method! signature_item i = match i.psig_desc with | Psig_extension ((({ txt = "i"; _ }, _) as ext), attrs) -> assert_no_attributes attrs; - self#typed (M.cast self ext None) "signature_item" + M.cast self ext None ~type_name:"signature_item" | _ -> super#signature_item i end end - module Expr = Make (struct type result = expression @@ -160,49 +171,55 @@ module Expr = Make (struct [@attr]) at the end of the list is consistent with other parts of ppxlib that accumulate attributes. *) - let add_quoted_attributes self e { quoted_attributes; field_name } ~loc = + let add_quoted_attributes self e { quoted_attributes; field_name } ~type_name + ~loc = match quoted_attributes with - | [] -> e + | [] -> self#typed e type_name | _ :: _ -> - let open Ppxlib_ast.Ast_helper in - let loc = { loc with loc_ghost = true } in - let mkloc x = Located.mk x ~loc in - let var = fresh_name () in - let var_expr = Exp.ident (mkloc (Lident var)) in - let field_name = mkloc (Lident field_name) in - let reified_attrs = self#attributes quoted_attributes in - (* append arg1 arg2 = [%expr Stdlib.List.append [%e arg1] [%e arg2]] *) - let append arg1 arg2 = - Exp.apply - (Exp.ident - (mkloc (Ldot (Ldot (Lident "Stdlib", "List"), "append")))) - [ Nolabel, arg1; Nolabel, arg2 ] - in - (* [%expr - let [%p var] = [%e e] in - { [%e var] with field_name = - [%e append [%expr [%e var].field_name] reified_attrs] - ] - - This comment lies a little bit: field_name is actually some other - literal string. - *) - Exp.let_ - Nonrecursive - [ Vb.mk (Pat.var (mkloc var)) e ] - (Exp.record - [ field_name, append (Exp.field var_expr field_name) reified_attrs ] - (Some var_expr)) - - let cast self ext attrs = + let loc = { loc with loc_ghost = true } in + let open (val Ast_builder.make loc) in + let var = fresh_name () in + let var_expr = pexp_ident (Located.mk (Lident var)) in + let field_name = Located.mk (Lident field_name) in + let reified_attrs = self#attributes quoted_attributes in + (* append arg1 arg2 = [%expr Stdlib.List.append [%e arg1] [%e arg2]] *) + let append arg1 arg2 = + pexp_apply + (pexp_ident + (Located.mk (Ldot (Ldot (Lident "Stdlib", "List"), "append")))) + [ (Nolabel, arg1); (Nolabel, arg2) ] + in + (* + Morally, + {[ + let var = ([%expr e] : [%type: type_name]) in + { var + with pexp_attributes = var.pexp_attributes @ [%e reified_attrs ] + } + ]} + *) + pexp_let Nonrecursive + [ + value_binding + ~pat:(ppat_var (Located.mk var)) + ~expr:(self#typed e type_name); + ] + (pexp_record + [ + ( field_name, + append (pexp_field var_expr field_name) reified_attrs ); + ] + (Some var_expr)) + + let cast self ext attrs ~type_name = match snd ext with - | PStr [ { pstr_desc = Pstr_eval (e, inner_attrs); _ } ] -> - assert_no_attributes inner_attrs; - (match attrs with - | None -> e - | Some quoted_attrs -> - add_quoted_attributes self e quoted_attrs - ~loc:(loc_of_extension ext)) + | PStr [ { pstr_desc = Pstr_eval (e, inner_attrs); _ } ] -> ( + assert_no_attributes inner_attrs; + match attrs with + | None -> self#typed e type_name + | Some quoted_attrs -> + add_quoted_attributes self e quoted_attrs ~type_name + ~loc:(loc_of_extension ext)) | _ -> Ast_builder.Default.( pexp_extension ~loc:(loc_of_extension ext) @@ -224,21 +241,18 @@ module Patt = Make (struct let annotate p core_type = ppat_constraint ~loc:core_type.ptyp_loc p core_type - let cast _ ext attrs = - begin - match attrs with - | None -> () - | Some { quoted_attributes; field_name = _ } -> - (* In theory, we could create a pattern where [quoted_attributes] - is consed to the front of [p.ppat_attributes]. But this is - inconsistent with [Expression.add_quoted_attributes], which appends - quoted attributes to the end -- and this wouldn't be a legal - pattern. - *) - assert_no_attributes quoted_attributes - end; + let cast self ext attrs ~type_name = match snd ext with - | PPat (p, None) -> p + | PPat (p, None) -> + (match attrs with + | None -> () + | Some { quoted_attributes; field_name = _ } -> + (* We can't construct a pattern that searches for [quoted_attributes] + at the end of [p]'s attribute list -- the pattern language isn't + expressive enough. Instead, we fail. + *) + assert_no_attributes quoted_attributes); + self#typed p type_name | PPat (_, Some e) -> Ast_builder.Default.( ppat_extension ~loc:e.pexp_loc diff --git a/test/metaquot/test.ml b/test/metaquot/test.ml index b56d98e5d..026e8dc2e 100644 --- a/test/metaquot/test.ml +++ b/test/metaquot/test.ml @@ -566,8 +566,39 @@ Error: This expression should not be a unit literal, the expected type is Ppxlib_ast.Ast.signature_item |}] +(* mistyped escapes (not producing ASTs at all) with attributes *) + let _ = [%expr [%e ()] [@attr]] [%%expect{| -Line _: -Error: This expression has type unit which is not a record type. +Line _, characters 19-21: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.expression +|}] + +let _ = [%pat? [%p ()] [@attr]] +[%%expect{| +Line _, characters 19-21: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.pattern +|}] + +let _ = [%type: [%t ()] [@attr]] +[%%expect{| +Line _, characters 20-22: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.core_type +|}] + +let _ = [%stri module M = [%m ()] [@attr]] +[%%expect{| +Line _, characters 30-32: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.module_expr +|}] + +let _ = [%sigi: module type M = [%m ()] [@attr]] +[%%expect{| +Line _, characters 36-38: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.module_type |}] diff --git a/test/metaquot/test_510.ml b/test/metaquot/test_510.ml index 949876c41..6382a965f 100644 --- a/test/metaquot/test_510.ml +++ b/test/metaquot/test_510.ml @@ -575,3 +575,40 @@ let _ = [%expr [%e ()] [@attr]] Line _: Error: This expression has type unit which is not a record type. |}] + +(* mistyped escapes (not producing ASTs at all) with attributes *) + +let _ = [%expr [%e ()] [@attr]] +[%%expect{| +Line _, characters 19-21: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.expression +|}] + +let _ = [%pat? [%p ()] [@attr]] +[%%expect{| +Line _, characters 19-21: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.pattern +|}] + +let _ = [%type: [%t ()] [@attr]] +[%%expect{| +Line _, characters 20-22: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.core_type +|}] + +let _ = [%stri module M = [%m ()] [@attr]] +[%%expect{| +Line _, characters 30-32: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.module_expr +|}] + +let _ = [%sigi: module type M = [%m ()] [@attr]] +[%%expect{| +Line _, characters 36-38: +Error: This expression should not be a unit literal, the expected type is + Ppxlib_ast.Ast.module_type +|}] From 0e0b552103e00dab5d183bc97a59abbddb3c711b Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Mon, 3 Jul 2023 12:23:05 -0400 Subject: [PATCH 06/26] Fix formatting and 510 test Signed-off-by: Nick Roberts --- metaquot/ppxlib_metaquot.ml | 18 +++++++++--------- test/metaquot/test_510.ml | 11 +++++------ 2 files changed, 14 insertions(+), 15 deletions(-) diff --git a/metaquot/ppxlib_metaquot.ml b/metaquot/ppxlib_metaquot.ml index 3c3e17a94..a7ffe4fe9 100644 --- a/metaquot/ppxlib_metaquot.ml +++ b/metaquot/ppxlib_metaquot.ml @@ -244,15 +244,15 @@ module Patt = Make (struct let cast self ext attrs ~type_name = match snd ext with | PPat (p, None) -> - (match attrs with - | None -> () - | Some { quoted_attributes; field_name = _ } -> - (* We can't construct a pattern that searches for [quoted_attributes] - at the end of [p]'s attribute list -- the pattern language isn't - expressive enough. Instead, we fail. - *) - assert_no_attributes quoted_attributes); - self#typed p type_name + (match attrs with + | None -> () + | Some { quoted_attributes; field_name = _ } -> + (* We can't construct a pattern that searches for [quoted_attributes] + at the end of [p]'s attribute list -- the pattern language isn't + expressive enough. Instead, we fail. + *) + assert_no_attributes quoted_attributes); + self#typed p type_name | PPat (_, Some e) -> Ast_builder.Default.( ppat_extension ~loc:e.pexp_loc diff --git a/test/metaquot/test_510.ml b/test/metaquot/test_510.ml index 6382a965f..4b6e64d18 100644 --- a/test/metaquot/test_510.ml +++ b/test/metaquot/test_510.ml @@ -570,16 +570,11 @@ Error: This expression should not be a unit literal, the expected type is Ppxlib_ast.Ast.signature_item |}] -let _ = [%expr [%e ()] [@attr]] -[%%expect{| -Line _: -Error: This expression has type unit which is not a record type. -|}] - (* mistyped escapes (not producing ASTs at all) with attributes *) let _ = [%expr [%e ()] [@attr]] [%%expect{| + Line _, characters 19-21: Error: This expression should not be a unit literal, the expected type is Ppxlib_ast.Ast.expression @@ -587,6 +582,7 @@ Error: This expression should not be a unit literal, the expected type is let _ = [%pat? [%p ()] [@attr]] [%%expect{| + Line _, characters 19-21: Error: This expression should not be a unit literal, the expected type is Ppxlib_ast.Ast.pattern @@ -594,6 +590,7 @@ Error: This expression should not be a unit literal, the expected type is let _ = [%type: [%t ()] [@attr]] [%%expect{| + Line _, characters 20-22: Error: This expression should not be a unit literal, the expected type is Ppxlib_ast.Ast.core_type @@ -601,6 +598,7 @@ Error: This expression should not be a unit literal, the expected type is let _ = [%stri module M = [%m ()] [@attr]] [%%expect{| + Line _, characters 30-32: Error: This expression should not be a unit literal, the expected type is Ppxlib_ast.Ast.module_expr @@ -608,6 +606,7 @@ Error: This expression should not be a unit literal, the expected type is let _ = [%sigi: module type M = [%m ()] [@attr]] [%%expect{| + Line _, characters 36-38: Error: This expression should not be a unit literal, the expected type is Ppxlib_ast.Ast.module_type From 5acaab50cd98640b4689ea01934cf4beaec25ad2 Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Wed, 5 Jul 2023 09:39:05 -0400 Subject: [PATCH 07/26] Fix CI error for OCaml 4.04 Signed-off-by: Nick Roberts --- metaquot/ppxlib_metaquot.ml | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/metaquot/ppxlib_metaquot.ml b/metaquot/ppxlib_metaquot.ml index a7ffe4fe9..66f7eb6a3 100644 --- a/metaquot/ppxlib_metaquot.ml +++ b/metaquot/ppxlib_metaquot.ml @@ -157,7 +157,7 @@ module Expr = Make (struct let fresh_name = let counter = ref 0 in fun () -> - let var = "_ppx_metaquot_helper_var%d" ^ Int.to_string !counter in + let var = "_ppx_metaquot_helper_var%d" ^ string_of_int !counter in incr counter; var @@ -177,7 +177,8 @@ module Expr = Make (struct | [] -> self#typed e type_name | _ :: _ -> let loc = { loc with loc_ghost = true } in - let open (val Ast_builder.make loc) in + let module Ast_builder_with_loc = (val Ast_builder.make loc) in + let open Ast_builder_with_loc in let var = fresh_name () in let var_expr = pexp_ident (Located.mk (Lident var)) in let field_name = Located.mk (Lident field_name) in From 37ec811b2627f1e2e59c406572a171734c956042 Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Thu, 6 Jul 2023 09:48:58 -0400 Subject: [PATCH 08/26] Add Changes Signed-off-by: Nick Roberts --- CHANGES.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGES.md b/CHANGES.md index d30021973..3186a4e24 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -9,6 +9,8 @@ unreleased - Expose migration/copying/etc. functions for all AST types needed by `Pprintast` (#454, @antalsz) +- Preserve quoted attributes on antiquotes in metaquot (#441, @ncik-roberts) + 0.30.0 (20/06/2023) ------------------- From 63345cee62c0038b29e2874a1d9bfdd6c05e4458 Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Wed, 16 Aug 2023 17:26:50 -0400 Subject: [PATCH 09/26] Respond to review comments Signed-off-by: Nick Roberts --- metaquot/ppxlib_metaquot.ml | 39 ++++++++++++++----------------------- 1 file changed, 15 insertions(+), 24 deletions(-) diff --git a/metaquot/ppxlib_metaquot.ml b/metaquot/ppxlib_metaquot.ml index 66f7eb6a3..2fb96b0f3 100644 --- a/metaquot/ppxlib_metaquot.ml +++ b/metaquot/ppxlib_metaquot.ml @@ -154,13 +154,6 @@ module Expr = Make (struct let annotate e core_type = pexp_constraint ~loc:core_type.ptyp_loc e core_type - let fresh_name = - let counter = ref 0 in - fun () -> - let var = "_ppx_metaquot_helper_var%d" ^ string_of_int !counter in - incr counter; - var - (* Append the quoted attributes to the attributes present on the antiquoted construct. Take this as example: @@ -177,17 +170,16 @@ module Expr = Make (struct | [] -> self#typed e type_name | _ :: _ -> let loc = { loc with loc_ghost = true } in - let module Ast_builder_with_loc = (val Ast_builder.make loc) in - let open Ast_builder_with_loc in - let var = fresh_name () in - let var_expr = pexp_ident (Located.mk (Lident var)) in - let field_name = Located.mk (Lident field_name) in + let var = gen_symbol ~prefix:"_ppx_metaquot_helper_var" () in + let var_expr = pexp_ident ~loc (Located.mk ~loc (Lident var)) in + let field_name = Located.mk ~loc (Lident field_name) in let reified_attrs = self#attributes quoted_attributes in (* append arg1 arg2 = [%expr Stdlib.List.append [%e arg1] [%e arg2]] *) let append arg1 arg2 = - pexp_apply - (pexp_ident - (Located.mk (Ldot (Ldot (Lident "Stdlib", "List"), "append")))) + pexp_apply ~loc + (pexp_ident ~loc + (Located.mk ~loc + (Ldot (Ldot (Lident "Stdlib", "List"), "append")))) [ (Nolabel, arg1); (Nolabel, arg2) ] in (* @@ -199,16 +191,16 @@ module Expr = Make (struct } ]} *) - pexp_let Nonrecursive + pexp_let Nonrecursive ~loc [ - value_binding - ~pat:(ppat_var (Located.mk var)) + value_binding ~loc + ~pat:(ppat_var ~loc (Located.mk ~loc var)) ~expr:(self#typed e type_name); ] - (pexp_record + (pexp_record ~loc [ ( field_name, - append (pexp_field var_expr field_name) reified_attrs ); + append (pexp_field ~loc var_expr field_name) reified_attrs ); ] (Some var_expr)) @@ -222,10 +214,9 @@ module Expr = Make (struct add_quoted_attributes self e quoted_attrs ~type_name ~loc:(loc_of_extension ext)) | _ -> - Ast_builder.Default.( - pexp_extension ~loc:(loc_of_extension ext) - (Location.error_extensionf ~loc:(loc_of_extension ext) - "expression expected")) + pexp_extension ~loc:(loc_of_extension ext) + (Location.error_extensionf ~loc:(loc_of_extension ext) + "expression expected") end) module Patt = Make (struct From 81cf1281c2ef79dcce6bcc124640e6c318c46fae Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Wed, 23 Aug 2023 09:21:05 -0400 Subject: [PATCH 10/26] Update metaquot/ppxlib_metaquot.ml Co-authored-by: panglesd Signed-off-by: Nick Roberts --- metaquot/ppxlib_metaquot.ml | 4 ---- 1 file changed, 4 deletions(-) diff --git a/metaquot/ppxlib_metaquot.ml b/metaquot/ppxlib_metaquot.ml index 2fb96b0f3..1de3eeff4 100644 --- a/metaquot/ppxlib_metaquot.ml +++ b/metaquot/ppxlib_metaquot.ml @@ -239,10 +239,6 @@ module Patt = Make (struct (match attrs with | None -> () | Some { quoted_attributes; field_name = _ } -> - (* We can't construct a pattern that searches for [quoted_attributes] - at the end of [p]'s attribute list -- the pattern language isn't - expressive enough. Instead, we fail. - *) assert_no_attributes quoted_attributes); self#typed p type_name | PPat (_, Some e) -> From fc172c5d99ad7aecf0f49ceabd39967308080f17 Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Thu, 6 Jul 2023 12:24:57 -0400 Subject: [PATCH 11/26] Add support for reserving subnamespaces, like foo.bar Signed-off-by: Nick Roberts --- src/name.ml | 55 ++++++++++++++++++++++++++----- test/driver/attributes/test.ml | 60 ++++++++++++++++++++++++++++++++++ 2 files changed, 106 insertions(+), 9 deletions(-) diff --git a/src/name.ml b/src/name.ml index 413b7bee8..3d9a35cb8 100644 --- a/src/name.ml +++ b/src/name.ml @@ -44,10 +44,15 @@ module Pattern = struct let matches t matched = String.Set.mem matched t.dot_suffixes end -let get_outer_namespace name = +(* On the namespace "a.NAMESPACE", return the pair ("a", NAMESPACE) *) +let split_outer_namespace name = match String.index_opt name '.' with | None -> None - | Some i -> Some (String.sub name ~pos:0 ~len:i) + | Some i -> + let n = String.length name in + let before_dot = String.sub name ~pos:0 ~len:i in + let after_dot = String.sub name ~pos:(i + 1) ~len:(n - i - 1) in + Some (before_dot, after_dot) module Allowlisted = struct (* Allow list the following attributes, as well as all their dot suffixes. @@ -114,8 +119,45 @@ module Allowlisted = struct end module Reserved_namespaces = struct - let tbl : (string, unit) Hashtbl.t = Hashtbl.create 16 - let reserve ns = Hashtbl.add_exn tbl ~key:ns ~data:() + type reserved = (string, sub_namespaces) Hashtbl.t + and sub_namespaces = All | Sub_namespaces of reserved + + (* If [tbl] contains a mapping from "x" to [All], then "x" and all paths that + * start with "x." are reserved with respect to [tbl] + * + * If [tbl] contains a mapping from "x" to [Sub_namespaces tbl'], and P is + * reserved with respect to [tbl'], then all paths "x.P" are reserved with + * respect to [tbl]. + *) + let create_reserved () : reserved = Hashtbl.create 16 + + let rec reserve ns tbl = + match split_outer_namespace ns with + | None -> Hashtbl.add_exn tbl ~key:ns ~data:All + | Some (outer_ns, rest_ns) -> ( + match + Hashtbl.find_or_add tbl outer_ns ~default:(fun () -> + Sub_namespaces (create_reserved ())) + with + | Sub_namespaces rest_tbl -> reserve rest_ns rest_tbl + | All -> ()) + + let rec is_in_reserved_namespaces name tbl = + match split_outer_namespace name with + | Some (ns, rest) -> ( + match Hashtbl.find_opt tbl ns with + | Some (Sub_namespaces rest_tbl) -> + is_in_reserved_namespaces rest rest_tbl + | Some All -> true + | None -> false) + | None -> ( + match Hashtbl.find_opt tbl name with + | Some All -> true + | Some (Sub_namespaces _) | None -> false) + + let tbl = create_reserved () + let reserve ns = reserve ns tbl + let is_in_reserved_namespaces name = is_in_reserved_namespaces name tbl let () = reserve "merlin" let () = reserve "reason" let () = reserve "refmt" (* reason *) @@ -124,11 +166,6 @@ module Reserved_namespaces = struct let () = reserve "metaocaml" let () = reserve "ocamlformat" - let is_in_reserved_namespaces name = - match get_outer_namespace name with - | Some ns -> Hashtbl.mem tbl ns - | None -> Hashtbl.mem tbl name - let check_not_reserved ~kind name = let kind, list = match kind with diff --git a/test/driver/attributes/test.ml b/test/driver/attributes/test.ml index 51b983a96..3b45a0b0e 100644 --- a/test/driver/attributes/test.ml +++ b/test/driver/attributes/test.ml @@ -92,3 +92,63 @@ type t2 = < > Line _, characters 17-20: Error: Attribute `foo' was not used |}] + +(* Reserved Namespaces *) + +(* ppxlib checks that unreserved attributes aren't dropped *) + +let x = (42 [@bar]) +[%%expect{| +Line _, characters 14-17: +Error: Attribute `bar' was silently dropped +|}] + +let x = (42 [@bar.baz]) +[%%expect{| +Line _, characters 14-21: +Error: Attribute `bar.baz' was silently dropped +|}] + +(* But reserving a namespace disables those checks. *) + +let () = Reserved_namespaces.reserve "bar" + +let x = (42 [@bar]) +let x = (42 [@bar.baz]) +[%%expect{| +val x : int = 42 +val x : int = 42 +|}] + +let x = (42 [@bar_not_proper_sub_namespace]) +[%%expect{| +Line _, characters 14-42: +Error: Attribute `bar_not_proper_sub_namespace' was silently dropped +|}] + +(* The namespace reservation process understands dots as namespace + separators. *) + +let () = Reserved_namespaces.reserve "baz.qux" + +let x = (42 [@baz]) +[%%expect{| +Line _, characters 14-17: +Error: Attribute `baz' was silently dropped +|}] + +let x = (42 [@baz.qux]) +[%%expect{| +val x : int = 42 +|}] + +let x = (42 [@baz.qux.quux]) +[%%expect{| +val x : int = 42 +|}] + +let x = (42 [@baz.qux_not_proper_sub_namespace]) +[%%expect{| +Line _, characters 14-46: +Error: Attribute `baz.qux_not_proper_sub_namespace' was silently dropped +|}] From e2e2cdab099a043a3d3b71d93b45e66a53edec43 Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Thu, 6 Jul 2023 12:27:32 -0400 Subject: [PATCH 12/26] Add a test for multiple subnamespaces Signed-off-by: Nick Roberts --- test/driver/attributes/test.ml | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/test/driver/attributes/test.ml b/test/driver/attributes/test.ml index 3b45a0b0e..22bfacdff 100644 --- a/test/driver/attributes/test.ml +++ b/test/driver/attributes/test.ml @@ -152,3 +152,20 @@ let x = (42 [@baz.qux_not_proper_sub_namespace]) Line _, characters 14-46: Error: Attribute `baz.qux_not_proper_sub_namespace' was silently dropped |}] + +(* You can reserve multiple subnamespaces under the same namespace *) + +let () = Reserved_namespaces.reserve "baz.qux2" + +let x = (42 [@baz.qux]) +let x = (42 [@baz.qux2]) +[%%expect{| +val x : int = 42 +val x : int = 42 +|}] + +let x = (42 [@baz.qux3]) +[%%expect{| +Line _, characters 14-22: +Error: Attribute `baz.qux3' was silently dropped +|}] From 434ec2bb161a59b7d3818cb0f6b901ede9a9480a Mon Sep 17 00:00:00 2001 From: Nick Roberts Date: Thu, 6 Jul 2023 13:59:23 -0400 Subject: [PATCH 13/26] Add 5.1.0 tests Signed-off-by: Nick Roberts --- test/driver/attributes/test_510.ml | 89 ++++++++++++++++++++++++++++++ 1 file changed, 89 insertions(+) diff --git a/test/driver/attributes/test_510.ml b/test/driver/attributes/test_510.ml index d8b6bed7e..a007312a7 100644 --- a/test/driver/attributes/test_510.ml +++ b/test/driver/attributes/test_510.ml @@ -103,3 +103,92 @@ type t2 = < > Line _, characters 17-20: Error: Attribute `foo' was not used |}] + +(* Reserved Namespaces *) + +(* ppxlib checks that unreserved attributes aren't dropped *) + +let x = (42 [@bar]) +[%%expect{| + +Line _, characters 14-17: +Error: Attribute `bar' was silently dropped +|}] + +let x = (42 [@bar.baz]) +[%%expect{| + +Line _, characters 14-21: +Error: Attribute `bar.baz' was silently dropped +|}] + +(* But reserving a namespace disables those checks. *) + +let () = Reserved_namespaces.reserve "bar" + +let x = (42 [@bar]) +let x = (42 [@bar.baz]) +[%%expect{| + +val x : int = 42 + +val x : int = 42 +|}] + +let x = (42 [@bar_not_proper_sub_namespace]) +[%%expect{| + +Line _, characters 14-42: +Error: Attribute `bar_not_proper_sub_namespace' was silently dropped +|}] + +(* The namespace reservation process understands dots as namespace + separators. *) + +let () = Reserved_namespaces.reserve "baz.qux" + +let x = (42 [@baz]) +[%%expect{| + +Line _, characters 14-17: +Error: Attribute `baz' was silently dropped +|}] + +let x = (42 [@baz.qux]) +[%%expect{| + +val x : int = 42 +|}] + +let x = (42 [@baz.qux.quux]) +[%%expect{| + +val x : int = 42 +|}] + +let x = (42 [@baz.qux_not_proper_sub_namespace]) +[%%expect{| + +Line _, characters 14-46: +Error: Attribute `baz.qux_not_proper_sub_namespace' was silently dropped +|}] + +(* You can reserve multiple subnamespaces under the same namespace *) + +let () = Reserved_namespaces.reserve "baz.qux2" + +let x = (42 [@baz.qux]) +let x = (42 [@baz.qux2]) +[%%expect{| + +val x : int = 42 + +val x : int = 42 +|}] + +let x = (42 [@baz.qux3]) +[%%expect{| + +Line _, characters 14-22: +Error: Attribute `baz.qux3' was silently dropped +|}] From 3e137a093f62f61287d8ef204ef6ea868166cf12 Mon Sep 17 00:00:00 2001 From: Sonja Heinze Date: Mon, 18 Sep 2023 21:12:22 +0200 Subject: [PATCH 14/26] Add changelog entry Signed-off-by: Sonja Heinze --- CHANGES.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGES.md b/CHANGES.md index 3186a4e24..41310895b 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -11,6 +11,8 @@ unreleased - Preserve quoted attributes on antiquotes in metaquot (#441, @ncik-roberts) +- Attribute namespaces: Fix semantics of reserving multi-component namespaces (#443, @ncik-roberts) + 0.30.0 (20/06/2023) ------------------- From ea93e632a4c4941881e79b91815d3ee46a1183fc Mon Sep 17 00:00:00 2001 From: Sonja Heinze Date: Fri, 23 Jun 2023 12:18:45 +0200 Subject: [PATCH 15/26] Make the migration tests deterministic Signed-off-by: Sonja Heinze --- test/501_migrations/normal_migrations.t | 135 +++++++++++++----------- 1 file changed, 76 insertions(+), 59 deletions(-) diff --git a/test/501_migrations/normal_migrations.t b/test/501_migrations/normal_migrations.t index c9d2c8d42..b49cf18c8 100644 --- a/test/501_migrations/normal_migrations.t +++ b/test/501_migrations/normal_migrations.t @@ -6,92 +6,105 @@ AST's resulting from We only expect a diff in one special case. $ echo "let x : int = 5" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo "let (x) : int = 5" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo "let _ : int = 5" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo "let f : type a b c. a -> b -> c = fun x y -> assert false" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo "let f = (fun (type a) (type b) (type c) -> (fun x y -> assert false : a -> b -> c))" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo "let _ = (fun (type a) (type b) (type c) -> (fun x y -> assert false : a -> b -> c))" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo "let f : type a . a -> a = fun x -> x" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo "let (x, y) : (int * int) = assert false" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo "let f : type a . a = assert false" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo 'let x : [`A] :> [`A | `B] = `A' > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo 'let x : [`A | `B] = (`A : [`A] :> [`A | `B])' > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo 'let x : :> = object method m = 0 method n = 1 end' > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] $ echo 'let x :> = object method m = 0 method n = 1 end' > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] Here might be a problem in the upward migration: the 5.1.0 parser parses the constraint as a pattern constraint. However, the upward migration makes a value binding constraint out of it. $ echo "let ((x,y) : (int*int)) = (assert false: int * int)" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe - 6,25c6,23 - < pattern (file.ml[1,0+4]..[1,0+23]) - < Ppat_constraint - < pattern (file.ml[1,0+5]..[1,0+10]) - < Ppat_tuple - < [ - < pattern (file.ml[1,0+6]..[1,0+7]) - < Ppat_var "x" (file.ml[1,0+6]..[1,0+7]) - < pattern (file.ml[1,0+8]..[1,0+9]) - < Ppat_var "y" (file.ml[1,0+8]..[1,0+9]) - < ] - < core_type (file.ml[1,0+14]..[1,0+21]) - < Ptyp_tuple - < [ - < core_type (file.ml[1,0+14]..[1,0+17]) - < Ptyp_constr "int" (file.ml[1,0+14]..[1,0+17]) - < [] - < core_type (file.ml[1,0+18]..[1,0+21]) - < Ptyp_constr "int" (file.ml[1,0+18]..[1,0+21]) - < [] - < ] - --- - > pattern (file.ml[1,0+5]..[1,0+10]) - > Ppat_tuple - > [ - > pattern (file.ml[1,0+6]..[1,0+7]) - > Ppat_var "x" (file.ml[1,0+6]..[1,0+7]) - > pattern (file.ml[1,0+8]..[1,0+9]) - > Ppat_var "y" (file.ml[1,0+8]..[1,0+9]) - > ] - > core_type (file.ml[1,0+14]..[1,0+21]) - > Ptyp_tuple - > [ - > core_type (file.ml[1,0+14]..[1,0+17]) - > Ptyp_constr "int" (file.ml[1,0+14]..[1,0+17]) - > [] - > core_type (file.ml[1,0+18]..[1,0+21]) - > Ptyp_constr "int" (file.ml[1,0+18]..[1,0+21]) - > [] - > ] + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + @@ -6,20 +6,18 @@ + - pattern (file.ml[1,0+4]..[1,0+23]) + - Ppat_constraint + - pattern (file.ml[1,0+5]..[1,0+10]) + - Ppat_tuple + - [ + - pattern (file.ml[1,0+6]..[1,0+7]) + - Ppat_var "x" (file.ml[1,0+6]..[1,0+7]) + - pattern (file.ml[1,0+8]..[1,0+9]) + - Ppat_var "y" (file.ml[1,0+8]..[1,0+9]) + - ] + - core_type (file.ml[1,0+14]..[1,0+21]) + - Ptyp_tuple + - [ + - core_type (file.ml[1,0+14]..[1,0+17]) + - Ptyp_constr "int" (file.ml[1,0+14]..[1,0+17]) + - [] + - core_type (file.ml[1,0+18]..[1,0+21]) + - Ptyp_constr "int" (file.ml[1,0+18]..[1,0+21]) + - [] + - ] + + pattern (file.ml[1,0+5]..[1,0+10]) + + Ppat_tuple + + [ + + pattern (file.ml[1,0+6]..[1,0+7]) + + Ppat_var "x" (file.ml[1,0+6]..[1,0+7]) + + pattern (file.ml[1,0+8]..[1,0+9]) + + Ppat_var "y" (file.ml[1,0+8]..[1,0+9]) + + ] + + core_type (file.ml[1,0+14]..[1,0+21]) + + Ptyp_tuple + + [ + + core_type (file.ml[1,0+14]..[1,0+17]) + + Ptyp_constr "int" (file.ml[1,0+14]..[1,0+17]) + + [] + + core_type (file.ml[1,0+18]..[1,0+21]) + + Ptyp_constr "int" (file.ml[1,0+18]..[1,0+21]) + + [] + + ] $ echo "let f: type a. a option -> _ = assert false" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] Here we may expect a diff (downwards migrating should yield the same as in the example right above). @@ -100,11 +113,13 @@ However, those case are recoverable. First, both $ echo "let f : 'a . 'a = (fun (type a) -> (assert false : a))" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] and $ echo "let f : type a . a = assert false" > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] are translated to the same 5.0 AST tree. But the locations on the expression constraint and pattern constraint are only the same in the second case. @@ -113,12 +128,14 @@ Thus, we can distinguish between the two. Similarly, the syntactic translation for $ echo 'let x :> [`A | `B] = `A' > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] and $ echo 'let x : [`A | `B] = (`A :> [ `A | `B ] )' > file.ml - $ ./compare_on.exe file.ml ./identity_driver.exe + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] are pretty close: The former is translated to "let (x: ø . [`A | `B]) = (`A :> [`A | `B])" whereas the latter is mapped to "let (x: ø . [`A | `B]) = ((`A :> [`A | `B]): [`A | `B]) ". From 9fbedce3e6fb9a2dcf0625193f71ed3febcc1fdd Mon Sep 17 00:00:00 2001 From: Sonja Heinze Date: Fri, 23 Jun 2023 13:05:00 +0200 Subject: [PATCH 16/26] 5.1 migrations: Add tests for generative functors Signed-off-by: Sonja Heinze --- test/501_migrations/normal_migrations.t | 41 +- test/501_migrations/one_migration.t | 558 +++++++++++++++++++++++ test/501_migrations/reverse_migrations.t | 30 ++ 3 files changed, 628 insertions(+), 1 deletion(-) diff --git a/test/501_migrations/normal_migrations.t b/test/501_migrations/normal_migrations.t index b49cf18c8..faf2289fb 100644 --- a/test/501_migrations/normal_migrations.t +++ b/test/501_migrations/normal_migrations.t @@ -3,7 +3,10 @@ The 501 parsetree contains a parsing modificacion. AST's resulting from 1. parsing on 5.1.0 directly 2. parsing on 5.1.0, migrating down to 5.0.0 and migrating back to 5.1.0 -We only expect a diff in one special case. + +------------------ + +Tests for the Parsetree change for type constraints in value bindings $ echo "let x : int = 5" > file.ml $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" @@ -146,3 +149,39 @@ Let's make sure that in the examples with diffs, the location invariants are still fulfilled. $ echo "let ((x,y) : (int*int)) = (assert false: int * int)" > file.ml $ ./identity_driver.exe -check -locations-check file.ml > /dev/null + +------------------ + +Tests for the Parsetree change for generative functor applications + + + $ cat > file.ml << EOF + > module F () = struct end + > module M = F () + > EOF + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + @@ -14 +14 @@ + - Pmod_apply_unit + + Pmod_apply + @@ -16,0 +17,3 @@ + + module_expr (_none_[0,0+-1]..[0,0+-1]) ghost + + Pmod_structure + + [] + @@ -18,0 +22,3 @@ + +File "_none_", line 1: + +Warning 73 [generative-application-expects-unit]: A generative functor + +should be applied to '()'; using '(struct end)' is deprecated. + + $ cat > file.ml << EOF + > module F () = struct end + > module M = F(struct end) + > EOF + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] + + $ cat > file.ml << EOF + > module F (N : sig end) = struct end + > module M = F (struct end) + > EOF + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] diff --git a/test/501_migrations/one_migration.t b/test/501_migrations/one_migration.t index 70a549b1f..48e844466 100644 --- a/test/501_migrations/one_migration.t +++ b/test/501_migrations/one_migration.t @@ -570,3 +570,561 @@ So let's only keep one example. ((loc_start ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 0))) (loc_end ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 15))) (loc_ghost false))))) + + $ cat > file.ml << EOF + > module F () = struct end + > module M = F () + > EOF + $ ./identity_driver.exe -dparsetree file.ml + (((pstr_desc + (Pstr_attribute + ((attr_name + ((txt ocaml.ppx.context) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_ghost true))))) + (attr_payload + (PStr + (((pstr_desc + (Pstr_eval + ((pexp_desc + (Pexp_record + ((((txt (Lident tool_name)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_constant + (Pconst_string ppxlib_driver + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)) + ()))) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident include_dirs)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident [])) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident load_path)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident [])) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident open_modules)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident [])) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident for_package)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident None)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident debug)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident false)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident use_threads)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident false)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident use_vmthreads)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident false)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident recursive_types)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident false)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident principal)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident false)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident transparent_modules)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident false)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident unboxed_types)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident false)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident unsafe_string)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident false)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ()))) + (((txt (Lident cookies)) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ((pexp_desc + (Pexp_construct + ((txt (Lident [])) + (loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true)))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) + (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ())))) + ())) + (pexp_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_ghost true))) + (pexp_loc_stack ()) (pexp_attributes ())) + ())) + (pstr_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_end + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_ghost true))))))) + (attr_loc + ((loc_start + ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_end ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_ghost true)))))) + (pstr_loc + ((loc_start ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_end ((pos_fname _none_) (pos_lnum 0) (pos_bol 0) (pos_cnum -1))) + (loc_ghost true)))) + ((pstr_desc + (Pstr_module + ((pmb_name + ((txt (F)) + (loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 7))) + (loc_end + ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 8))) + (loc_ghost false))))) + (pmb_expr + ((pmod_desc + (Pmod_functor Unit + ((pmod_desc (Pmod_structure ())) + (pmod_loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 14))) + (loc_end + ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 24))) + (loc_ghost false))) + (pmod_attributes ())))) + (pmod_loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 9))) + (loc_end + ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 24))) + (loc_ghost false))) + (pmod_attributes ()))) + (pmb_attributes ()) + (pmb_loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 0))) + (loc_end ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 24))) + (loc_ghost false)))))) + (pstr_loc + ((loc_start ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 0))) + (loc_end ((pos_fname file.ml) (pos_lnum 1) (pos_bol 0) (pos_cnum 24))) + (loc_ghost false)))) + ((pstr_desc + (Pstr_module + ((pmb_name + ((txt (M)) + (loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 32))) + (loc_end + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 33))) + (loc_ghost false))))) + (pmb_expr + ((pmod_desc + (Pmod_apply + ((pmod_desc + (Pmod_ident + ((txt (Lident F)) + (loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 36))) + (loc_end + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 37))) + (loc_ghost false)))))) + (pmod_loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 36))) + (loc_end + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 37))) + (loc_ghost false))) + (pmod_attributes ())) + ((pmod_desc (Pmod_structure ())) + (pmod_loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 36))) + (loc_end + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 40))) + (loc_ghost false))) + (pmod_attributes ())))) + (pmod_loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 36))) + (loc_end + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 40))) + (loc_ghost false))) + (pmod_attributes ()))) + (pmb_attributes ()) + (pmb_loc + ((loc_start + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 25))) + (loc_end + ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 40))) + (loc_ghost false)))))) + (pstr_loc + ((loc_start ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 25))) + (loc_end ((pos_fname file.ml) (pos_lnum 2) (pos_bol 25) (pos_cnum 40))) + (loc_ghost false))))) diff --git a/test/501_migrations/reverse_migrations.t b/test/501_migrations/reverse_migrations.t index 9120ed91e..0e3dda23f 100644 --- a/test/501_migrations/reverse_migrations.t +++ b/test/501_migrations/reverse_migrations.t @@ -4,6 +4,10 @@ AST's resulting from 1. parsing on 5.0.0 directly 2. parsing on 5.0.0, migrating up to 5.1.0 and migrating back to 5.0.0 +------------------ + +Tests for the Parsetree change for type constraints in value bindings + $ echo "let x : int = 5" > file.ml $ ./compare_on.exe file.ml ./reverse_migrations.exe | grep -v "without_migrations" | grep -v "with_migrations" [1] @@ -162,3 +166,29 @@ the location invariants are still fulfilled. $ echo "let (x) : int = 5" > file.ml $ ./reverse_migrations.exe -check -locations-check file.ml > /dev/null + + +------------------ + +Tests for the Parsetree change for generative functor applications + + $ cat > file.ml << EOF + > module F () = struct end + > module M = F () + > EOF + $ ./compare_on.exe file.ml ./reverse_migrations.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] + + $ cat > file.ml << EOF + > module F () = struct end + > module M = F(struct end) + > EOF + $ ./compare_on.exe file.ml ./reverse_migrations.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] + + $ cat > file.ml << EOF + > module F (N : sig end) = struct end + > module M = F (struct end) + > EOF + $ ./compare_on.exe file.ml ./reverse_migrations.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] From 59e0193ce4a351990823bce27d36d1128e8aa806 Mon Sep 17 00:00:00 2001 From: Paul-Elliot Date: Wed, 6 Sep 2023 12:35:49 +0200 Subject: [PATCH 17/26] Fix generative functor location in 501 -> 500 migration Signed-off-by: Paul-Elliot --- astlib/migrate_501_500.ml | 11 ++++++----- test/501_migrations/normal_migrations.t | 8 +++++--- test/code_path/test_510.ml | 2 +- 3 files changed, 12 insertions(+), 9 deletions(-) diff --git a/astlib/migrate_501_500.ml b/astlib/migrate_501_500.ml index ac53b3203..344a8887f 100644 --- a/astlib/migrate_501_500.ml +++ b/astlib/migrate_501_500.ml @@ -769,14 +769,15 @@ and copy_module_expr : Ast_501.Parsetree.pmod_desc; Ast_501.Parsetree.pmod_loc; Ast_501.Parsetree.pmod_attributes; - } -> + } -> + let loc = copy_location pmod_loc in { - Ast_500.Parsetree.pmod_desc = copy_module_expr_desc pmod_desc; - Ast_500.Parsetree.pmod_loc = copy_location pmod_loc; + Ast_500.Parsetree.pmod_desc = copy_module_expr_desc loc pmod_desc; + Ast_500.Parsetree.pmod_loc = loc; Ast_500.Parsetree.pmod_attributes = copy_attributes pmod_attributes; } -and copy_module_expr_desc : +and copy_module_expr_desc loc : Ast_501.Parsetree.module_expr_desc -> Ast_500.Parsetree.module_expr_desc = function | Ast_501.Parsetree.Pmod_ident x0 -> @@ -793,7 +794,7 @@ and copy_module_expr_desc : Ast_500.Parsetree. { pmod_desc = Pmod_structure []; - pmod_loc = Location.none; + pmod_loc = loc; pmod_attributes = []; } in diff --git a/test/501_migrations/normal_migrations.t b/test/501_migrations/normal_migrations.t index faf2289fb..e2cd2608f 100644 --- a/test/501_migrations/normal_migrations.t +++ b/test/501_migrations/normal_migrations.t @@ -164,11 +164,13 @@ Tests for the Parsetree change for generative functor applications - Pmod_apply_unit + Pmod_apply @@ -16,0 +17,3 @@ - + module_expr (_none_[0,0+-1]..[0,0+-1]) ghost + + module_expr (file.ml[2,25+11]..[2,25+15]) + Pmod_structure + [] - @@ -18,0 +22,3 @@ - +File "_none_", line 1: + @@ -18,0 +22,5 @@ + +File "file.ml", line 2, characters 11-15: + +2 | module M = F () + + ^^^^ +Warning 73 [generative-application-expects-unit]: A generative functor +should be applied to '()'; using '(struct end)' is deprecated. diff --git a/test/code_path/test_510.ml b/test/code_path/test_510.ml index 976aac8ef..7be253b88 100644 --- a/test/code_path/test_510.ml +++ b/test/code_path/test_510.ml @@ -91,7 +91,7 @@ end let _ = let module M = Functor() in !M.code_path [%%expect{| module Functor : functor () -> sig val code_path : string ref end -Line _: +Line _, characters 23-32: Error (warning 73 [generative-application-expects-unit]): A generative functor should be applied to '()'; using '(struct end)' is deprecated. |}] From 1adb6f82f95efa38c928387a7dc1be259c9cb763 Mon Sep 17 00:00:00 2001 From: Paul-Elliot Date: Wed, 6 Sep 2023 13:44:48 +0200 Subject: [PATCH 18/26] Migration: Use an attribute for generative functor migration Two 501 constructs are represented the same way in 500. One of them raise a warning. During the 501 -> 500 -> 501 migration, we need to be careful to keep the warning where it was originally. We use attributes to remember which 501 representation to choose. The reserved namespace is "ppxlib.migration". Signed-off-by: Paul-Elliot --- astlib/migrate_500_501.ml | 22 ++++++++++++++++++++- astlib/migrate_501_500.ml | 25 +++++++++++++++++++++--- src/name.ml | 1 + src/name.mli | 3 ++- test/501_migrations/normal_migrations.t | 14 +------------ test/501_migrations/reverse_migrations.t | 11 +++++++++-- 6 files changed, 56 insertions(+), 20 deletions(-) diff --git a/astlib/migrate_500_501.ml b/astlib/migrate_500_501.ml index bb26e4f38..c3cd1499e 100644 --- a/astlib/migrate_500_501.ml +++ b/astlib/migrate_500_501.ml @@ -751,7 +751,27 @@ and copy_module_expr_desc : Ast_501.Parsetree.Pmod_functor (copy_functor_parameter x0, copy_module_expr x1) | Ast_500.Parsetree.Pmod_apply (x0, x1) -> - Ast_501.Parsetree.Pmod_apply (copy_module_expr x0, copy_module_expr x1) + let x1, is_unit = + match x1.pmod_desc with + | Pmod_structure [] -> + let rec extract_attr acc : Ast_500.Parsetree.attributes -> _ = + function + | [] -> (List.rev acc, true) + | { + attr_name = { txt = "ppxlib.migration.keep_structure"; _ }; + _; + } + :: q -> + (List.rev_append acc q, false) + | hd :: tl -> extract_attr (hd :: acc) tl + in + let pmod_attributes, b = extract_attr [] x1.pmod_attributes in + ({ x1 with pmod_attributes }, b) + | _ -> (x1, false) + in + if is_unit then Ast_501.Parsetree.Pmod_apply_unit (copy_module_expr x0) + else + Ast_501.Parsetree.Pmod_apply (copy_module_expr x0, copy_module_expr x1) | Ast_500.Parsetree.Pmod_constraint (x0, x1) -> Ast_501.Parsetree.Pmod_constraint (copy_module_expr x0, copy_module_type x1) diff --git a/astlib/migrate_501_500.ml b/astlib/migrate_501_500.ml index 344a8887f..da4759ab8 100644 --- a/astlib/migrate_501_500.ml +++ b/astlib/migrate_501_500.ml @@ -769,8 +769,8 @@ and copy_module_expr : Ast_501.Parsetree.pmod_desc; Ast_501.Parsetree.pmod_loc; Ast_501.Parsetree.pmod_attributes; - } -> - let loc = copy_location pmod_loc in + } -> + let loc = copy_location pmod_loc in { Ast_500.Parsetree.pmod_desc = copy_module_expr_desc loc pmod_desc; Ast_500.Parsetree.pmod_loc = loc; @@ -788,7 +788,26 @@ and copy_module_expr_desc loc : Ast_500.Parsetree.Pmod_functor (copy_functor_parameter x0, copy_module_expr x1) | Ast_501.Parsetree.Pmod_apply (x0, x1) -> - Ast_500.Parsetree.Pmod_apply (copy_module_expr x0, copy_module_expr x1) + let x1 = copy_module_expr x1 in + let x1 = + match x1.pmod_desc with + | Pmod_structure [] -> + let pmod_attributes = + { + Ast_500.Parsetree.attr_name = + { + txt = "ppxlib.migration.keep_structure"; + loc = { x1.pmod_loc with loc_ghost = true }; + }; + attr_payload = Ast_500.Parsetree.PStr []; + attr_loc = Location.none; + } + :: x1.pmod_attributes + in + { x1 with pmod_attributes } + | _ -> x1 + in + Ast_500.Parsetree.Pmod_apply (copy_module_expr x0, x1) | Ast_501.Parsetree.Pmod_apply_unit x0 -> let empty_struct = Ast_500.Parsetree. diff --git a/src/name.ml b/src/name.ml index 3d9a35cb8..357b7a1c0 100644 --- a/src/name.ml +++ b/src/name.ml @@ -165,6 +165,7 @@ module Reserved_namespaces = struct let () = reserve "res" (* rescript *) let () = reserve "metaocaml" let () = reserve "ocamlformat" + let () = reserve "ppxlib.migration" let check_not_reserved ~kind name = let kind, list = diff --git a/src/name.mli b/src/name.mli index d8b5c25aa..0ec02f8fd 100644 --- a/src/name.mli +++ b/src/name.mli @@ -109,7 +109,8 @@ module Reserved_namespaces : sig This is here to insure that the rewriter cohabits well with other rewriter or tools (e.g. merlin) which might leave attribute on the AST. - N.B. the "merlin" namespace is reserved by default. *) + N.B. the following namespaces are reserved by default: [merlin], [reason], + [refmt] [ns], [res], [metaocaml], [ocamlformat] and [ppxlib]. *) val is_in_reserved_namespaces : string -> bool end diff --git a/test/501_migrations/normal_migrations.t b/test/501_migrations/normal_migrations.t index e2cd2608f..78965c17a 100644 --- a/test/501_migrations/normal_migrations.t +++ b/test/501_migrations/normal_migrations.t @@ -160,19 +160,7 @@ Tests for the Parsetree change for generative functor applications > module M = F () > EOF $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" - @@ -14 +14 @@ - - Pmod_apply_unit - + Pmod_apply - @@ -16,0 +17,3 @@ - + module_expr (file.ml[2,25+11]..[2,25+15]) - + Pmod_structure - + [] - @@ -18,0 +22,5 @@ - +File "file.ml", line 2, characters 11-15: - +2 | module M = F () - + ^^^^ - +Warning 73 [generative-application-expects-unit]: A generative functor - +should be applied to '()'; using '(struct end)' is deprecated. + [1] $ cat > file.ml << EOF > module F () = struct end diff --git a/test/501_migrations/reverse_migrations.t b/test/501_migrations/reverse_migrations.t index 0e3dda23f..81bee9734 100644 --- a/test/501_migrations/reverse_migrations.t +++ b/test/501_migrations/reverse_migrations.t @@ -179,16 +179,23 @@ Tests for the Parsetree change for generative functor applications $ ./compare_on.exe file.ml ./reverse_migrations.exe | grep -v "without_migrations" | grep -v "with_migrations" [1] +When going up, F(struct end) is turned into F(), which makes the location be lost. +It could be stored in an attribute, or turned into F(struct end [@warning "-73"]). + $ cat > file.ml << EOF > module F () = struct end > module M = F(struct end) > EOF $ ./compare_on.exe file.ml ./reverse_migrations.exe | grep -v "without_migrations" | grep -v "with_migrations" - [1] + @@ -17 +17 @@ + - module_expr (file.ml[2,25+13]..[2,25+23]) + + module_expr (file.ml[2,25+11]..[2,25+24]) $ cat > file.ml << EOF > module F (N : sig end) = struct end > module M = F (struct end) > EOF $ ./compare_on.exe file.ml ./reverse_migrations.exe | grep -v "without_migrations" | grep -v "with_migrations" - [1] + @@ -20 +20 @@ + - module_expr (file.ml[2,36+14]..[2,36+24]) + + module_expr (file.ml[2,36+11]..[2,36+25]) From f882c41bb29022f8695a31671a7431ddb0e59579 Mon Sep 17 00:00:00 2001 From: Paul-Elliot Date: Wed, 6 Sep 2023 14:05:23 +0200 Subject: [PATCH 19/26] test is now identical between 510 and earlier Signed-off-by: Paul-Elliot --- test/code_path/dune | 22 +----- test/code_path/test_510.ml | 152 ------------------------------------- 2 files changed, 1 insertion(+), 173 deletions(-) delete mode 100644 test/code_path/test_510.ml diff --git a/test/code_path/dune b/test/code_path/dune index b88be2aec..06179b3fc 100644 --- a/test/code_path/dune +++ b/test/code_path/dune @@ -1,9 +1,7 @@ (rule (alias runtest) (enabled_if - (and - (>= %{ocaml_version} "4.10.0") - (< %{ocaml_version} "5.1.0"))) + (>= %{ocaml_version} "4.10.0")) (deps (:test test.ml) (package ppxlib)) @@ -13,21 +11,3 @@ (progn (run expect-test %{test}) (diff? %{test} %{test}.corrected))))) - -(rule - (alias runtest) - (enabled_if - (>= %{ocaml_version} "5.1.0")) - (deps - (:test test.ml) - (:t test_510.ml) - (package ppxlib)) - (action - (chdir - %{project_root} - (progn - (run mv %{t} %{t}.old) - (run cp %{test} %{t}) - (run expect-test %{t}) - (run mv %{t}.old %{t}) - (diff? %{t} %{t}.corrected))))) diff --git a/test/code_path/test_510.ml b/test/code_path/test_510.ml deleted file mode 100644 index 7be253b88..000000000 --- a/test/code_path/test_510.ml +++ /dev/null @@ -1,152 +0,0 @@ -open Ppxlib - -let sexp_of_code_path code_path = - Sexplib0.Sexp.message - "code_path" - [ "main_module_name", Sexplib0.Sexp_conv.sexp_of_string (Code_path.main_module_name code_path) - ; "submodule_path", Sexplib0.Sexp_conv.sexp_of_list Sexplib0.Sexp_conv.sexp_of_string (Code_path.submodule_path code_path) - ; "enclosing_module", Sexplib0.Sexp_conv.sexp_of_string (Code_path.enclosing_module code_path) - ; "enclosing_value", Sexplib0.Sexp_conv.sexp_of_option Sexplib0.Sexp_conv.sexp_of_string (Code_path.enclosing_value code_path) - ; "value", Sexplib0.Sexp_conv.sexp_of_option Sexplib0.Sexp_conv.sexp_of_string (Code_path.value code_path) - ; "fully_qualified_path", Sexplib0.Sexp_conv.sexp_of_string (Code_path.fully_qualified_path code_path) - ] - -let () = - Driver.register_transformation "test" - ~extensions:[ - Extension.V3.declare "code_path" - Expression - Ast_pattern.(pstr nil) - (fun ~ctxt -> - let loc = Expansion_context.Extension.extension_point_loc ctxt in - let code_path = Expansion_context.Extension.code_path ctxt in - Ast_builder.Default.estring ~loc - (Sexplib0.Sexp.to_string (sexp_of_code_path code_path))) - ] -[%%expect{| -val sexp_of_code_path : Code_path.t -> Sexplib0.Sexp.t = -|}] - -let s = - let module A = struct - module A' = struct - let a = - let module B = struct - module B' = struct - let b = - let module C = struct - module C' = struct - let c = [%code_path] - end - end - in C.C'.c - end - end - in B.B'.b - end - end - in A.A'.a -;; -[%%expect{| -val s : string = - "(code_path(main_module_name Test_510)(submodule_path())(enclosing_module C')(enclosing_value(c))(value(s))(fully_qualified_path Test_510.s))" -|}] - -let module M = struct - let m = [%code_path] - end - in - M.m -[%%expect{| -- : string = -"(code_path(main_module_name Test_510)(submodule_path())(enclosing_module M)(enclosing_value(m))(value())(fully_qualified_path Test_510))" -|}] - -module Outer = struct - module Inner = struct - let code_path = [%code_path] - end -end -let _ = Outer.Inner.code_path -[%%expect{| -module Outer : sig module Inner : sig val code_path : string end end -- : string = -"(code_path(main_module_name Test_510)(submodule_path(Outer Inner))(enclosing_module Inner)(enclosing_value(code_path))(value(code_path))(fully_qualified_path Test_510.Outer.Inner.code_path))" -|}] - -module Functor() = struct - let code_path = ref "" - module _ = struct - let x = - let module First_class = struct - code_path := [%code_path] - end in - let module _ = First_class in - () - ;; - - ignore x - end -end -let _ = let module M = Functor() in !M.code_path -[%%expect{| -module Functor : functor () -> sig val code_path : string ref end -Line _, characters 23-32: -Error (warning 73 [generative-application-expects-unit]): A generative functor -should be applied to '()'; using '(struct end)' is deprecated. -|}] - -module Actual = struct - let code_path = [%code_path] -end [@enter_module Dummy] -let _ = Actual.code_path -[%%expect{| - -module Actual : sig val code_path : string end - -- : string = -"(code_path(main_module_name Test_510)(submodule_path(Actual Dummy))(enclosing_module Dummy)(enclosing_value(code_path))(value(code_path))(fully_qualified_path Test_510.Actual.Dummy.code_path))" -|}] - -module Ignore_me = struct - let code_path = [%code_path] -end [@@do_not_enter_module] -let _ = Ignore_me.code_path -[%%expect{| - -module Ignore_me : sig val code_path : string end - -- : string = -"(code_path(main_module_name Test_510)(submodule_path())(enclosing_module Test_510)(enclosing_value(code_path))(value(code_path))(fully_qualified_path Test_510.code_path))" -|}] - -let _ = - (let module Ignore_me = struct - let code_path = [%code_path] - end - in - Ignore_me.code_path) - [@do_not_enter_module] -[%%expect{| - -- : string = -"(code_path(main_module_name Test_510)(submodule_path())(enclosing_module Test_510)(enclosing_value(code_path))(value())(fully_qualified_path Test_510))" -|}] - -let _ = ([%code_path] [@ppxlib.enter_value dummy]) -[%%expect{| - -- : string = -"(code_path(main_module_name Test_510)(submodule_path())(enclosing_module Test_510)(enclosing_value(dummy))(value(dummy))(fully_qualified_path Test_510.dummy))" -|}] - -let _ = - let ignore_me = [%code_path] - [@@do_not_enter_value] - in - ignore_me -[%%expect{| - -- : string = -"(code_path(main_module_name Test_510)(submodule_path())(enclosing_module Test_510)(enclosing_value())(value())(fully_qualified_path Test_510))" -|}] From a66679b86bbbb2409783d2a9b73823603792c35e Mon Sep 17 00:00:00 2001 From: Paul-Elliot Date: Tue, 19 Sep 2023 09:08:24 +0200 Subject: [PATCH 20/26] 501<->500 migration: review comments Signed-off-by: Paul-Elliot --- astlib/migrate_501_500.ml | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/astlib/migrate_501_500.ml b/astlib/migrate_501_500.ml index da4759ab8..b77cc75a4 100644 --- a/astlib/migrate_501_500.ml +++ b/astlib/migrate_501_500.ml @@ -792,15 +792,16 @@ and copy_module_expr_desc loc : let x1 = match x1.pmod_desc with | Pmod_structure [] -> + let loc = { x1.pmod_loc with loc_ghost = true } in let pmod_attributes = { Ast_500.Parsetree.attr_name = { txt = "ppxlib.migration.keep_structure"; - loc = { x1.pmod_loc with loc_ghost = true }; + loc; }; attr_payload = Ast_500.Parsetree.PStr []; - attr_loc = Location.none; + attr_loc = loc; } :: x1.pmod_attributes in From 08aed5c61740a821aa984779e03457d472ed4ac5 Mon Sep 17 00:00:00 2001 From: Paul-Elliot Date: Tue, 19 Sep 2023 09:12:23 +0200 Subject: [PATCH 21/26] 501<->500 migration: changelog entry Signed-off-by: Paul-Elliot --- CHANGES.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/CHANGES.md b/CHANGES.md index 41310895b..54560e8c3 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -1,6 +1,10 @@ unreleased ------------------- +- Fix support for OCaml 5.1: migrated code preserves generative + functor warnings, without creating more. Locations are better + preserved. (#432, @pitag-ha, @panglesd) + - Driver: Add `-unused-code-warnings` command-line flag. (#444, @ceastlund) - Add `?warning` flag to `Deriving.Generator.make`. (#440, @jacksonzou123 via @ceastlund) From a0d0c351abdc37de4cf10de4a825ce2d92d65d8d Mon Sep 17 00:00:00 2001 From: Paul-Elliot Date: Tue, 19 Sep 2023 09:15:12 +0200 Subject: [PATCH 22/26] 501<->500 migrations: adding a test with attributes Signed-off-by: Paul-Elliot --- test/501_migrations/normal_migrations.t | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/test/501_migrations/normal_migrations.t b/test/501_migrations/normal_migrations.t index 78965c17a..8e0de5f89 100644 --- a/test/501_migrations/normal_migrations.t +++ b/test/501_migrations/normal_migrations.t @@ -162,6 +162,13 @@ Tests for the Parsetree change for generative functor applications $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" [1] + $ cat > file.ml << EOF + > module F () = struct end + > module M = F [@attr1] () [@attr2] + > EOF + $ ./compare_on.exe file.ml ./identity_driver.exe | grep -v "without_migrations" | grep -v "with_migrations" + [1] + $ cat > file.ml << EOF > module F () = struct end > module M = F(struct end) From 4026b795d9b9bd44beaf11b790a7f9a26fc0aa63 Mon Sep 17 00:00:00 2001 From: Paul-Elliot Date: Tue, 19 Sep 2023 10:02:53 +0200 Subject: [PATCH 23/26] formatting Signed-off-by: Paul-Elliot --- astlib/migrate_501_500.ml | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/astlib/migrate_501_500.ml b/astlib/migrate_501_500.ml index b77cc75a4..a0a776e1c 100644 --- a/astlib/migrate_501_500.ml +++ b/astlib/migrate_501_500.ml @@ -792,14 +792,11 @@ and copy_module_expr_desc loc : let x1 = match x1.pmod_desc with | Pmod_structure [] -> - let loc = { x1.pmod_loc with loc_ghost = true } in + let loc = { x1.pmod_loc with loc_ghost = true } in let pmod_attributes = { Ast_500.Parsetree.attr_name = - { - txt = "ppxlib.migration.keep_structure"; - loc; - }; + { txt = "ppxlib.migration.keep_structure"; loc }; attr_payload = Ast_500.Parsetree.PStr []; attr_loc = loc; } From 4d0a186f7047a0512bf10abab06bffb81fdd2f69 Mon Sep 17 00:00:00 2001 From: Nicolas van Kempen Date: Wed, 20 Sep 2023 11:12:49 +0100 Subject: [PATCH 24/26] Fix minor error in README Signed-off-by: Nicolas van Kempen --- examples/simple-extension-rewriter/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/simple-extension-rewriter/README.md b/examples/simple-extension-rewriter/README.md index 69eba73de..2f0c087bf 100644 --- a/examples/simple-extension-rewriter/README.md +++ b/examples/simple-extension-rewriter/README.md @@ -7,7 +7,7 @@ as a string. E.g., assuming we set `MY_VAR="foo"`, it will turn: ```ocaml -let () = print_string [%get_env "foo"] +let () = print_string [%get_env "MY_VAR"] ``` into: From e765a30151347f8044ce077d103d3828db8d5409 Mon Sep 17 00:00:00 2001 From: Sonja Heinze Date: Thu, 21 Sep 2023 15:09:46 +0200 Subject: [PATCH 25/26] Prepare release 0.31.0 Signed-off-by: Sonja Heinze --- CHANGES.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGES.md b/CHANGES.md index 54560e8c3..931467031 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -1,4 +1,4 @@ -unreleased +0.31.0 (2023/09/21) ------------------- - Fix support for OCaml 5.1: migrated code preserves generative From af1756cfcfb6ebabd2ebf2234109cd50fce3f2f8 Mon Sep 17 00:00:00 2001 From: dianaoigo <127844136+dianaoigo@users.noreply.github.com> Date: Tue, 26 Sep 2023 13:00:38 +0300 Subject: [PATCH 26/26] Add API to consider attributes as flag (#408) Signed-off-by: dianaoigo Co-authored-by: Paul-Elliot --- CHANGES.md | 5 ++++ src/attribute.ml | 17 ++++++++++++++ src/attribute.mli | 18 +++++++++++++++ test/driver/attributes/test.ml | 33 ++++++++++++++++++++++++++ test/driver/attributes/test_510.ml | 37 ++++++++++++++++++++++++++++++ 5 files changed, 110 insertions(+) diff --git a/CHANGES.md b/CHANGES.md index 931467031..bffcc50a6 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -1,3 +1,8 @@ +unreleased +------------------- + +- Add API to manipulate attributes that are used as flags (#404, @dianaoigo) + 0.31.0 (2023/09/21) ------------------- diff --git a/src/attribute.ml b/src/attribute.ml index 07cfa808a..2672715fa 100644 --- a/src/attribute.ml +++ b/src/attribute.ml @@ -297,6 +297,13 @@ let declare_with_attr_loc name context pattern k = declare_with_all_args name context pattern (fun ~attr_loc ~name_loc:_ -> k ~attr_loc) +type 'a flag = ('a, unit) t + +let declare_flag name context = + let payload_pattern = Ast_pattern.(pstr nil) in + let continuation ~attr_loc:_ ~name_loc:_ = () in + declare_with_all_args name context payload_pattern continuation + module Attribute_table = Caml.Hashtbl.Make (struct type t = string loc @@ -356,6 +363,16 @@ let get t ?mark_as_seen:do_mark_as_seen x = get_res t ?mark_as_seen:do_mark_as_seen x |> Result.handle_error ~f:(fun (err, _) -> Location.Error.raise err) +let has_flag_res t ?mark_as_seen x = + match get_res ?mark_as_seen t x with + | Ok (Some ()) -> Ok true + | Ok None -> Ok false + | Error _ as e -> e + +let has_flag t ?mark_as_seen x = + has_flag_res t ?mark_as_seen x + |> Result.handle_error ~f:(fun (err, _) -> Location.Error.raise err) + let consume_res t x = let open Result in let attrs = Context.get_attributes t.context x in diff --git a/src/attribute.mli b/src/attribute.mli index 12f8d487d..fad528f23 100644 --- a/src/attribute.mli +++ b/src/attribute.mli @@ -135,6 +135,13 @@ val declare_with_attr_loc : ('a, 'c) t (** Same as [declare] but the callback receives the location of the attribute. *) +type 'a flag = ('a, unit) t +(** Types for attributes without payload. *) + +val declare_flag : string -> 'a Context.t -> 'a flag +(** Same as {!declare}, but the payload is expected to be empty. It is supposed + to be used in conjunction with {!has_flag}. *) + val name : _ t -> string val context : ('a, _) t -> 'a Context.t @@ -150,6 +157,17 @@ val get : ('a, 'b) t -> ?mark_as_seen:bool (** default [true] *) -> 'a -> 'b option (** See {!get_res}. Raises a located error if the attribute is duplicated *) +val has_flag_res : + 'a flag -> + ?mark_as_seen:bool (** default [true] *) -> + 'a -> + (bool, Location.Error.t NonEmptyList.t) result +(** Answers whether the given flag is attached as an attribute. See {!get_res} + for the meaning of [mark_as_seen]. *) + +val has_flag : 'a flag -> ?mark_as_seen:bool (** default [true] *) -> 'a -> bool +(** See {!has_flag_res}. Raises a located error if the attribute is duplicated. *) + val consume_res : ('a, 'b) t -> 'a -> (('a * 'b) option, Location.Error.t NonEmptyList.t) result (** [consume_res t x] returns the value associated to attribute [t] on [x] if diff --git a/test/driver/attributes/test.ml b/test/driver/attributes/test.ml index 22bfacdff..6e5f1bb7d 100644 --- a/test/driver/attributes/test.ml +++ b/test/driver/attributes/test.ml @@ -169,3 +169,36 @@ let x = (42 [@baz.qux3]) Line _, characters 14-22: Error: Attribute `baz.qux3' was silently dropped |}] + +(* Testing flags *) + +let flag = Attribute.declare_flag "flag" Attribute.Context.expression +[%%expect{| +val flag : expression Attribute.flag = +|}] + +let replace_flagged = object + inherit Ast_traverse.map as super + + method! expression e = + match Attribute.has_flag_res flag e with + | Ok true -> Ast_builder.Default.estring ~loc:e.pexp_loc "Found flag" + | Ok false -> super#expression e + | Error (err, _) -> Ast_builder.Default.estring ~loc:e.pexp_loc (Location.Error.message err) +end +[%%expect{| +val replace_flagged : Ast_traverse.map = +|}] + +let () = + Driver.register_transformation "" ~impl:replace_flagged#structure + +let e1 = "flagged" [@flag] +[%%expect{| +val e1 : string = "Found flag" +|}] + +let e1 = "flagged" [@flag 12] +[%%expect{| +val e1 : string = "[] expected" +|}] diff --git a/test/driver/attributes/test_510.ml b/test/driver/attributes/test_510.ml index a007312a7..7f96affa5 100644 --- a/test/driver/attributes/test_510.ml +++ b/test/driver/attributes/test_510.ml @@ -192,3 +192,40 @@ let x = (42 [@baz.qux3]) Line _, characters 14-22: Error: Attribute `baz.qux3' was silently dropped |}] + +(* Testing flags *) + +let flag = Attribute.declare_flag "flag" Attribute.Context.expression +[%%expect{| + +val flag : expression Attribute.flag = +|}] + +let replace_flagged = object + inherit Ast_traverse.map as super + + method! expression e = + match Attribute.has_flag_res flag e with + | Ok true -> Ast_builder.Default.estring ~loc:e.pexp_loc "Found flag" + | Ok false -> super#expression e + | Error (err, _) -> Ast_builder.Default.estring ~loc:e.pexp_loc (Location.Error.message err) +end +[%%expect{| + +val replace_flagged : Ast_traverse.map = +|}] + +let () = + Driver.register_transformation "" ~impl:replace_flagged#structure + +let e1 = "flagged" [@flag] +[%%expect{| + +val e1 : string = "Found flag" +|}] + +let e1 = "flagged" [@flag 12] +[%%expect{| + +val e1 : string = "[] expected" +|}]