Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat(dot/parachain): Add CompactStatement type #4424

Merged
merged 5 commits into from
Dec 20, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
171 changes: 151 additions & 20 deletions dot/parachain/types/statement.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,14 +4,18 @@
package parachaintypes

import (
"bytes"
"fmt"
"io"

"github.com/ChainSafe/gossamer/lib/common"
"github.com/ChainSafe/gossamer/lib/crypto/sr25519"
"github.com/ChainSafe/gossamer/lib/keystore"
"github.com/ChainSafe/gossamer/pkg/scale"
)

var BACKING_STATEMENT_MAGIC = [4]byte{'B', 'K', 'N', 'G'}
axaysagathiya marked this conversation as resolved.
Show resolved Hide resolved

// Statement is a result of candidate validation. It could be either `Valid` or `Seconded`.
type StatementVDTValues interface {
Valid | Seconded
Expand Down Expand Up @@ -80,33 +84,60 @@ type Seconded CommittedCandidateReceipt
// Valid represents a statement that a validator has deemed a candidate valid.
type Valid CandidateHash

// statementVDTAndSigningContext is just a wrapper struct to hold both the statement and the signing context.
type statementVDTAndSigningContext struct {
Statement StatementVDT
Context SigningContext
// encodeSignData encodes the statement and signing context into a byte slice.
func encodeSignData(statement StatementVDT, signingContext SigningContext) ([]byte, error) {
axaysagathiya marked this conversation as resolved.
Show resolved Hide resolved
buffer := bytes.NewBuffer(nil)
encoder := scale.NewEncoder(buffer)

compact, err := statement.CompactStatement()
if err != nil {
return nil, fmt.Errorf("getting compact statement: %w", err)
}

err = encoder.Encode(compact)
if err != nil {
return nil, fmt.Errorf("encoding compact statement: %w", err)
}

err = encoder.Encode(signingContext)
if err != nil {
return nil, fmt.Errorf("encoding signing context: %w", err)
}

return buffer.Bytes(), nil
}

// CompactStatement returns a compact representation of the statement.
func (s StatementVDT) CompactStatement() (any, error) {
switch s := s.inner.(type) {
case Valid:
return CompactStatement[Valid]{Value: s}, nil
case Seconded:
hash, err := GetCandidateHash(CommittedCandidateReceipt(s))
if err != nil {
return nil, fmt.Errorf("getting candidate hash: %w", err)
}
return CompactStatement[SecondedCandidateHash]{Value: SecondedCandidateHash(hash)}, nil
}
return nil, fmt.Errorf("unsupported type")
}

func (s *StatementVDT) Sign(
keystore keystore.Keystore,
signingContext SigningContext,
key ValidatorID,
) (*ValidatorSignature, error) {
statementAndSigningCtx := statementVDTAndSigningContext{
Statement: *s,
Context: signingContext,
}

encodedData, err := scale.Marshal(statementAndSigningCtx)
data, err := encodeSignData(*s, signingContext)
if err != nil {
return nil, fmt.Errorf("marshalling statement and signing-context: %w", err)
return nil, fmt.Errorf("encoding data to sign: %w", err)
}

validatorPublicKey, err := sr25519.NewPublicKey(key[:])
if err != nil {
return nil, fmt.Errorf("getting public key: %w", err)
}

signatureBytes, err := keystore.GetKeypair(validatorPublicKey).Sign(encodedData)
signatureBytes, err := keystore.GetKeypair(validatorPublicKey).Sign(data)
if err != nil {
return nil, fmt.Errorf("signing data: %w", err)
}
Expand All @@ -123,22 +154,17 @@ func (s *StatementVDT) VerifySignature(
signingContext SigningContext,
validatorSignature ValidatorSignature,
) (bool, error) {
statementAndSigningCtx := statementVDTAndSigningContext{
Statement: *s,
Context: signingContext,
}

encodedMsg, err := scale.Marshal(statementAndSigningCtx)
data, err := encodeSignData(*s, signingContext)
if err != nil {
return false, fmt.Errorf("marshalling statement and signing-context: %w", err)
return false, fmt.Errorf("encoding signed data: %w", err)
}

publicKey, err := sr25519.NewPublicKey(validator[:])
if err != nil {
return false, fmt.Errorf("getting public key: %w", err)
}

return publicKey.Verify(encodedMsg, validatorSignature[:])
return publicKey.Verify(data, validatorSignature[:])
}

// UncheckedSignedFullStatement is a Variant of `SignedFullStatement` where the signature has not yet been verified.
Expand Down Expand Up @@ -177,3 +203,108 @@ type SignedFullStatementWithPVD struct {
// otherwise, it should be nil.
PersistedValidationData *PersistedValidationData
}

type SecondedCandidateHash CandidateHash

type CompactStatementValues interface {
Valid | SecondedCandidateHash
}

// compactStatementInner is a helper struct that is used to encode/decode CompactStatement.
type compactStatementInner struct {
inner any
}

func setCompactStatement[Value CompactStatementValues](mvdt *compactStatementInner, value Value) {
mvdt.inner = value
}

func (mvdt *compactStatementInner) SetValue(value any) (err error) {
switch value := value.(type) {
case Valid:
setCompactStatement(mvdt, value)
return
case SecondedCandidateHash:
setCompactStatement(mvdt, value)
return
default:
return fmt.Errorf("unsupported type")
}
}

func (mvdt compactStatementInner) IndexValue() (index uint, value any, err error) {
switch mvdt.inner.(type) {
case Valid:
return 2, mvdt.inner, nil
case SecondedCandidateHash:
return 1, mvdt.inner, nil
}
return 0, nil, scale.ErrUnsupportedVaryingDataTypeValue
}

func (mvdt compactStatementInner) Value() (value any, err error) {
_, value, err = mvdt.IndexValue()
return
}

func (mvdt compactStatementInner) ValueAt(index uint) (value any, err error) {
switch index {
case 2:
return Valid{}, nil
case 1:
return SecondedCandidateHash{}, nil
}
return nil, scale.ErrUnknownVaryingDataTypeValue
}

// CompactStatement is a compact representation of a statement that can be made about parachain candidates.
// this is the actual value that is signed.
type CompactStatement[T CompactStatementValues] struct {
Value T
}

func (c CompactStatement[CompactStatementValues]) MarshalSCALE() ([]byte, error) {
inner := compactStatementInner{}
err := inner.SetValue(c.Value)
if err != nil {
return nil, fmt.Errorf("setting value: %w", err)
}

buffer := bytes.NewBuffer(BACKING_STATEMENT_MAGIC[:])
encoder := scale.NewEncoder(buffer)

err = encoder.Encode(inner)
if err != nil {
return nil, err
}

return buffer.Bytes(), nil
}

func (c *CompactStatement[CompactStatementValues]) UnmarshalSCALE(reader io.Reader) error {
decoder := scale.NewDecoder(reader)

var magicBytes [4]byte
err := decoder.Decode(&magicBytes)
if err != nil {
return err
}

if !bytes.Equal(magicBytes[:], BACKING_STATEMENT_MAGIC[:]) {
return fmt.Errorf("invalid magic bytes")
}

var inner compactStatementInner
err = decoder.Decode(&inner)
if err != nil {
return fmt.Errorf("decoding compactStatementInner: %w", err)
}

value, err := inner.Value()
if err != nil {
return fmt.Errorf("getting value: %w", err)
}

c.Value = value.(CompactStatementValues)
return nil
}
62 changes: 62 additions & 0 deletions dot/parachain/types/statement_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -169,3 +169,65 @@ func TestStatementVDT_Sign(t *testing.T) {
require.NoError(t, err)
require.True(t, ok)
}

func TestCompactStatement(t *testing.T) {
t.Parallel()

testCases := []struct {
name string
compactStatement any
encodingValue []byte
expectedErr error
}{
{
name: "SecondedCandidateHash",
compactStatement: CompactStatement[SecondedCandidateHash]{
Value: SecondedCandidateHash{Value: getDummyHash(6)},
},
encodingValue: []byte{66, 75, 78, 71, 1,
6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6},
},
{
name: "Valid",
compactStatement: CompactStatement[Valid]{
Value: Valid{Value: getDummyHash(7)},
},
encodingValue: []byte{
66, 75, 78, 71, 2,
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7},
},
}

for _, c := range testCases {
c := c
t.Run(c.name, func(t *testing.T) {
t.Parallel()

t.Run("marshal", func(t *testing.T) {
t.Parallel()

compactStatementBytes, err := scale.Marshal(c.compactStatement)
require.NoError(t, err)
require.Equal(t, c.encodingValue, compactStatementBytes)
})

t.Run("unmarshal", func(t *testing.T) {
t.Parallel()

switch expectedSatetement := c.compactStatement.(type) {
case CompactStatement[Valid]:
var actualStatement CompactStatement[Valid]
err := scale.Unmarshal(c.encodingValue, &actualStatement)
require.NoError(t, err)
require.EqualValues(t, expectedSatetement, actualStatement)
case CompactStatement[SecondedCandidateHash]:
var actualStatement CompactStatement[SecondedCandidateHash]
err := scale.Unmarshal(c.encodingValue, &actualStatement)
require.NoError(t, err)
require.EqualValues(t, expectedSatetement, actualStatement)
}
})

})
}
}
Loading