Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

experiment: attach optional migration expression to actor (class) as actor [exp]? #4812

Open
wants to merge 54 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
54 commits
Select commit Hold shift + click to select a range
03476d6
attach optional migration expression to actor (class) as actor [exp]?
crusso Dec 10, 2024
67425f7
first draft of logic
crusso Dec 13, 2024
ee545f5
test (not working)
crusso Dec 13, 2024
bb81d3f
working with loose ends
crusso Dec 17, 2024
e3621d6
working (without type-checking)
crusso Dec 18, 2024
5b75bd9
add typechecking (wip)
crusso Dec 18, 2024
bb965a5
Apply suggestions from code review
crusso Dec 18, 2024
7f5b963
cleanup
crusso Dec 18, 2024
bc23bcb
Merge branch 'claudio/migration' of github.com:dfinity/motoko into cl…
crusso Dec 18, 2024
406d244
ocamlformat
crusso Dec 18, 2024
7d4737e
eop changes (WIP)
crusso Dec 19, 2024
9964bb3
add rts_in_install prim; working
crusso Dec 19, 2024
9ca998c
extend signatures to record pre and post when required
crusso Dec 20, 2024
d28733b
fix printer
crusso Dec 20, 2024
c23fd70
fix printer
crusso Dec 20, 2024
8202cb0
refine versioning
crusso Jan 8, 2025
45b6c07
fix grammar
crusso Jan 8, 2025
858826e
remove fix me
crusso Jan 8, 2025
76b5faf
Claudio/migration eop sigs refactor (#4842)
crusso Jan 8, 2025
19487b1
simplify; delete trailing whitespace
crusso Jan 8, 2025
568d090
Update test/run-drun/upgrade-migration/Migration1.mo
crusso Jan 8, 2025
20be12c
check for hash collisions in induced pre signature
crusso Jan 8, 2025
1906f91
refine static checks on migration function:
crusso Jan 8, 2025
ebe108d
implement scoping properly
crusso Jan 9, 2025
86272d3
refactor to use syntactic exp_opt@
crusso Jan 10, 2025
91a73b5
update test output
crusso Jan 10, 2025
079700b
add test for lexical scoping
crusso Jan 10, 2025
ebe63ca
refactor typechecking code
crusso Jan 10, 2025
ea99627
tidy some todos
crusso Jan 10, 2025
55b56fd
loose ends
crusso Jan 10, 2025
d7b67c2
renamings@
crusso Jan 10, 2025
2b686f8
Update src/mo_types/type.ml
crusso Jan 10, 2025
c70ed43
adjust definedness (not too sure about this tbh)
crusso Jan 10, 2025
1a89789
Merge branch 'claudio/migration' of github.com:dfinity/motoko into cl…
crusso Jan 10, 2025
9077c9c
custom error codes
crusso Jan 10, 2025
edffd72
missing test
crusso Jan 10, 2025
b9c844b
implement interprete rts_in_install prim; check migration exxpression…
crusso Jan 13, 2025
6c918b2
simplify upgrade-migration.drun; add corresponding upgrade-class-migr…
crusso Jan 13, 2025
4383f81
add comments
crusso Jan 13, 2025
45ffad5
port to class tests
crusso Jan 13, 2025
a2f7ea4
fix tests
crusso Jan 13, 2025
f3994a4
update test output
crusso Jan 14, 2025
9bc4b78
positive tests (derived from drun/upgrade-migration.drun
crusso Jan 15, 2025
843043a
add more cmp test and correct calculation of pre-sig to remove migrat…
crusso Jan 15, 2025
85c46e9
fix typo in test
crusso Jan 15, 2025
cb1416d
Apply suggestions from code review
crusso Jan 15, 2025
e8f7b2c
remove negative test since eop doesn't allow field deletion
crusso Jan 15, 2025
21f1f23
remove debug spew
crusso Jan 15, 2025
9cc961b
Merge branch 'master' into claudio/migration
crusso Jan 21, 2025
0467131
test migration expression ingored on (re)install
crusso Jan 21, 2025
543e53b
Update src/mo_frontend/typing.ml
crusso Jan 21, 2025
1a47278
Update src/mo_frontend/typing.ml
crusso Jan 21, 2025
3d8803e
Revert "Update src/mo_frontend/typing.ml"
crusso Jan 21, 2025
9e8b2e4
Revert "Update src/mo_frontend/typing.ml"
crusso Jan 21, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion doc/md/examples/grammar.txt
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@

<obj_sort> ::=
'object'
'persistent'? 'actor'
'persistent'? 'actor' ('[' <exp> ']')?
Copy link
Contributor

@luc-blaeser luc-blaeser Jan 24, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I like the syntax although some users might be confuse with array syntax. An alternative could be ('migrate' <exp>)? or, not sure, if we want to keep existing reserved keywords maybe('with' <exp>)?.

'module'

<query> ::=
Expand Down
14 changes: 11 additions & 3 deletions src/docs/extract.ml
Original file line number Diff line number Diff line change
Expand Up @@ -140,7 +140,7 @@ struct
_;
} -> (
match rhs with
| Source.{ it = Syntax.ObjBlockE (sort, _, fields); _ } ->
| Source.{ it = Syntax.ObjBlockE (sort, _, _, fields); _ } ->
let mk_field_xref xref = mk_xref (Xref.XClass (name, xref)) in
Some
( mk_xref (Xref.XType name),
Expand All @@ -155,7 +155,7 @@ struct
)
| Source.{ it = Syntax.VarD ({ it = name; _ }, rhs); _ } -> (
match rhs with
| Source.{ it = Syntax.ObjBlockE (sort, _, fields); _ } ->
| Source.{ it = Syntax.ObjBlockE (sort, _, _, fields); _ } ->
let mk_field_xref xref = mk_xref (Xref.XClass (name, xref)) in
Some
( mk_xref (Xref.XType name),
Expand Down Expand Up @@ -184,7 +184,15 @@ struct
{
it =
Syntax.ClassD
(shared_pat, name, type_args, ctor, _, obj_sort, _, fields);
( shared_pat,
exp_opt,
name,
type_args,
ctor,
_,
obj_sort,
_,
fields );
_;
} ->
let mk_field_xref xref = mk_xref (Xref.XClass (name.it, xref)) in
Expand Down
4 changes: 2 additions & 2 deletions src/docs/namespace.ml
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ let from_module =
| Syntax.ExpD _ -> acc
| Syntax.LetD
( { it = Syntax.VarP id; _ },
{ it = Syntax.ObjBlockE (_, _, decs); _ },
{ it = Syntax.ObjBlockE (_, _, _, decs); _ },
_ ) ->
let mk_nested x = mk_xref (Xref.XNested (id.it, x)) in
{
Expand Down Expand Up @@ -69,7 +69,7 @@ let from_module =
(mk_xref (Xref.XValue id.it), None)
acc.values;
}
| Syntax.ClassD (_, id, _, _, _, _, _, _) ->
| Syntax.ClassD (_, _, id, _, _, _, _, _, _) ->
{
acc with
types = StringMap.add id.it (mk_xref (Xref.XType id.it)) acc.types;
Expand Down
2 changes: 1 addition & 1 deletion src/ir_def/check_ir.ml
Original file line number Diff line number Diff line change
Expand Up @@ -241,7 +241,7 @@ let rec check_typ env typ : unit =
if not (Lib.List.is_strictly_ordered T.compare_field fields) then
error env no_region "variant type's fields are not distinct and sorted %s" (T.string_of_typ typ)
| T.Mut typ ->
error env no_region "unexpected T.Mut"
error env no_region "unexpected T.Mut %s" (T.string_of_typ typ)
| T.Typ c ->
error env no_region "unexpected T.Typ"

Expand Down
34 changes: 34 additions & 0 deletions src/ir_def/construct.ml
Original file line number Diff line number Diff line change
Expand Up @@ -790,8 +790,42 @@ let objE sort typ_flds flds =
in
go [] [] [] flds


let recordE flds = objE T.Object [] flds

let objectE sort flds (tfs : T.field list) =
let rec go ds fields = function
| [] ->
blockE
(List.rev ds)
(newObjE sort fields
(T.Obj (sort, List.sort T.compare_field tfs)))
| (lab, exp)::flds ->
let v, typ, ds =
match T.lookup_val_field_opt lab tfs with
| None -> assert false
| Some typ ->
if T.is_mut typ
then
let v = fresh_var lab typ in
v, typ, varD v exp :: ds
else
match exp.it with
| VarE (Const, v) ->
var v typ, typ, ds
| _ ->
let v = fresh_var lab typ in
v, typ, letD v exp :: ds
in
let field = {
it = {name = lab; var = id_of_var v};
at = no_region;
note = typ
} in
go ds (field::fields) flds
in
go [] [] flds

let check_call_perform_status success mk_failure =
ifE
(callE
Expand Down
2 changes: 2 additions & 0 deletions src/ir_def/construct.mli
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,8 @@ val (-*-) : exp -> exp -> exp (* application *)

val objE : obj_sort -> (lab * con) list -> (lab * exp) list -> exp

val objectE : obj_sort -> (lab * exp) list -> field list -> exp

(* Records *)

val recordE : (lab * exp) list -> exp
Expand Down
2 changes: 1 addition & 1 deletion src/languageServer/declaration_index.ml
Original file line number Diff line number Diff line change
Expand Up @@ -251,7 +251,7 @@ let populate_definitions (project_root : string) (libs : Syntax.lib list)
let is_type_def dec_field =
match dec_field.it.Syntax.dec.it with
| Syntax.TypD (typ_id, _, _) -> Some typ_id
| Syntax.ClassD (_, typ_id, _, _, _, _, _, _) -> Some typ_id
| Syntax.ClassD (_, _, typ_id, _, _, _, _, _, _) -> Some typ_id
| _ -> None
in
let extract_binders env (pat : Syntax.pat) = gather_pat env pat in
Expand Down
84 changes: 69 additions & 15 deletions src/lowering/desugar.ml
Original file line number Diff line number Diff line change
Expand Up @@ -90,8 +90,8 @@ and exp' at note = function
(breakE "!" (nullE()))
(* case ? v : *)
(varP v) (varE v) ty).it
| S.ObjBlockE (s, (self_id_opt, _), dfs) ->
obj_block at s self_id_opt dfs note.Note.typ
| S.ObjBlockE (s, exp_opt, (self_id_opt, _), dfs) ->
obj_block at s exp_opt self_id_opt dfs note.Note.typ
| S.ObjE (bs, efs) ->
obj note.Note.typ efs bs
| S.TagE (c, e) -> (tagE c.it (exp e)).it
Expand Down Expand Up @@ -326,12 +326,12 @@ and mut m = match m.it with
| S.Const -> Ir.Const
| S.Var -> Ir.Var

and obj_block at s self_id dfs obj_typ =
and obj_block at s exp_opt self_id dfs obj_typ =
match s.it with
| T.Object | T.Module ->
build_obj at s.it self_id dfs obj_typ
| T.Actor ->
build_actor at [] self_id dfs obj_typ
build_actor at [] exp_opt self_id dfs obj_typ
| T.Memory -> assert false

and build_field {T.lab; T.typ;_} =
Expand Down Expand Up @@ -532,7 +532,7 @@ and export_runtime_information self_id =
)],
[{ it = I.{ name = lab; var = v }; at = no_region; note = typ }])

and build_actor at ts self_id es obj_typ =
and build_actor at ts exp_opt self_id es obj_typ =
let candid = build_candid ts obj_typ in
let fs = build_fields obj_typ in
let es = List.filter (fun ef -> is_not_typD ef.it.S.dec) es in
Expand All @@ -550,8 +550,60 @@ and build_actor at ts self_id es obj_typ =
let state = fresh_var "state" (T.Mut (T.Opt ty)) in
let get_state = fresh_var "getState" (T.Func(T.Local, T.Returns, [], [], [ty])) in
let ds = List.map (fun mk_d -> mk_d get_state) mk_ds in
let migration = match exp_opt with
| None -> primE (I.ICStableRead ty) [] (* as before *)
| Some exp0 ->
let e = exp exp0 in
let [@warning "-8"] (_s,_c, [], [dom], [rng]) = T.as_func (exp0.note.S.note_typ) in
let [@warning "-8"] (T.Object, dom_fields) = T.as_obj dom in
let [@warning "-8"] (T.Object, rng_fields) = T.as_obj rng in
ifE (primE (Ir.RelPrim (T.nat, Operator.EqOp)) [
primE (I.OtherPrim "rts_stable_memory_size") [];
natE Numerics.Nat.zero])
(primE (I.ICStableRead ty) [])
(let fields' =
List.map
(fun (i,t) ->
T.{lab = i; typ = T.Opt (T.as_immut t); src = T.empty_src})
((List.map (fun T.{lab;typ;_} -> (lab,typ)) dom_fields) @
(List.filter_map
(fun (i,t) ->
match T.lookup_val_field_opt i dom_fields with
| Some t -> None (* ignore overriden *)
| None -> Some (i, t) (* retain others *))
ids)) in
let ty' = T.Obj (T.Memory, List.sort T.compare_field fields') in
let v = fresh_var "v" ty' in
let v_dom = fresh_var "v_dom" dom in
let v_rng = fresh_var "v_rng" rng in
letE v (primE (I.ICStableRead ty') [])
(letE v_dom
(objectE T.Object
(List.map (fun T.{lab=i;typ=t;_} ->
let vi = fresh_var ("v_"^i) (T.as_immut t) in
(i, switch_optE (dotE (varE v) i (T.Opt (T.as_immut t)))
(primE (Ir.OtherPrim "trap")
[textE (Printf.sprintf
"stable variable `%s` of type `%s` expected but not found" i (T.string_of_typ t))])
(varP vi) (varE vi)
(T.as_immut t))) dom_fields) dom_fields)
(letE v_rng (callE e [] (varE v_dom))
(objectE T.Memory
(List.map (fun T.{lab=i;typ=t;_} ->
i,
match T.lookup_val_field_opt i rng_fields with
(* produced by migration *)
| Some t -> optE (dotE (varE v_rng) i (T.as_immut t)) (* wrap in ?_*)
| None ->
(* not produced by migration *)
match T.lookup_val_field_opt i dom_fields with
| Some t -> nullE() (* consumed by migration (not produced) *)
(*TBR: could also reuse if compatible *)
| None -> dotE (varE v) i t)
fields) fields))))
in
let ds =
varD state (optE (primE (I.ICStableRead ty) []))
varD state (optE migration)
::
nary_funcD get_state []
(let v = fresh_var "v" ty in
Expand Down Expand Up @@ -812,7 +864,8 @@ and dec' at n = function
end
| S.VarD (i, e) -> I.VarD (i.it, e.note.S.note_typ, exp e)
| S.TypD _ -> assert false
| S.ClassD (sp, id, tbs, p, _t_opt, s, self_id, dfs) ->
| S.ClassD (sp, exp_opt, id, tbs, p, _t_opt, s, self_id, dfs) ->
(* TODO exp_opt *)
let id' = {id with note = ()} in
let sort, _, _, _, _ = Type.as_func n.S.note_typ in
let op = match sp.it with
Expand All @@ -839,13 +892,13 @@ and dec' at n = function
let (_, _, obj_typ) = T.as_async rng_typ in
let c = Cons.fresh T.default_scope_var (T.Abs ([], T.scope_bound)) in
asyncE T.Fut (typ_arg c T.Scope T.scope_bound) (* TBR *)
(wrap { it = obj_block at s (Some self_id) dfs (T.promote obj_typ);
(wrap { it = obj_block at s exp_opt (Some self_id) dfs (T.promote obj_typ);
at = at;
note = Note.{def with typ = obj_typ } })
(List.hd inst)
else
wrap
{ it = obj_block at s (Some self_id) dfs rng_typ;
{ it = obj_block at s exp_opt (Some self_id) dfs rng_typ;
at = at;
note = Note.{ def with typ = rng_typ } }
in
Expand Down Expand Up @@ -1023,7 +1076,7 @@ let import_compiled_class (lib : S.comp_unit) wasm : import_declaration =
let f = lib.note.filename in
let { body; _ } = lib.it in
let id = match body.it with
| S.ActorClassU (_, id, _, _, _, _, _) -> id.it
| S.ActorClassU (_, _, id, _, _, _, _, _) -> id.it
| _ -> assert false
in
let fun_typ = T.normalize body.note.S.note_typ in
Expand Down Expand Up @@ -1118,7 +1171,8 @@ let transform_unit_body (u : S.comp_unit_body) : Ir.comp_unit =
I.LibU ([], {
it = build_obj u.at T.Module self_id fields u.note.S.note_typ;
at = u.at; note = typ_note u.note})
| S.ActorClassU (sp, typ_id, _tbs, p, _, self_id, fields) ->
| S.ActorClassU (sp, exp_opt, typ_id, _tbs, p, _, self_id, fields) ->
(* TODO exp_opt *)
let fun_typ = u.note.S.note_typ in
let op = match sp.it with
| T.Local -> None
Expand All @@ -1134,7 +1188,7 @@ let transform_unit_body (u : S.comp_unit_body) : Ir.comp_unit =
T.promote rng
| _ -> assert false
in
let actor_expression = build_actor u.at ts (Some self_id) fields obj_typ in
let actor_expression = build_actor u.at ts exp_opt (Some self_id) fields obj_typ in
let e = wrap {
it = actor_expression;
at = no_region;
Expand All @@ -1145,8 +1199,8 @@ let transform_unit_body (u : S.comp_unit_body) : Ir.comp_unit =
I.ActorU (Some args, ds, fs, u, t)
| _ -> assert false
end
| S.ActorU (self_id, fields) ->
let actor_expression = build_actor u.at [] self_id fields u.note.S.note_typ in
| S.ActorU (exp_opt, self_id, fields) ->
let actor_expression = build_actor u.at [] exp_opt self_id fields u.note.S.note_typ in
begin match actor_expression with
| I.ActorE (ds, fs, u, t) ->
I.ActorU (None, ds, fs, u, t)
Expand Down Expand Up @@ -1182,7 +1236,7 @@ let import_unit (u : S.comp_unit) : import_declaration =
raise (Invalid_argument "Desugar: Cannot import actor")
| I.ActorU (Some as_, ds, fs, up, actor_t) ->
let id = match body.it with
| S.ActorClassU (_, id, _, _, _, _, _) -> id.it
| S.ActorClassU (_, _, id, _, _, _, _, _) -> id.it
| _ -> assert false
in
let s, cntrl, tbs, ts1, ts2 = T.as_func t in
Expand Down
10 changes: 7 additions & 3 deletions src/mo_def/arrange.ml
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,10 @@ module Make (Cfg : Config) = struct
| FromCandidE e -> "FromCandidE" $$ [exp e]
| TupE es -> "TupE" $$ exps es
| ProjE (e, i) -> "ProjE" $$ [exp e; Atom (string_of_int i)]
| ObjBlockE (s, nt, dfs) -> "ObjBlockE" $$ [obj_sort s;
| ObjBlockE (s, po, nt, dfs) -> "ObjBlockE" $$ [obj_sort s;
(match po with
| None -> Atom "_"
| Some e -> exp e);
match nt with
| None, None -> Atom "_"
| None, Some t -> typ t
Expand Down Expand Up @@ -267,8 +270,9 @@ module Make (Cfg : Config) = struct
| VarD (x, e) -> "VarD" $$ [id x; exp e]
| TypD (x, tp, t) ->
"TypD" $$ [id x] @ List.map typ_bind tp @ [typ t]
| ClassD (sp, x, tp, p, rt, s, i', dfs) ->
"ClassD" $$ shared_pat sp :: id x :: List.map typ_bind tp @ [
| ClassD (sp, po, x, tp, p, rt, s, i', dfs) ->
"ClassD" $$ shared_pat sp :: id x :: List.map typ_bind tp @ [
(match po with None -> Atom "_" | Some e -> exp e);
pat p;
(match rt with None -> Atom "_" | Some t -> typ t);
obj_sort s; id i'
Expand Down
Loading
Loading