From 4f32fd95d6af22ac18379d5f7a3dc5b3c57bf327 Mon Sep 17 00:00:00 2001 From: Hsiao-Wei Wang Date: Mon, 28 Nov 2022 18:51:38 +0800 Subject: [PATCH 1/3] Enable EIP4844 lint and fix Pylint --- Makefile | 8 ++++---- linter.ini | 5 +++++ setup.py | 4 ++++ specs/eip4844/beacon-chain.md | 1 + specs/eip4844/validator.md | 1 + 5 files changed, 15 insertions(+), 4 deletions(-) diff --git a/Makefile b/Makefile index c38d39a6d..73450562b 100644 --- a/Makefile +++ b/Makefile @@ -105,12 +105,12 @@ install_test: # Testing against `minimal` config by default test: pyspec . venv/bin/activate; cd $(PY_SPEC_DIR); \ - python3 -m pytest -n 4 --disable-bls --cov=eth2spec.phase0.minimal --cov=eth2spec.altair.minimal --cov=eth2spec.bellatrix.minimal --cov=eth2spec.capella.minimal --cov-report="html:$(COV_HTML_OUT)" --cov-branch eth2spec + python3 -m pytest -n 4 --disable-bls --cov=eth2spec.phase0.minimal --cov=eth2spec.altair.minimal --cov=eth2spec.bellatrix.minimal --cov=eth2spec.capella.minimal --cov=eth2spec.eip4844.minimal --cov-report="html:$(COV_HTML_OUT)" --cov-branch eth2spec # Testing against `minimal` config by default find_test: pyspec . venv/bin/activate; cd $(PY_SPEC_DIR); \ - python3 -m pytest -k=$(K) --disable-bls --cov=eth2spec.phase0.minimal --cov=eth2spec.altair.minimal --cov=eth2spec.bellatrix.minimal --cov=eth2spec.capella.minimal --cov-report="html:$(COV_HTML_OUT)" --cov-branch eth2spec + python3 -m pytest -k=$(K) --disable-bls --cov=eth2spec.phase0.minimal --cov=eth2spec.altair.minimal --cov=eth2spec.bellatrix.minimal --cov=eth2spec.capella.minimal --cov=eth2spec.eip4844.minimal --cov-report="html:$(COV_HTML_OUT)" --cov-branch eth2spec citest: pyspec mkdir -p $(TEST_REPORT_DIR); @@ -142,8 +142,8 @@ codespell: lint: pyspec . venv/bin/activate; cd $(PY_SPEC_DIR); \ flake8 --config $(LINTER_CONFIG_FILE) ./eth2spec \ - && pylint --disable=all --enable unused-argument ./eth2spec/phase0 ./eth2spec/altair ./eth2spec/bellatrix ./eth2spec/capella \ - && mypy --config-file $(LINTER_CONFIG_FILE) -p eth2spec.phase0 -p eth2spec.altair -p eth2spec.bellatrix -p eth2spec.capella + && pylint --rcfile $(LINTER_CONFIG_FILE) ./eth2spec/phase0 ./eth2spec/altair ./eth2spec/bellatrix ./eth2spec/capella ./eth2spec/eip4844 \ + && mypy --config-file $(LINTER_CONFIG_FILE) -p eth2spec.phase0 -p eth2spec.altair -p eth2spec.bellatrix -p eth2spec.capella -p eth2spec.eip4844 lint_generators: pyspec . venv/bin/activate; cd $(TEST_GENERATORS_DIR); \ diff --git a/linter.ini b/linter.ini index 6575642f1..52a3aec0e 100644 --- a/linter.ini +++ b/linter.ini @@ -11,3 +11,8 @@ warn_unused_configs = True warn_redundant_casts = True ignore_missing_imports = True + +# pylint +[MESSAGES CONTROL] +disable = all +enable = unused-argument diff --git a/setup.py b/setup.py index de446bb1c..13ef3c701 100644 --- a/setup.py +++ b/setup.py @@ -588,6 +588,7 @@ class NoopExecutionEngine(ExecutionEngine): pass def get_payload(self: ExecutionEngine, payload_id: PayloadId) -> ExecutionPayload: + # pylint: disable=unused-argument raise NotImplementedError("no default block production") @@ -643,12 +644,14 @@ T = TypeVar('T') # For generic function def no_op(fn): # type: ignore + # pylint: disable=unused-argument def wrapper(*args, **kw): # type: ignore return None return wrapper def get_empty_list_result(fn): # type: ignore + # pylint: disable=unused-argument def wrapper(*args, **kw): # type: ignore return [] return wrapper @@ -664,6 +667,7 @@ get_expected_withdrawals = get_empty_list_result(get_expected_withdrawals) # def retrieve_blobs_sidecar(slot: Slot, beacon_block_root: Root) -> Optional[BlobsSidecar]: + # pylint: disable=unused-argument return "TEST"''' @classmethod diff --git a/specs/eip4844/beacon-chain.md b/specs/eip4844/beacon-chain.md index 600793a3f..1afd21e88 100644 --- a/specs/eip4844/beacon-chain.md +++ b/specs/eip4844/beacon-chain.md @@ -284,6 +284,7 @@ def process_execution_payload(state: BeaconState, payload: ExecutionPayload, exe ```python def process_blob_kzg_commitments(state: BeaconState, body: BeaconBlockBody): + # pylint: disable=unused-argument assert verify_kzg_commitments_against_transactions(body.execution_payload.transactions, body.blob_kzg_commitments) ``` diff --git a/specs/eip4844/validator.md b/specs/eip4844/validator.md index 57a5610ce..1258358a2 100644 --- a/specs/eip4844/validator.md +++ b/specs/eip4844/validator.md @@ -46,6 +46,7 @@ Implementers may also retrieve blobs individually per transaction. ```python def get_blobs_and_kzg_commitments(payload_id: PayloadId) -> Tuple[Sequence[BLSFieldElement], Sequence[KZGCommitment]]: + # pylint: disable=unused-argument ... ``` From b3a176689d0d9712172a1a06a701e1ae9aa41764 Mon Sep 17 00:00:00 2001 From: Hsiao-Wei Wang Date: Mon, 28 Nov 2022 20:16:18 +0800 Subject: [PATCH 2/3] WIP. Fixing mypy errors --- setup.py | 15 +------------ specs/eip4844/beacon-chain.md | 13 ++++++----- specs/eip4844/polynomial-commitments.md | 30 ++++++++++++------------- 3 files changed, 23 insertions(+), 35 deletions(-) diff --git a/setup.py b/setup.py index 13ef3c701..f15e4ca6d 100644 --- a/setup.py +++ b/setup.py @@ -666,7 +666,7 @@ get_expected_withdrawals = get_empty_list_result(get_expected_withdrawals) # End # -def retrieve_blobs_sidecar(slot: Slot, beacon_block_root: Root) -> Optional[BlobsSidecar]: +def retrieve_blobs_sidecar(slot: Slot, beacon_block_root: Root) -> PyUnion[BlobsSidecar, str]: # pylint: disable=unused-argument return "TEST"''' @@ -686,10 +686,6 @@ spec_builders = { } -def is_spec_defined_type(value: str) -> bool: - return value.startswith(('ByteList', 'Union', 'Vector', 'List')) - - def objects_to_spec(preset_name: str, spec_object: SpecObject, builder: SpecBuilder, @@ -702,15 +698,6 @@ def objects_to_spec(preset_name: str, [ f"class {key}({value}):\n pass\n" for key, value in spec_object.custom_types.items() - if not is_spec_defined_type(value) - ] - ) - + ('\n\n' if len([key for key, value in spec_object.custom_types.items() if is_spec_defined_type(value)]) > 0 else '') - + '\n\n'.join( - [ - f"{key} = {value}\n" - for key, value in spec_object.custom_types.items() - if is_spec_defined_type(value) ] ) ) diff --git a/specs/eip4844/beacon-chain.md b/specs/eip4844/beacon-chain.md index 1afd21e88..63a6cd987 100644 --- a/specs/eip4844/beacon-chain.md +++ b/specs/eip4844/beacon-chain.md @@ -175,10 +175,13 @@ but MUST NOT be considered valid until a valid `BlobsSidecar` has been downloade def is_data_available(slot: Slot, beacon_block_root: Root, blob_kzg_commitments: Sequence[KZGCommitment]) -> bool: # `retrieve_blobs_sidecar` is implementation dependent, raises an exception if not available. sidecar = retrieve_blobs_sidecar(slot, beacon_block_root) - if sidecar == "TEST": - return True # For testing; remove once we have a way to inject `BlobsSidecar` into tests - validate_blobs_sidecar(slot, beacon_block_root, blob_kzg_commitments, sidecar) + # For testing, `retrieve_blobs_sidecar` returns "TEST. + # TODO: Remove it once we have a way to inject `BlobsSidecar` into tests. + if isinstance(sidecar, str): + return True + + validate_blobs_sidecar(slot, beacon_block_root, blob_kzg_commitments, sidecar) return True ``` @@ -216,7 +219,7 @@ def tx_peek_blob_versioned_hashes(opaque_tx: Transaction) -> Sequence[VersionedH ```python def verify_kzg_commitments_against_transactions(transactions: Sequence[Transaction], kzg_commitments: Sequence[KZGCommitment]) -> bool: - all_versioned_hashes = [] + all_versioned_hashes: List[VersionedHash] = [] for tx in transactions: if tx[0] == BLOB_TX_TYPE: all_versioned_hashes += tx_peek_blob_versioned_hashes(tx) @@ -283,7 +286,7 @@ def process_execution_payload(state: BeaconState, payload: ExecutionPayload, exe #### Blob KZG commitments ```python -def process_blob_kzg_commitments(state: BeaconState, body: BeaconBlockBody): +def process_blob_kzg_commitments(state: BeaconState, body: BeaconBlockBody) -> None: # pylint: disable=unused-argument assert verify_kzg_commitments_against_transactions(body.execution_payload.transactions, body.blob_kzg_commitments) ``` diff --git a/specs/eip4844/polynomial-commitments.md b/specs/eip4844/polynomial-commitments.md index 74c0b3cc5..726947134 100644 --- a/specs/eip4844/polynomial-commitments.md +++ b/specs/eip4844/polynomial-commitments.md @@ -144,7 +144,7 @@ def bytes_to_bls_field(b: Bytes32) -> BLSFieldElement: """ Convert 32-byte value to a BLS field scalar. The output is not uniform over the BLS field. """ - return int.from_bytes(b, ENDIANNESS) % BLS_MODULUS + return BLSFieldElement(int.from_bytes(b, ENDIANNESS) % BLS_MODULUS) ``` #### `blob_to_polynomial` @@ -210,7 +210,7 @@ def bls_modular_inverse(x: BLSFieldElement) -> BLSFieldElement: Compute the modular inverse of x i.e. return y such that x * y % BLS_MODULUS == 1 and return 0 for x == 0 """ - return pow(x, -1, BLS_MODULUS) if x != 0 else 0 + return BLSFieldElement(pow(x, -1, BLS_MODULUS)) if x != 0 else BLSFieldElement(0) ``` #### `div` @@ -220,7 +220,7 @@ def div(x: BLSFieldElement, y: BLSFieldElement) -> BLSFieldElement: """ Divide two field elements: ``x`` by `y``. """ - return (int(x) * int(bls_modular_inverse(y))) % BLS_MODULUS + return BLSFieldElement((int(x) * int(bls_modular_inverse(y))) % BLS_MODULUS) ``` #### `g1_lincomb` @@ -251,7 +251,7 @@ def poly_lincomb(polys: Sequence[Polynomial], for v, s in zip(polys, scalars): for i, x in enumerate(v): result[i] = (result[i] + int(s) * int(x)) % BLS_MODULUS - return [BLSFieldElement(x) for x in result] + return Polynomial([BLSFieldElement(x) for x in result]) ``` #### `compute_powers` @@ -284,7 +284,7 @@ def evaluate_polynomial_in_evaluation_form(polynomial: Polynomial, """ width = len(polynomial) assert width == FIELD_ELEMENTS_PER_BLOB - inverse_width = bls_modular_inverse(width) + inverse_width = bls_modular_inverse(BLSFieldElement(width)) # Make sure we won't divide by zero during division assert z not in ROOTS_OF_UNITY @@ -293,9 +293,11 @@ def evaluate_polynomial_in_evaluation_form(polynomial: Polynomial, result = 0 for i in range(width): - result += div(int(polynomial[i]) * int(roots_of_unity_brp[i]), (int(z) - int(roots_of_unity_brp[i]))) - result = result * (pow(z, width, BLS_MODULUS) - 1) * inverse_width % BLS_MODULUS - return result + a = BLSFieldElement(int(polynomial[i]) * int(roots_of_unity_brp[i]) % BLS_MODULUS) + b = BLSFieldElement((int(BLS_MODULUS) + int(z) - int(roots_of_unity_brp[i])) % BLS_MODULUS) + result += int(div(a, b) % BLS_MODULUS) + result = result * int(pow(z, width, BLS_MODULUS) - 1) * int(inverse_width) + return BLSFieldElement(result % BLS_MODULUS) ``` ### KZG @@ -355,17 +357,13 @@ def compute_kzg_proof(polynomial: Polynomial, z: BLSFieldElement) -> KZGProof: Compute KZG proof at point `z` with `polynomial` being in evaluation form Do this by computing the quotient polynomial in evaluation form: q(x) = (p(x) - p(z)) / (x - z) """ - - # To avoid SSZ overflow/underflow, convert element into int - polynomial = [int(i) for i in polynomial] - z = int(z) - y = evaluate_polynomial_in_evaluation_form(polynomial, z) - polynomial_shifted = [(p - int(y)) % BLS_MODULUS for p in polynomial] + polynomial_shifted = [BLSFieldElement((int(p) - int(y)) % BLS_MODULUS) for p in polynomial] # Make sure we won't divide by zero during division assert z not in ROOTS_OF_UNITY - denominator_poly = [(int(x) - z) % BLS_MODULUS for x in bit_reversal_permutation(ROOTS_OF_UNITY)] + denominator_poly = [BLSFieldElement((int(x) - int(z)) % BLS_MODULUS) + for x in bit_reversal_permutation(ROOTS_OF_UNITY)] # Calculate quotient polynomial by doing point-by-point division quotient_polynomial = [div(a, b) for a, b in zip(polynomial_shifted, denominator_poly)] @@ -392,7 +390,7 @@ def compute_aggregated_poly_and_commitment( r_powers, evaluation_challenge = compute_challenges(polynomials, kzg_commitments) # Create aggregated polynomial in evaluation form - aggregated_poly = Polynomial(poly_lincomb(polynomials, r_powers)) + aggregated_poly = poly_lincomb(polynomials, r_powers) # Compute commitment to aggregated polynomial aggregated_poly_commitment = KZGCommitment(g1_lincomb(kzg_commitments, r_powers)) From edde563b3ac19aea4ff6290ce1e14c279b75b0df Mon Sep 17 00:00:00 2001 From: Hsiao-Wei Wang Date: Mon, 28 Nov 2022 21:06:22 +0800 Subject: [PATCH 3/3] Workaround: ignore Invalid base class "ByteVector" error --- setup.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/setup.py b/setup.py index f15e4ca6d..432a41fe4 100644 --- a/setup.py +++ b/setup.py @@ -686,6 +686,10 @@ spec_builders = { } +def is_byte_vector(value: str) -> bool: + return value.startswith(('ByteVector')) + + def objects_to_spec(preset_name: str, spec_object: SpecObject, builder: SpecBuilder, @@ -696,7 +700,7 @@ def objects_to_spec(preset_name: str, new_type_definitions = ( '\n\n'.join( [ - f"class {key}({value}):\n pass\n" + f"class {key}({value}):\n pass\n" if not is_byte_vector(value) else f"class {key}({value}): # type: ignore\n pass\n" for key, value in spec_object.custom_types.items() ] )