Merge branch 'engine-versioned-hashes-explicit' into engine-versioned-hashes
This commit is contained in:
commit
0a90b58080
62
setup.py
62
setup.py
|
@ -336,6 +336,10 @@ class SpecBuilder(ABC):
|
||||||
"""
|
"""
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def execution_engine_cls(cls) -> str:
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def hardcoded_ssz_dep_constants(cls) -> Dict[str, str]:
|
def hardcoded_ssz_dep_constants(cls) -> Dict[str, str]:
|
||||||
|
@ -469,6 +473,12 @@ get_attesting_indices = cache_this(
|
||||||
),
|
),
|
||||||
_get_attesting_indices, lru_size=SLOTS_PER_EPOCH * MAX_COMMITTEES_PER_SLOT * 3)'''
|
_get_attesting_indices, lru_size=SLOTS_PER_EPOCH * MAX_COMMITTEES_PER_SLOT * 3)'''
|
||||||
|
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def execution_engine_cls(cls) -> str:
|
||||||
|
return ""
|
||||||
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def hardcoded_ssz_dep_constants(cls) -> Dict[str, str]:
|
def hardcoded_ssz_dep_constants(cls) -> Dict[str, str]:
|
||||||
return {}
|
return {}
|
||||||
|
@ -573,12 +583,14 @@ def get_execution_state(_execution_state_root: Bytes32) -> ExecutionState:
|
||||||
|
|
||||||
|
|
||||||
def get_pow_chain_head() -> PowBlock:
|
def get_pow_chain_head() -> PowBlock:
|
||||||
pass
|
pass"""
|
||||||
|
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def execution_engine_cls(cls) -> str:
|
||||||
|
return "\n\n" + """
|
||||||
class NoopExecutionEngine(ExecutionEngine):
|
class NoopExecutionEngine(ExecutionEngine):
|
||||||
|
|
||||||
def notify_new_payload(self: ExecutionEngine, new_payload_request: NewPayloadRequest) -> bool:
|
def notify_new_payload(self: ExecutionEngine, execution_payload: ExecutionPayload) -> bool:
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def notify_forkchoice_updated(self: ExecutionEngine,
|
def notify_forkchoice_updated(self: ExecutionEngine,
|
||||||
|
@ -658,6 +670,39 @@ def retrieve_blobs_and_proofs(beacon_block_root: Root) -> PyUnion[Tuple[Blob, KZ
|
||||||
# pylint: disable=unused-argument
|
# pylint: disable=unused-argument
|
||||||
return ("TEST", "TEST")'''
|
return ("TEST", "TEST")'''
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def execution_engine_cls(cls) -> str:
|
||||||
|
return "\n\n" + """
|
||||||
|
class NoopExecutionEngine(ExecutionEngine):
|
||||||
|
|
||||||
|
def notify_new_payload(self: ExecutionEngine, execution_payload: ExecutionPayload) -> bool:
|
||||||
|
return True
|
||||||
|
|
||||||
|
def notify_forkchoice_updated(self: ExecutionEngine,
|
||||||
|
head_block_hash: Hash32,
|
||||||
|
safe_block_hash: Hash32,
|
||||||
|
finalized_block_hash: Hash32,
|
||||||
|
payload_attributes: Optional[PayloadAttributes]) -> Optional[PayloadId]:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def get_payload(self: ExecutionEngine, payload_id: PayloadId) -> GetPayloadResponse:
|
||||||
|
# pylint: disable=unused-argument
|
||||||
|
raise NotImplementedError("no default block production")
|
||||||
|
|
||||||
|
def is_valid_block_hash(self: ExecutionEngine, execution_payload: ExecutionPayload) -> bool:
|
||||||
|
return True
|
||||||
|
|
||||||
|
def is_valid_versioned_hashes(self: ExecutionEngine, new_payload_request: NewPayloadRequest) -> bool:
|
||||||
|
return True
|
||||||
|
|
||||||
|
def verify_and_notify_new_payload(self: ExecutionEngine,
|
||||||
|
new_payload_request: NewPayloadRequest) -> bool:
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
EXECUTION_ENGINE = NoopExecutionEngine()"""
|
||||||
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def hardcoded_custom_type_dep_constants(cls, spec_object) -> str:
|
def hardcoded_custom_type_dep_constants(cls, spec_object) -> str:
|
||||||
constants = {
|
constants = {
|
||||||
|
@ -691,6 +736,11 @@ def is_byte_vector(value: str) -> bool:
|
||||||
return value.startswith(('ByteVector'))
|
return value.startswith(('ByteVector'))
|
||||||
|
|
||||||
|
|
||||||
|
def make_function_abstract(protocol_def: ProtocolDefinition, key: str):
|
||||||
|
function = protocol_def.functions[key].split('"""')
|
||||||
|
protocol_def.functions[key] = function[0] + "..."
|
||||||
|
|
||||||
|
|
||||||
def objects_to_spec(preset_name: str,
|
def objects_to_spec(preset_name: str,
|
||||||
spec_object: SpecObject,
|
spec_object: SpecObject,
|
||||||
builder: SpecBuilder,
|
builder: SpecBuilder,
|
||||||
|
@ -708,6 +758,11 @@ def objects_to_spec(preset_name: str,
|
||||||
)
|
)
|
||||||
|
|
||||||
def format_protocol(protocol_name: str, protocol_def: ProtocolDefinition) -> str:
|
def format_protocol(protocol_name: str, protocol_def: ProtocolDefinition) -> str:
|
||||||
|
abstract_functions = ["verify_and_notify_new_payload"]
|
||||||
|
for key in protocol_def.functions.keys():
|
||||||
|
if key in abstract_functions:
|
||||||
|
make_function_abstract(protocol_def, key)
|
||||||
|
|
||||||
protocol = f"class {protocol_name}(Protocol):"
|
protocol = f"class {protocol_name}(Protocol):"
|
||||||
for fn_source in protocol_def.functions.values():
|
for fn_source in protocol_def.functions.values():
|
||||||
fn_source = fn_source.replace("self: "+protocol_name, "self")
|
fn_source = fn_source.replace("self: "+protocol_name, "self")
|
||||||
|
@ -783,6 +838,7 @@ def objects_to_spec(preset_name: str,
|
||||||
+ ('\n\n\n' + protocols_spec if protocols_spec != '' else '')
|
+ ('\n\n\n' + protocols_spec if protocols_spec != '' else '')
|
||||||
+ '\n\n\n' + functions_spec
|
+ '\n\n\n' + functions_spec
|
||||||
+ '\n\n' + builder.sundry_functions()
|
+ '\n\n' + builder.sundry_functions()
|
||||||
|
+ builder.execution_engine_cls()
|
||||||
# Since some constants are hardcoded in setup.py, the following assertions verify that the hardcoded constants are
|
# Since some constants are hardcoded in setup.py, the following assertions verify that the hardcoded constants are
|
||||||
# as same as the spec definition.
|
# as same as the spec definition.
|
||||||
+ ('\n\n\n' + ssz_dep_constants_verification if ssz_dep_constants_verification != '' else '')
|
+ ('\n\n\n' + ssz_dep_constants_verification if ssz_dep_constants_verification != '' else '')
|
||||||
|
|
|
@ -249,7 +249,7 @@ def process_execution_payload(state: BeaconState, body: BeaconBlockBody, executi
|
||||||
assert payload.timestamp == compute_timestamp_at_slot(state, state.slot)
|
assert payload.timestamp == compute_timestamp_at_slot(state, state.slot)
|
||||||
# Verify the execution payload is valid
|
# Verify the execution payload is valid
|
||||||
versioned_hashes = [kzg_commitment_to_versioned_hash(commitment) for commitment in body.blob_kzg_commitments]
|
versioned_hashes = [kzg_commitment_to_versioned_hash(commitment) for commitment in body.blob_kzg_commitments]
|
||||||
assert execution_engine.notify_new_payload(
|
assert execution_engine.verify_and_notify_new_payload(
|
||||||
NewPayloadRequest(execution_payload=payload, versioned_hashes=versioned_hashes)
|
NewPayloadRequest(execution_payload=payload, versioned_hashes=versioned_hashes)
|
||||||
)
|
)
|
||||||
# Cache execution payload header
|
# Cache execution payload header
|
||||||
|
|
|
@ -328,9 +328,9 @@ The Engine API may be used to implement this and similarly defined functions via
|
||||||
#### `notify_new_payload`
|
#### `notify_new_payload`
|
||||||
|
|
||||||
```python
|
```python
|
||||||
def notify_new_payload(self: ExecutionEngine, new_payload_request: NewPayloadRequest) -> bool:
|
def notify_new_payload(self: ExecutionEngine, execution_payload: ExecutionPayload) -> bool:
|
||||||
"""
|
"""
|
||||||
Return ``True`` if and only if ``new_payload_request`` is valid with respect to ``self.execution_state``.
|
Return ``True`` if and only if ``execution_payload`` is valid with respect to ``self.execution_state``.
|
||||||
"""
|
"""
|
||||||
...
|
...
|
||||||
```
|
```
|
||||||
|
@ -366,7 +366,7 @@ def process_execution_payload(state: BeaconState, body: BeaconBlockBody, executi
|
||||||
# Verify timestamp
|
# Verify timestamp
|
||||||
assert payload.timestamp == compute_timestamp_at_slot(state, state.slot)
|
assert payload.timestamp == compute_timestamp_at_slot(state, state.slot)
|
||||||
# Verify the execution payload is valid
|
# Verify the execution payload is valid
|
||||||
assert execution_engine.notify_new_payload(NewPayloadRequest(execution_payload=payload))
|
assert execution_engine.notify_new_payload(payload)
|
||||||
# Cache execution payload header
|
# Cache execution payload header
|
||||||
state.latest_execution_payload_header = ExecutionPayloadHeader(
|
state.latest_execution_payload_header = ExecutionPayloadHeader(
|
||||||
parent_hash=payload.parent_hash,
|
parent_hash=payload.parent_hash,
|
||||||
|
|
|
@ -418,7 +418,7 @@ def process_execution_payload(state: BeaconState, body: BeaconBlockBody, executi
|
||||||
# Verify timestamp
|
# Verify timestamp
|
||||||
assert payload.timestamp == compute_timestamp_at_slot(state, state.slot)
|
assert payload.timestamp == compute_timestamp_at_slot(state, state.slot)
|
||||||
# Verify the execution payload is valid
|
# Verify the execution payload is valid
|
||||||
assert execution_engine.notify_new_payload(NewPayloadRequest(execution_payload=payload))
|
assert execution_engine.notify_new_payload(payload)
|
||||||
# Cache execution payload header
|
# Cache execution payload header
|
||||||
state.latest_execution_payload_header = ExecutionPayloadHeader(
|
state.latest_execution_payload_header = ExecutionPayloadHeader(
|
||||||
parent_hash=payload.parent_hash,
|
parent_hash=payload.parent_hash,
|
||||||
|
|
|
@ -29,7 +29,9 @@
|
||||||
- [Request data](#request-data)
|
- [Request data](#request-data)
|
||||||
- [Modified `NewPayloadRequest`](#modified-newpayloadrequest)
|
- [Modified `NewPayloadRequest`](#modified-newpayloadrequest)
|
||||||
- [Engine APIs](#engine-apis)
|
- [Engine APIs](#engine-apis)
|
||||||
- [Modified `notify_new_payload`](#modified-notify_new_payload)
|
- [`is_valid_block_hash`](#is_valid_block_hash)
|
||||||
|
- [`is_valid_versioned_hashes`](#is_valid_versioned_hashes)
|
||||||
|
- [`verify_and_notify_new_payload`](#verify_and_notify_new_payload)
|
||||||
- [Block processing](#block-processing)
|
- [Block processing](#block-processing)
|
||||||
- [Execution payload](#execution-payload)
|
- [Execution payload](#execution-payload)
|
||||||
- [`process_execution_payload`](#process_execution_payload)
|
- [`process_execution_payload`](#process_execution_payload)
|
||||||
|
@ -177,14 +179,40 @@ class NewPayloadRequest(object):
|
||||||
|
|
||||||
#### Engine APIs
|
#### Engine APIs
|
||||||
|
|
||||||
#### Modified `notify_new_payload`
|
#### `is_valid_block_hash`
|
||||||
|
|
||||||
```python
|
```python
|
||||||
def notify_new_payload(self: ExecutionEngine, new_payload_request: NewPayloadRequest) -> bool:
|
def is_valid_block_hash(self: ExecutionEngine, execution_payload: ExecutionPayload) -> bool:
|
||||||
|
"""
|
||||||
|
Return ``True`` if and only if ``execution_payload.block_hash`` is computed correctly.
|
||||||
|
"""
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
#### `is_valid_versioned_hashes`
|
||||||
|
|
||||||
|
```python
|
||||||
|
def is_valid_versioned_hashes(self: ExecutionEngine, new_payload_request: NewPayloadRequest) -> bool:
|
||||||
|
"""
|
||||||
|
Return ``True`` if and only if the version hashes computed by the blob transactions of
|
||||||
|
``new_payload_request.execution_payload`` matches ``new_payload_request.version_hashes``.
|
||||||
|
"""
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
#### `verify_and_notify_new_payload`
|
||||||
|
|
||||||
|
```python
|
||||||
|
def verify_and_notify_new_payload(self: ExecutionEngine,
|
||||||
|
new_payload_request: NewPayloadRequest) -> bool:
|
||||||
"""
|
"""
|
||||||
Return ``True`` if and only if ``new_payload_request`` is valid with respect to ``self.execution_state``.
|
Return ``True`` if and only if ``new_payload_request`` is valid with respect to ``self.execution_state``.
|
||||||
"""
|
"""
|
||||||
|
assert self.is_valid_block_hash(new_payload_request.execution_payload)
|
||||||
|
assert self.is_valid_versioned_hashes(new_payload_request)
|
||||||
|
assert self.notify_new_payload(new_payload_request.execution_payload)
|
||||||
...
|
...
|
||||||
|
return True
|
||||||
```
|
```
|
||||||
|
|
||||||
### Block processing
|
### Block processing
|
||||||
|
@ -207,7 +235,7 @@ def process_execution_payload(state: BeaconState, body: BeaconBlockBody, executi
|
||||||
# Verify the execution payload is valid
|
# Verify the execution payload is valid
|
||||||
# [Modified in Deneb]
|
# [Modified in Deneb]
|
||||||
versioned_hashes = [kzg_commitment_to_versioned_hash(commitment) for commitment in body.blob_kzg_commitments]
|
versioned_hashes = [kzg_commitment_to_versioned_hash(commitment) for commitment in body.blob_kzg_commitments]
|
||||||
assert execution_engine.notify_new_payload(
|
assert execution_engine.verify_and_notify_new_payload(
|
||||||
NewPayloadRequest(execution_payload=payload, versioned_hashes=versioned_hashes)
|
NewPayloadRequest(execution_payload=payload, versioned_hashes=versioned_hashes)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -31,7 +31,13 @@ def run_execution_payload_processing(spec, state, execution_payload, valid=True,
|
||||||
called_new_block = False
|
called_new_block = False
|
||||||
|
|
||||||
class TestEngine(spec.NoopExecutionEngine):
|
class TestEngine(spec.NoopExecutionEngine):
|
||||||
def notify_new_payload(self, new_payload_request) -> bool:
|
def notify_new_payload(self, payload) -> bool:
|
||||||
|
nonlocal called_new_block, execution_valid
|
||||||
|
called_new_block = True
|
||||||
|
assert payload == execution_payload
|
||||||
|
return execution_valid
|
||||||
|
|
||||||
|
def verify_and_notify_new_payload(self, new_payload_request) -> bool:
|
||||||
nonlocal called_new_block, execution_valid
|
nonlocal called_new_block, execution_valid
|
||||||
called_new_block = True
|
called_new_block = True
|
||||||
assert new_payload_request.execution_payload == body.execution_payload
|
assert new_payload_request.execution_payload == body.execution_payload
|
||||||
|
|
|
@ -38,7 +38,7 @@ def run_execution_payload_processing(spec, state, execution_payload, blob_kzg_co
|
||||||
called_new_block = False
|
called_new_block = False
|
||||||
|
|
||||||
class TestEngine(spec.NoopExecutionEngine):
|
class TestEngine(spec.NoopExecutionEngine):
|
||||||
def notify_new_payload(self, new_payload_request) -> bool:
|
def verify_and_notify_new_payload(self, new_payload_request) -> bool:
|
||||||
nonlocal called_new_block, execution_valid
|
nonlocal called_new_block, execution_valid
|
||||||
called_new_block = True
|
called_new_block = True
|
||||||
assert new_payload_request.execution_payload == body.execution_payload
|
assert new_payload_request.execution_payload == body.execution_payload
|
||||||
|
|
Loading…
Reference in New Issue