Compare commits
13 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d66b8a0af9 | ||
|
|
2abb15cceb | ||
|
|
057861e9cf | ||
|
|
7ffc96b68a | ||
|
|
a1be8cbc14 | ||
|
|
3822bb69c9 | ||
|
|
6506fc91bd | ||
|
|
68a0c3d623 | ||
|
|
d3fb598506 | ||
|
|
4ed2bd0cd1 | ||
|
|
0caf5a61c7 | ||
|
|
7704d82c6f | ||
|
|
a6f58a95e2 |
2
.gitignore
vendored
2
.gitignore
vendored
@ -8,7 +8,9 @@ cancer
|
||||
erl_crash.dump
|
||||
ebin/*.beam
|
||||
doc/*.html
|
||||
doc/*.css
|
||||
doc/edoc-info
|
||||
doc/erlang.png
|
||||
rel/example_project
|
||||
.concrete/DEV_MODE
|
||||
.rebar
|
||||
|
||||
BIN
doc/erlang.png
BIN
doc/erlang.png
Binary file not shown.
|
Before Width: | Height: | Size: 3.3 KiB |
@ -1,79 +1,77 @@
|
||||
@author Craig Everett <craigeverett@qpq.swiss> [https://git.qpq.swiss/QPQ-AG/hakuzaru]
|
||||
@version 0.8.0
|
||||
@title Hakuzaru: Gajumaru blockchain bindings for Erlang
|
||||
@author Craig Everett <ceverett@zxq9.com> [https://gitlab.com/zxq9/zj]
|
||||
@version 0.3.0
|
||||
@title Vanillae: Aeternity blockchain bindings for Erlang
|
||||
|
||||
@doc
|
||||
This Erlang application provides bindings for the Gajumaru blockchain and basic utilities for manipulating Gajumaru-related data.
|
||||
This Erlang application provides bindings for the Erlang blockchain.
|
||||
The primary goal is for usage to be easy to understand and as simple as possible to use.
|
||||
The secondary goal is to enable real-world projects to more easily connect with the blockchain in an obvious way and provide a clear path for them to provide feedback regarding areas that are difficult to understand, functionality that is lacking, and explain their use cases to us so we can more easily provide needed features and usage examples to make adoption easier.
|
||||
|
||||
== To Start or Not To Start ==
|
||||
Starting the `hakuzaru' application is only required if you need to query the chain.
|
||||
== Basic operation ==
|
||||
All external interfaces expected to be used by authors of programs that use Vanillae are built into the `vanillae' module.
|
||||
|
||||
The application can be started via a call to `application', or with an explicit call to `hz:start()'.
|
||||
When Vanillae is started as an application a named process called `vanillae_man' is spawned that manages interactions with and the state of the service, as well as a simple-one-for-one supervisor that manages the lifecycle of Vanillae workers (defined in `vanillae_fetcher').
|
||||
|
||||
Hakuzaru can also be run as a local application from the shell by invoking it with `zxh run hakuzaru' if you have `zx' installed.
|
||||
After startup `vanillae_man' must be given the address and port of a list of Aeternity nodes that are available to service requests. Note that the service nodes will need to have the "dry run" endpoint enabled and the internal service query port made available in order to provide "dry run" and mempool TX submission functionality.
|
||||
|
||||
== Operation ==
|
||||
All blockchain-specific operations are accessible from the main interface modulle: `hz'
|
||||
The `vanillae_man' will round-robin requests to however many Aeternity nodes are provided in its configuration. Note that this congiruation is dynamic and can be changed completely at runtime.
|
||||
|
||||
When Hakuzaru is started as an application a named process called `hz_man' is spawned that manages interactions with chain nodes, as well as a simple-one-for-one supervisor that manages the lifecycle of workers (defined in `hz_fetcher').
|
||||
== Functions ==
|
||||
The `vanillae' module exposes one function per blockchain feature provided. Most of these are actually wrappers for blockchain endpoint functions, others provide functionality specific to accomplishing a local processing task related to chain data.
|
||||
|
||||
After startup `hz_man' must be given the address and port of a list of Gajumaru nodes that are available to service requests.
|
||||
Note that the service nodes will need to have the dry-run endpoint enabled and the internal service query port made available in order to provide dry-runs and transaction submission.
|
||||
== Initialization ==
|
||||
When Vanillae is first started the vanillae_man is started but does not yet know what Aeternity nodes to use to service queries. You will need to provide it with at least one node and port where it can make Aeternity endpoint calls.
|
||||
|
||||
When configuring chain nodes a list of nodes should be provided.
|
||||
To avoid sync issues in the case of fast transaction formation/submission to the chain, only one node from the list of chain nodes is used for submitting transactions and querying `next_nonce/1`.
|
||||
This node is called "the sticky node".
|
||||
Note that if you will need to make read-only calls to contracts that are deployed on chain (to queery their state or perform specific read-only operations provided by the contract) the backend nodes you configure will need to be configured with "dry-run" enabled.
|
||||
|
||||
The first node in the list of chain nodes provided during configuration is designated as the sticky node.
|
||||
If you also want to use the sticky node as a query endpoint, include it twice in the list.
|
||||
|
||||
The `hz_man' will round-robin requests to however many additional Gajumaru nodes are provided in the configuration.
|
||||
Note that this configuration is dynamic and can be changed at runtime, so your service can adapt to node availability on the fly if needed.
|
||||
Example of a shell session where vanillae is started and initialized manually with an AE node in the local network at 192.168.10.10:3013:
|
||||
|
||||
```
|
||||
ceverett@steak:~$ zxh run hakuzaru
|
||||
Erlang/OTP 27 [erts-15.2] [source] [64-bit] [smp:16:16] [ds:16:16:10] [async-threads:1] [jit:ns]
|
||||
|
||||
Eshell V15.2 (press Ctrl+G to abort, type help(). for help)
|
||||
Fetching otpr-hakuzaru-0.8.0
|
||||
[100.00%]
|
||||
Recompile: src/hz_sup
|
||||
Recompile: src/hz_man
|
||||
Recompile: src/hz_key_master
|
||||
Recompile: src/hz_grids
|
||||
Recompile: src/hz_format
|
||||
Recompile: src/hz_fetcher
|
||||
Recompile: src/hz
|
||||
Recompile: src/hakuzaru
|
||||
Starting otpr-hakuzaru-0.8.0.
|
||||
hz_man starting.
|
||||
Started [hakuzaru]
|
||||
1> hz:chain_nodes([{"groot.testnet.gajumaru.io", 3013}]).
|
||||
1> vanillae:start().
|
||||
Starting.
|
||||
ok
|
||||
2> hz:status().
|
||||
{ok,#{"difficulty" => 2877405482,
|
||||
"finalized" =>
|
||||
#{"hash" =>
|
||||
"kh_PDSn6Xru5JVdpJfdDCNpfsL8gUZvvjyhTYuzgndoy98G5oLLR",
|
||||
"height" => 277454,"type" => "height"},
|
||||
2> vanillae:status().
|
||||
{error,no_nodes}
|
||||
3> vanillae:ae_nodes([{"192.168.7.7", 3013}]).
|
||||
ok
|
||||
4> vanillae:status().
|
||||
{ok,#{"difficulty" => 59729882,
|
||||
"genesis_key_block_hash" =>
|
||||
"kh_Qdi5MTuuhJm7xzn5JUAbYG12cX3qoLMnXrBxPGzBkMWJ4K8vq",
|
||||
"hashrate" => 864394,"listening" => true,
|
||||
"network_id" => "groot.testnet",
|
||||
"kh_wUCideEB8aDtUaiHCtKcfywU6oHZW6gnyci8Mw6S1RSTCnCRu",
|
||||
"listening" => true,"network_id" => "ae_uat",
|
||||
"node_revision" =>
|
||||
"7b3cc1db3bb36053023167b86f7d6f2d5dcbd01d",
|
||||
"node_version" => "0.1.0+203.7b3cc1db3",
|
||||
"peer_connections" => #{"inbound" => 1,"outbound" => 3},
|
||||
"peer_count" => 5,
|
||||
"3a08153c635c53d92029a617f2e784731ba367c6",
|
||||
"node_version" => "6.7.0",
|
||||
"peer_connections" => #{"inbound" => 25,"outbound" => 10},
|
||||
"peer_count" => 50,
|
||||
"peer_pubkey" =>
|
||||
"pp_2nQHucGyEt5wkYruNuRkg19cbZuEeyR9BZfvtv49F3AoyNSYMT",
|
||||
"pending_transactions_count" => 0,
|
||||
"pp_fCBqobeSwhdnrzC8DoSsmWbf2GzDK61CJujmsCEd3RUkmh9Ny",
|
||||
"pending_transactions_count" => 2,
|
||||
"protocols" =>
|
||||
[#{"effective_at_height" => 0,"version" => 1}],
|
||||
[#{"effective_at_height" => 425900,"version" => 5},
|
||||
#{"effective_at_height" => 154300,"version" => 4},
|
||||
#{"effective_at_height" => 82900,"version" => 3},
|
||||
#{"effective_at_height" => 40900,"version" => 2},
|
||||
#{"effective_at_height" => 0,"version" => 1}],
|
||||
"solutions" => 0,"sync_progress" => 100.0,
|
||||
"syncing" => false,"top_block_height" => 277555,
|
||||
"top_hash" =>
|
||||
"kh_2vuNc8eG77aTmHcQDcievjKufFwR4MSSuZbEMWwW5TqUzSQy71",
|
||||
"syncing" => false,"top_block_height" => 802644,
|
||||
"top_key_block_hash" =>
|
||||
"kh_2vuNc8eG77aTmHcQDcievjKufFwR4MSSuZbEMWwW5TqUzSQy71"}}
|
||||
"kh_28LZSvHZPCGqeWsMsqtSjxQjQHKW1pHzoBex97oMT7U2HcLPgV"}}
|
||||
'''
|
||||
|
||||
Alternatively, here is a start function for an application using Vanillae that initializes vanillae_man with a list of nodes provided by a configuration file:
|
||||
|
||||
```
|
||||
start(normal, _Args) ->
|
||||
ok = application:ensure_started(sasl),
|
||||
{ok, Started} = application:ensure_all_started(cowboy),
|
||||
ok = application:ensure_started(vanillae),
|
||||
Nodes = proplists:get_value(ae_nodes, read_config(), []),
|
||||
ok = vanillae:ae_nodes(Nodes),
|
||||
ok = log(info, "Started: ~p~n", [[vanillae | Started]]),
|
||||
Routes = [{'_', [{"/", count_top, []}]}],
|
||||
Dispatch = cowboy_router:compile(Routes),
|
||||
Env = #{env => #{dispatch => Dispatch}},
|
||||
{ok, _} = cowboy:start_clear(count_listener, [{port, 8080}], Env),
|
||||
count_sup:start_link().
|
||||
'''
|
||||
|
||||
@ -1,75 +0,0 @@
|
||||
/* standard EDoc style sheet */
|
||||
body {
|
||||
font-family: Verdana, Arial, Helvetica, sans-serif;
|
||||
margin-left: .25in;
|
||||
margin-right: .2in;
|
||||
margin-top: 0.2in;
|
||||
margin-bottom: 0.2in;
|
||||
color: #696969;
|
||||
background-color: #ffffff;
|
||||
}
|
||||
a:link{
|
||||
color: #000000;
|
||||
}
|
||||
a:visited{
|
||||
color: #000000;
|
||||
}
|
||||
a:hover{
|
||||
color: #d8613c;
|
||||
}
|
||||
h1,h2 {
|
||||
margin-left: -0.2in;
|
||||
}
|
||||
div.navbar {
|
||||
background-color: #000000;
|
||||
padding: 0.2em;
|
||||
}
|
||||
h2.indextitle {
|
||||
padding: 0.4em;
|
||||
color: #dfdfdf;
|
||||
background-color: #000000;
|
||||
}
|
||||
div.navbar a:link {
|
||||
color: #dfdfdf;
|
||||
}
|
||||
div.navbar a:visited {
|
||||
color: #dfdfdf;
|
||||
}
|
||||
div.navbar a:hover {
|
||||
color: #d8613c;
|
||||
}
|
||||
h3.function,h3.typedecl {
|
||||
background-color: #000000;
|
||||
color: #dfdfdf;
|
||||
padding-left: 1em;
|
||||
}
|
||||
div.spec {
|
||||
margin-left: 2em;
|
||||
background-color: #eeeeee;
|
||||
}
|
||||
a.module {
|
||||
text-decoration:none
|
||||
}
|
||||
a.module:hover {
|
||||
background-color: #eeeeee;
|
||||
}
|
||||
ul.definitions {
|
||||
list-style-type: none;
|
||||
}
|
||||
ul.index {
|
||||
list-style-type: none;
|
||||
background-color: #eeeeee;
|
||||
}
|
||||
|
||||
/*
|
||||
* Minor style tweaks
|
||||
*/
|
||||
ul {
|
||||
list-style-type: square;
|
||||
}
|
||||
table {
|
||||
border-collapse: collapse;
|
||||
}
|
||||
td {
|
||||
padding: 3
|
||||
}
|
||||
@ -3,7 +3,7 @@
|
||||
{included_applications,[]},
|
||||
{applications,[stdlib,kernel]},
|
||||
{description,"Gajumaru interoperation library"},
|
||||
{vsn,"0.8.2"},
|
||||
{modules,[hakuzaru,hz,hz_fetcher,hz_format,hz_grids,
|
||||
hz_key_master,hz_man,hz_sup]},
|
||||
{vsn,"0.6.1"},
|
||||
{modules,[hakuzaru,hz,hz_fetcher,hz_grids,hz_key_master,hz_man,
|
||||
hz_sup]},
|
||||
{mod,{hakuzaru,[]}}]}.
|
||||
|
||||
@ -6,7 +6,7 @@
|
||||
%%% @end
|
||||
|
||||
-module(hakuzaru).
|
||||
-vsn("0.8.2").
|
||||
-vsn("0.6.1").
|
||||
-author("Craig Everett <ceverett@tsuriai.jp>").
|
||||
-copyright("Craig Everett <ceverett@tsuriai.jp>").
|
||||
-license("GPL-3.0-or-later").
|
||||
|
||||
207
src/hz.erl
207
src/hz.erl
@ -9,7 +9,7 @@
|
||||
%%%
|
||||
%%% The get/set admin functions are for setting or checking things like the Gajumaru
|
||||
%%% "network ID" and list of addresses of nodes you want to use for answering
|
||||
%%% queries to the blockchain. Get functions are arity 0, and set functions are arity 1.
|
||||
%%% queries to the blockchain.
|
||||
%%%
|
||||
%%% The JSON query interface functions are the blockchain query functions themselves
|
||||
%%% which are translated to network queries and return Erlang messages as responses.
|
||||
@ -18,12 +18,12 @@
|
||||
%%% a desired call to a smart contract on the chain to call data serialized in a form
|
||||
%%% that a Gajumaru compatible wallet or library can sign and submit to a Gajumaru node.
|
||||
%%%
|
||||
%%% NOTE:
|
||||
%%% This module does not implement the OTP application behavior. Refer to hakuzaru.erl.
|
||||
%%% This module does not implement the OTP application behavior.
|
||||
%%% helper functions.
|
||||
%%% @end
|
||||
|
||||
-module(hz).
|
||||
-vsn("0.8.2").
|
||||
-vsn("0.6.1").
|
||||
-author("Craig Everett <ceverett@tsuriai.jp>").
|
||||
-copyright("Craig Everett <ceverett@tsuriai.jp>").
|
||||
-license("GPL-3.0-or-later").
|
||||
@ -66,8 +66,6 @@
|
||||
contract_create/8,
|
||||
prepare_contract/1,
|
||||
prepare_aaci/1,
|
||||
cache_aaci/2,
|
||||
lookup_aaci/1,
|
||||
aaci_lookup_spec/2,
|
||||
contract_call/5,
|
||||
contract_call/6,
|
||||
@ -75,8 +73,8 @@
|
||||
decode_bytearray_fate/1, decode_bytearray/2,
|
||||
spend/5, spend/10,
|
||||
sign_tx/2, sign_tx/3,
|
||||
sign_message/2, verify_signature/3,
|
||||
sign_binary/2, verify_bin_signature/3]).
|
||||
sign_message/2,
|
||||
verify_signature/3]).
|
||||
|
||||
|
||||
%%% Types
|
||||
@ -227,7 +225,7 @@
|
||||
NetworkID :: string(),
|
||||
Reason :: term().
|
||||
%% @doc
|
||||
%% Returns the network ID or the atom `none' if unavailable.
|
||||
%% Returns the network ID or the atom `none' if it is unset.
|
||||
%% Checking this is not normally necessary, but if network ID assignment is dynamic
|
||||
%% in your system it may be necessary to call this before attempting to form
|
||||
%% call data or perform other actions on chain that require a signature.
|
||||
@ -243,9 +241,7 @@ network_id() ->
|
||||
%% @doc
|
||||
%% Returns the list of currently assigned nodes.
|
||||
%% The normal reason to call this is in preparation for altering the nodes list or
|
||||
%% checking the current list in debugging. Note that the first node in the list is
|
||||
%% the "sticky" node: the one that will be used for submitting transactions and
|
||||
%% querying `next_nonce'.
|
||||
%% checking the current list in debugging.
|
||||
|
||||
chain_nodes() ->
|
||||
hz_man:chain_nodes().
|
||||
@ -255,26 +251,19 @@ chain_nodes() ->
|
||||
when List :: [chain_node()],
|
||||
Reason :: {invalid, [term()]}.
|
||||
%% @doc
|
||||
%% Sets the chain nodes that will be queried whenever you communicate with the chain.
|
||||
%% Sets the nodes that are intended to be used as your interface to the peer
|
||||
%% network. The common situation is that your project runs a non-mining node as
|
||||
%% part of your backend infrastructure. Typically one or two nodes is plenty, but
|
||||
%% this may need to expand depending on how much query load your application generates.
|
||||
%% The Hakuzaru manager will load balance by round-robin distribution.
|
||||
%%
|
||||
%% The common situation is that a project runs a non-mining node as part of the backend
|
||||
%% infrastructure. Typically one or two nodes is plenty, but this may need to expand
|
||||
%% depending on how much query load your application generates.
|
||||
%%
|
||||
%% There are two situations: one node, or multiple nodes.
|
||||
%%
|
||||
%% Single node:
|
||||
%% In the case of a single node, everything passes through that one node. Duh.
|
||||
%%
|
||||
%% Multiple nodes:
|
||||
%% In the case of multiple nodes a distinction is made between the node to which
|
||||
%% transactions that update the chain state are made and to which `next_nonce' queries
|
||||
%% are made, and nodes that are used for read-only queries. The node to which stateful
|
||||
%% transactions are submitted is called the "sticky node". This is the first node
|
||||
%% (head position) in the list of nodes submitted to the chain when `chain_nodes/1'
|
||||
%% is called. If using multiple nodes but the sticky node should also be used for
|
||||
%% read-only queries, submit the sticky node at the head of the list and again in
|
||||
%% the tail.
|
||||
%% NOTE: When load balancing in this way be aware that there can be race conditions
|
||||
%% among the backend nodes with regard to a single account's current nonce when performing
|
||||
%% contract calls in quick succession. Round robin distribution is extremely useful when
|
||||
%% performing rapid lookups to the chain, but does not work well when submitting many
|
||||
%% transactions to the chain from a single user in a short period of time. A future version
|
||||
%% of this library will allow the caller to designate a single node as "sticky" to be used
|
||||
%% exclusively in the case of nonce reads and TX submissions.
|
||||
|
||||
chain_nodes(List) when is_list(List) ->
|
||||
hz_man:chain_nodes(List).
|
||||
@ -282,16 +271,7 @@ chain_nodes(List) when is_list(List) ->
|
||||
|
||||
-spec tls() -> boolean().
|
||||
%% @doc
|
||||
%% Check whether TLS is in use. The typical situation is to not use TLS as nodes that
|
||||
%% serve as part of the backend of an application are typically run in the same
|
||||
%% backend network as the application service. When accessing chain nodes over the WAN
|
||||
%% however, TLS is strongly recommended to avoid a MITM attack.
|
||||
%%
|
||||
%% In this version of Hakuzaru TLS is either on or off for all nodes, making a mixed
|
||||
%% infrastructure complicated to support without two Hakuzaru instances. This will
|
||||
%% likely become a per-node setting in the future.
|
||||
%%
|
||||
%% TLS defaults to `false'.
|
||||
%% Check whether TLS is in use.
|
||||
|
||||
tls() ->
|
||||
hz_man:tls().
|
||||
@ -301,8 +281,6 @@ tls() ->
|
||||
%% @doc
|
||||
%% Set TLS true or false. That's what a boolean is, by the way, `true' or `false'.
|
||||
%% This is a condescending comment. That means I am talking down to you.
|
||||
%%
|
||||
%% TLS defaults to `false'.
|
||||
|
||||
tls(Boolean) ->
|
||||
hz_man:tls(Boolean).
|
||||
@ -313,8 +291,6 @@ tls(Boolean) ->
|
||||
when Timeout :: pos_integer() | infinity.
|
||||
%% @doc
|
||||
%% Returns the current request timeout setting in milliseconds.
|
||||
%% The default timeout is 5,000ms.
|
||||
%% The max timeout is 120,000ms.
|
||||
|
||||
timeout() ->
|
||||
hz_man:timeout().
|
||||
@ -324,8 +300,6 @@ timeout() ->
|
||||
when MS :: pos_integer() | infinity.
|
||||
%% @doc
|
||||
%% Sets the request timeout in milliseconds.
|
||||
%% The default timeout is 5,000ms.
|
||||
%% The max timeout is 120,000ms.
|
||||
|
||||
timeout(MS) ->
|
||||
hz_man:timeout(MS).
|
||||
@ -602,18 +576,18 @@ acc_pending_txs(AccountID) ->
|
||||
%% Retrieve the next nonce for the given account
|
||||
|
||||
next_nonce(AccountID) ->
|
||||
case request_sticky(["/v3/accounts/", AccountID, "/next-nonce"]) of
|
||||
{ok, #{"next_nonce" := Nonce}} -> {ok, Nonce};
|
||||
{ok, #{"reason" := "Account not found"}} -> {ok, 1};
|
||||
{ok, #{"reason" := Reason}} -> {error, Reason};
|
||||
Error -> Error
|
||||
end.
|
||||
% case request_sticky(["/v3/accounts/", AccountID]) of
|
||||
% {ok, #{"nonce" := Nonce}} -> {ok, Nonce + 1};
|
||||
% case request(["/v3/accounts/", AccountID, "/next-nonce"]) of
|
||||
% {ok, #{"next_nonce" := Nonce}} -> {ok, Nonce};
|
||||
% {ok, #{"reason" := "Account not found"}} -> {ok, 1};
|
||||
% {ok, #{"reason" := Reason}} -> {error, Reason};
|
||||
% Error -> Error
|
||||
% end.
|
||||
case request(["/v3/accounts/", AccountID]) of
|
||||
{ok, #{"nonce" := Nonce}} -> {ok, Nonce + 1};
|
||||
{ok, #{"reason" := "Account not found"}} -> {ok, 1};
|
||||
{ok, #{"reason" := Reason}} -> {error, Reason};
|
||||
Error -> Error
|
||||
end.
|
||||
|
||||
|
||||
-spec dry_run(TX) -> {ok, Result} | {error, Reason}
|
||||
@ -755,7 +729,7 @@ tx_info(ID) ->
|
||||
|
||||
post_tx(Data) when is_binary(Data) ->
|
||||
JSON = zj:binary_encode(#{tx => Data}),
|
||||
request_sticky("/v3/transactions", JSON);
|
||||
request("/v3/transactions", JSON);
|
||||
post_tx(Data) when is_list(Data) ->
|
||||
post_tx(list_to_binary(Data)).
|
||||
|
||||
@ -867,14 +841,6 @@ status_chainends() ->
|
||||
request("/v3/status/chain-ends").
|
||||
|
||||
|
||||
request_sticky(Path) ->
|
||||
hz_man:request_sticky(unicode:characters_to_list(Path)).
|
||||
|
||||
|
||||
request_sticky(Path, Payload) ->
|
||||
hz_man:request_sticky(unicode:characters_to_list(Path), Payload).
|
||||
|
||||
|
||||
request(Path) ->
|
||||
hz_man:request(unicode:characters_to_list(Path)).
|
||||
|
||||
@ -924,7 +890,7 @@ contract_create(CreatorID, Path, InitArgs) ->
|
||||
when CreatorID :: pubkey(),
|
||||
Nonce :: pos_integer(),
|
||||
Amount :: non_neg_integer(),
|
||||
TTL :: non_neg_integer(),
|
||||
TTL :: pos_integer(),
|
||||
Gas :: pos_integer(),
|
||||
GasPrice :: pos_integer(),
|
||||
Path :: file:filename(),
|
||||
@ -970,7 +936,6 @@ contract_create(CreatorID, Path, InitArgs) ->
|
||||
%% of course there are very good reasons why it should be set to a non-zero value
|
||||
%% in the case of calls related to contract-governed payment systems.
|
||||
%% </li>
|
||||
%% <li>
|
||||
%% <b>TTL:</b>
|
||||
%% This stands for "Time-To-Live", meaning the height beyond which this element is
|
||||
%% considered to be eligible for garbage collection (and therefore inaccessible!).
|
||||
@ -1076,7 +1041,7 @@ contract_create2(CreatorID, Nonce, Amount, TTL, Gas, GasPrice, Source, Options,
|
||||
InitArgs :: [string()],
|
||||
Result :: {ok, CreateTX} | {error, Reason},
|
||||
CreateTX :: binary(),
|
||||
Reason :: file:posix() | bad_fun_name | aaci_not_found | term().
|
||||
Reason :: file:posix() | term().
|
||||
%% @doc
|
||||
%% This function takes the compiler output (instead of starting from source),
|
||||
%% and returns the unsigned create contract call data with default values.
|
||||
@ -1178,10 +1143,9 @@ assemble_calldata2(OwnerID, Nonce, Amount, TTL, Gas, GasPrice, Compiled, CallDat
|
||||
read_aci(Path) ->
|
||||
case file:read_file(Path) of
|
||||
{ok, Bin} ->
|
||||
try
|
||||
{ok, binary_to_term(Bin, [safe])}
|
||||
catch
|
||||
error:badarg -> {error, bad_aci}
|
||||
case zx_lib:b_to_ts(Bin) of
|
||||
error -> {error, bad_aci};
|
||||
OK -> OK
|
||||
end;
|
||||
Error ->
|
||||
Error
|
||||
@ -1190,7 +1154,7 @@ read_aci(Path) ->
|
||||
|
||||
-spec contract_call(CallerID, AACI, ConID, Fun, Args) -> Result
|
||||
when CallerID :: unicode:chardata(),
|
||||
AACI :: aaci() | {aaci, Label :: term()},
|
||||
AACI :: aaci(),
|
||||
ConID :: unicode:chardata(),
|
||||
Fun :: string(),
|
||||
Args :: [string()],
|
||||
@ -1225,7 +1189,7 @@ contract_call(CallerID, AACI, ConID, Fun, Args) ->
|
||||
-spec contract_call(CallerID, Gas, AACI, ConID, Fun, Args) -> Result
|
||||
when CallerID :: unicode:chardata(),
|
||||
Gas :: pos_integer(),
|
||||
AACI :: aaci() | {aaci, Label :: term()},
|
||||
AACI :: aaci(),
|
||||
ConID :: unicode:chardata(),
|
||||
Fun :: string(),
|
||||
Args :: [string()],
|
||||
@ -1262,8 +1226,8 @@ contract_call(CallerID, Gas, AACI, ConID, Fun, Args) ->
|
||||
Gas :: pos_integer(),
|
||||
GasPrice :: pos_integer(),
|
||||
Amount :: non_neg_integer(),
|
||||
TTL :: non_neg_integer(),
|
||||
AACI :: aaci() | {aaci, Label :: term()},
|
||||
TTL :: pos_integer(),
|
||||
AACI :: aaci(),
|
||||
ConID :: unicode:chardata(),
|
||||
Fun :: string(),
|
||||
Args :: [string()],
|
||||
@ -1891,11 +1855,9 @@ coerce({_, _, contract}, {contract, Bin}, from_fate) ->
|
||||
Address = gmser_api_encoder:encode(contract_pubkey, Bin),
|
||||
{ok, unicode:characters_to_list(Address)};
|
||||
coerce({_, _, signature}, S, to_fate) when is_binary(S) andalso (byte_size(S) =:= 64) ->
|
||||
% Usually to pass a binary in, you need to wrap it as {raw, Binary}, but
|
||||
% since sg_... strings OR hex blobs can be used as signatures in Sophia, we
|
||||
% special case this case based on the length. Even if a binary starts with
|
||||
% "sg_", 64 characters is not enough to represent a 64 byte signature, so
|
||||
% the most optimistic interpretation is to use the binary directly.
|
||||
% If it is a binary of 64 bytes then it can be used as is... If it is an
|
||||
% sg_... string of 64 bytes, then it is too short to be valid, so just
|
||||
% interpret it as a binary directly.
|
||||
{ok, S};
|
||||
coerce({O, N, signature}, S, to_fate) ->
|
||||
coerce_chain_object(O, N, signature, signature, S);
|
||||
@ -2011,8 +1973,6 @@ coerce_bytes(O, N, Count, Bytes) when byte_size(Bytes) /= Count ->
|
||||
coerce_bytes(_, _, _, Bytes) ->
|
||||
{ok, Bytes}.
|
||||
|
||||
coerce_chain_object(_, _, _, _, {raw, Binary}) ->
|
||||
{ok, Binary};
|
||||
coerce_chain_object(O, N, T, Tag, S) ->
|
||||
case decode_chain_object(Tag, S) of
|
||||
{ok, Data} -> {ok, coerce_chain_object2(T, Data)};
|
||||
@ -2210,34 +2170,11 @@ zip_record_field({Name, Type}, {Remaining, Missing}) ->
|
||||
{missing, {Remaining, [Name | Missing]}}
|
||||
end.
|
||||
|
||||
|
||||
-spec cache_aaci(Label, AACI) -> ok
|
||||
when Label :: term(),
|
||||
AACI :: aaci().
|
||||
%% @doc
|
||||
%% Caches an AACI for future reference in calls that would otherwise require
|
||||
%% the AACI as an argument. Once cached, a pre-built AACI can be referenced in
|
||||
%% later calls by substituting the AACI argument with `{aaci, Label}'.
|
||||
|
||||
cache_aaci(Label, AACI) ->
|
||||
hz_man:cache_aaci(Label, AACI).
|
||||
|
||||
|
||||
-spec lookup_aaci(Label) -> Result
|
||||
when Label :: term(),
|
||||
Result :: {ok, aaci()} | error.
|
||||
%% @doc
|
||||
%% Retrieve a previously prepared and cached AACI.
|
||||
|
||||
lookup_aaci(Label) ->
|
||||
hz_man:lookup_aaci(Label).
|
||||
|
||||
|
||||
-spec aaci_lookup_spec(AACI, Fun) -> {ok, Type} | {error, Reason}
|
||||
when AACI :: aaci() | {aaci, Label :: term()},
|
||||
when AACI :: aaci(),
|
||||
Fun :: binary() | string(),
|
||||
Type :: {term(), term()}, % FIXME
|
||||
Reason :: bad_fun_name | aaci_not_found.
|
||||
Reason :: bad_fun_name.
|
||||
|
||||
%% @doc
|
||||
%% Look up the type information of a given function, in the AACI provided by
|
||||
@ -2248,11 +2185,6 @@ aaci_lookup_spec({aaci, _, FunDefs, _}, Fun) ->
|
||||
case maps:find(Fun, FunDefs) of
|
||||
A = {ok, _} -> A;
|
||||
error -> {error, bad_fun_name}
|
||||
end;
|
||||
aaci_lookup_spec({aaci, Label}, Fun) ->
|
||||
case hz_man:lookup_aaci(Label) of
|
||||
{ok, AACI} -> aaci_lookup_spec(AACI, Fun);
|
||||
error -> {error, aaci_not_found}
|
||||
end.
|
||||
|
||||
-spec min_gas_price() -> integer().
|
||||
@ -2282,12 +2214,7 @@ min_gas() ->
|
||||
encode_call_data({aaci, _ContractName, FunDefs, _TypeDefs}, Fun, Args) ->
|
||||
case maps:find(Fun, FunDefs) of
|
||||
{ok, {ArgDef, _ResultDef}} -> encode_call_data2(ArgDef, Fun, Args);
|
||||
error -> {error, bad_fun_name}
|
||||
end;
|
||||
encode_call_data({aaci, Label}, Fun, Args) ->
|
||||
case hz_man:lookup_aaci(Label) of
|
||||
{ok, AACI} -> encode_call_data(AACI, Fun, Args);
|
||||
error -> {error, aaci_not_found}
|
||||
error -> {error, bad_fun_name}
|
||||
end.
|
||||
|
||||
encode_call_data2(ArgDef, Fun, Args) ->
|
||||
@ -2467,7 +2394,7 @@ spend3(DSenderID,
|
||||
Sig :: binary().
|
||||
|
||||
sign_message(Message, SecKey) ->
|
||||
Prefix = message_sig_prefix(),
|
||||
Prefix = <<"Gajumaru Signed Message:\n">>,
|
||||
{ok, PSize} = vencode(byte_size(Prefix)),
|
||||
{ok, MSize} = vencode(byte_size(Message)),
|
||||
Smashed = iolist_to_binary([PSize, Prefix, MSize, Message]),
|
||||
@ -2476,7 +2403,7 @@ sign_message(Message, SecKey) ->
|
||||
|
||||
|
||||
-spec verify_signature(Sig, Message, PubKey) -> Result
|
||||
when Sig :: string(), % base64 encoded signature,
|
||||
when Sig :: binary(),
|
||||
Message :: iodata(),
|
||||
PubKey :: pubkey(),
|
||||
Result :: {ok, Outcome :: boolean()}
|
||||
@ -2501,7 +2428,7 @@ verify_signature2(Sig, Message, PK) ->
|
||||
% the user from accidentally signing a transaction disguised as a message.
|
||||
%
|
||||
% Salt the message then hash with blake2b.
|
||||
Prefix = message_sig_prefix(),
|
||||
Prefix = <<"Gajumaru Signed Message:\n">>,
|
||||
{ok, PSize} = vencode(byte_size(Prefix)),
|
||||
{ok, MSize} = vencode(byte_size(Message)),
|
||||
Smashed = iolist_to_binary([PSize, Prefix, MSize, Message]),
|
||||
@ -2511,7 +2438,6 @@ verify_signature2(Sig, Message, PK) ->
|
||||
Result = ecu_eddsa:sign_verify_detached(Signature, Hashed, PK),
|
||||
{ok, Result}.
|
||||
|
||||
message_sig_prefix() -> <<"Gajumaru Signed Message:\n">>.
|
||||
|
||||
% This is Bitcoin's variable-length unsigned integer encoding
|
||||
% See: https://en.bitcoin.it/wiki/Protocol_documentation#Variable_length_integer
|
||||
@ -2539,42 +2465,6 @@ eu(N, Size) ->
|
||||
<<Bytes/binary, ExtraZeros/binary>>.
|
||||
|
||||
|
||||
-spec sign_binary(Binary, SecKey) -> Sig
|
||||
when Binary :: binary(),
|
||||
SecKey :: binary(),
|
||||
Sig :: binary().
|
||||
|
||||
sign_binary(Binary, SecKey) ->
|
||||
Prefix = binary_sig_prefix(),
|
||||
Target = <<Prefix/binary, Binary/binary>>,
|
||||
{ok, Hash} = eblake2:blake2b(32, Target),
|
||||
ecu_eddsa:sign_detached(Hash, SecKey).
|
||||
|
||||
|
||||
-spec verify_bin_signature(Sig, Binary, PubKey) -> Result
|
||||
when Sig :: string(), % base64 encoded signature,
|
||||
Binary :: binary(),
|
||||
PubKey :: pubkey(),
|
||||
Result :: {ok, Outcome :: boolean()}
|
||||
| {error, Reason :: term()}.
|
||||
|
||||
verify_bin_signature(Sig, Binary, PubKey) ->
|
||||
case gmser_api_encoder:decode(PubKey) of
|
||||
{account_pubkey, PK} -> verify_bin_signature2(Sig, Binary, PK);
|
||||
Other -> {error, {bad_key, Other}}
|
||||
end.
|
||||
|
||||
verify_bin_signature2(Sig, Binary, PK) ->
|
||||
Prefix = binary_sig_prefix(),
|
||||
Target = <<Prefix/binary, Binary/binary>>,
|
||||
{ok, Hash} = eblake2:blake2b(32, Target),
|
||||
Signature = base64:decode(Sig),
|
||||
Result = ecu_eddsa:sign_verify_detached(Signature, Hash, PK),
|
||||
{ok, Result}.
|
||||
|
||||
binary_sig_prefix() -> <<"Gajumaru Signed Binary:">>.
|
||||
|
||||
|
||||
%%% Debug functionality
|
||||
|
||||
% debug_network() ->
|
||||
@ -2673,7 +2563,6 @@ coerce_signature_binary_test() ->
|
||||
169,85,212,142,14,12,233,252,97,50,193,158,229,51,123,206,222,
|
||||
249,2,3,85,173,106,150,243,253,89,128,248,52,195,140,95,114,
|
||||
233,110,119,143,206,137,124,36,63,154,85,7>>,
|
||||
{ok, Binary} = coerce(Type, {raw, Binary}, to_fate),
|
||||
{ok, Binary} = coerce(Type, Binary, to_fate),
|
||||
ok.
|
||||
|
||||
|
||||
@ -1,10 +1,12 @@
|
||||
-module(hz_fetcher).
|
||||
-vsn("0.8.2").
|
||||
-vsn("0.6.1").
|
||||
-author("Craig Everett <ceverett@tsuriai.jp>").
|
||||
-copyright("Craig Everett <ceverett@tsuriai.jp>").
|
||||
-license("MIT").
|
||||
|
||||
-export([connect/4, connect_slowly/4]).
|
||||
-export([connect/4, slowly_connect/4]).
|
||||
|
||||
-include("$zx_include/zx_logger.hrl").
|
||||
|
||||
|
||||
connect(Node = {Host, Port}, Request, From, Timeout) ->
|
||||
@ -76,7 +78,7 @@ parse(Received, Sock, From, Timer) ->
|
||||
<<"HTTP/1.1 500 Internal Server Error\r\n", Tail/binary>> ->
|
||||
parse2(500, Tail, Sock, From, Timer);
|
||||
_ ->
|
||||
ok = disconnect(Sock),
|
||||
ok = zx_net:disconnect(Sock),
|
||||
ok = erlang:cancel_timer(Timer, [{async, true}]),
|
||||
gen_server:reply(From, {error, {received, Received}})
|
||||
end.
|
||||
@ -113,7 +115,7 @@ consume2(Length, Received, Sock, From, Timer) ->
|
||||
if
|
||||
Size == Length ->
|
||||
ok = erlang:cancel_timer(Timer, [{async, true}]),
|
||||
ok = disconnect(Sock),
|
||||
ok = zx_net:disconnect(Sock),
|
||||
Result = zj:decode(Received),
|
||||
gen_server:reply(From, Result);
|
||||
Size < Length ->
|
||||
@ -206,7 +208,7 @@ read_hval(_, Received, _, _, _) ->
|
||||
{error, headers}.
|
||||
|
||||
|
||||
connect_slowly(Node, {get, Path}, From, Timeout) ->
|
||||
slowly_connect(Node, {get, Path}, From, Timeout) ->
|
||||
HttpOptions = [{connect_timeout, 3000}, {timeout, Timeout}],
|
||||
URL = lists:flatten(url(Node, Path)),
|
||||
Request = {URL, []},
|
||||
@ -217,7 +219,7 @@ connect_slowly(Node, {get, Path}, From, Timeout) ->
|
||||
BAD -> {error, BAD}
|
||||
end,
|
||||
gen_server:reply(From, Result);
|
||||
connect_slowly(Node, {post, Path, Payload}, From, Timeout) ->
|
||||
slowly_connect(Node, {post, Path, Payload}, From, Timeout) ->
|
||||
HttpOptions = [{connect_timeout, 3000}, {timeout, Timeout}],
|
||||
URL = lists:flatten(url(Node, Path)),
|
||||
Request = {URL, [], "application/json", Payload},
|
||||
@ -234,45 +236,3 @@ url({Node, Port}, Path) when is_list(Node) ->
|
||||
["https://", Node, ":", integer_to_list(Port), Path];
|
||||
url({Node, Port}, Path) when is_tuple(Node) ->
|
||||
["https://", inet:ntoa(Node), ":", integer_to_list(Port), Path].
|
||||
|
||||
|
||||
disconnect(Socket) ->
|
||||
case peername(Socket) of
|
||||
{ok, {Addr, Port}} ->
|
||||
Host = inet:ntoa(Addr),
|
||||
disconnect(Socket, Host, Port);
|
||||
{error, Reason} ->
|
||||
log(warning, "Disconnect failed with: ~w", [Reason])
|
||||
end.
|
||||
|
||||
disconnect(Socket, Host, Port) ->
|
||||
case gen_tcp:shutdown(Socket, read_write) of
|
||||
ok ->
|
||||
ok;
|
||||
{error, enotconn} ->
|
||||
receive
|
||||
{tcp_closed, Socket} -> ok
|
||||
after 0 -> ok
|
||||
end;
|
||||
{error, E} ->
|
||||
ok = log(warning, "~ts:~w disconnect failed with: ~w", [Host, Port, E]),
|
||||
receive
|
||||
{tcp_closed, Socket} -> ok
|
||||
after 0 -> ok
|
||||
end
|
||||
end.
|
||||
|
||||
peername(Socket) ->
|
||||
case inet:peername(Socket) of
|
||||
{ok, {{0, 0, 0, 0, 0, 65535, X, Y}, Port}} ->
|
||||
<<A:8, B:8, C:8, D:8>> = <<X:16, Y:16>>,
|
||||
{ok, {{A, B, C, D}, Port}};
|
||||
Other ->
|
||||
Other
|
||||
end.
|
||||
|
||||
|
||||
log(Level, Format, Args) ->
|
||||
Raw = io_lib:format("~w ~w: " ++ Format, [?MODULE, self() | Args]),
|
||||
Entry = unicode:characters_to_list(Raw),
|
||||
logger:log(Level, Entry).
|
||||
|
||||
@ -1,728 +0,0 @@
|
||||
%%% @doc
|
||||
%%% Formatting and reading functions for Gaju and Puck quantities
|
||||
%%%
|
||||
%%% The numbers involved in dealing with blockchain amounts are enormous
|
||||
%%% by comparison to legacy forms of currency. It isn't so much that
|
||||
%%% thousands of Gajus is hard to reason about, but rather that quadrillions
|
||||
%%% of Pucks is quite hard to even lock on to visually.
|
||||
%%%
|
||||
%%% A normal commas and underscores method of decimal formatting is provided, as
|
||||
%%% `us' formatting along with two additional approaches:
|
||||
%%% - Japanese traditional myriad structure (`jp' style)
|
||||
%%% - An internationalized variant inspired by the Japanese technique over periods
|
||||
%%% (`metric' for SI prefixes, and `legacy' for Anglicized prefixes)
|
||||
%%%
|
||||
%%% These are all accessible via the `amount/N' functions.
|
||||
%%%
|
||||
%%% The `read/1' function can accept any of the output variants as a string and
|
||||
%%% will return the number of pucks indicated by the provided string, allowing for
|
||||
%%% simple copy/paste functionality as well as direct input using any of the
|
||||
%%% supported notations.
|
||||
%%% @end
|
||||
|
||||
-module(hz_format).
|
||||
-vsn("0.8.2").
|
||||
-author("Craig Everett <ceverett@tsuriai.jp>").
|
||||
-copyright("Craig Everett <ceverett@tsuriai.jp>").
|
||||
-license("GPL-3.0-or-later").
|
||||
|
||||
-export([amount/1, amount/2, amount/3,
|
||||
approx_amount/2, approx_amount/3,
|
||||
read/1,
|
||||
one/1, mark/1,
|
||||
price_to_string/1, string_to_price/1]).
|
||||
|
||||
-spec amount(Pucks) -> Formatted
|
||||
when Pucks :: integer(),
|
||||
Formatted :: string().
|
||||
%% @doc
|
||||
%% A convenience formatting function.
|
||||
%% ```
|
||||
%% hz_format:amount(1) ->
|
||||
%% 木0.000,000,000,000,000,001
|
||||
%%
|
||||
%% hz_format:amount(5000) ->
|
||||
%% 木0.000,000,000,000,005
|
||||
%%
|
||||
%% hz_format:amount(5000000000000000000) ->
|
||||
%% 木5
|
||||
%%
|
||||
%% hz_format:amount(500000123000000000000000) ->
|
||||
%% 木500,000.123
|
||||
%% '''
|
||||
%% @equiv amount(us, Pucks)
|
||||
|
||||
amount(Pucks) ->
|
||||
amount(us, Pucks).
|
||||
|
||||
|
||||
-spec amount(Style, Pucks) -> Formatted
|
||||
when Style :: us | jp | metric | legacy | {Separator, Span},
|
||||
Separator :: $, | $_,
|
||||
Span :: 3 | 4,
|
||||
Pucks :: integer(),
|
||||
Formatted :: string().
|
||||
%% @doc
|
||||
%% A money formatting function.
|
||||
%% ```
|
||||
%% hz_format:amount(us, 100500040123000000000000000) ->
|
||||
%% 木100,500,040.123
|
||||
%%
|
||||
%% hz_format:amount(jp, 100500040123000000000000000) ->
|
||||
%% 1億50万40木 12京3000兆本
|
||||
%%
|
||||
%% hz_format:amount(metric, 100500040123000000000000000) ->
|
||||
%% 木100m 500k 40 G 123p P
|
||||
%%
|
||||
%% hz_format:amount(legacy, 100500040123000000000000000) ->
|
||||
%% 木100m 500k 40 G 123q P
|
||||
%%
|
||||
%% hz_format:amount({$_, 3}, 100500040123000000000000000) ->
|
||||
%% 木100_500_040.123
|
||||
%%
|
||||
%% hz_format:amount({$_, 4}, 100500040123000000000000000) ->
|
||||
%% 木1_0050_0040.123
|
||||
%% '''
|
||||
%% @equiv amount(gaju, Style, Pucks)
|
||||
|
||||
amount(Style, Pucks) ->
|
||||
amount(gaju, Style, Pucks).
|
||||
|
||||
|
||||
-spec amount(Unit, Style, Pucks) -> Formatted
|
||||
when Unit :: gaju | puck,
|
||||
Style :: us | jp | metric | legacy | {Separator, Span},
|
||||
Separator :: $, | $_,
|
||||
Span :: 3 | 4,
|
||||
Pucks :: integer(),
|
||||
Formatted :: string().
|
||||
%% @doc
|
||||
%% A simplified format function covering the most common formats desired.
|
||||
%% ```
|
||||
%% hz_format:amount(gaju, us, 100500040123000004500000000) ->
|
||||
%% 木100,500,040.123,000,004,5
|
||||
%%
|
||||
%% hz_format:amount(puck, us, 100500040123000004500000000) ->
|
||||
%% 本100,500,040,123,000,004,500,000,000
|
||||
%%
|
||||
%% hz_format:amount(gaju, jp, 100500040123000004500000000) ->
|
||||
%% 1億50万40木 12京3000兆45億本
|
||||
%%
|
||||
%% hz_format:amount(puck, jp, 100500040123000004500000000) ->
|
||||
%% 100秭5000垓4012京3000兆45億本
|
||||
%%
|
||||
%% hz_format:amount(gaju, metric, 100500040123000004500000000) ->
|
||||
%% 木100m 500k 40 G 123p 4g 500m P
|
||||
%%
|
||||
%% hz_format:amount(puck, metric, 100500040123000004500000000) ->
|
||||
%% 本100y 500z 40e 123p 4g 500m P
|
||||
%%
|
||||
%% hz_format:amount(gaju, legacy, 100500040123000004500000000) ->
|
||||
%% 木100m 500k 40 G 123q 4b 500m P
|
||||
%%
|
||||
%% hz_format:amount(puck, legacy, 100500040123000004500000000) ->
|
||||
%% 本100y 500z 40e 123q 4b 500m P
|
||||
%% '''
|
||||
|
||||
amount(gaju, us, Pucks) ->
|
||||
western($,, $., 3, all, Pucks);
|
||||
amount(puck, us, Pucks) ->
|
||||
western($,, 3, Pucks);
|
||||
amount(Unit, jp, Pucks) ->
|
||||
jp(Unit, all, Pucks);
|
||||
amount(Unit, metric, Pucks) ->
|
||||
bestern(Unit, ranks(metric), all, Pucks);
|
||||
amount(Unit, legacy, Pucks) ->
|
||||
bestern(Unit, ranks(heresy), all, Pucks);
|
||||
amount(gaju, {Separator, Span}, Pucks) ->
|
||||
western(Separator, $., Span, all, Pucks);
|
||||
amount(puck, {Separator, Span}, Pucks) ->
|
||||
western(Separator, Span, Pucks).
|
||||
|
||||
|
||||
|
||||
-spec approx_amount(Precision, Pucks) -> Serialized
|
||||
when Precision :: all | 0..18,
|
||||
Pucks :: integer(),
|
||||
Serialized :: string().
|
||||
%% A formatter for decimal notation which permits a precision
|
||||
%% value to be applied to the puck side of the format.
|
||||
%% ```
|
||||
%% hz_format:approx_amount(3, 100500040123000004500000001) ->
|
||||
%% 木100,500,040.123
|
||||
%%
|
||||
%% hz_format:approx_amount(13, 100500040123000004500000001) ->
|
||||
%% 木100,500,040.123,000,004,5...
|
||||
%%
|
||||
%% hz_format:approx_amount(all, 100500040123000004500000001) ->
|
||||
%% 木100,500,040.123,000,004,500,000,001
|
||||
%% '''
|
||||
%% @equiv approx_amount(us, Precision, Pucks)
|
||||
|
||||
approx_amount(Precision, Pucks) ->
|
||||
approx_amount(us, Precision, Pucks).
|
||||
|
||||
|
||||
-spec approx_amount(Style, Precision, Pucks) -> Serialized
|
||||
when Style :: us | {Separator, Span},
|
||||
Precision :: all | 0..18,
|
||||
Separator :: $, | $_,
|
||||
Span :: 3 | 4,
|
||||
Pucks :: integer(),
|
||||
Serialized :: string().
|
||||
%% @doc
|
||||
%% A formatter for decimal notation which permits a precision
|
||||
%% value to be applied to the puck side of the format.
|
||||
%% ```
|
||||
%% hz_format:approx_amount({$_, 3}, 3, 100500040123000004500000001) ->
|
||||
%% 木100_500_040.123...
|
||||
%%
|
||||
%% hz_format:approx_amount({$_, 4}, 12, 100500040123000004500000001) ->
|
||||
%% 木1_0050_0040.1230_0000_45...
|
||||
%% '''
|
||||
|
||||
approx_amount(us, Precision, Pucks) ->
|
||||
western($,, $., 3, Precision, Pucks);
|
||||
approx_amount({Separator, Span}, Precision, Pucks) ->
|
||||
western(Separator, $., Span, Precision, Pucks).
|
||||
|
||||
|
||||
western(Separator, Span, Pucks) when Pucks >= 0 ->
|
||||
western2(Separator, Span, Pucks);
|
||||
western(Separator, Span, Pucks) when Pucks < 0 ->
|
||||
[$- | western2(Separator, Span, Pucks * -1)].
|
||||
|
||||
western2(Separator, Span, Pucks) ->
|
||||
P = lists:reverse(integer_to_list(Pucks)),
|
||||
[mark(puck) | separate(Separator, Span, P)].
|
||||
|
||||
|
||||
western(Separator, Break, Span, Precision, Pucks) when Pucks >= 0 ->
|
||||
western2(Separator, Break, Span, Precision, Pucks);
|
||||
western(Separator, Break, Span, Precision, Pucks) when Pucks < 0 ->
|
||||
[$- | western2(Separator, Break, Span, Precision, Pucks * -1)].
|
||||
|
||||
|
||||
western2(Separator, _, Span, 0, Pucks) ->
|
||||
G = lists:reverse(integer_to_list(Pucks div one(gaju))),
|
||||
[mark(gaju) | separate(Separator, Span, G)];
|
||||
western2(Separator, Break, Span, Precision, Pucks) ->
|
||||
SP = integer_to_list(Pucks),
|
||||
Length = length(SP),
|
||||
Over18 = Length > 18,
|
||||
NoPucks = (Pucks rem one(gaju)) =:= 0,
|
||||
case {Over18, NoPucks} of
|
||||
{true, true} ->
|
||||
Gs = lists:reverse(lists:sublist(SP, Length - 18)),
|
||||
[mark(gaju) | separate(Separator, Span, Gs)];
|
||||
{true, false} ->
|
||||
{PChars, GChars} = lists:split(18, lists:reverse(SP)),
|
||||
H = [mark(gaju) | separate(Separator, Span, GChars)],
|
||||
{P, E} = decimal_pucks(Precision, lists:reverse(PChars)),
|
||||
T = lists:reverse(separate(Separator, Span, P)),
|
||||
lists:flatten([H, Break, T, E]);
|
||||
{false, true} ->
|
||||
[mark(gaju), $0];
|
||||
{false, false} ->
|
||||
PChars = lists:flatten(string:pad(SP, 18, leading, $0)),
|
||||
{P, E} = decimal_pucks(Precision, PChars),
|
||||
T = lists:reverse(separate(Separator, Span, P)),
|
||||
lists:flatten([mark(gaju), $0, Break, T, E])
|
||||
end.
|
||||
|
||||
decimal_pucks(all, PChars) ->
|
||||
RTrailing = lists:reverse(PChars),
|
||||
{lists:reverse(lists:dropwhile(fun(C) -> C =:= $0 end, RTrailing)), ""};
|
||||
decimal_pucks(Precision, PChars) ->
|
||||
{Significant, Rest} = lists:split(min(Precision, 18), PChars),
|
||||
RTrailing = lists:reverse(Significant),
|
||||
Trailing = lists:reverse(lists:dropwhile(fun(C) -> C =:= $0 end, RTrailing)),
|
||||
case lists:all(fun(C) -> C =:= $0 end, Rest) of
|
||||
true -> {Trailing, ""};
|
||||
false -> {Trailing, "..."}
|
||||
end.
|
||||
|
||||
separate(_, _, "") ->
|
||||
"";
|
||||
separate(S, P, G) ->
|
||||
separate(S, P, 1, G, []).
|
||||
|
||||
separate(_, _, _, [H], A) ->
|
||||
[H | A];
|
||||
separate(S, P, P, [H | T], A) ->
|
||||
separate(S, P, 1, T, [S, H | A]);
|
||||
separate(S, P, N, [H | T], A) ->
|
||||
separate(S, P, N + 1, T, [H | A]).
|
||||
|
||||
|
||||
bestern(gaju, Ranks, Precision, Pucks) when Pucks >= 0 ->
|
||||
[mark(gaju), bestern2(gaju, Ranks, 3, Precision, Pucks)];
|
||||
bestern(gaju, Ranks, Precision, Pucks) when Pucks < 0 ->
|
||||
[$-, mark(gaju), bestern2(gaju, Ranks, 3, Precision, Pucks * -1)];
|
||||
bestern(puck, Ranks, Precision, Pucks) when Pucks >= 0 ->
|
||||
[mark(puck), bestern2(puck, Ranks, 3, Precision, Pucks)];
|
||||
bestern(puck, Ranks, Precision, Pucks) when Pucks < 0 ->
|
||||
[$-, mark(puck), bestern2(puck, Ranks, 3, Precision, Pucks * -1)].
|
||||
|
||||
jp(Unit, Precision, Pucks) when Pucks >= 0 ->
|
||||
bestern2(Unit, ranks(jp), 4, Precision, Pucks);
|
||||
jp(Unit, Precision, Pucks) when Pucks < 0 ->
|
||||
[$-, bestern2(Unit, ranks(jp), 4, Precision, Pucks * -1)].
|
||||
|
||||
bestern2(gaju, Ranks, Span, 0, Pucks) ->
|
||||
G = lists:reverse(integer_to_list(Pucks div one(gaju))),
|
||||
case Span of
|
||||
3 -> period("G", Ranks, G);
|
||||
4 -> myriad(mark(gaju), Ranks, G)
|
||||
end;
|
||||
bestern2(gaju, Ranks, Span, all, Pucks) ->
|
||||
P = lists:flatten(string:pad(integer_to_list(Pucks rem one(gaju)), 18, leading, $0)),
|
||||
Zilch = lists:all(fun(C) -> C =:= $0 end, P),
|
||||
{H, T} =
|
||||
case {Span, Zilch} of
|
||||
{3, false} -> {bestern2(gaju, Ranks, 3, 0, Pucks), period("P", Ranks, lists:reverse(P))};
|
||||
{4, false} -> {jp(gaju, 0, Pucks), myriad(mark(puck), Ranks, lists:reverse(P))};
|
||||
{3, true} -> {bestern2(gaju, Ranks, 3, 0, Pucks), ""};
|
||||
{4, true} -> {jp(gaju, 0, Pucks), ""}
|
||||
end,
|
||||
lists:flatten([H, " ", T]);
|
||||
bestern2(gaju, Ranks, Span, Precision, Pucks) ->
|
||||
P = lists:flatten(string:pad(integer_to_list(Pucks rem one(gaju)), 18, leading, $0)),
|
||||
H =
|
||||
case Span of
|
||||
3 -> bestern2(gaju, Ranks, 3, 0, Pucks);
|
||||
4 -> jp(gaju, 0, Pucks)
|
||||
end,
|
||||
Digits = min(Precision, 18),
|
||||
T =
|
||||
case length(P) < Digits of
|
||||
false ->
|
||||
ReverseP = lists:reverse(lists:sublist(P, Digits)),
|
||||
PuckingString = lists:flatten(string:pad(ReverseP, 18, leading, $0)),
|
||||
case lists:all(fun(C) -> C =:= $0 end, PuckingString) of
|
||||
false ->
|
||||
case Span of
|
||||
3 -> period("P", Ranks, PuckingString);
|
||||
4 -> myriad(mark(puck), Ranks, PuckingString)
|
||||
end;
|
||||
true ->
|
||||
""
|
||||
end;
|
||||
true ->
|
||||
[]
|
||||
end,
|
||||
lists:flatten([H, " ", T]);
|
||||
bestern2(puck, Ranks, Span, all, Pucks) ->
|
||||
P = lists:reverse(integer_to_list(Pucks)),
|
||||
case lists:all(fun(C) -> C =:= $0 end, P) of
|
||||
false ->
|
||||
case Span of
|
||||
3 -> period("P", Ranks, P);
|
||||
4 -> myriad(mark(puck), Ranks, P)
|
||||
end;
|
||||
true ->
|
||||
case Span of
|
||||
3 -> [$0, " P"];
|
||||
4 -> [$0, mark(puck)]
|
||||
end
|
||||
end;
|
||||
bestern2(puck, Ranks, Span, Precision, Pucks) ->
|
||||
Digits = min(Precision, 18),
|
||||
P = lists:flatten(string:pad(integer_to_list(Pucks), 18, leading, $0)),
|
||||
case length(P) < Digits of
|
||||
true ->
|
||||
case Span of
|
||||
3 -> [$0, " P"];
|
||||
4 -> [$0, mark(puck)]
|
||||
end;
|
||||
false ->
|
||||
PucksToGive = lists:sublist(P, Digits),
|
||||
PuckingString = lists:flatten(string:pad(lists:reverse(PucksToGive), 18, leading, $0)),
|
||||
case lists:all(fun(C) -> C =:= $0 end, PuckingString) of
|
||||
false ->
|
||||
case Span of
|
||||
3 -> period("P", Ranks, PuckingString);
|
||||
4 -> myriad(mark(puck), Ranks, PuckingString)
|
||||
end;
|
||||
true ->
|
||||
case Span of
|
||||
3 -> [$0, " P"];
|
||||
4 -> [$0, mark(puck)]
|
||||
end
|
||||
end
|
||||
end.
|
||||
|
||||
|
||||
period(Symbol, Ranks, [$0, $0, $0 | PT]) ->
|
||||
rank3(Ranks, PT, [Symbol]);
|
||||
period(Symbol, Ranks, [P3, $0, $0 | PT]) ->
|
||||
rank3(Ranks, PT, [P3, 32, Symbol]);
|
||||
period(Symbol, Ranks, [P3, P2, $0 | PT]) ->
|
||||
rank3(Ranks, PT, [P2, P3, 32, Symbol]);
|
||||
period(Symbol, Ranks, [P3, P2, P1 | PT]) ->
|
||||
rank3(Ranks, PT, [P1, P2, P3, 32, Symbol]);
|
||||
period(Symbol, _, [P3]) ->
|
||||
[P3, 32, Symbol];
|
||||
period(Symbol, _, [P3, P2]) ->
|
||||
[P2, P3, 32, Symbol].
|
||||
|
||||
rank3([_ | RT], [$0, $0, $0 | PT], A) ->
|
||||
rank3(RT, PT, A);
|
||||
rank3([RH | RT], [P3, $0, $0 | PT], A) ->
|
||||
rank3(RT, PT, [P3, RH | A]);
|
||||
rank3([RH | RT], [P3, P2, $0 | PT], A) ->
|
||||
rank3(RT, PT, [P2, P3, RH | A]);
|
||||
rank3([RH | RT], [P3, P2, P1 | PT], A) ->
|
||||
rank3(RT, PT, [P1, P2, P3, RH | A]);
|
||||
rank3(_, [$0, $0, $0], A) ->
|
||||
A;
|
||||
rank3(_, [$0, $0], A) ->
|
||||
A;
|
||||
rank3(_, [$0], A) ->
|
||||
A;
|
||||
rank3(_, [], A) ->
|
||||
A;
|
||||
rank3([RH | _], [P3, $0, $0], A) ->
|
||||
[P3, RH | A];
|
||||
rank3([RH | _], [P3, $0], A) ->
|
||||
[P3, RH | A];
|
||||
rank3([RH | _], [P3], A) ->
|
||||
[P3, RH | A];
|
||||
rank3([RH | _], [P3, P2, $0], A) ->
|
||||
[P2, P3, RH | A];
|
||||
rank3([RH | _], [P3, P2], A) ->
|
||||
[P2, P3, RH | A];
|
||||
rank3([RH | _], [P3, P2, P1], A) ->
|
||||
[P1, P2, P3, RH | A].
|
||||
|
||||
|
||||
myriad(Symbol, Ranks, [$0, $0, $0, $0 | PT]) ->
|
||||
rank4(Ranks, PT, [Symbol]);
|
||||
myriad(Symbol, Ranks, [P4, $0, $0, $0 | PT]) ->
|
||||
rank4(Ranks, PT, [P4, Symbol]);
|
||||
myriad(Symbol, Ranks, [P4, P3, $0, $0 | PT]) ->
|
||||
rank4(Ranks, PT, [P3, P4, Symbol]);
|
||||
myriad(Symbol, Ranks, [P4, P3, P2, $0 | PT]) ->
|
||||
rank4(Ranks, PT, [P2, P3, P4, Symbol]);
|
||||
myriad(Symbol, Ranks, [P4, P3, P2, P1 | PT]) ->
|
||||
rank4(Ranks, PT, [P1, P2, P3, P4, Symbol]);
|
||||
myriad(Symbol, _, [P4]) ->
|
||||
[P4, Symbol];
|
||||
myriad(Symbol, _, [P4, P3]) ->
|
||||
[P3, P4, Symbol];
|
||||
myriad(Symbol, _, [P4, P3, P2]) ->
|
||||
[P2, P3, P4, Symbol].
|
||||
|
||||
rank4([_ | RT], [$0, $0, $0, $0 | PT], A) ->
|
||||
rank4(RT, PT, A);
|
||||
rank4([RH | RT], [P4, $0, $0, $0 | PT], A) ->
|
||||
rank4(RT, PT, [P4, RH | A]);
|
||||
rank4([RH | RT], [P4, P3, $0, $0 | PT], A) ->
|
||||
rank4(RT, PT, [P3, P4, RH | A]);
|
||||
rank4([RH | RT], [P4, P3, P2, $0 | PT], A) ->
|
||||
rank4(RT, PT, [P2, P3, P4, RH | A]);
|
||||
rank4([RH | RT], [P4, P3, P2, P1 | PT], A) ->
|
||||
rank4(RT, PT, [P1, P2, P3, P4, RH | A]);
|
||||
rank4(_, [$0, $0, $0, $0], A) ->
|
||||
A;
|
||||
rank4(_, [$0, $0, $0], A) ->
|
||||
A;
|
||||
rank4(_, [$0, $0], A) ->
|
||||
A;
|
||||
rank4(_, [$0], A) ->
|
||||
A;
|
||||
rank4(_, [], A) ->
|
||||
A;
|
||||
rank4([RH | _], [P4, $0, $0, $0], A) ->
|
||||
[P4, RH | A];
|
||||
rank4([RH | _], [P4, $0, $0], A) ->
|
||||
[P4, RH | A];
|
||||
rank4([RH | _], [P4, $0], A) ->
|
||||
[P4, RH | A];
|
||||
rank4([RH | _], [P4], A) ->
|
||||
[P4, RH | A];
|
||||
rank4([RH | _], [P4, P3, $0, $0], A) ->
|
||||
[P3, P4, RH | A];
|
||||
rank4([RH | _], [P4, P3, $0], A) ->
|
||||
[P3, P4, RH | A];
|
||||
rank4([RH | _], [P4, P3], A) ->
|
||||
[P3, P4, RH | A];
|
||||
rank4([RH | _], [P4, P3, P2, $0], A) ->
|
||||
[P2, P3, P4, RH | A];
|
||||
rank4([RH | _], [P4, P3, P2], A) ->
|
||||
[P2, P3, P4, RH | A];
|
||||
rank4([RH | _], [P4, P3, P2, P1], A) ->
|
||||
[P1, P2, P3, P4, RH | A].
|
||||
|
||||
ranks(jp) ->
|
||||
"万億兆京垓秭穣溝澗正載極";
|
||||
ranks(metric) ->
|
||||
["k ", "m ", "g ", "t ", "p ", "e ", "z ", "y ", "r ", "Q "];
|
||||
ranks(heresy) ->
|
||||
["k ", "m ", "b ", "t ", "q ", "e ", "z ", "y ", "r ", "Q "].
|
||||
|
||||
|
||||
mark(gaju) -> $木;
|
||||
mark(puck) -> $本.
|
||||
|
||||
one(gaju) -> 1_000_000_000_000_000_000;
|
||||
one(puck) -> 1.
|
||||
|
||||
|
||||
-spec read(Format) -> Result
|
||||
when Format :: string(),
|
||||
Result :: {ok, Pucks} | error,
|
||||
Pucks :: integer().
|
||||
%% @doc
|
||||
%% Convert any valid string formatted representation and output a value in pucks.
|
||||
%% NOTE: This function does not accept approximated values.
|
||||
%% ```
|
||||
%% 1> hz_format:read("木100,500,040.123,000,004,5").
|
||||
%% {ok,100500040123000004500000000}
|
||||
%% 2> hz_format:read("本100,500,040,123,000,004,500,000,000").
|
||||
%% {ok,100500040123000004500000000}
|
||||
%% 3> hz_format:read("1億50万40木 12京3000兆45億本").
|
||||
%% {ok,100500040123000004500000000}
|
||||
%% 4> hz_format:read("100秭5000垓4012京3000兆45億本").
|
||||
%% {ok,100500040123000004500000000}
|
||||
%% 5> hz_format:read("木100m 500k 40 G 123p 4g 500m P").
|
||||
%% {ok,100500040123000004500000000}
|
||||
%% 6> hz_format:read("本100y 500z 40e 123p 4g 500m P").
|
||||
%% {ok,100500040123000004500000000}
|
||||
%% 7> hz_format:read("木100m 500k 40 G 123q 4b 500m P").
|
||||
%% {ok,100500040123000004500000000}
|
||||
%% 8> hz_format:read("本100y 500z 40e 123q 4b 500m P").
|
||||
%% {ok,100500040123000004500000000}
|
||||
%% '''
|
||||
|
||||
read([$木 | Rest]) ->
|
||||
read_w_gajus(Rest, []);
|
||||
read([$本 | Rest]) ->
|
||||
read_w_pucks(Rest, []);
|
||||
read([C | Rest])
|
||||
when C =:= $- orelse
|
||||
C =:= $− orelse
|
||||
C =:= $- ->
|
||||
case read(Rest) of
|
||||
{ok, Pucks} -> {ok, Pucks * -1};
|
||||
Error -> Error
|
||||
end;
|
||||
read([C | Rest])
|
||||
when C =:= 32 orelse % ASCII space
|
||||
C =:= 12288 orelse % full-width space
|
||||
C =:= $\t orelse
|
||||
C =:= $\r orelse
|
||||
C =:= $\n ->
|
||||
read(Rest);
|
||||
read([C | Rest]) when $0 =< C andalso C =< $9 ->
|
||||
read(Rest, [C], []);
|
||||
read([C | Rest]) when $0 =< C andalso C =< $9 ->
|
||||
NumC = C - $0 + $0,
|
||||
read(Rest, [NumC], []);
|
||||
read(String) when is_binary(String) ->
|
||||
read(binary_to_list(String));
|
||||
read(_) ->
|
||||
error.
|
||||
|
||||
read_w_gajus([C | Rest], A) when $0 =< C andalso C =< $9 ->
|
||||
read_w_gajus(Rest, [C | A]);
|
||||
read_w_gajus([C | Rest], A) when $0 =< C andalso C =< $9 ->
|
||||
NumC = C - $0 + $0,
|
||||
read_w_gajus(Rest, [NumC | A]);
|
||||
read_w_gajus([$, | Rest], A) ->
|
||||
read_w_gajus(Rest, A);
|
||||
read_w_gajus([$_ | Rest], A) ->
|
||||
read_w_gajus(Rest, A);
|
||||
read_w_gajus([$. | Rest], A) ->
|
||||
case read_w_pucks(Rest, []) of
|
||||
{ok, P} ->
|
||||
G = list_to_integer(lists:reverse(A)) * one(gaju),
|
||||
{ok, G + P};
|
||||
Error ->
|
||||
Error
|
||||
end;
|
||||
read_w_gajus([], A) ->
|
||||
G = list_to_integer(lists:reverse(A)) * one(gaju),
|
||||
{ok, G};
|
||||
read_w_gajus([C, 32 | Rest], A) ->
|
||||
read(Rest, [], [{C, A}]);
|
||||
read_w_gajus([32, $G, 32 | Rest], A) ->
|
||||
read(Rest, [], [{$G, A}], []);
|
||||
read_w_gajus([32, $G], A) ->
|
||||
calc([{$G, A}], []);
|
||||
read_w_gajus([32, $P], A) ->
|
||||
calc([], [{$P, A}]);
|
||||
read_w_gajus(_, _) ->
|
||||
error.
|
||||
|
||||
read_w_pucks([C | Rest], A) when $0 =< C andalso C =< $9 ->
|
||||
read_w_pucks(Rest, [C | A]);
|
||||
read_w_pucks([C | Rest], A) when $0 =< C andalso C =< $9 ->
|
||||
NumC = C - $0 + $0,
|
||||
read_w_pucks(Rest, [NumC | A]);
|
||||
read_w_pucks([$, | Rest], A) ->
|
||||
read_w_pucks(Rest, A);
|
||||
read_w_pucks([$_ | Rest], A) ->
|
||||
read_w_pucks(Rest, A);
|
||||
read_w_pucks([C, 32 | Rest], A) ->
|
||||
read(Rest, [], [], [{C, A}]);
|
||||
read_w_pucks([32, $P], A) ->
|
||||
calc([], [{$P, A}]);
|
||||
read_w_pucks([], A) ->
|
||||
Padded = lists:flatten(string:pad(lists:reverse(A), 18, trailing, $0)),
|
||||
{ok, list_to_integer(Padded)}.
|
||||
|
||||
|
||||
read([C | Rest], A, G) when $0 =< C andalso C =< $9 ->
|
||||
read(Rest, [C | A], G);
|
||||
read([C | Rest], A, G) when $0 =< C andalso C =< $9 ->
|
||||
NumC = C - $0 + $0,
|
||||
read(Rest, [NumC | A], G);
|
||||
read([$木], A, G) ->
|
||||
calc([{$G, A} | G], []);
|
||||
read([$G], A, G) ->
|
||||
calc([{$G, A} | G], []);
|
||||
read([32, $G], A, G) ->
|
||||
calc([{$G, A} | G], []);
|
||||
read([32, $G, 32 | Rest], A, G) ->
|
||||
read(Rest, [], [{$G, A} | G], []);
|
||||
read([$木, 32 | Rest], A, G) ->
|
||||
read(Rest, [], [{$G, A} | G], []);
|
||||
read([$G, 32 | Rest], A, G) ->
|
||||
read(Rest, [], [{$G, A} | G], []);
|
||||
read([$本], A, P) ->
|
||||
calc([], [{$P, A} | P]);
|
||||
read([$P], A, P) ->
|
||||
calc([], [{$P, A} | P]);
|
||||
read([32, $P], A, P) ->
|
||||
calc([], [{$P, A} | P]);
|
||||
read([C, 32 | Rest], A, G) ->
|
||||
read(Rest, [], [{C, A} | G]);
|
||||
read([$, | Rest], A, []) ->
|
||||
read_w_gajus(Rest, A);
|
||||
read([$_ | Rest], A, []) ->
|
||||
read_w_gajus(Rest, A);
|
||||
read([$. | Rest], A, []) ->
|
||||
case read_w_pucks(Rest, []) of
|
||||
{ok, P} ->
|
||||
G = list_to_integer(lists:reverse(A)) * one(gaju),
|
||||
{ok, G + P};
|
||||
Error ->
|
||||
Error
|
||||
end;
|
||||
read([C | Rest], A, G) ->
|
||||
read(Rest, [], [{C, A} | G]);
|
||||
read([], A, []) ->
|
||||
read_w_gajus([], A);
|
||||
read(_, _, _) ->
|
||||
error.
|
||||
|
||||
|
||||
read([C | Rest], A, G, P) when $0 =< C andalso C =< $9 ->
|
||||
read(Rest, [C | A], G, P);
|
||||
read([C | Rest], A, G, P) when $0 =< C andalso C =< $9 ->
|
||||
NumC = C - $0 + $0,
|
||||
read(Rest, [NumC | A], G, P);
|
||||
read([$本], A, G, P) ->
|
||||
calc(G, [{$P, A} | P]);
|
||||
read([$P], A, G, P) ->
|
||||
calc(G, [{$P, A} | P]);
|
||||
read([32, $P], A, G, P) ->
|
||||
calc(G, [{$P, A} | P]);
|
||||
read([C, 32 | Rest], A, G, P) ->
|
||||
read(Rest, [], G, [{C, A} | P]);
|
||||
read([C | Rest], A, G, P) ->
|
||||
read(Rest, [], G, [{C, A} | P]);
|
||||
read(_, _, _, _) ->
|
||||
error.
|
||||
|
||||
calc(G, P) ->
|
||||
case calc(gaju, G, 0) of
|
||||
{ok, Gajus} ->
|
||||
case calc(puck, P, 0) of
|
||||
{ok, Pucks} -> {ok, Gajus + Pucks};
|
||||
error -> error
|
||||
end;
|
||||
error ->
|
||||
error
|
||||
end.
|
||||
|
||||
calc(U, [{_, []} | S], A) ->
|
||||
calc(U, S, A);
|
||||
calc(U, [{M, Cs} | S], A) ->
|
||||
case magnitude(M) of
|
||||
{ok, J} ->
|
||||
N = list_to_integer(lists:reverse(Cs)) * J * one(U),
|
||||
calc(U, S, A + N);
|
||||
Error ->
|
||||
Error
|
||||
end;
|
||||
calc(_, [], A) ->
|
||||
{ok, A}.
|
||||
|
||||
|
||||
magnitude($G) ->
|
||||
{ok, 1};
|
||||
magnitude($P) ->
|
||||
{ok, 1};
|
||||
magnitude(Mark) ->
|
||||
case rank(Mark, ranks(jp), 1_0000, 1) of
|
||||
{ok, J} ->
|
||||
{ok, J};
|
||||
error ->
|
||||
case rank([Mark, 32], ranks(metric), 1_000, 1) of
|
||||
{ok, J} -> {ok, J};
|
||||
error -> rank([Mark, 32], ranks(heresy), 1_000, 1)
|
||||
end
|
||||
end.
|
||||
|
||||
rank(Mark, [Mark | _], Magnitude, Sum) ->
|
||||
{ok, Sum * Magnitude};
|
||||
rank(Mark, [_ | Rest], Magnitude, Sum) ->
|
||||
rank(Mark, Rest, Magnitude, Sum * Magnitude);
|
||||
rank(_, [], _, _) ->
|
||||
error.
|
||||
|
||||
|
||||
-spec price_to_string(Pucks) -> Gajus
|
||||
when Pucks :: integer(),
|
||||
Gajus :: string().
|
||||
%% @doc
|
||||
%% A simplified formatting function that converts an integer value in Pucks to a string representation
|
||||
%% in Gajus. Useful for formatting generic output for UI elements
|
||||
|
||||
price_to_string(Pucks) ->
|
||||
Gaju = one(gaju),
|
||||
H = integer_to_list(Pucks div Gaju),
|
||||
R = Pucks rem Gaju,
|
||||
case string:strip(lists:flatten(io_lib:format("~18..0w", [R])), right, $0) of
|
||||
[] -> H;
|
||||
T -> string:join([H, T], ".")
|
||||
end.
|
||||
|
||||
|
||||
-spec string_to_price(Gajus) -> Pucks
|
||||
when Gajus :: string(),
|
||||
Pucks :: integer().
|
||||
%% @doc
|
||||
%% A simplified formatting function that converts a Gaju value represented as a string to an
|
||||
%% integer value in Pucks.
|
||||
|
||||
string_to_price(String) ->
|
||||
case string:split(String, ".") of
|
||||
[H] -> join_price(H, "0");
|
||||
[H, T] -> join_price(H, T);
|
||||
_ -> {error, bad_price}
|
||||
end.
|
||||
|
||||
join_price(H, T) ->
|
||||
try
|
||||
Parts = [H, string:pad(T, 18, trailing, $0)],
|
||||
Price = list_to_integer(unicode:characters_to_list(Parts)),
|
||||
case Price < 0 of
|
||||
false -> {ok, Price};
|
||||
true -> {error, negative_price}
|
||||
end
|
||||
catch
|
||||
error:R -> {error, R}
|
||||
end.
|
||||
@ -37,15 +37,15 @@
|
||||
%%% @end
|
||||
|
||||
-module(hz_grids).
|
||||
-vsn("0.8.2").
|
||||
-export([url/2, url/3, url/4, parse/1, req/2, req/3]).
|
||||
-vsn("0.6.1").
|
||||
-export([url/2, parse/1, req/2, req/3]).
|
||||
|
||||
|
||||
-spec url(Instruction, HTTP) -> Result
|
||||
when Instruction :: spend | transfer | sign,
|
||||
HTTP :: uri_string:uri_string(),
|
||||
Result :: {ok, GRIDS} | uri_string:uri_error(),
|
||||
GRIDS :: uri_string:uri_string().
|
||||
GRIDS :: uri_string:uri_string(),
|
||||
Result :: {ok, GRIDS} | uri_string:uri_error().
|
||||
%% @doc
|
||||
%% Takes
|
||||
|
||||
@ -66,63 +66,6 @@ url2(Instruction, URL = #{path := Path}) ->
|
||||
{ok, uri_string:recompose(GRIDS)}.
|
||||
|
||||
|
||||
-spec url(Instruction, Recipient, Amount) -> GRIDS
|
||||
when Instruction :: {spend, Network} | {transfer, Node},
|
||||
Network :: string(),
|
||||
Node :: {inet:ip_address() | inet:hostname(), inet:port_number()}
|
||||
| uri_string:uri_string(),
|
||||
Recipient :: string(),
|
||||
Amount :: non_neg_integer(),
|
||||
GRIDS :: uri_string:uri_string().
|
||||
%% @doc
|
||||
%% Forms a GRIDS URL for spends or transfers.
|
||||
%% @equiv uri(Instruction, Recipient, Amount, "")
|
||||
|
||||
url(Instruction, Recipient, Amount) ->
|
||||
url(Instruction, Recipient, Amount, "").
|
||||
|
||||
|
||||
-spec url(Instruction, Recipient, Amount, Payload) -> GRIDS
|
||||
when Instruction :: {spend, Network} | {transfer, Node},
|
||||
Network :: string(),
|
||||
Node :: {inet:ip_address() | inet:hostname(), inet:port_number()}
|
||||
| uri_string:uri_string(), % "http://..." | "https://..."
|
||||
Recipient :: string(),
|
||||
Amount :: non_neg_integer() | none,
|
||||
Payload :: binary(),
|
||||
GRIDS :: uri_string:uri_string().
|
||||
%% @doc
|
||||
%% Forms a GRIDS URL for spends or transfers.
|
||||
|
||||
url({spend, Network}, Recipient, Amount, Payload) ->
|
||||
Elements = ["grids://", Network, "/1/s/", Recipient, qwargs(Amount, Payload)],
|
||||
unicode:characters_to_list(Elements);
|
||||
url({transfer, Node}, Recipient, Amount, Payload) ->
|
||||
Prefix =
|
||||
case Node of
|
||||
{H, P} -> ["grid://", h_to_s(H), ":", integer_to_list(P)];
|
||||
"https://" ++ H -> ["grids://", H];
|
||||
"http://" ++ H -> ["grid://", H];
|
||||
<<"https://", H/binary>> -> ["grids://", H];
|
||||
<<"http://", H/binary>> -> ["grid://", H]
|
||||
end,
|
||||
unicode:characters_to_list([Prefix, "/1/t/", Recipient, qwargs(Amount, Payload)]).
|
||||
|
||||
h_to_s(Host) when is_list(Host) -> Host;
|
||||
h_to_s(Host) when is_binary(Host) -> Host;
|
||||
h_to_s(Host) when is_tuple(Host) -> inet:ntoa(Host);
|
||||
h_to_s(Host) when is_atom(Host) -> atom_to_list(Host).
|
||||
|
||||
qwargs(none, "") ->
|
||||
[];
|
||||
qwargs(Amount, "") ->
|
||||
["?a=", integer_to_list(Amount)];
|
||||
qwargs(none, Payload) ->
|
||||
[$? | uri_string:compose_query([{"p", Payload}])];
|
||||
qwargs(Amount, Payload) ->
|
||||
[$? | uri_string:compose_query([{"a", integer_to_list(Amount)}, {"p", Payload}])].
|
||||
|
||||
|
||||
-spec parse(GRIDS) -> Result
|
||||
when GRIDS :: string(),
|
||||
Result :: {ok, Instruction} | uri_string:error(),
|
||||
|
||||
@ -8,7 +8,7 @@
|
||||
%%% @end
|
||||
|
||||
-module(hz_key_master).
|
||||
-vsn("0.8.2").
|
||||
-vsn("0.6.1").
|
||||
|
||||
|
||||
-export([make_key/1, encode/1, decode/1]).
|
||||
@ -91,8 +91,9 @@ chunksize(N, C, A) -> chunksize(N div C, C, A + 1).
|
||||
|
||||
|
||||
read_words() ->
|
||||
ModPath = code:which(?MODULE),
|
||||
Path = filename:join([filename:dirname(filename:dirname(ModPath)), "priv", "words4096.txt"]),
|
||||
{ok, V} = zx_lib:string_to_version(proplists:get_value(vsn, module_info(attributes))),
|
||||
HZ_Lib = zx_lib:ppath(lib, {"otpr", "hakuzaru", V}),
|
||||
Path = filename:join([HZ_Lib, "priv", "words4096.txt"]),
|
||||
{ok, Bin} = file:read_file(Path),
|
||||
string:lexemes(Bin, "\n").
|
||||
|
||||
|
||||
143
src/hz_man.erl
143
src/hz_man.erl
@ -9,7 +9,7 @@
|
||||
%%% @end
|
||||
|
||||
-module(hz_man).
|
||||
-vsn("0.8.2").
|
||||
-vsn("0.6.1").
|
||||
-behavior(gen_server).
|
||||
-author("Craig Everett <ceverett@tsuriai.jp>").
|
||||
-copyright("Craig Everett <ceverett@tsuriai.jp>").
|
||||
@ -20,17 +20,17 @@
|
||||
chain_nodes/0, chain_nodes/1,
|
||||
timeout/0, timeout/1]).
|
||||
|
||||
%% Contract caching
|
||||
-export([cache_aaci/2, lookup_aaci/1]).
|
||||
|
||||
%% The whole point of this module:
|
||||
-export([request_sticky/1, request_sticky/2, request/1, request/2]).
|
||||
-export([request/1, request/2]).
|
||||
|
||||
%% gen_server goo
|
||||
-export([start_link/0]).
|
||||
-export([init/1, handle_call/3, handle_cast/2, handle_info/2,
|
||||
code_change/3, terminate/2]).
|
||||
|
||||
%% TODO: Make logging more flexible
|
||||
-include("$zx_include/zx_logger.hrl").
|
||||
|
||||
|
||||
%%% Type and Record Definitions
|
||||
|
||||
@ -43,12 +43,11 @@
|
||||
req = none :: none | binary()}).
|
||||
|
||||
-record(s,
|
||||
{tls = false :: boolean(),
|
||||
chain_nodes = {[], []} :: {[hz:chain_node()], [hz:chain_node()]},
|
||||
sticky = none :: none | hz:chain_node(),
|
||||
fetchers = [] :: [#fetcher{}],
|
||||
timeout = 5000 :: pos_integer(),
|
||||
cache = #{} :: #{Label :: term() := AACI :: hz:aaci()}}).
|
||||
{tls = false :: boolean(),
|
||||
chain_nodes = {[], []} :: {[hz:chain_node()], [hz:chain_node()]},
|
||||
sticky = none :: none | hz:chain_node(),
|
||||
fetchers = [] :: [#fetcher{}],
|
||||
timeout = 5000 :: pos_integer()}).
|
||||
|
||||
|
||||
-type state() :: #s{}.
|
||||
@ -98,41 +97,6 @@ timeout(Value) when 0 < Value, Value =< 120000 ->
|
||||
gen_server:cast(?MODULE, {timeout, Value}).
|
||||
|
||||
|
||||
-spec cache_aaci(Label, AACI) -> ok
|
||||
when Label :: term(),
|
||||
AACI :: hz:aaci().
|
||||
|
||||
cache_aaci(Label, AACI) ->
|
||||
gen_server:call(?MODULE, {cache, Label, AACI}).
|
||||
|
||||
|
||||
-spec lookup_aaci(Label) -> Result
|
||||
when Label :: term(),
|
||||
Result :: {ok, hz:aaci()} | error.
|
||||
|
||||
lookup_aaci(Label) ->
|
||||
gen_server:call(?MODULE, {lookup, Label}).
|
||||
|
||||
|
||||
-spec request_sticky(Path) -> {ok, Value} | {error, Reason}
|
||||
when Path :: unicode:charlist(),
|
||||
Value :: map(),
|
||||
Reason :: hz:chain_error().
|
||||
|
||||
request_sticky(Path) ->
|
||||
gen_server:call(?MODULE, {request_sticky, {get, Path}}, infinity).
|
||||
|
||||
|
||||
-spec request_sticky(Path, Data) -> {ok, Value} | {error, Reason}
|
||||
when Path :: unicode:charlist(),
|
||||
Data :: unicode:charlist(),
|
||||
Value :: map(),
|
||||
Reason :: hz:chain_error().
|
||||
|
||||
request_sticky(Path, Data) ->
|
||||
gen_server:call(?MODULE, {request_sticky, {post, Path, Data}}, infinity).
|
||||
|
||||
|
||||
-spec request(Path) -> {ok, Value} | {error, Reason}
|
||||
when Path :: unicode:charlist(),
|
||||
Value :: map(),
|
||||
@ -184,19 +148,10 @@ init(none) ->
|
||||
handle_call({request, Request}, From, State) ->
|
||||
NewState = do_request(Request, From, State),
|
||||
{noreply, NewState};
|
||||
handle_call({request_sticky, Request}, From, State) ->
|
||||
NewState = do_request_sticky(Request, From, State),
|
||||
{noreply, NewState};
|
||||
handle_call({lookup, Label}, _, State) ->
|
||||
Result = do_lookup(Label, State),
|
||||
{reply, Result, State};
|
||||
handle_call({cache, Label, AACI}, _, State) ->
|
||||
NewState = do_cache_aaci(Label, AACI, State),
|
||||
{reply, ok, NewState};
|
||||
handle_call(tls, _, State = #s{tls = TLS}) ->
|
||||
{reply, TLS, State};
|
||||
handle_call(chain_nodes, _, State) ->
|
||||
Nodes = do_chain_nodes(State),
|
||||
handle_call(chain_nodes, _, State = #s{chain_nodes = {Wait, Used}}) ->
|
||||
Nodes = lists:append(Wait, Used),
|
||||
{reply, Nodes, State};
|
||||
handle_call(timeout, _, State = #s{timeout = Value}) ->
|
||||
{reply, Value, State};
|
||||
@ -208,9 +163,10 @@ handle_call(Unexpected, From, State) ->
|
||||
handle_cast({tls, Boolean}, State) ->
|
||||
NewState = do_tls(Boolean, State),
|
||||
{noreply, NewState};
|
||||
handle_cast({chain_nodes, List}, State) ->
|
||||
NewState = do_chain_nodes(List, State),
|
||||
{noreply, NewState};
|
||||
handle_cast({chain_nodes, []}, State) ->
|
||||
{noreply, State#s{chain_nodes = {[], []}}};
|
||||
handle_cast({chain_nodes, ToUse}, State) ->
|
||||
{noreply, State#s{chain_nodes = {ToUse, []}}};
|
||||
handle_cast({timeout, Value}, State) ->
|
||||
{noreply, State#s{timeout = Value}};
|
||||
handle_cast(Unexpected, State) ->
|
||||
@ -265,23 +221,6 @@ terminate(_, _) ->
|
||||
|
||||
%%% Doer Functions
|
||||
|
||||
do_chain_nodes(#s{sticky = none, chain_nodes = {Wait, Used}}) ->
|
||||
lists:append(Wait, Used);
|
||||
do_chain_nodes(#s{sticky = Sticky, chain_nodes = {Wait, Used}}) ->
|
||||
case lists:append(Wait, Used) of
|
||||
[Sticky] -> [Sticky];
|
||||
Nodes -> [Sticky | Nodes]
|
||||
end.
|
||||
|
||||
|
||||
do_chain_nodes([], State) ->
|
||||
State#s{sticky = none, chain_nodes = {[], []}};
|
||||
do_chain_nodes(List = [Sticky], State) ->
|
||||
State#s{sticky = Sticky, chain_nodes = {List, []}};
|
||||
do_chain_nodes([Sticky | List], State) ->
|
||||
State#s{sticky = Sticky, chain_nodes = {List, []}}.
|
||||
|
||||
|
||||
do_tls(true, State) ->
|
||||
ok = ssl:start(),
|
||||
State#s{tls = true};
|
||||
@ -291,30 +230,17 @@ do_tls(_, State) ->
|
||||
State.
|
||||
|
||||
|
||||
do_cache_aaci(Label, AACI, State = #s{cache = Cache}) ->
|
||||
NewCache = maps:put(Label, AACI, Cache),
|
||||
State#s{cache = NewCache}.
|
||||
|
||||
|
||||
do_lookup(Label, #s{cache = Cache}) ->
|
||||
maps:find(Label, Cache).
|
||||
|
||||
|
||||
do_request_sticky(_, From, State = #s{sticky = none}) ->
|
||||
do_request(_, From, State = #s{chain_nodes = {[], []}}) ->
|
||||
ok = gen_server:reply(From, {error, no_nodes}),
|
||||
State;
|
||||
do_request_sticky(Request,
|
||||
From,
|
||||
State = #s{tls = TLS,
|
||||
fetchers = Fetchers,
|
||||
sticky = Node,
|
||||
timeout = Timeout}) ->
|
||||
do_request(Request,
|
||||
From,
|
||||
State = #s{tls = false,
|
||||
fetchers = Fetchers,
|
||||
chain_nodes = {[Node | Rest], Used},
|
||||
timeout = Timeout}) ->
|
||||
Now = erlang:system_time(nanosecond),
|
||||
Fetcher =
|
||||
case TLS of
|
||||
true -> fun() -> hz_fetcher:connect_slowly(Node, Request, From, Timeout) end;
|
||||
false -> fun() -> hz_fetcher:connect(Node, Request, From, Timeout) end
|
||||
end,
|
||||
Fetcher = fun() -> hz_fetcher:connect(Node, Request, From, Timeout) end,
|
||||
{PID, Mon} = spawn_monitor(Fetcher),
|
||||
New = #fetcher{pid = PID,
|
||||
mon = Mon,
|
||||
@ -322,24 +248,15 @@ do_request_sticky(Request,
|
||||
node = Node,
|
||||
from = From,
|
||||
req = Request},
|
||||
State#s{fetchers = [New | Fetchers]}.
|
||||
|
||||
|
||||
do_request(_, From, State = #s{chain_nodes = {[], []}}) ->
|
||||
ok = gen_server:reply(From, {error, no_nodes}),
|
||||
State;
|
||||
State#s{fetchers = [New | Fetchers], chain_nodes = {Rest, [Node | Used]}};
|
||||
do_request(Request,
|
||||
From,
|
||||
State = #s{tls = TLS,
|
||||
State = #s{tls = true,
|
||||
fetchers = Fetchers,
|
||||
chain_nodes = {[Node | Rest], Used},
|
||||
timeout = Timeout}) ->
|
||||
Now = erlang:system_time(nanosecond),
|
||||
Fetcher =
|
||||
case TLS of
|
||||
true -> fun() -> hz_fetcher:connect_slowly(Node, Request, From, Timeout) end;
|
||||
false -> fun() -> hz_fetcher:connect(Node, Request, From, Timeout) end
|
||||
end,
|
||||
Fetcher = fun() -> hz_fetcher:slowly_connect(Node, Request, From, Timeout) end,
|
||||
{PID, Mon} = spawn_monitor(Fetcher),
|
||||
New = #fetcher{pid = PID,
|
||||
mon = Mon,
|
||||
@ -351,9 +268,3 @@ do_request(Request,
|
||||
do_request(Request, From, State = #s{chain_nodes = {[], Used}}) ->
|
||||
Fresh = lists:reverse(Used),
|
||||
do_request(Request, From, State#s{chain_nodes = {Fresh, []}}).
|
||||
|
||||
|
||||
log(Level, Format, Args) ->
|
||||
Raw = io_lib:format("~w ~w: " ++ Format, [?MODULE, self() | Args]),
|
||||
Entry = unicode:characters_to_list(Raw),
|
||||
logger:log(Level, Entry).
|
||||
|
||||
@ -9,7 +9,7 @@
|
||||
%%% @end
|
||||
|
||||
-module(hz_sup).
|
||||
-vsn("0.8.2").
|
||||
-vsn("0.6.1").
|
||||
-behaviour(supervisor).
|
||||
-author("Craig Everett <zxq9@zxq9.com>").
|
||||
-copyright("Craig Everett <zxq9@zxq9.com>").
|
||||
|
||||
@ -4,7 +4,7 @@
|
||||
{prefix,"hz"}.
|
||||
{desc,"Gajumaru interoperation library"}.
|
||||
{author,"Craig Everett"}.
|
||||
{package_id,{"otpr","hakuzaru",{0,8,2}}}.
|
||||
{package_id,{"otpr","hakuzaru",{0,6,1}}}.
|
||||
{deps,[{"otpr","sophia",{9,0,0}},
|
||||
{"otpr","gmserialization",{0,1,3}},
|
||||
{"otpr","gmbytecode",{3,4,1}},
|
||||
@ -19,6 +19,6 @@
|
||||
{copyright,"Craig Everett"}.
|
||||
{file_exts,[]}.
|
||||
{license,"MIT"}.
|
||||
{repo_url,"https://git.qpq.swiss/QPQ-AG/hakuzaru"}.
|
||||
{repo_url,"https://gitlab.com/ioecs/hakuzaru"}.
|
||||
{tags,["qpq","gajumaru","blockchain","hakuzaru","crypto","defi"]}.
|
||||
{ws_url,"https://git.qpq.swiss/QPQ-AG/hakuzaru"}.
|
||||
{ws_url,"https://gitlab.com/ioecs/hakuzaru"}.
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user