diff --git a/src/stages/1-ast_imperative/combinators.mli b/src/stages/1-ast_imperative/combinators.mli index a35375610..75b65404f 100644 --- a/src/stages/1-ast_imperative/combinators.mli +++ b/src/stages/1-ast_imperative/combinators.mli @@ -49,6 +49,7 @@ val t_michelson_pair : ?loc:Location.t -> type_expression -> michelson_prct_anno val t_operator : ?loc:Location.t -> type_operator -> type_expression list -> type_expression result val t_set : ?loc:Location.t -> type_expression -> type_expression +val t_contract : ?loc:Location.t -> type_expression -> type_expression val make_e : ?loc:Location.t -> expression_content -> expression diff --git a/src/test/test.ml b/src/test/test.ml index c21411092..01d8a78f6 100644 --- a/src/test/test.ml +++ b/src/test/test.ml @@ -18,5 +18,6 @@ let () = Hash_lock_tests.main ; Time_lock_repeat_tests.main ; Pledge_tests.main ; + Tzip12_tests.main ; ] ; () diff --git a/src/test/tzip12_tests.ml b/src/test/tzip12_tests.ml new file mode 100644 index 000000000..db5996c9e --- /dev/null +++ b/src/test/tzip12_tests.ml @@ -0,0 +1,190 @@ +open Trace +open Test_helpers + +let file_FA12 = "./contracts/FA1.2.ligo" +let mfile_FA12 = "./contracts/FA1.2.mligo" +let refile_FA12 = "./contracts/FA1.2.religo" + +let type_file f s = + let%bind typed,state = Ligo.Compile.Utils.type_file f s (Contract "main") in + ok @@ (typed,state) + +let get_program f st = + let s = ref None in + fun () -> match !s with + | Some s -> ok s + | None -> ( + let%bind program = type_file f st in + s := Some program ; + ok program + ) + +let compile_main f s () = + let%bind typed_prg,_ = get_program f s () in + let%bind mini_c_prg = Ligo.Compile.Of_typed.compile typed_prg in + let%bind michelson_prg = Ligo.Compile.Of_mini_c.aggregate_and_compile_contract mini_c_prg "main" in + let%bind (_contract: Tezos_utils.Michelson.michelson) = + (* fails if the given entry point is not a valid contract *) + Ligo.Compile.Of_michelson.build_contract michelson_prg in + ok () + +open Ast_imperative + + +let (sender , contract) = + let open Proto_alpha_utils.Memory_proto_alpha in + let id = List.nth dummy_environment.identities 0 in + let kt = id.implicit_contract in + Protocol.Alpha_context.Contract.to_b58check kt , kt + +let external_contract = + let open Proto_alpha_utils.Memory_proto_alpha in + let id = List.nth dummy_environment.identities 4 in + let kh = id.public_key_hash in + Tezos_utils.Signature.Public_key_hash.to_string kh + +let from_ = e_address @@ addr 5 +let to_ = e_address @@ addr 2 +let sender = e_address @@ sender +let external_contract = e_annotation (e_constant C_IMPLICIT_ACCOUNT [e_key_hash external_contract]) (t_contract (t_nat ())) + +let transfer f s () = + let%bind program,_ = get_program f s () in + let storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 100); (to_, e_nat 100)]); + ("allowances", e_big_map [(e_pair sender from_, e_nat 100)]); + ("total_amount",e_nat 300); + ] in + let parameter = e_record_ez [("address_from", from_);("address_to",to_); ("value",e_nat 10)] in + let new_storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 90); (to_, e_nat 110)]); + ("allowances", e_big_map [(e_pair sender from_, e_nat 90)]); + ("total_amount",e_nat 300); + ] in + let input = e_pair parameter storage in + let expected = e_pair (e_typed_list [] (t_operation ())) new_storage in + let options = Proto_alpha_utils.Memory_proto_alpha.make_options () in + expect_eq program ~options "transfer" input expected + +let transfer_not_e_allowance f s () = + let%bind program,_ = get_program f s () in + let storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 100); (to_, e_nat 100)]); + ("allowances", e_big_map [(e_pair sender from_, e_nat 0)]); + ("total_amount",e_nat 300); + ] in + let parameter = e_record_ez [("address_from", from_);("address_to",to_); ("value",e_nat 10)] in + let input = e_pair parameter storage in + let options = Proto_alpha_utils.Memory_proto_alpha.make_options () in + expect_string_failwith ~options program "transfer" input + "Not Enough Allowance" + +let transfer_not_e_balance f s () = + let%bind program,_ = get_program f s () in + let storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 0); (to_, e_nat 100)]); + ("allowances", e_big_map [(e_pair sender from_, e_nat 100)]); + ("total_amount",e_nat 300); + ] in + let parameter = e_record_ez [("address_from", from_);("address_to",to_); ("value",e_nat 10)] in + let input = e_pair parameter storage in + let options = Proto_alpha_utils.Memory_proto_alpha.make_options () in + expect_string_failwith ~options program "transfer" input + "Not Enough Balance" + +let approve f s () = + let%bind program,_ = get_program f s () in + let storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 100); (to_, e_nat 100)]); + ("allowances", e_big_map [(e_pair from_ sender, e_nat 0)]); + ("total_amount",e_nat 300); + ] in + let parameter = e_record_ez [("spender", from_);("value",e_nat 100)] in + let new_storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 100); (to_, e_nat 100)]); + ("allowances", e_big_map [(e_pair from_ sender, e_nat 100)]); + ("total_amount",e_nat 300); + ] in + let input = e_pair parameter storage in + let expected = e_pair (e_typed_list [] (t_operation ())) new_storage in + let options = Proto_alpha_utils.Memory_proto_alpha.make_options () in + expect_eq program ~options "approve" input expected + +let approve_unsafe f s () = + let%bind program,_ = get_program f s () in + let storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 100); (to_, e_nat 100)]); + ("allowances", e_big_map [(e_pair from_ sender, e_nat 100)]); + ("total_amount",e_nat 300); + ] in + let parameter = e_record_ez [("spender", from_);("value",e_nat 100)] in + let input = e_pair parameter storage in + let options = Proto_alpha_utils.Memory_proto_alpha.make_options () in + expect_string_failwith ~options program "approve" input + "Unsafe Allowance Change" + +let get_allowance f s () = + let%bind program,_ = get_program f s () in + let storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 100); (to_, e_nat 100)]); + ("allowances", e_big_map [(e_pair from_ sender, e_nat 100)]); + ("total_amount",e_nat 300); + ] in + let parameter = e_record_ez [("owner", from_);("spender",sender); ("callback", external_contract)] in + let input = e_pair parameter storage in + let expected = e_pair (e_typed_list [] (t_operation ())) storage in + let options = Proto_alpha_utils.Memory_proto_alpha.make_options () in + expect_eq program ~options "getAllowance" input expected + +let get_balance f s () = + let%bind program,_ = get_program f s () in + let storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 100); (to_, e_nat 100)]); + ("allowances", e_big_map [(e_pair from_ sender, e_nat 100)]); + ("total_amount",e_nat 300); + ] in + let parameter = e_record_ez [("owner", from_);("callback", external_contract)] in + let input = e_pair parameter storage in + let expected = e_pair (e_typed_list [] (t_operation ())) storage in + let options = Proto_alpha_utils.Memory_proto_alpha.make_options () in + expect_eq program ~options "getBalance" input expected + +let get_total_supply f s () = + let%bind program,_ = get_program f s () in + let storage = e_record_ez [ + ("tokens", e_big_map [(sender, e_nat 100); (from_, e_nat 100); (to_, e_nat 100)]); + ("allowances", e_big_map [(e_pair from_ sender, e_nat 100)]); + ("total_amount",e_nat 300); + ] in + let parameter = e_record_ez [("callback", external_contract)] in + let input = e_pair parameter storage in + let expected = e_pair (e_typed_list [] (t_operation ())) storage in + let options = Proto_alpha_utils.Memory_proto_alpha.make_options () in + expect_eq program ~options "getTotalSupply" input expected + +let main = test_suite "tzip-12" [ + test "transfer" (transfer file_FA12 "pascaligo"); + test "transfer (not enough allowance)" (transfer_not_e_allowance file_FA12 "pascaligo"); + test "transfer (not enough balance)" (transfer_not_e_balance file_FA12 "pascaligo"); + test "approve" (approve file_FA12 "pascaligo"); + test "approve (unsafe allowance change)" (approve_unsafe file_FA12 "pascaligo"); + (* test "getAllowance" (get_allowance file_FA12 "pascaligo"); + test "getBalance" (get_balance file_FA12 "pascaligo"); + test "getTotalSupply" (get_total_supply file_FA12 "pascaligo"); waiting for a dummy_contract with type nat contractt*) + test "transfer" (transfer mfile_FA12 "cameligo"); + test "transfer (not enough allowance)" (transfer_not_e_allowance mfile_FA12 "cameligo"); + test "transfer (not enough balance)" (transfer_not_e_balance mfile_FA12 "cameligo"); + test "approve" (approve mfile_FA12 "cameligo"); + test "approve (unsafe allowance change)" (approve_unsafe mfile_FA12 "cameligo"); + (* test "getAllowance" (get_allowance mfile_FA12 "cameligo"); + test "getBalance" (get_balance mfile_FA12 "cameligo"); + test "getTotalSupply" (get_total_supply mfile_FA12 "cameligo"); waiting for a dummy_contract with type nat contractt*) + test "transfer" (transfer refile_FA12 "reasonligo"); + test "transfer (not enough allowance)" (transfer_not_e_allowance refile_FA12 "reasonligo"); + test "transfer (not enough balance)" (transfer_not_e_balance refile_FA12 "reasonligo"); + test "approve" (approve refile_FA12 "reasonligo"); + test "approve (unsafe allowance change)" (approve_unsafe refile_FA12 "reasonligo"); + (* test "getAllowance" (get_allowance refile_FA12 "reasonligo"); + test "getBalance" (get_balance refile_FA12 "reasonligo"); + test "getTotalSupply" (get_total_supply refile_FA12 "reasonligo"); waiting for a dummy_contract with type nat contractt*) + ]