2024-07-18 17:01:40 +08:00
|
|
|
# Fluffy
|
|
|
|
# Copyright (c) 2024 Status Research & Development GmbH
|
|
|
|
# Licensed and distributed under either of
|
|
|
|
# * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT).
|
|
|
|
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
|
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
|
|
|
{.push raises: [].}
|
|
|
|
|
|
|
|
import
|
|
|
|
chronicles,
|
|
|
|
stew/byteutils,
|
|
|
|
stint,
|
|
|
|
eth/common/[eth_types, eth_types_rlp],
|
|
|
|
../../../rpc/rpc_calls/rpc_trace_calls,
|
|
|
|
../portal_bridge_common
|
|
|
|
|
|
|
|
type
|
|
|
|
DiffType* = enum
|
|
|
|
unchanged
|
|
|
|
create
|
|
|
|
update
|
|
|
|
delete
|
|
|
|
|
|
|
|
Code = seq[byte]
|
|
|
|
StateValue* = UInt256 | AccountNonce | Code
|
|
|
|
|
|
|
|
StateValueDiff*[StateValue] = object
|
|
|
|
kind*: DiffType
|
|
|
|
before*: StateValue
|
|
|
|
after*: StateValue
|
|
|
|
|
2024-07-30 22:56:21 +08:00
|
|
|
SlotDiff* = tuple[slotKey: UInt256, slotValueDiff: StateValueDiff[UInt256]]
|
|
|
|
|
|
|
|
AccountDiff* = object
|
|
|
|
address*: EthAddress
|
|
|
|
balanceDiff*: StateValueDiff[UInt256]
|
|
|
|
nonceDiff*: StateValueDiff[AccountNonce]
|
|
|
|
storageDiff*: seq[SlotDiff]
|
|
|
|
codeDiff*: StateValueDiff[Code]
|
|
|
|
|
|
|
|
TransactionDiff* = seq[AccountDiff]
|
2024-07-18 17:01:40 +08:00
|
|
|
|
|
|
|
proc toStateValue(T: type UInt256, hex: string): T {.raises: [ValueError].} =
|
|
|
|
UInt256.fromHex(hex)
|
|
|
|
|
|
|
|
proc toStateValue(T: type AccountNonce, hex: string): T {.raises: [ValueError].} =
|
|
|
|
UInt256.fromHex(hex).truncate(uint64)
|
|
|
|
|
|
|
|
proc toStateValue(T: type Code, hex: string): T {.raises: [ValueError].} =
|
|
|
|
hexToSeqByte(hex)
|
|
|
|
|
|
|
|
proc toStateValueDiff(
|
|
|
|
diffJson: JsonNode, T: type StateValue
|
|
|
|
): StateValueDiff[T] {.raises: [ValueError].} =
|
|
|
|
if diffJson.kind == JString and diffJson.getStr() == "=":
|
|
|
|
return StateValueDiff[T](kind: unchanged)
|
|
|
|
elif diffJson.kind == JObject:
|
|
|
|
if diffJson{"+"} != nil:
|
|
|
|
return
|
|
|
|
StateValueDiff[T](kind: create, after: T.toStateValue(diffJson{"+"}.getStr()))
|
|
|
|
elif diffJson{"-"} != nil:
|
|
|
|
return
|
|
|
|
StateValueDiff[T](kind: delete, before: T.toStateValue(diffJson{"-"}.getStr()))
|
|
|
|
elif diffJson{"*"} != nil:
|
|
|
|
return StateValueDiff[T](
|
|
|
|
kind: update,
|
|
|
|
before: T.toStateValue(diffJson{"*"}{"from"}.getStr()),
|
|
|
|
after: T.toStateValue(diffJson{"*"}{"to"}.getStr()),
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
doAssert false # unreachable
|
|
|
|
else:
|
|
|
|
doAssert false # unreachable
|
|
|
|
|
2024-07-30 22:56:21 +08:00
|
|
|
proc toTransactionDiff(
|
|
|
|
stateDiffJson: JsonNode
|
|
|
|
): TransactionDiff {.raises: [ValueError].} =
|
|
|
|
var txDiff = newSeqOfCap[AccountDiff](stateDiffJson.len())
|
2024-07-18 17:01:40 +08:00
|
|
|
|
2024-07-30 22:56:21 +08:00
|
|
|
for addrJson, accJson in stateDiffJson:
|
|
|
|
let storageDiffJson = accJson["storage"]
|
|
|
|
var storageDiff = newSeqOfCap[SlotDiff](storageDiffJson.len())
|
2024-07-18 17:01:40 +08:00
|
|
|
|
2024-07-30 22:56:21 +08:00
|
|
|
for slotKeyJson, slotValueJson in storageDiffJson:
|
|
|
|
storageDiff.add(
|
|
|
|
(UInt256.fromHex(slotKeyJson), toStateValueDiff(slotValueJson, UInt256))
|
|
|
|
)
|
2024-07-18 17:01:40 +08:00
|
|
|
|
2024-07-30 22:56:21 +08:00
|
|
|
let accountDiff = AccountDiff(
|
|
|
|
address: EthAddress.fromHex(addrJson),
|
|
|
|
balanceDiff: toStateValueDiff(accJson["balance"], UInt256),
|
|
|
|
nonceDiff: toStateValueDiff(accJson["nonce"], AccountNonce),
|
|
|
|
storageDiff: storageDiff,
|
|
|
|
codeDiff: toStateValueDiff(accJson["code"], Code),
|
|
|
|
)
|
|
|
|
txDiff.add(accountDiff)
|
2024-07-18 17:01:40 +08:00
|
|
|
|
2024-07-30 22:56:21 +08:00
|
|
|
txDiff
|
2024-07-18 17:01:40 +08:00
|
|
|
|
2024-07-30 22:56:21 +08:00
|
|
|
proc toTransactionDiffs(
|
2024-07-18 17:01:40 +08:00
|
|
|
blockTraceJson: JsonNode
|
2024-07-30 22:56:21 +08:00
|
|
|
): seq[TransactionDiff] {.raises: [ValueError].} =
|
|
|
|
var txDiffs = newSeqOfCap[TransactionDiff](blockTraceJson.len())
|
2024-07-18 17:01:40 +08:00
|
|
|
for blockTrace in blockTraceJson:
|
2024-07-30 22:56:21 +08:00
|
|
|
txDiffs.add(blockTrace["stateDiff"].toTransactionDiff())
|
2024-07-18 17:01:40 +08:00
|
|
|
|
2024-07-30 22:56:21 +08:00
|
|
|
txDiffs
|
2024-07-18 17:01:40 +08:00
|
|
|
|
|
|
|
proc getStateDiffsByBlockNumber*(
|
|
|
|
client: RpcClient, blockId: BlockIdentifier
|
2024-07-30 22:56:21 +08:00
|
|
|
): Future[Result[seq[TransactionDiff], string]] {.async: (raises: []).} =
|
2024-07-18 17:01:40 +08:00
|
|
|
const traceOpts = @["stateDiff"]
|
|
|
|
|
|
|
|
try:
|
|
|
|
let blockTraceJson = await client.trace_replayBlockTransactions(blockId, traceOpts)
|
|
|
|
if blockTraceJson.isNil:
|
|
|
|
return err("EL failed to provide requested state diff")
|
2024-07-30 22:56:21 +08:00
|
|
|
ok(blockTraceJson.toTransactionDiffs())
|
2024-07-18 17:01:40 +08:00
|
|
|
except CatchableError as e:
|
|
|
|
return err("EL JSON-RPC trace_replayBlockTransactions failed: " & e.msg)
|