Skip to content

Commit d406386

Browse files
cmwatersrootulp
andauthored
fix: share shouldn't validate namespace versions (#98)
Whilst updating `celestia-app`, I realised that we falsely verify the namespace of the shares using the criteria that it must be a "usable namespace". This validation shouldn't be here because the protocol uses arbitrary namespaces when erasure coding. This resolves the problem --------- Co-authored-by: Rootul P <[email protected]>
1 parent 384e43d commit d406386

7 files changed

+43
-98
lines changed

share/compact_shares_test.go

+3-39
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ func TestCompactShareSplitter(t *testing.T) {
2525
shares, err := css.Export()
2626
require.NoError(t, err)
2727

28-
resTxs, err := parseCompactShares(shares, SupportedShareVersions)
28+
resTxs, err := parseCompactShares(shares)
2929
require.NoError(t, err)
3030

3131
assert.Equal(t, txs, resTxs)
@@ -80,7 +80,7 @@ func Test_processCompactShares(t *testing.T) {
8080
shares, _, err := splitTxs(txs)
8181
require.NoError(t, err)
8282

83-
parsedTxs, err := parseCompactShares(shares, SupportedShareVersions)
83+
parsedTxs, err := parseCompactShares(shares)
8484
if err != nil {
8585
t.Error(err)
8686
}
@@ -97,7 +97,7 @@ func Test_processCompactShares(t *testing.T) {
9797

9898
txShares, _, err := splitTxs(txs)
9999
require.NoError(t, err)
100-
parsedTxs, err := parseCompactShares(txShares, SupportedShareVersions)
100+
parsedTxs, err := parseCompactShares(txShares)
101101
if err != nil {
102102
t.Error(err)
103103
}
@@ -212,42 +212,6 @@ func TestContiguousCompactShareContainsInfoByte(t *testing.T) {
212212
assert.Equal(t, byte(want), infoByte)
213213
}
214214

215-
func Test_parseCompactSharesErrors(t *testing.T) {
216-
type testCase struct {
217-
name string
218-
shares []Share
219-
}
220-
221-
txs := generateRandomTxs(2, ContinuationCompactShareContentSize*4)
222-
txShares, _, err := splitTxs(txs)
223-
require.NoError(t, err)
224-
rawShares := ToBytes(txShares)
225-
226-
unsupportedShareVersion := 5
227-
infoByte, _ := NewInfoByte(uint8(unsupportedShareVersion), true)
228-
shareWithUnsupportedShareVersionBytes := rawShares[0]
229-
shareWithUnsupportedShareVersionBytes[NamespaceSize] = byte(infoByte)
230-
231-
shareWithUnsupportedShareVersion, err := NewShare(shareWithUnsupportedShareVersionBytes)
232-
if err != nil {
233-
t.Fatal(err)
234-
}
235-
236-
testCases := []testCase{
237-
{
238-
"share with unsupported share version",
239-
[]Share{*shareWithUnsupportedShareVersion},
240-
},
241-
}
242-
243-
for _, tt := range testCases {
244-
t.Run(tt.name, func(t *testing.T) {
245-
_, err := parseCompactShares(tt.shares, SupportedShareVersions)
246-
assert.Error(t, err)
247-
})
248-
}
249-
}
250-
251215
func generateRandomlySizedTxs(count, maxSize int) [][]byte {
252216
txs := make([][]byte, count)
253217
for i := 0; i < count; i++ {

share/namespace.go

+10-4
Original file line numberDiff line numberDiff line change
@@ -9,9 +9,10 @@ type Namespace struct {
99
data []byte
1010
}
1111

12-
// NewNamespace returns a new namespace with the provided version and id.
12+
// NewNamespace validates the provided version and id and returns a new namespace.
13+
// This should be used for user specified namespaces.
1314
func NewNamespace(version uint8, id []byte) (Namespace, error) {
14-
if err := validate(version, id); err != nil {
15+
if err := ValidateUserNamespace(version, id); err != nil {
1516
return Namespace{}, err
1617
}
1718

@@ -38,11 +39,12 @@ func MustNewNamespace(version uint8, id []byte) Namespace {
3839
}
3940

4041
// NewNamespaceFromBytes returns a new namespace from the provided byte slice.
42+
// This is for user specified namespaces.
4143
func NewNamespaceFromBytes(bytes []byte) (Namespace, error) {
4244
if len(bytes) != NamespaceSize {
4345
return Namespace{}, fmt.Errorf("invalid namespace length: %d. Must be %d bytes", len(bytes), NamespaceSize)
4446
}
45-
if err := validate(bytes[VersionIndex], bytes[NamespaceVersionSize:]); err != nil {
47+
if err := ValidateUserNamespace(bytes[VersionIndex], bytes[NamespaceVersionSize:]); err != nil {
4648
return Namespace{}, err
4749
}
4850

@@ -90,7 +92,11 @@ func (n Namespace) ID() []byte {
9092
return n.data[NamespaceVersionSize:]
9193
}
9294

93-
func validate(version uint8, id []byte) error {
95+
// ValidateUserNamespace returns an error if the provided version is not
96+
// supported or the provided id does not meet the requirements
97+
// for the provided version. This should be used for validating
98+
// user specified namespaces
99+
func ValidateUserNamespace(version uint8, id []byte) error {
94100
err := validateVersionSupported(version)
95101
if err != nil {
96102
return err

share/parse.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ import (
88
// ParseTxs collects all of the transactions from the shares provided
99
func ParseTxs(shares []Share) ([][]byte, error) {
1010
// parse the shares
11-
rawTxs, err := parseCompactShares(shares, SupportedShareVersions)
11+
rawTxs, err := parseCompactShares(shares)
1212
if err != nil {
1313
return nil, err
1414
}

share/parse_compact_shares.go

+1-18
Original file line numberDiff line numberDiff line change
@@ -6,16 +6,11 @@ package share
66
// an error is returned. The returned data [][]byte does not have namespaces,
77
// info bytes, data length delimiter, or unit length delimiters and are ready to
88
// be unmarshalled.
9-
func parseCompactShares(shares []Share, supportedShareVersions []uint8) (data [][]byte, err error) {
9+
func parseCompactShares(shares []Share) (data [][]byte, err error) {
1010
if len(shares) == 0 {
1111
return nil, nil
1212
}
1313

14-
err = validateShareVersions(shares, supportedShareVersions)
15-
if err != nil {
16-
return nil, err
17-
}
18-
1914
rawData, err := extractRawData(shares)
2015
if err != nil {
2116
return nil, err
@@ -29,18 +24,6 @@ func parseCompactShares(shares []Share, supportedShareVersions []uint8) (data []
2924
return data, nil
3025
}
3126

32-
// validateShareVersions returns an error if the shares contain a share with an
33-
// unsupported share version. Returns nil if all shares contain supported share
34-
// versions.
35-
func validateShareVersions(shares []Share, supportedShareVersions []uint8) error {
36-
for i := 0; i < len(shares); i++ {
37-
if err := shares[i].DoesSupportVersions(supportedShareVersions); err != nil {
38-
return err
39-
}
40-
}
41-
return nil
42-
}
43-
4427
// parseRawData returns the units (transactions, PFB transactions, intermediate
4528
// state roots) contained in raw data by parsing the unit length delimiter
4629
// prefixed to each unit.

share/parse_sparse_shares_test.go

-32
Original file line numberDiff line numberDiff line change
@@ -91,38 +91,6 @@ func Test_parseSparseShares(t *testing.T) {
9191
}
9292
}
9393

94-
func Test_parseSparseSharesErrors(t *testing.T) {
95-
type testCase struct {
96-
name string
97-
shares []Share
98-
}
99-
100-
unsupportedShareVersion := 5
101-
infoByte, _ := NewInfoByte(uint8(unsupportedShareVersion), true)
102-
103-
rawShare := RandomNamespace().Bytes()
104-
rawShare = append(rawShare, byte(infoByte))
105-
rawShare = append(rawShare, bytes.Repeat([]byte{0}, ShareSize-len(rawShare))...)
106-
share, err := NewShare(rawShare)
107-
if err != nil {
108-
t.Fatal(err)
109-
}
110-
111-
tests := []testCase{
112-
{
113-
"share with unsupported share version",
114-
[]Share{*share},
115-
},
116-
}
117-
118-
for _, tt := range tests {
119-
t.Run(tt.name, func(*testing.T) {
120-
_, err := parseSparseShares(tt.shares, SupportedShareVersions)
121-
assert.Error(t, err)
122-
})
123-
}
124-
}
125-
12694
func Test_parseSparseSharesWithNamespacedPadding(t *testing.T) {
12795
sss := NewSparseShareSplitter()
12896
randomSmallBlob := generateRandomBlob(ContinuationSparseShareContentSize / 2)

share/shares.go share/share.go

+5-4
Original file line numberDiff line numberDiff line change
@@ -23,10 +23,11 @@ func NewShare(data []byte) (*Share, error) {
2323
if err := validateSize(data); err != nil {
2424
return nil, err
2525
}
26-
if err := validate(data[0], data[1:NamespaceSize]); err != nil {
26+
sh := &Share{data}
27+
if err := sh.doesSupportVersions(SupportedShareVersions); err != nil {
2728
return nil, err
2829
}
29-
return &Share{data}, nil
30+
return sh, nil
3031
}
3132

3233
func validateSize(data []byte) error {
@@ -53,8 +54,8 @@ func (s *Share) Version() uint8 {
5354
return s.InfoByte().Version()
5455
}
5556

56-
// DoesSupportVersions checks if the share version is supported
57-
func (s *Share) DoesSupportVersions(supportedShareVersions []uint8) error {
57+
// doesSupportVersions checks if the share version is supported
58+
func (s *Share) doesSupportVersions(supportedShareVersions []uint8) error {
5859
ver := s.Version()
5960
if !bytes.Contains(supportedShareVersions, []byte{ver}) {
6061
return fmt.Errorf("unsupported share version %v is not present in the list of supported share versions %v", ver, supportedShareVersions)

share/shares_test.go share/share_test.go

+23
Original file line numberDiff line numberDiff line change
@@ -214,3 +214,26 @@ func TestIsPadding(t *testing.T) {
214214
})
215215
}
216216
}
217+
218+
func TestUnsupportedShareVersion(t *testing.T) {
219+
unsupportedShareVersion := 5
220+
infoByte, _ := NewInfoByte(uint8(unsupportedShareVersion), true)
221+
222+
rawShare := RandomNamespace().Bytes()
223+
rawShare = append(rawShare, byte(infoByte))
224+
rawShare = append(rawShare, bytes.Repeat([]byte{0}, ShareSize-len(rawShare))...)
225+
_, err := NewShare(rawShare)
226+
require.Error(t, err)
227+
}
228+
229+
func TestShareToBytesAndFromBytes(t *testing.T) {
230+
blobs := []*Blob{generateRandomBlob(580), generateRandomBlob(380), generateRandomBlob(1100)}
231+
SortBlobs(blobs)
232+
shares, err := splitBlobs(blobs...)
233+
require.NoError(t, err)
234+
235+
shareBytes := ToBytes(shares)
236+
reconstructedShares, err := FromBytes(shareBytes)
237+
require.NoError(t, err)
238+
assert.Equal(t, shares, reconstructedShares)
239+
}

0 commit comments

Comments
 (0)