Skip to content

Commit 6be9632

Browse files
authored
chore: fix share and blob validation (#101)
I've now integrated celestia-app locally. These are the last set of changes in order to make everything work. These are: - Remove validation of share verision. Erasure coded data doesn't comply with this rule. The validation has been added to the parsing function - Check for empty namespaces in the blob constructor. There's a bit of a footgun here where users could set an empty namespace.
1 parent 7c3a449 commit 6be9632

11 files changed

+74
-38
lines changed

inclusion/commitment_test.go

-7
Original file line numberDiff line numberDiff line change
@@ -88,13 +88,6 @@ func TestCreateCommitment(t *testing.T) {
8888
shareVersion: share.ShareVersionOne,
8989
signer: bytes.Repeat([]byte{1}, share.SignerSize),
9090
},
91-
{
92-
name: "blob with unsupported share version should return error",
93-
namespace: ns1,
94-
blob: bytes.Repeat([]byte{0xFF}, share.AvailableBytesFromSparseShares(2)),
95-
expectErr: true,
96-
shareVersion: uint8(2), // unsupported share version
97-
},
9891
}
9992
for _, tt := range tests {
10093
t.Run(tt.name, func(t *testing.T) {

share/blob.go

+27-8
Original file line numberDiff line numberDiff line change
@@ -25,14 +25,25 @@ func NewBlob(ns Namespace, data []byte, shareVersion uint8, signer []byte) (*Blo
2525
if len(data) == 0 {
2626
return nil, errors.New("data can not be empty")
2727
}
28-
if shareVersion == 0 && signer != nil {
29-
return nil, errors.New("share version 0 does not support signer")
28+
if ns.IsEmpty() {
29+
return nil, errors.New("namespace can not be empty")
3030
}
31-
if shareVersion == 1 && len(signer) != SignerSize {
32-
return nil, fmt.Errorf("share version 1 requires signer of size %d bytes", SignerSize)
31+
if ns.Version() != NamespaceVersionZero {
32+
return nil, fmt.Errorf("namespace version must be %d got %d", NamespaceVersionZero, ns.Version())
3333
}
34-
if shareVersion > MaxShareVersion {
35-
return nil, fmt.Errorf("share version can not be greater than MaxShareVersion %d", MaxShareVersion)
34+
switch shareVersion {
35+
case ShareVersionZero:
36+
if signer != nil {
37+
return nil, errors.New("share version 0 does not support signer")
38+
}
39+
case ShareVersionOne:
40+
if len(signer) != SignerSize {
41+
return nil, fmt.Errorf("share version 1 requires signer of size %d bytes", SignerSize)
42+
}
43+
// Note that we don't specifically check that shareVersion is less than 128 as this is caught
44+
// by the default case
45+
default:
46+
return nil, fmt.Errorf("share version %d not supported. Please use 0 or 1", shareVersion)
3647
}
3748
return &Blob{
3849
namespace: ns,
@@ -79,8 +90,8 @@ func NewBlobFromProto(pb *v1.BlobProto) (*Blob, error) {
7990
if pb.NamespaceVersion > NamespaceVersionMax {
8091
return nil, errors.New("namespace version can not be greater than MaxNamespaceVersion")
8192
}
82-
if len(pb.Data) == 0 {
83-
return nil, errors.New("blob data can not be empty")
93+
if pb.ShareVersion > MaxShareVersion {
94+
return nil, fmt.Errorf("share version can not be greater than MaxShareVersion %d", MaxShareVersion)
8495
}
8596
ns, err := NewNamespace(uint8(pb.NamespaceVersion), pb.NamespaceId)
8697
if err != nil {
@@ -124,6 +135,14 @@ func (b *Blob) Compare(other *Blob) int {
124135
return b.namespace.Compare(other.namespace)
125136
}
126137

138+
// IsEmpty returns true if the blob is empty. This is an invalid
139+
// construction that can only occur if using the nil value. We
140+
// only check that the data is empty but this also implies that
141+
// all other fields would have their zero value
142+
func (b *Blob) IsEmpty() bool {
143+
return len(b.data) == 0
144+
}
145+
127146
// Sort sorts the blobs by their namespace.
128147
func SortBlobs(blobs []*Blob) {
129148
sort.SliceStable(blobs, func(i, j int) bool {

share/blob_test.go

+16-2
Original file line numberDiff line numberDiff line change
@@ -48,7 +48,21 @@ func TestBlobConstructor(t *testing.T) {
4848

4949
_, err = NewBlob(ns, data, 128, nil)
5050
require.Error(t, err)
51-
require.Contains(t, err.Error(), "share version can not be greater than MaxShareVersion")
51+
require.Contains(t, err.Error(), "share version 128 not supported")
52+
53+
_, err = NewBlob(ns, data, 2, nil)
54+
require.Error(t, err)
55+
require.Contains(t, err.Error(), "share version 2 not supported")
56+
57+
_, err = NewBlob(Namespace{}, data, 1, signer)
58+
require.Error(t, err)
59+
require.Contains(t, err.Error(), "namespace can not be empty")
60+
61+
ns2, err := NewNamespace(NamespaceVersionMax, ns.ID())
62+
require.NoError(t, err)
63+
_, err = NewBlob(ns2, data, 0, nil)
64+
require.Error(t, err)
65+
require.Contains(t, err.Error(), "namespace version must be 0")
5266
}
5367

5468
func TestNewBlobFromProto(t *testing.T) {
@@ -86,7 +100,7 @@ func TestNewBlobFromProto(t *testing.T) {
86100
ShareVersion: 0,
87101
Data: []byte{},
88102
},
89-
expectedErr: "blob data can not be empty",
103+
expectedErr: "data can not be empty",
90104
},
91105
{
92106
name: "invalid namespace ID length",

share/namespace.go

+5
Original file line numberDiff line numberDiff line change
@@ -125,6 +125,11 @@ func validateID(version uint8, id []byte) error {
125125
return nil
126126
}
127127

128+
// IsEmpty returns true if the namespace is empty
129+
func (n Namespace) IsEmpty() bool {
130+
return len(n.data) == 0
131+
}
132+
128133
// IsReserved returns true if the namespace is reserved
129134
// for the Celestia state machine
130135
func (n Namespace) IsReserved() bool {

share/parse.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ import (
77

88
// ParseTxs collects all of the transactions from the shares provided
99
func ParseTxs(shares []Share) ([][]byte, error) {
10-
// parse the shares
10+
// parse the shares. Only share version 0 is supported for transactions
1111
rawTxs, err := parseCompactShares(shares)
1212
if err != nil {
1313
return nil, err
@@ -18,7 +18,7 @@ func ParseTxs(shares []Share) ([][]byte, error) {
1818

1919
// ParseBlobs collects all blobs from the shares provided
2020
func ParseBlobs(shares []Share) ([]*Blob, error) {
21-
blobList, err := parseSparseShares(shares, SupportedShareVersions)
21+
blobList, err := parseSparseShares(shares)
2222
if err != nil {
2323
return []*Blob{}, err
2424
}

share/parse_compact_shares.go

+8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
package share
22

3+
import "fmt"
4+
35
// parseCompactShares returns data (transactions or intermediate state roots
46
// based on the contents of rawShares and supportedShareVersions. If rawShares
57
// contains a share with a version that isn't present in supportedShareVersions,
@@ -11,6 +13,12 @@ func parseCompactShares(shares []Share) (data [][]byte, err error) {
1113
return nil, nil
1214
}
1315

16+
for _, share := range shares {
17+
if share.Version() != ShareVersionZero {
18+
return nil, fmt.Errorf("unsupported share version for compact shares %v", share.Version())
19+
}
20+
}
21+
1422
rawData, err := extractRawData(shares)
1523
if err != nil {
1624
return nil, err

share/parse_sparse_shares.go

+3-3
Original file line numberDiff line numberDiff line change
@@ -16,16 +16,16 @@ type sequence struct {
1616
// parseSparseShares iterates through rawShares and parses out individual
1717
// blobs. It returns an error if a rawShare contains a share version that
1818
// isn't present in supportedShareVersions.
19-
func parseSparseShares(shares []Share, supportedShareVersions []uint8) (blobs []*Blob, err error) {
19+
func parseSparseShares(shares []Share) (blobs []*Blob, err error) {
2020
if len(shares) == 0 {
2121
return nil, nil
2222
}
2323
sequences := make([]sequence, 0)
2424

2525
for _, share := range shares {
2626
version := share.Version()
27-
if !bytes.Contains(supportedShareVersions, []byte{version}) {
28-
return nil, fmt.Errorf("unsupported share version %v is not present in supported share versions %v", version, supportedShareVersions)
27+
if !bytes.Contains(SupportedShareVersions, []byte{version}) {
28+
return nil, fmt.Errorf("unsupported share version %v is not present in supported share versions %v", version, SupportedShareVersions)
2929
}
3030

3131
if share.IsPadding() {

share/parse_sparse_shares_test.go

+4-4
Original file line numberDiff line numberDiff line change
@@ -60,7 +60,7 @@ func Test_parseSparseShares(t *testing.T) {
6060

6161
shares, err := splitBlobs(blobs...)
6262
require.NoError(t, err)
63-
parsedBlobs, err := parseSparseShares(shares, SupportedShareVersions)
63+
parsedBlobs, err := parseSparseShares(shares)
6464
if err != nil {
6565
t.Error(err)
6666
}
@@ -77,7 +77,7 @@ func Test_parseSparseShares(t *testing.T) {
7777
blobs := generateRandomlySizedBlobs(tc.blobCount, tc.blobSize)
7878
shares, err := splitBlobs(blobs...)
7979
require.NoError(t, err)
80-
parsedBlobs, err := parseSparseShares(shares, SupportedShareVersions)
80+
parsedBlobs, err := parseSparseShares(shares)
8181
if err != nil {
8282
t.Error(err)
8383
}
@@ -114,7 +114,7 @@ func Test_parseSparseSharesWithNamespacedPadding(t *testing.T) {
114114
require.NoError(t, err)
115115

116116
shares := sss.Export()
117-
pblobs, err := parseSparseShares(shares, SupportedShareVersions)
117+
pblobs, err := parseSparseShares(shares)
118118
require.NoError(t, err)
119119
require.Equal(t, blobs, pblobs)
120120
}
@@ -125,7 +125,7 @@ func Test_parseShareVersionOne(t *testing.T) {
125125
v1shares, err := splitBlobs(v1blob)
126126
require.NoError(t, err)
127127

128-
parsedBlobs, err := parseSparseShares(v1shares, SupportedShareVersions)
128+
parsedBlobs, err := parseSparseShares(v1shares)
129129
require.NoError(t, err)
130130
require.Equal(t, v1blob, parsedBlobs[0])
131131
require.Len(t, parsedBlobs, 1)

share/share.go

+5-9
Original file line numberDiff line numberDiff line change
@@ -23,11 +23,7 @@ func NewShare(data []byte) (*Share, error) {
2323
if err := validateSize(data); err != nil {
2424
return nil, err
2525
}
26-
sh := &Share{data}
27-
if err := sh.doesSupportVersions(SupportedShareVersions); err != nil {
28-
return nil, err
29-
}
30-
return sh, nil
26+
return &Share{data}, nil
3127
}
3228

3329
func validateSize(data []byte) error {
@@ -54,11 +50,11 @@ func (s *Share) Version() uint8 {
5450
return s.InfoByte().Version()
5551
}
5652

57-
// doesSupportVersions checks if the share version is supported
58-
func (s *Share) doesSupportVersions(supportedShareVersions []uint8) error {
53+
// CheckVersionSupported checks if the share version is supported
54+
func (s *Share) CheckVersionSupported() error {
5955
ver := s.Version()
60-
if !bytes.Contains(supportedShareVersions, []byte{ver}) {
61-
return fmt.Errorf("unsupported share version %v is not present in the list of supported share versions %v", ver, supportedShareVersions)
56+
if !bytes.Contains(SupportedShareVersions, []byte{ver}) {
57+
return fmt.Errorf("unsupported share version %v is not present in the list of supported share versions %v", ver, SupportedShareVersions)
6258
}
6359
return nil
6460
}

share/share_test.go

+3-2
Original file line numberDiff line numberDiff line change
@@ -222,8 +222,9 @@ func TestUnsupportedShareVersion(t *testing.T) {
222222
rawShare := RandomNamespace().Bytes()
223223
rawShare = append(rawShare, byte(infoByte))
224224
rawShare = append(rawShare, bytes.Repeat([]byte{0}, ShareSize-len(rawShare))...)
225-
_, err := NewShare(rawShare)
226-
require.Error(t, err)
225+
share, err := NewShare(rawShare)
226+
require.NoError(t, err)
227+
require.Error(t, share.CheckVersionSupported())
227228
}
228229

229230
func TestShareToBytesAndFromBytes(t *testing.T) {

tx/blob_tx.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -59,7 +59,7 @@ func MarshalBlobTx(tx []byte, blobs ...*share.Blob) ([]byte, error) {
5959
}
6060
// nil check
6161
for i, b := range blobs {
62-
if b == nil {
62+
if b == nil || b.IsEmpty() {
6363
return nil, fmt.Errorf("blob %d is nil", i)
6464
}
6565
}

0 commit comments

Comments
 (0)