From 22554499de7662bb9808a3f34286f09d46c2458a Mon Sep 17 00:00:00 2001 From: Charles Cooper Date: Mon, 1 Apr 2019 18:36:29 -0700 Subject: [PATCH 1/4] Add blockhash as option for some JSON-RPC methods --- EIPS/blockhash-eip-draft.md | 69 +++++++++++++++++++++++++++++++++++++ 1 file changed, 69 insertions(+) create mode 100644 EIPS/blockhash-eip-draft.md diff --git a/EIPS/blockhash-eip-draft.md b/EIPS/blockhash-eip-draft.md new file mode 100644 index 00000000000000..abe133685e5235 --- /dev/null +++ b/EIPS/blockhash-eip-draft.md @@ -0,0 +1,69 @@ +--- +eip: +title: Add `blockHash` to JSON-RPC methods which accept a default block parameter. +author: Charles Cooper (@charles-cooper) +type: Standards Track +category: Interface +status: Draft +created: 2019-04-01 +--- + +## Simple Summary + +For JSON-RPC methods which currently accept a default block parameter, additionally allow the parameter to be a block hash. + +## Abstract + +This EIP can be considered a generalization of [EIP-234](https://github.com/ethereum/EIPs/blob/d053eb66921c5915f3e16d72c7566289e2d7c151/EIPS/eip-234.md). It would enable clients to unambiguously specify the block they want to query for certain JSON-RPC methods, even if the block is not in the canonical chain. This allows clients to maintain a coherent picture of blockchain state that they are interested in, even in the presence of reorgs, without requiring that the node maintain a persistent connection with the client or store any client-specific state. + +## Specification + +The following JSON-RPC methods are affected: +- `eth_getBalance` +- `eth_getStorageAt` +- `eth_getTransactionCount` +- `eth_getCode` +- `eth_call` +- `eth_getProof` + +The following options, quoted from the [JSON-RPC spec](https://github.com/ethereum/wiki/wiki/JSON-RPC#the-default-block-parameter), are currently possible for the defaultBlock parameter: +> - HEX String - an integer block number +> - String "earliest" for the earliest/genesis block +> - String "latest" - for the latest mined block +> - String "pending" - for the pending state/transactions + +Since there is no way to clearly distinguish between a DATA parameter and a QUANTITY parameter, this EIP proposes a new scheme for the block parameter. The following option is additionally allowed: +- OBJECT + - `blockNumber`: QUANTITY - a block number + - `blockHash`: DATA - a block hash + +To maintain backwards compatibility, the block number may be specified either as a hex string or using the new block parameter scheme. In other words, the following are equivalent for the default block parameter: +- `"earliest"` +- `"0x0"` +- `{ "blockNumber": "0x0" }` +- `{ "blockHash": "0xd4e56740f876aef8c010b86a40d5f56745a118d0906a34e69aec8c0db1cb8fa3" }` (hash of the genesis block on the Ethereum main chain) + +## Rationale + +Currently, the state-querying JSON-RPC methods specified above have no option to unambiguously specify which block to query the state for. This can cause issues for applications which need to make multiple calls to the RPC. For instance, a wallet which just executed a transfer may want to display the balances of both the sender and recipient. If there is a re-org in between when the balance of the sender is queried via `eth_getBalance` and when the balance of the recipient is queried, the balances may not reconcile. As a slightly more complicated example, the UI for a decentralized exchange (which hosts orders on-chain) may walk a list of orders by calling `eth_call` for each of them to get the order data. Another type of use case is where an application needs to make a decision based on multiple pieces of state, e.g. a payout predicated on simultaneous ownership of two NFTs. + +In order to ensure that the state is coherent (i.e., `eth_call` was called with exactly the same block for every call), the application may currently use one of several strategies: +- Decide on a block number to use (e.g., the latest block number known to the application). After each `eth_call` using that block number, call `eth_getBlockByNumber`, also with that block number. If the block hash does not match the known hash for that block number, rollback the current activity and retry from the beginning. This adds `O(n)` invocations as baseline overhead and another `O(n)` invocations for every retry needed. Moreover, there is no way to detect the (unlikely but possible) case that the relevant block was reorged out before `eth_call`, and then reorged back in before `eth_getBlockByNumber`. +- Rely on logs, which *can* be queried unambiguously thanks to the `blockHash` parameter. However, this requires semantic support from the smart contract; if the smart contract does not emit appropriate events, the client will not be able to reconstruct the specific state it is interested in. +- Rely on non-standard extensions like `parity_subscribe`. This requires a persistent connection between the client and node (via IPC or websockets), increases coupling between the client and the node, and cannot handle use cases where there are dependencies between invocations of `eth_call`, for example, walking a linked list. + +Allowing `eth_call` and friends to unambiguously specify the block to be queried give the application developer a robust and intuitive way to solve these problems. Multiple sequential queries will query the same state, enabling the application developer to not worry about inconsistencies in their view of the blockchain state. + +## Backwards Compatibility + +Backwards compatible. + +## Test Cases + +## Implementation + +None yet. + +## Copyright + +Copyright and related rights waived via [CC0](https://creativecommons.org/publicdomain/zero/1.0/). From b9e8e1d1a989a2caff9624f0fdab60eee6b5d751 Mon Sep 17 00:00:00 2001 From: Charles Cooper Date: Wed, 3 Apr 2019 09:56:55 -0700 Subject: [PATCH 2/4] Add the requires field --- EIPS/blockhash-eip-draft.md | 1 + 1 file changed, 1 insertion(+) diff --git a/EIPS/blockhash-eip-draft.md b/EIPS/blockhash-eip-draft.md index abe133685e5235..492aef6cc303a7 100644 --- a/EIPS/blockhash-eip-draft.md +++ b/EIPS/blockhash-eip-draft.md @@ -4,6 +4,7 @@ title: Add `blockHash` to JSON-RPC methods which accept a default block paramete author: Charles Cooper (@charles-cooper) type: Standards Track category: Interface +requires: 234 status: Draft created: 2019-04-01 --- From 6008418b81a10d73ffd00a79625cd83c4c337f99 Mon Sep 17 00:00:00 2001 From: Charles Cooper Date: Wed, 3 Apr 2019 09:58:10 -0700 Subject: [PATCH 3/4] Set the EIP number --- EIPS/{blockhash-eip-draft.md => eip-1898.md} | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) rename EIPS/{blockhash-eip-draft.md => eip-1898.md} (99%) diff --git a/EIPS/blockhash-eip-draft.md b/EIPS/eip-1898.md similarity index 99% rename from EIPS/blockhash-eip-draft.md rename to EIPS/eip-1898.md index 492aef6cc303a7..1a7935c293cf70 100644 --- a/EIPS/blockhash-eip-draft.md +++ b/EIPS/eip-1898.md @@ -1,5 +1,5 @@ --- -eip: +eip: 1898 title: Add `blockHash` to JSON-RPC methods which accept a default block parameter. author: Charles Cooper (@charles-cooper) type: Standards Track From 9622fdf849ef076a019cdbdcba87a1201bd0a022 Mon Sep 17 00:00:00 2001 From: Charles Cooper Date: Wed, 3 Apr 2019 09:58:59 -0700 Subject: [PATCH 4/4] Move requires to the bottom --- EIPS/eip-1898.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/EIPS/eip-1898.md b/EIPS/eip-1898.md index 1a7935c293cf70..7ea8da1acb231c 100644 --- a/EIPS/eip-1898.md +++ b/EIPS/eip-1898.md @@ -4,9 +4,9 @@ title: Add `blockHash` to JSON-RPC methods which accept a default block paramete author: Charles Cooper (@charles-cooper) type: Standards Track category: Interface -requires: 234 status: Draft created: 2019-04-01 +requires: 234 --- ## Simple Summary