Merge branch 'engine-versioned-hashes-explicit' into engine-versioned-hashes

This commit is contained in:
Hsiao-Wei Wang 2023-05-24 10:40:36 +08:00
commit 0a90b58080
No known key found for this signature in database
GPG Key ID: AE3D6B174F971DE4
7 changed files with 104 additions and 14 deletions

View File

@ -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 '')

View File

@ -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

View File

@ -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,

View File

@ -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,

View File

@ -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)
) )

View File

@ -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

View File

@ -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