From f3426b03037ee700e82147547c1db2d525913a5b Mon Sep 17 00:00:00 2001 From: Awea Date: Thu, 10 Oct 2024 17:40:05 +0200 Subject: [PATCH] fixup! fix(forge_operation): add error handling --- lib/forge_operation.ex | 412 ++++++++++++++++++----------------------- 1 file changed, 185 insertions(+), 227 deletions(-) diff --git a/lib/forge_operation.ex b/lib/forge_operation.ex index e0a173d..4494710 100644 --- a/lib/forge_operation.ex +++ b/lib/forge_operation.ex @@ -92,54 +92,44 @@ defmodule Tezex.ForgeOperation do |> Base.encode16(case: :lower) {:ok, content} - else - err -> - err end end @spec activate_account(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def activate_account(content) do - case validate_required_keys(content, ~w(kind pkh secret)) do - :ok -> - content = - [ - forge_tag(@operation_tags[content["kind"]]), - binary_slice(Forge.forge_address(content["pkh"]), 2..-1//1), - Base.decode16!(content["secret"], case: :mixed) - ] - |> IO.iodata_to_binary() - - {:ok, content} - - err -> - err + with :ok <- validate_required_keys(content, ~w(kind pkh secret)) do + content = + [ + forge_tag(@operation_tags[content["kind"]]), + binary_slice(Forge.forge_address(content["pkh"]), 2..-1//1), + Base.decode16!(content["secret"], case: :mixed) + ] + |> IO.iodata_to_binary() + + {:ok, content} end end @spec reveal(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def reveal(content) do - case validate_required_keys( - content, - ~w(kind source fee counter gas_limit storage_limit public_key) - ) do - :ok -> - content = - [ - forge_tag(@operation_tags[content["kind"]]), - Forge.forge_address(content["source"], :bytes, true), - Forge.forge_nat(String.to_integer(content["fee"])), - Forge.forge_nat(String.to_integer(content["counter"])), - Forge.forge_nat(String.to_integer(content["gas_limit"])), - Forge.forge_nat(String.to_integer(content["storage_limit"])), - Forge.forge_public_key(content["public_key"]) - ] - |> IO.iodata_to_binary() - - {:ok, content} - - err -> - err + with :ok <- + validate_required_keys( + content, + ~w(kind source fee counter gas_limit storage_limit public_key) + ) do + content = + [ + forge_tag(@operation_tags[content["kind"]]), + Forge.forge_address(content["source"], :bytes, true), + Forge.forge_nat(String.to_integer(content["fee"])), + Forge.forge_nat(String.to_integer(content["counter"])), + Forge.forge_nat(String.to_integer(content["gas_limit"])), + Forge.forge_nat(String.to_integer(content["storage_limit"])), + Forge.forge_public_key(content["public_key"]) + ] + |> IO.iodata_to_binary() + + {:ok, content} end end @@ -186,105 +176,92 @@ defmodule Tezex.ForgeOperation do @spec origination(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def origination(content) do - case validate_required_keys( - content, - ~w(kind source fee counter gas_limit storage_limit balance) - ) do - :ok -> - content = - [ - forge_tag(@operation_tags[content["kind"]]), - Forge.forge_address(content["source"], :bytes, true), - Forge.forge_nat(String.to_integer(content["fee"])), - Forge.forge_nat(String.to_integer(content["counter"])), - Forge.forge_nat(String.to_integer(content["gas_limit"])), - Forge.forge_nat(String.to_integer(content["storage_limit"])), - Forge.forge_nat(String.to_integer(content["balance"])), - case Map.get(content, "delegate") do - nil -> - Forge.forge_bool(false) - - delegate -> - [Forge.forge_bool(true), Forge.forge_address(delegate, :bytes, true)] - end, - Forge.forge_script(content["script"]) - ] - |> IO.iodata_to_binary() - - {:ok, content} - - err -> - err + with :ok <- + validate_required_keys( + content, + ~w(kind source fee counter gas_limit storage_limit balance) + ) do + content = + [ + forge_tag(@operation_tags[content["kind"]]), + Forge.forge_address(content["source"], :bytes, true), + Forge.forge_nat(String.to_integer(content["fee"])), + Forge.forge_nat(String.to_integer(content["counter"])), + Forge.forge_nat(String.to_integer(content["gas_limit"])), + Forge.forge_nat(String.to_integer(content["storage_limit"])), + Forge.forge_nat(String.to_integer(content["balance"])), + case Map.get(content, "delegate") do + nil -> + Forge.forge_bool(false) + + delegate -> + [Forge.forge_bool(true), Forge.forge_address(delegate, :bytes, true)] + end, + Forge.forge_script(content["script"]) + ] + |> IO.iodata_to_binary() + + {:ok, content} end end @spec delegation(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def delegation(content) do - case validate_required_keys( - content, - ~w(kind source fee counter gas_limit storage_limit) - ) do - :ok -> - content = - [ - forge_tag(@operation_tags[content["kind"]]), - Forge.forge_address(content["source"], :bytes, true), - Forge.forge_nat(String.to_integer(content["fee"])), - Forge.forge_nat(String.to_integer(content["counter"])), - Forge.forge_nat(String.to_integer(content["gas_limit"])), - Forge.forge_nat(String.to_integer(content["storage_limit"])), - case Map.get(content, "delegate") do - nil -> - Forge.forge_bool(false) - - delegate -> - [Forge.forge_bool(true), Forge.forge_address(delegate, :bytes, true)] - end - ] - |> IO.iodata_to_binary() - - {:ok, content} - - err -> - err + with :ok <- + validate_required_keys( + content, + ~w(kind source fee counter gas_limit storage_limit) + ) do + content = + [ + forge_tag(@operation_tags[content["kind"]]), + Forge.forge_address(content["source"], :bytes, true), + Forge.forge_nat(String.to_integer(content["fee"])), + Forge.forge_nat(String.to_integer(content["counter"])), + Forge.forge_nat(String.to_integer(content["gas_limit"])), + Forge.forge_nat(String.to_integer(content["storage_limit"])), + case Map.get(content, "delegate") do + nil -> + Forge.forge_bool(false) + + delegate -> + [Forge.forge_bool(true), Forge.forge_address(delegate, :bytes, true)] + end + ] + |> IO.iodata_to_binary() + + {:ok, content} end end @spec endorsement(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def endorsement(content) do - case validate_required_keys(content, ~w(kind level)) do - :ok -> - content = - [forge_tag(content["kind"]), Forge.forge_int32(String.to_integer(content["level"]))] - |> IO.iodata_to_binary() - - {:ok, content} + with :ok <- validate_required_keys(content, ~w(kind level)) do + content = + [forge_tag(content["kind"]), Forge.forge_int32(String.to_integer(content["level"]))] + |> IO.iodata_to_binary() - err -> - err + {:ok, content} end end @spec inline_endorsement(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def inline_endorsement(content) do - case validate_required_keys( - content, - ~w(branch operations operations.kind operations.level signature) - ) do - :ok -> - content = - [ - Forge.forge_base58(content["branch"]), - Forge.forge_nat(@operation_tags[content["operations"]["kind"]]), - Forge.forge_int32(String.to_integer(content["operations"]["level"])), - Forge.forge_base58(content["signature"]) - ] - |> IO.iodata_to_binary() - - {:ok, content} - - err -> - err + with :ok <- + validate_required_keys( + content, + ~w(branch operations operations.kind operations.level signature) + ) do + content = + [ + Forge.forge_base58(content["branch"]), + Forge.forge_nat(@operation_tags[content["operations"]["kind"]]), + Forge.forge_int32(String.to_integer(content["operations"]["level"])), + Forge.forge_base58(content["signature"]) + ] + |> IO.iodata_to_binary() + + {:ok, content} end end @@ -301,141 +278,122 @@ defmodule Tezex.ForgeOperation do |> IO.iodata_to_binary() {:ok, content} - else - err -> - err end end @spec failing_noop(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def failing_noop(content) do - case validate_required_keys(content, ~w(kind arbitrary)) do - :ok -> - content = - [forge_tag(content["kind"]), Forge.forge_array(content["arbitrary"])] - |> IO.iodata_to_binary() - - {:ok, content} + with :ok <- validate_required_keys(content, ~w(kind arbitrary)) do + content = + [forge_tag(content["kind"]), Forge.forge_array(content["arbitrary"])] + |> IO.iodata_to_binary() - err -> - err + {:ok, content} end end @spec register_global_constant(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def register_global_constant(content) do - case validate_required_keys( - content, - ~w(kind source fee counter gas_limit storage_limit value) - ) do - :ok -> - content = - [ - forge_tag(@operation_tags[content["kind"]]), - Forge.forge_address(content["source"], :bytes, true), - Forge.forge_nat(String.to_integer(content["fee"])), - Forge.forge_nat(String.to_integer(content["counter"])), - Forge.forge_nat(String.to_integer(content["gas_limit"])), - Forge.forge_nat(String.to_integer(content["storage_limit"])), - Forge.forge_array(Forge.forge_micheline(content["value"])) - ] - |> IO.iodata_to_binary() - - {:ok, content} - - err -> - err + with :ok <- + validate_required_keys( + content, + ~w(kind source fee counter gas_limit storage_limit value) + ) do + content = + [ + forge_tag(@operation_tags[content["kind"]]), + Forge.forge_address(content["source"], :bytes, true), + Forge.forge_nat(String.to_integer(content["fee"])), + Forge.forge_nat(String.to_integer(content["counter"])), + Forge.forge_nat(String.to_integer(content["gas_limit"])), + Forge.forge_nat(String.to_integer(content["storage_limit"])), + Forge.forge_array(Forge.forge_micheline(content["value"])) + ] + |> IO.iodata_to_binary() + + {:ok, content} end end @spec transfer_ticket(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def transfer_ticket(content) do - case validate_required_keys( - content, - ~w(kind source fee counter gas_limit storage_limit ticket_contents ticket_ty ticket_ticketer ticket_amount destination entrypoint) - ) do - :ok -> - content = - [ - forge_tag(@operation_tags[content["kind"]]), - Forge.forge_address(content["source"], :bytes, true), - Forge.forge_nat(String.to_integer(content["fee"])), - Forge.forge_nat(String.to_integer(content["counter"])), - Forge.forge_nat(String.to_integer(content["gas_limit"])), - Forge.forge_nat(String.to_integer(content["storage_limit"])), - Forge.forge_array(Forge.forge_micheline(content["ticket_contents"])), - Forge.forge_array(Forge.forge_micheline(content["ticket_ty"])), - Forge.forge_address(content["ticket_ticketer"]), - Forge.forge_nat(String.to_integer(content["ticket_amount"])), - Forge.forge_address(content["destination"]), - Forge.forge_array(content["entrypoint"]) - ] - |> IO.iodata_to_binary() - - {:ok, content} - - err -> - err + with :ok <- + validate_required_keys( + content, + ~w(kind source fee counter gas_limit storage_limit ticket_contents ticket_ty ticket_ticketer ticket_amount destination entrypoint) + ) do + content = + [ + forge_tag(@operation_tags[content["kind"]]), + Forge.forge_address(content["source"], :bytes, true), + Forge.forge_nat(String.to_integer(content["fee"])), + Forge.forge_nat(String.to_integer(content["counter"])), + Forge.forge_nat(String.to_integer(content["gas_limit"])), + Forge.forge_nat(String.to_integer(content["storage_limit"])), + Forge.forge_array(Forge.forge_micheline(content["ticket_contents"])), + Forge.forge_array(Forge.forge_micheline(content["ticket_ty"])), + Forge.forge_address(content["ticket_ticketer"]), + Forge.forge_nat(String.to_integer(content["ticket_amount"])), + Forge.forge_address(content["destination"]), + Forge.forge_array(content["entrypoint"]) + ] + |> IO.iodata_to_binary() + + {:ok, content} end end @spec smart_rollup_add_messages(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def smart_rollup_add_messages(content) do - case validate_required_keys( - content, - ~w(kind source fee counter gas_limit storage_limit message) - ) do - :ok -> - content = - [ - forge_tag(@operation_tags[content["kind"]]), - Forge.forge_address(content["source"], :bytes, true), - Forge.forge_nat(String.to_integer(content["fee"])), - Forge.forge_nat(String.to_integer(content["counter"])), - Forge.forge_nat(String.to_integer(content["gas_limit"])), - Forge.forge_nat(String.to_integer(content["storage_limit"])), - Forge.forge_array( - Enum.join( - Enum.map(content["message"], &Forge.forge_array(:binary.decode_hex(&1))), - "" - ) + with :ok <- + validate_required_keys( + content, + ~w(kind source fee counter gas_limit storage_limit message) + ) do + content = + [ + forge_tag(@operation_tags[content["kind"]]), + Forge.forge_address(content["source"], :bytes, true), + Forge.forge_nat(String.to_integer(content["fee"])), + Forge.forge_nat(String.to_integer(content["counter"])), + Forge.forge_nat(String.to_integer(content["gas_limit"])), + Forge.forge_nat(String.to_integer(content["storage_limit"])), + Forge.forge_array( + Enum.join( + Enum.map(content["message"], &Forge.forge_array(:binary.decode_hex(&1))), + "" ) - ] - |> IO.iodata_to_binary() - - {:ok, content} + ) + ] + |> IO.iodata_to_binary() - err -> - err + {:ok, content} end end @spec smart_rollup_execute_outbox_message(map()) :: {:ok, nonempty_binary()} | {:error, nonempty_binary()} def smart_rollup_execute_outbox_message(content) do - case validate_required_keys( - content, - ~w(kind source fee counter gas_limit storage_limit rollup cemented_commitment output_proof) - ) do - :ok -> - content = - [ - forge_tag(@operation_tags[content["kind"]]), - Forge.forge_address(content["source"], :bytes, true), - Forge.forge_nat(String.to_integer(content["fee"])), - Forge.forge_nat(String.to_integer(content["counter"])), - Forge.forge_nat(String.to_integer(content["gas_limit"])), - Forge.forge_nat(String.to_integer(content["storage_limit"])), - Forge.forge_base58(content["rollup"]), - Forge.forge_base58(content["cemented_commitment"]), - Forge.forge_array(:binary.decode_hex(content["output_proof"])) - ] - |> IO.iodata_to_binary() - - {:ok, content} - - err -> - err + with :ok <- + validate_required_keys( + content, + ~w(kind source fee counter gas_limit storage_limit rollup cemented_commitment output_proof) + ) do + content = + [ + forge_tag(@operation_tags[content["kind"]]), + Forge.forge_address(content["source"], :bytes, true), + Forge.forge_nat(String.to_integer(content["fee"])), + Forge.forge_nat(String.to_integer(content["counter"])), + Forge.forge_nat(String.to_integer(content["gas_limit"])), + Forge.forge_nat(String.to_integer(content["storage_limit"])), + Forge.forge_base58(content["rollup"]), + Forge.forge_base58(content["cemented_commitment"]), + Forge.forge_array(:binary.decode_hex(content["output_proof"])) + ] + |> IO.iodata_to_binary() + + {:ok, content} end end