Da Dispersal (#80)
* Added common bls types * Added verifier attestation build * Implement verification and aggregation in dispersal * Added certificate building tests * Added dispersal test
This commit is contained in:
parent
7ffb7cc7ed
commit
8c34f8a39e
14
da/common.py
14
da/common.py
|
@ -2,12 +2,15 @@ from dataclasses import dataclass
|
||||||
from itertools import chain, zip_longest
|
from itertools import chain, zip_longest
|
||||||
from typing import List, Generator, Self
|
from typing import List, Generator, Self
|
||||||
|
|
||||||
from eth2spec.eip7594.mainnet import Bytes32
|
|
||||||
|
|
||||||
|
from eth2spec.eip7594.mainnet import Bytes32, KZGCommitment as Commitment
|
||||||
|
|
||||||
|
|
||||||
class NodeId(Bytes32):
|
class NodeId(Bytes32):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
class Chunk(Bytes32):
|
class Chunk(Bytes32):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -31,14 +34,19 @@ class ChunksMatrix(List[Row | Column]):
|
||||||
return ChunksMatrix(self.columns)
|
return ChunksMatrix(self.columns)
|
||||||
|
|
||||||
|
|
||||||
|
BLSPublickey = bytes
|
||||||
|
BLSPrivateKey = int
|
||||||
|
BLSSignature = bytes
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class Attestation:
|
class Attestation:
|
||||||
pass
|
signature: BLSSignature
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class Certificate:
|
class Certificate:
|
||||||
pass
|
aggregated_signatures: BLSSignature
|
||||||
|
aggregated_column_commitment: Commitment
|
||||||
|
row_commitments: List[Commitment]
|
||||||
|
|
||||||
|
|
|
@ -1,50 +1,77 @@
|
||||||
from typing import List, Optional, Generator
|
from dataclasses import dataclass
|
||||||
|
from hashlib import sha3_256
|
||||||
|
from typing import List, Optional, Generator, Sequence
|
||||||
|
|
||||||
from da.common import Certificate, NodeId
|
from py_ecc.bls import G2ProofOfPossession as bls_pop
|
||||||
|
|
||||||
|
from da.common import Certificate, NodeId, BLSPublickey
|
||||||
from da.encoder import EncodedData
|
from da.encoder import EncodedData
|
||||||
from da.verifier import DABlob, Attestation
|
from da.verifier import DABlob, Attestation
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class DispersalSettings:
|
||||||
|
nodes_ids: List[NodeId]
|
||||||
|
nodes_pubkey: List[BLSPublickey]
|
||||||
|
threshold: int
|
||||||
|
|
||||||
|
|
||||||
class Dispersal:
|
class Dispersal:
|
||||||
def __init__(self, nodes: List[NodeId], threshold: int):
|
def __init__(self, settings: DispersalSettings):
|
||||||
self.nodes = nodes
|
self.settings = settings
|
||||||
self.threshold = threshold
|
|
||||||
|
|
||||||
def _prepare_data(self, encoded_data: EncodedData) -> Generator[DABlob, None, None]:
|
def _prepare_data(self, encoded_data: EncodedData) -> Generator[DABlob, None, None]:
|
||||||
assert len(encoded_data.row_commitments) == len(self.nodes)
|
assert len(encoded_data.column_commitments) == len(self.settings.nodes_ids)
|
||||||
assert len(encoded_data.row_proofs) == len(self.nodes)
|
assert len(encoded_data.aggregated_column_proofs) == len(self.settings.nodes_ids)
|
||||||
columns = encoded_data.extended_matrix.columns
|
columns = encoded_data.extended_matrix.columns
|
||||||
column_commitments = encoded_data.column_commitments
|
column_commitments = encoded_data.column_commitments
|
||||||
row_commitments = encoded_data.row_commitments
|
row_commitments = encoded_data.row_commitments
|
||||||
rows_proofs = encoded_data.row_proofs
|
rows_proofs = encoded_data.row_proofs
|
||||||
aggregated_column_commitment = encoded_data.aggregated_column_commitment
|
aggregated_column_commitment = encoded_data.aggregated_column_commitment
|
||||||
aggregated_column_proof = encoded_data.aggregated_column_proof
|
aggregated_column_proofs = encoded_data.aggregated_column_proofs
|
||||||
for index, (column, column_commitment, row_proofs) in enumerate(zip(columns, column_commitments, rows_proofs)):
|
blobs_data = enumerate(zip(columns, column_commitments, zip(*rows_proofs), aggregated_column_proofs))
|
||||||
|
for index, (column, column_commitment, row_proofs, column_proof) in blobs_data:
|
||||||
blob = DABlob(
|
blob = DABlob(
|
||||||
index,
|
index,
|
||||||
column,
|
column,
|
||||||
column_commitment,
|
column_commitment,
|
||||||
aggregated_column_commitment,
|
aggregated_column_commitment,
|
||||||
aggregated_column_proof,
|
column_proof,
|
||||||
row_commitments,
|
row_commitments,
|
||||||
row_proofs
|
row_proofs
|
||||||
)
|
)
|
||||||
yield blob
|
yield blob
|
||||||
|
|
||||||
def _send_and_await_response(self, node, encoded_data: EncodedData) -> Optional[Attestation]:
|
def _send_and_await_response(self, node: NodeId, blob: DABlob) -> Optional[Attestation]:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def _build_certificate(self, attestations: List[Attestation]):
|
def _build_certificate(self, encoded_data: EncodedData, attestations: Sequence[Attestation]) -> Certificate:
|
||||||
pass
|
assert len(attestations) >= self.settings.threshold
|
||||||
|
aggregated = bls_pop.Aggregate([attestation.signature for attestation in attestations])
|
||||||
|
return Certificate(
|
||||||
|
aggregated_signatures=aggregated,
|
||||||
|
aggregated_column_commitment=encoded_data.aggregated_column_commitment,
|
||||||
|
row_commitments=encoded_data.row_commitments
|
||||||
|
)
|
||||||
|
|
||||||
def _verify_attestation(self, attestation: Attestation) -> bool:
|
@staticmethod
|
||||||
pass
|
def _verify_attestation(public_key: BLSPublickey, attested_message: bytes, attestation: Attestation) -> bool:
|
||||||
|
return bls_pop.Verify(public_key, attested_message, attestation.signature)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _build_attestation_message(encoded_data: EncodedData) -> bytes:
|
||||||
|
hasher = sha3_256()
|
||||||
|
hasher.update(bytes(encoded_data.aggregated_column_commitment))
|
||||||
|
for c in encoded_data.row_commitments:
|
||||||
|
hasher.update(bytes(c))
|
||||||
|
return hasher.digest()
|
||||||
|
|
||||||
def disperse(self, encoded_data: EncodedData) -> Optional[Certificate]:
|
def disperse(self, encoded_data: EncodedData) -> Optional[Certificate]:
|
||||||
attestations = []
|
attestations = []
|
||||||
for node, blob in zip(self.nodes, self._prepare_data(encoded_data)):
|
attested_message = self._build_attestation_message(encoded_data)
|
||||||
|
for node, pk, blob in zip(self.settings.nodes_ids, self.settings.nodes_pubkey, self._prepare_data(encoded_data)):
|
||||||
if attestation := self._send_and_await_response(node, blob):
|
if attestation := self._send_and_await_response(node, blob):
|
||||||
if self._verify_attestation(attestation):
|
if self._verify_attestation(pk, attested_message, attestation):
|
||||||
attestations.append(attestation)
|
attestations.append(attestation)
|
||||||
if len(attestations) >= self.threshold:
|
if len(attestations) >= self.settings.threshold:
|
||||||
return self._build_certificate(attestations)
|
return self._build_certificate(encoded_data, attestations)
|
||||||
|
|
|
@ -1,20 +1,70 @@
|
||||||
|
from hashlib import sha3_256
|
||||||
from unittest import TestCase
|
from unittest import TestCase
|
||||||
from da.dispersal import Dispersal, DABlob, EncodedData
|
|
||||||
|
from .encoder import DAEncoderParams, DAEncoder
|
||||||
|
from .test_encoder import TestEncoder
|
||||||
|
|
||||||
|
from da.common import NodeId, Attestation
|
||||||
|
from da.dispersal import Dispersal, EncodedData, DispersalSettings
|
||||||
|
from py_ecc.bls import G2ProofOfPossession as bls_pop
|
||||||
|
|
||||||
|
from .verifier import DAVerifier, DABlob
|
||||||
|
|
||||||
|
|
||||||
class TestDispersal(TestCase):
|
class TestDispersal(TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
self.n_nodes = 16
|
||||||
|
self.nodes_ids = [NodeId(x.to_bytes(length=32, byteorder='big')) for x in range(self.n_nodes)]
|
||||||
|
self.secret_keys = list(range(1, self.n_nodes+1))
|
||||||
|
self.public_keys = [bls_pop.SkToPk(sk) for sk in self.secret_keys]
|
||||||
|
dispersal_settings = DispersalSettings(
|
||||||
|
self.nodes_ids,
|
||||||
|
self.public_keys,
|
||||||
|
self.n_nodes // 2 + 1
|
||||||
|
)
|
||||||
|
self.dispersal = Dispersal(dispersal_settings)
|
||||||
|
self.encoder_test = TestEncoder()
|
||||||
|
self.encoder_test.setUp()
|
||||||
|
|
||||||
def test_build_certificate_insufficient_attestations(self):
|
def test_build_certificate_insufficient_attestations(self):
|
||||||
pass
|
with self.assertRaises(AssertionError):
|
||||||
|
self.dispersal._build_certificate(None, [])
|
||||||
|
|
||||||
def test_build_certificate_enough_attestations(self):
|
def test_build_certificate_enough_attestations(self):
|
||||||
pass
|
mock_encoded_data = EncodedData(
|
||||||
|
None, None, None, [], [], [], bytes(b"f"*48), []
|
||||||
def test_prepare_data(self):
|
)
|
||||||
pass
|
mock_message = sha3_256(mock_encoded_data.aggregated_column_commitment).digest()
|
||||||
|
mock_attestations = [Attestation(bls_pop.Sign(sk, mock_message)) for sk in self.secret_keys]
|
||||||
def test_verify_attestation(self):
|
certificate = self.dispersal._build_certificate(mock_encoded_data, mock_attestations)
|
||||||
pass
|
self.assertIsNotNone(certificate)
|
||||||
|
self.assertEqual(certificate.aggregated_column_commitment, mock_encoded_data.aggregated_column_commitment)
|
||||||
|
self.assertEqual(certificate.row_commitments, [])
|
||||||
|
self.assertIsNotNone(certificate.aggregated_signatures)
|
||||||
|
self.assertTrue(
|
||||||
|
bls_pop.AggregateVerify(self.public_keys, [mock_message]*len(self.public_keys), certificate.aggregated_signatures)
|
||||||
|
)
|
||||||
|
|
||||||
def test_disperse(self):
|
def test_disperse(self):
|
||||||
pass
|
data = self.encoder_test.data
|
||||||
|
encoding_params = DAEncoderParams(column_count=self.n_nodes // 2, bytes_per_field_element=32)
|
||||||
|
encoded_data = DAEncoder(encoding_params).encode(data)
|
||||||
|
|
||||||
|
# mock send and await method with local verifiers
|
||||||
|
def __send_and_await_response(node: NodeId, blob: DABlob):
|
||||||
|
sk = self.secret_keys[int.from_bytes(node)]
|
||||||
|
verifier = DAVerifier(sk)
|
||||||
|
return verifier.verify(blob)
|
||||||
|
# inject mock send and await method
|
||||||
|
self.dispersal._send_and_await_response = __send_and_await_response
|
||||||
|
|
||||||
|
certificate = self.dispersal.disperse(encoded_data)
|
||||||
|
self.assertIsNotNone(certificate)
|
||||||
|
self.assertTrue(
|
||||||
|
bls_pop.AggregateVerify(
|
||||||
|
self.public_keys[:self.dispersal.settings.threshold],
|
||||||
|
[self.dispersal._build_attestation_message(encoded_data)]*self.dispersal.settings.threshold,
|
||||||
|
certificate.aggregated_signatures
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
|
@ -56,7 +56,6 @@ class TestEncoder(TestCase):
|
||||||
ROOTS_OF_UNITY
|
ROOTS_OF_UNITY
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def test_chunkify(self):
|
def test_chunkify(self):
|
||||||
encoder_settings = DAEncoderParams(column_count=2, bytes_per_field_element=32)
|
encoder_settings = DAEncoderParams(column_count=2, bytes_per_field_element=32)
|
||||||
elements = 10
|
elements = 10
|
||||||
|
|
|
@ -11,7 +11,7 @@ from da.verifier import Attestation, DAVerifier, DABlob
|
||||||
class TestVerifier(TestCase):
|
class TestVerifier(TestCase):
|
||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
self.verifier = DAVerifier(b"")
|
self.verifier = DAVerifier(1987)
|
||||||
|
|
||||||
def test_verify_column(self):
|
def test_verify_column(self):
|
||||||
column = Column(int.to_bytes(i, length=32) for i in range(8))
|
column = Column(int.to_bytes(i, length=32) for i in range(8))
|
||||||
|
|
|
@ -7,9 +7,9 @@ from eth2spec.eip7594.mainnet import (
|
||||||
KZGCommitment as Commitment,
|
KZGCommitment as Commitment,
|
||||||
KZGProof as Proof,
|
KZGProof as Proof,
|
||||||
)
|
)
|
||||||
from itertools import batched
|
from py_ecc.bls import G2ProofOfPossession as bls_pop
|
||||||
|
|
||||||
from da.common import Column, Chunk, Attestation
|
from da.common import Column, Chunk, Attestation, BLSPrivateKey
|
||||||
from da.encoder import DAEncoder
|
from da.encoder import DAEncoder
|
||||||
from da.kzg_rs import kzg
|
from da.kzg_rs import kzg
|
||||||
from da.kzg_rs.common import ROOTS_OF_UNITY, GLOBAL_PARAMETERS, BLS_MODULUS
|
from da.kzg_rs.common import ROOTS_OF_UNITY, GLOBAL_PARAMETERS, BLS_MODULUS
|
||||||
|
@ -27,7 +27,7 @@ class DABlob:
|
||||||
|
|
||||||
|
|
||||||
class DAVerifier:
|
class DAVerifier:
|
||||||
def __init__(self, sk: bytes):
|
def __init__(self, sk: BLSPrivateKey):
|
||||||
self.sk = sk
|
self.sk = sk
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
|
@ -70,8 +70,13 @@ class DAVerifier:
|
||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def _build_attestation(self, _blob: DABlob) -> Attestation:
|
def _build_attestation(self, blob: DABlob) -> Attestation:
|
||||||
return Attestation()
|
hasher = sha3_256()
|
||||||
|
hasher.update(bytes(blob.aggregated_column_commitment))
|
||||||
|
for c in blob.rows_commitments:
|
||||||
|
hasher.update(bytes(c))
|
||||||
|
message = hasher.digest()
|
||||||
|
return Attestation(signature=bls_pop.Sign(self.sk, message))
|
||||||
|
|
||||||
def verify(self, blob: DABlob) -> Optional[Attestation]:
|
def verify(self, blob: DABlob) -> Optional[Attestation]:
|
||||||
is_column_verified = DAVerifier._verify_column(
|
is_column_verified = DAVerifier._verify_column(
|
||||||
|
|
Loading…
Reference in New Issue