forked from celestiaorg/celestia-node
-
Notifications
You must be signed in to change notification settings - Fork 0
/
namespace.go
184 lines (158 loc) · 6.47 KB
/
namespace.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
package share
import (
"bytes"
"encoding/hex"
"fmt"
appns "github.com/celestiaorg/celestia-app/pkg/namespace"
"github.com/celestiaorg/nmt/namespace"
)
// NamespaceSize is a system-wide size for NMT namespaces.
const NamespaceSize = appns.NamespaceSize
// Various reserved namespaces.
var (
// MaxPrimaryReservedNamespace is the highest primary reserved namespace.
// Namespaces lower than this are reserved for protocol use.
MaxPrimaryReservedNamespace = Namespace(appns.MaxPrimaryReservedNamespace.Bytes())
// MinSecondaryReservedNamespace is the lowest secondary reserved namespace
// reserved for protocol use. Namespaces higher than this are reserved for
// protocol use.
MinSecondaryReservedNamespace = Namespace(appns.MinSecondaryReservedNamespace.Bytes())
ParitySharesNamespace = Namespace(appns.ParitySharesNamespace.Bytes())
TailPaddingNamespace = Namespace(appns.TailPaddingNamespace.Bytes())
PrimaryReservedPaddingNamespace = Namespace(appns.PrimaryReservedPaddingNamespace.Bytes())
TxNamespace = Namespace(appns.TxNamespace.Bytes())
PayForBlobNamespace = Namespace(appns.PayForBlobNamespace.Bytes())
ISRNamespace = Namespace(appns.IntermediateStateRootsNamespace.Bytes())
)
// Namespace represents namespace of a Share.
// Consists of version byte and namespace ID.
type Namespace []byte
// NewBlobNamespaceV0 takes a variable size byte slice and creates a valid version 0 Blob Namespace.
// The byte slice must be <= 10 bytes.
// If it is less than 10 bytes, it will be left padded to size 10 with 0s.
// Use predefined namespaces above, if non-blob namespace is needed.
func NewBlobNamespaceV0(id []byte) (Namespace, error) {
if len(id) == 0 || len(id) > appns.NamespaceVersionZeroIDSize {
return nil, fmt.Errorf(
"namespace id must be > 0 && <= %d, but it was %d bytes", appns.NamespaceVersionZeroIDSize, len(id))
}
n := make(Namespace, NamespaceSize)
// version and zero padding are already set as zero,
// so simply copying subNID to the end is enough to comply the V0 spec
copy(n[len(n)-len(id):], id)
return n, n.ValidateForBlob()
}
// NamespaceFromBytes converts bytes into Namespace and validates it.
func NamespaceFromBytes(b []byte) (Namespace, error) {
n := Namespace(b)
return n, n.Validate()
}
// Version reports version of the Namespace.
func (n Namespace) Version() byte {
return n[appns.NamespaceVersionSize-1]
}
// ID reports ID of the Namespace.
func (n Namespace) ID() namespace.ID {
return namespace.ID(n[appns.NamespaceVersionSize:])
}
// ToNMT converts the whole Namespace(both Version and ID parts) into NMT's namespace.ID
// NOTE: Once https://github.com/celestiaorg/nmt/issues/206 is closed Namespace should become NNT's
// type.
func (n Namespace) ToNMT() namespace.ID {
return namespace.ID(n)
}
// ToAppNamespace converts the Namespace to App's definition of Namespace.
// TODO: Unify types between node and app
func (n Namespace) ToAppNamespace() appns.Namespace {
return appns.Namespace{Version: n.Version(), ID: n.ID()}
}
// Len reports the total length of the namespace.
func (n Namespace) Len() int {
return len(n)
}
// String stringifies the Namespace.
func (n Namespace) String() string {
return hex.EncodeToString(n)
}
// Equals compares two Namespaces.
func (n Namespace) Equals(target Namespace) bool {
return bytes.Equal(n, target)
}
// Validate checks if the namespace is correct.
func (n Namespace) Validate() error {
if n.Len() != NamespaceSize {
return fmt.Errorf("invalid namespace length: expected %d, got %d", NamespaceSize, n.Len())
}
if n.Version() != appns.NamespaceVersionZero && n.Version() != appns.NamespaceVersionMax {
return fmt.Errorf("invalid namespace version %v", n.Version())
}
if len(n.ID()) != appns.NamespaceIDSize {
return fmt.Errorf("invalid namespace id length: expected %d, got %d", appns.NamespaceIDSize, n.ID().Size())
}
if n.Version() == appns.NamespaceVersionZero && !bytes.HasPrefix(n.ID(), appns.NamespaceVersionZeroPrefix) {
return fmt.Errorf("invalid namespace id: expect %d leading zeroes", len(appns.NamespaceVersionZeroPrefix))
}
return nil
}
// ValidateForData checks if the Namespace is of real/useful data.
func (n Namespace) ValidateForData() error {
if err := n.Validate(); err != nil {
return err
}
if n.Equals(ParitySharesNamespace) || n.Equals(TailPaddingNamespace) {
return fmt.Errorf("invalid data namespace(%s): parity and tail padding namespace are forbidden", n)
}
if n.Version() != appns.NamespaceVersionZero {
return fmt.Errorf("invalid data namespace(%s): only version 0 is supported", n)
}
return nil
}
// ValidateForBlob checks if the Namespace is valid blob namespace.
func (n Namespace) ValidateForBlob() error {
if err := n.ValidateForData(); err != nil {
return err
}
if bytes.Compare(n, MaxPrimaryReservedNamespace) < 1 {
return fmt.Errorf("invalid blob namespace(%s): reserved namespaces are forbidden", n)
}
if bytes.Compare(n, MinSecondaryReservedNamespace) > -1 {
return fmt.Errorf("invalid blob namespace(%s): reserved namespaces are forbidden", n)
}
return nil
}
// IsAboveMax checks if the namespace is above the maximum namespace of the given hash.
func (n Namespace) IsAboveMax(nodeHash []byte) bool {
return !n.IsLessOrEqual(nodeHash[n.Len() : n.Len()*2])
}
// IsBelowMin checks if the target namespace is below the minimum namespace of the given hash.
func (n Namespace) IsBelowMin(nodeHash []byte) bool {
return n.IsLess(nodeHash[:n.Len()])
}
// IsOutsideRange checks if the namespace is outside the min-max range of the given hashes.
func (n Namespace) IsOutsideRange(leftNodeHash, rightNodeHash []byte) bool {
return n.IsBelowMin(leftNodeHash) || n.IsAboveMax(rightNodeHash)
}
// Repeat copies the Namespace t times.
func (n Namespace) Repeat(t int) []Namespace {
ns := make([]Namespace, t)
for i := 0; i < t; i++ {
ns[i] = n
}
return ns
}
// IsLess reports if the Namespace is less than the target.
func (n Namespace) IsLess(target Namespace) bool {
return bytes.Compare(n, target) == -1
}
// IsLessOrEqual reports if the Namespace is less than the target.
func (n Namespace) IsLessOrEqual(target Namespace) bool {
return bytes.Compare(n, target) < 1
}
// IsGreater reports if the Namespace is greater than the target.
func (n Namespace) IsGreater(target Namespace) bool {
return bytes.Compare(n, target) == 1
}
// IsGreaterOrEqualThan reports if the Namespace is greater or equal than the target.
func (n Namespace) IsGreaterOrEqualThan(target Namespace) bool {
return bytes.Compare(n, target) > -1
}