%%%------------------------------------------------------------------- %%% @copyright (C) 2017, Aeternity Anstalt %%% @doc Decoding fate data to AST %%% @end %%%------------------------------------------------------------------- -module(so_vm_decode). -export([ from_fate/2 ]). -include_lib("gmbytecode/include/gmb_fate_data.hrl"). -spec from_fate(so_syntax:type(), gmb_fate_data:fate_type()) -> so_syntax:expr(). from_fate({id, _, "address"}, ?FATE_ADDRESS(Bin)) -> {account_pubkey, [], Bin}; from_fate({id, _, "signature"}, ?FATE_BYTES(Bin)) -> {signature, [], Bin}; from_fate({id, _, "hash"}, ?FATE_BYTES(Bin)) -> {bytes, [], Bin}; from_fate({id, _, "unit"}, ?FATE_UNIT) -> {tuple, [], []}; from_fate({app_t, _, {id, _, "oracle"}, _}, ?FATE_ORACLE(Bin)) -> {oracle_pubkey, [], Bin}; from_fate({app_t, _, {id, _, "oracle_query"}, _}, ?FATE_ORACLE_Q(Bin)) -> {oracle_query_id, [], Bin}; from_fate({con, _, _Name}, ?FATE_CONTRACT(Bin)) -> {contract_pubkey, [], Bin}; from_fate({bytes_t, _, any}, ?FATE_BYTES(Bin)) -> make_any_bytes(Bin); from_fate({bytes_t, _, N}, ?FATE_BYTES(Bin)) when byte_size(Bin) == N -> {bytes, [], Bin}; from_fate({id, _, "bits"}, ?FATE_BITS(N)) -> make_bits(N); from_fate({id, _, "int"}, N) when is_integer(N) -> if N < 0 -> {app, [{format, prefix}], {'-', []}, [{int, [], -N}]}; true -> {int, [], N} end; from_fate({id, _, "bool"}, B) when is_boolean(B) -> {bool, [], B}; from_fate({id, _, "string"}, S) when is_binary(S) -> {string, [], S}; from_fate({app_t, _, {id, _, "list"}, [Type]}, List) when is_list(List) -> {list, [], [from_fate(Type, X) || X <- List]}; from_fate({app_t, _, {id, _, "option"}, [Type]}, Val) -> case Val of {variant, [0, 1], 0, {}} -> {con, [], "None"}; {variant, [0, 1], 1, {X}} -> {app, [], {con, [], "Some"}, [from_fate(Type, X)]} end; from_fate({tuple_t, _, []}, ?FATE_UNIT) -> {tuple, [], []}; from_fate({tuple_t, _, Types}, ?FATE_TUPLE(Val)) when length(Types) == tuple_size(Val) -> {tuple, [], [from_fate(Type, X) || {Type, X} <- lists:zip(Types, tuple_to_list(Val))]}; from_fate({record_t, [{field_t, _, FName, FType}]}, Val) -> {record, [], [{field, [], [{proj, [], FName}], from_fate(FType, Val)}]}; from_fate({record_t, Fields}, ?FATE_TUPLE(Val)) when length(Fields) == tuple_size(Val) -> {record, [], [ {field, [], [{proj, [], FName}], from_fate(FType, X)} || {{field_t, _, FName, FType}, X} <- lists:zip(Fields, tuple_to_list(Val)) ]}; from_fate({app_t, _, {id, _, "map"}, [KeyType, ValType]}, Map) when is_map(Map) -> {map, [], [ {from_fate(KeyType, Key), from_fate(ValType, Val)} || {Key, Val} <- maps:to_list(Map) ]}; from_fate({variant_t, Cons}, {variant, Ar, Tag, Args}) when length(Cons) > Tag -> ConType = lists:nth(Tag + 1, Cons), Arity = lists:nth(Tag + 1, Ar), case tuple_to_list(Args) of ArgList when length(ArgList) == Arity -> from_fate(ConType, ArgList); _ -> throw(cannot_translate_to_sophia) end; from_fate({constr_t, _, Con, []}, []) -> Con; from_fate({constr_t, _, Con, Types}, Args) when length(Types) == length(Args) -> {app, [], Con, [ from_fate(Type, Arg) || {Type, Arg} <- lists:zip(Types, Args) ]}; from_fate({qid, _, QType}, Val) -> from_fate_builtin(QType, Val); from_fate(_Type, _Data) -> throw(cannot_translate_to_sophia). from_fate_builtin(QType, Val) -> Con = fun([Name | _] = Names) when is_list(Name) -> {qcon, [], Names}; (Name) -> {con, [], Name} end, App = fun(Name, []) -> Con(Name); (Name, Value) -> {app, [], Con(Name), Value} end, Chk = fun(Type, Value) -> from_fate(Type, Value) end, Int = {id, [], "int"}, Str = {id, [], "string"}, Adr = {id, [], "address"}, Hsh = {bytes_t, [], 32}, I32 = {bytes_t, [], 32}, I48 = {bytes_t, [], 48}, Bts = {bytes_t, [], any}, Qid = fun(Name) -> {qid, [], Name} end, Map = fun(KT, VT) -> {app_t, [], {id, [], "map"}, [KT, VT]} end, ChainTxArities = [3, 0, 0, 0, 0, 0, 1, 1, 1, 2, 1, 2, 2, 1, 1, 1, 1, 1, 1, 1, 2, 0], case {QType, Val} of {["Chain", "ttl"], {variant, [1, 1], 0, {X}}} -> App("RelativeTTL", [Chk(Int, X)]); {["Chain", "ttl"], {variant, [1, 1], 1, {X}}} -> App("FixedTTL", [Chk(Int, X)]); {["AENS", "name"], {variant, [3], 0, {Addr, TTL, Ptrs}}} -> App(["AENS","Name"], [Chk(Adr, Addr), Chk(Qid(["Chain", "ttl"]), TTL), Chk(Map(Str, Qid(["AENS", "pointee"])), Ptrs)]); {["AENS", "pointee"], {variant, [1, 1, 1, 1], 0, {Addr}}} -> App(["AENS","AccountPt"], [Chk(Adr, Addr)]); {["AENS", "pointee"], {variant, [1, 1, 1, 1], 1, {Addr}}} -> App(["AENS","OraclePt"], [Chk(Adr, Addr)]); {["AENS", "pointee"], {variant, [1, 1, 1, 1], 2, {Addr}}} -> App(["AENS","ContractPt"], [Chk(Adr, Addr)]); {["AENS", "pointee"], {variant, [1, 1, 1, 1], 3, {Addr}}} -> App(["AENS","ChannelPt"], [Chk(Adr, Addr)]); {["AENSv2", "name"], {variant, [3], 0, {Addr, TTL, Ptrs}}} -> App(["AENSv2","Name"], [Chk(Adr, Addr), Chk(Qid(["Chain", "ttl"]), TTL), Chk(Map(Str, Qid(["AENSv2", "pointee"])), Ptrs)]); {["AENSv2", "pointee"], {variant, [1, 1, 1, 1, 1], 0, {Value}}} -> App(["AENSv2","AccountPt"], [Chk(Adr, Value)]); {["AENSv2", "pointee"], {variant, [1, 1, 1, 1, 1], 1, {Value}}} -> App(["AENSv2","OraclePt"], [Chk(Adr, Value)]); {["AENSv2", "pointee"], {variant, [1, 1, 1, 1, 1], 2, {Value}}} -> App(["AENSv2","ContractPt"], [Chk(Adr, Value)]); {["AENSv2", "pointee"], {variant, [1, 1, 1, 1, 1], 3, {Value}}} -> App(["AENSv2","ChannelPt"], [Chk(Adr, Value)]); {["AENSv2", "pointee"], {variant, [1, 1, 1, 1, 1], 4, {Value}}} -> App(["AENSv2","DataPt"], [Chk(Bts, Value)]); {["Chain", "ga_meta_tx"], {variant, [2], 0, {Addr, X}}} -> App(["Chain","GAMetaTx"], [Chk(Adr, Addr), Chk(Int, X)]); {["Chain", "paying_for_tx"], {variant, [2], 0, {Addr, X}}} -> App(["Chain","PayingForTx"], [Chk(Adr, Addr), Chk(Int, X)]); {["Chain", "base_tx"], {variant, ChainTxArities, 0, {Addr, Fee, Payload}}} -> App(["Chain","SpendTx"], [Chk(Adr, Addr), Chk(Int, Fee), Chk(Str, Payload)]); {["Chain", "base_tx"], {variant, ChainTxArities, 1, {}}} -> App(["Chain","OracleRegisterTx"], []); {["Chain", "base_tx"], {variant, ChainTxArities, 2, {}}} -> App(["Chain","OracleQueryTx"], []); {["Chain", "base_tx"], {variant, ChainTxArities, 3, {}}} -> App(["Chain","OracleResponseTx"], []); {["Chain", "base_tx"], {variant, ChainTxArities, 4, {}}} -> App(["Chain","OracleExtendTx"], []); {["Chain", "base_tx"], {variant, ChainTxArities, 5, {}}} -> App(["Chain","NamePreclaimTx"], []); {["Chain", "base_tx"], {variant, ChainTxArities, 6, {Name}}} -> App(["Chain","NameClaimTx"], [Chk(Str, Name)]); {["Chain", "base_tx"], {variant, ChainTxArities, 7, {NameHash}}} -> App(["Chain","NameUpdateTx"], [Chk(Hsh, NameHash)]); {["Chain", "base_tx"], {variant, ChainTxArities, 8, {NameHash}}} -> App(["Chain","NameRevokeTx"], [Chk(Hsh, NameHash)]); {["Chain", "base_tx"], {variant, ChainTxArities, 9, {NewOwner, NameHash}}} -> App(["Chain","NameTransferTx"], [Chk(Adr, NewOwner), Chk(Hsh, NameHash)]); {["Chain", "base_tx"], {variant, ChainTxArities, 10, {Addr}}} -> App(["Chain","ChannelCreateTx"], [Chk(Adr, Addr)]); {["Chain", "base_tx"], {variant, ChainTxArities, 11, {Addr, Amount}}} -> App(["Chain","ChannelDepositTx"], [Chk(Adr, Addr), Chk(Int, Amount)]); {["Chain", "base_tx"], {variant, ChainTxArities, 12, {Addr, Amount}}} -> App(["Chain","ChannelWithdrawTx"], [Chk(Adr, Addr), Chk(Int, Amount)]); {["Chain", "base_tx"], {variant, ChainTxArities, 13, {Addr}}} -> App(["Chain","ChannelForceProgressTx"], [Chk(Adr, Addr)]); {["Chain", "base_tx"], {variant, ChainTxArities, 14, {Addr}}} -> App(["Chain","ChannelCloseMutualTx"], [Chk(Adr, Addr)]); {["Chain", "base_tx"], {variant, ChainTxArities, 15, {Addr}}} -> App(["Chain","ChannelCloseSoloTx"], [Chk(Adr, Addr)]); {["Chain", "base_tx"], {variant, ChainTxArities, 16, {Addr}}} -> App(["Chain","ChannelSlashTx"], [Chk(Adr, Addr)]); {["Chain", "base_tx"], {variant, ChainTxArities, 17, {Addr}}} -> App(["Chain","ChannelSettleTx"], [Chk(Adr, Addr)]); {["Chain", "base_tx"], {variant, ChainTxArities, 18, {Addr}}} -> App(["Chain","ChannelSnapshotSoloTx"], [Chk(Adr, Addr)]); {["Chain", "base_tx"], {variant, ChainTxArities, 19, {Amount}}} -> App(["Chain","ContractCreateTx"], [Chk(Int, Amount)]); {["Chain", "base_tx"], {variant, ChainTxArities, 20, {Addr, Amount}}} -> App(["Chain","ContractCallTx"], [Chk(Adr, Addr), Chk(Int, Amount)]); {["Chain", "base_tx"], {variant, ChainTxArities, 21, {}}} -> App(["Chain","GAAttachTx"], []); {["MCL_BLS12_381", "fp"], X} -> App(["MCL_BLS12_381", "fp"], [Chk(I32, X)]); {["MCL_BLS12_381", "fr"], X} -> App(["MCL_BLS12_381", "fr"], [Chk(I48, X)]); _ -> throw(cannot_translate_to_sophia) end. make_bits(N) -> Id = fun(F) -> {qid, [], ["Bits", F]} end, if N < 0 -> make_bits(Id("clear"), Id("all"), 0, bnot N); true -> make_bits(Id("set"), Id("none"), 0, N) end. make_bits(_Set, Zero, _I, 0) -> Zero; make_bits(Set, Zero, I, N) when 0 == N rem 2 -> make_bits(Set, Zero, I + 1, N div 2); make_bits(Set, Zero, I, N) -> {app, [], Set, [make_bits(Set, Zero, I + 1, N div 2), {int, [], I}]}. make_any_bytes(Bin) -> {app, [], {qid, [], ["Bytes", "to_any_size"]}, [{bytes, [], Bin}]}.