[#170] checksum: Drop Empty method

Signed-off-by: Pavel Karpy <carpawell@nspcc.ru>
This commit is contained in:
Pavel Karpy 2021-03-22 13:56:18 +03:00 committed by LeL
parent 9c502a9cae
commit fd13e61266
11 changed files with 184 additions and 252 deletions

View file

@ -1,15 +1,23 @@
package checksum package checksum
import ( import (
"bytes"
"crypto/sha256" "crypto/sha256"
"encoding/hex" "encoding/hex"
"fmt" "fmt"
"github.com/nspcc-dev/neofs-api-go/v2/refs" "github.com/nspcc-dev/neofs-api-go/v2/refs"
"github.com/nspcc-dev/tzhash/tz"
) )
// Checksum represents v2-compatible checksum. // Checksum represents checksum of some digital data.
//
// Checksum is mutually compatible with github.com/nspcc-dev/neofs-api-go/v2/refs.Checksum
// message. See ReadFromV2 / WriteToV2 methods.
//
// Instances can be created using built-in var declaration.
//
// Note that direct typecast is not safe and may result in loss of compatibility:
// _ = Checksum(refs.Checksum{}) // not recommended
type Checksum refs.Checksum type Checksum refs.Checksum
// Type represents the enumeration // Type represents the enumeration
@ -23,29 +31,33 @@ const (
// SHA256 is a SHA256 checksum type. // SHA256 is a SHA256 checksum type.
SHA256 SHA256
// TZ is a Tillich-Zemor checksum type. // TZ is a Tillich-Zémor checksum type.
TZ TZ
) )
// NewFromV2 wraps v2 Checksum message to Checksum. // ReadFromV2 reads Checksum from the refs.Checksum message.
// //
// Nil refs.Checksum converts to nil. // See also WriteToV2.
func NewFromV2(cV2 *refs.Checksum) *Checksum { func (c *Checksum) ReadFromV2(m refs.Checksum) {
return (*Checksum)(cV2) *c = Checksum(m)
} }
// New creates and initializes blank Checksum. // WriteToV2 writes Checksum to the refs.Checksum message.
// The message must not be nil.
// //
// Defaults: // See also ReadFromV2.
// - sum: nil; func (c Checksum) WriteToV2(m *refs.Checksum) {
// - type: Unknown. *m = (refs.Checksum)(c)
func New() *Checksum {
return NewFromV2(new(refs.Checksum))
} }
// Type returns checksum type. // Type returns checksum type.
func (c *Checksum) Type() Type { //
switch (*refs.Checksum)(c).GetType() { // Zero Checksum has Unknown checksum type.
//
// See also SetTillichZemor and SetSHA256.
func (c Checksum) Type() Type {
v2 := (refs.Checksum)(c)
switch v2.GetType() {
case refs.SHA256: case refs.SHA256:
return SHA256 return SHA256
case refs.TillichZemor: case refs.TillichZemor:
@ -55,93 +67,77 @@ func (c *Checksum) Type() Type {
} }
} }
// Sum returns checksum bytes. // Value returns checksum bytes. Return value
func (c *Checksum) Sum() []byte { // MUST NOT be mutated.
return (*refs.Checksum)(c).GetSum() //
// Zero Checksum has nil sum.
//
// See also SetTillichZemor and SetSHA256.
func (c Checksum) Value() []byte {
v2 := (refs.Checksum)(c)
return v2.GetSum()
} }
// SetSHA256 sets checksum to SHA256 hash. // SetSHA256 sets checksum to SHA256 hash.
//
// See also Calculate.
func (c *Checksum) SetSHA256(v [sha256.Size]byte) { func (c *Checksum) SetSHA256(v [sha256.Size]byte) {
checksum := (*refs.Checksum)(c) v2 := (*refs.Checksum)(c)
checksum.SetType(refs.SHA256) v2.SetType(refs.SHA256)
checksum.SetSum(v[:]) v2.SetSum(v[:])
} }
// SetTillichZemor sets checksum to Tillich-Zemor hash. // Calculate calculates checksum and sets it
func (c *Checksum) SetTillichZemor(v [64]byte) { // to the passed checksum. Checksum must not be nil.
checksum := (*refs.Checksum)(c)
checksum.SetType(refs.TillichZemor)
checksum.SetSum(v[:])
}
// ToV2 converts Checksum to v2 Checksum message.
// //
// Nil Checksum converts to nil. // Does nothing if the passed type is not one of the:
func (c *Checksum) ToV2() *refs.Checksum { // * SHA256;
return (*refs.Checksum)(c) // * TZ.
} //
// Does not mutate the passed value.
func Equal(cs1, cs2 *Checksum) bool { //
return cs1.Type() == cs2.Type() && bytes.Equal(cs1.Sum(), cs2.Sum()) // See also SetSHA256, SetTillichZemor.
} func Calculate(c *Checksum, t Type, v []byte) {
switch t {
// Marshal marshals Checksum into a protobuf binary form. case SHA256:
func (c *Checksum) Marshal() ([]byte, error) { c.SetSHA256(sha256.Sum256(v))
return (*refs.Checksum)(c).StableMarshal(nil) case TZ:
} c.SetTillichZemor(tz.Sum(v))
// Unmarshal unmarshals protobuf binary representation of Checksum.
func (c *Checksum) Unmarshal(data []byte) error {
return (*refs.Checksum)(c).Unmarshal(data)
}
// MarshalJSON encodes Checksum to protobuf JSON format.
func (c *Checksum) MarshalJSON() ([]byte, error) {
return (*refs.Checksum)(c).MarshalJSON()
}
// UnmarshalJSON decodes Checksum from protobuf JSON format.
func (c *Checksum) UnmarshalJSON(data []byte) error {
return (*refs.Checksum)(c).UnmarshalJSON(data)
}
func (c *Checksum) String() string {
return hex.EncodeToString((*refs.Checksum)(c).GetSum())
}
// Parse parses Checksum from its string representation.
func (c *Checksum) Parse(s string) error {
data, err := hex.DecodeString(s)
if err != nil {
return err
}
var typ refs.ChecksumType
switch ln := len(data); ln {
default: default:
return fmt.Errorf("unsupported checksum length %d", ln)
case sha256.Size:
typ = refs.SHA256
case 64:
typ = refs.TillichZemor
} }
cV2 := (*refs.Checksum)(c)
cV2.SetType(typ)
cV2.SetSum(data)
return nil
} }
// String returns string representation of Type. // SetTillichZemor sets checksum to Tillich-Zémor hash.
// //
// String mapping: // See also Calculate.
// * TZ: TZ; func (c *Checksum) SetTillichZemor(v [tz.Size]byte) {
// * SHA256: SHA256; v2 := (*refs.Checksum)(c)
// * Unknown, default: CHECKSUM_TYPE_UNSPECIFIED.
v2.SetType(refs.TillichZemor)
v2.SetSum(v[:])
}
// String implements fmt.Stringer.
//
// String is designed to be human-readable, and its format MAY differ between
// SDK versions.
func (c Checksum) String() string {
v2 := (refs.Checksum)(c)
return fmt.Sprintf("%s:%s", c.Type(), hex.EncodeToString(v2.GetSum()))
}
// Empty returns true if it is called on
// zero checksum.
func (c Checksum) Empty() bool {
v2 := (refs.Checksum)(c)
return v2.GetSum() == nil
}
// String implements fmt.Stringer.
//
// String is designed to be human-readable, and its format MAY differ between
// SDK versions.
func (m Type) String() string { func (m Type) String() string {
var m2 refs.ChecksumType var m2 refs.ChecksumType
@ -156,26 +152,3 @@ func (m Type) String() string {
return m2.String() return m2.String()
} }
// FromString parses Type from a string representation.
// It is a reverse action to String().
//
// Returns true if s was parsed successfully.
func (m *Type) FromString(s string) bool {
var g refs.ChecksumType
ok := g.FromString(s)
if ok {
switch g {
default:
*m = Unknown
case refs.TillichZemor:
*m = TZ
case refs.SHA256:
*m = SHA256
}
}
return ok
}

View file

@ -6,19 +6,12 @@ import (
"testing" "testing"
"github.com/nspcc-dev/neofs-api-go/v2/refs" "github.com/nspcc-dev/neofs-api-go/v2/refs"
"github.com/nspcc-dev/tzhash/tz"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
) )
func randSHA256(t *testing.T) [sha256.Size]byte {
cSHA256 := [sha256.Size]byte{}
_, err := rand.Read(cSHA256[:])
require.NoError(t, err)
return cSHA256
}
func TestChecksum(t *testing.T) { func TestChecksum(t *testing.T) {
c := New() var c Checksum
cSHA256 := [sha256.Size]byte{} cSHA256 := [sha256.Size]byte{}
_, _ = rand.Read(cSHA256[:]) _, _ = rand.Read(cSHA256[:])
@ -26,150 +19,62 @@ func TestChecksum(t *testing.T) {
c.SetSHA256(cSHA256) c.SetSHA256(cSHA256)
require.Equal(t, SHA256, c.Type()) require.Equal(t, SHA256, c.Type())
require.Equal(t, cSHA256[:], c.Sum()) require.Equal(t, cSHA256[:], c.Value())
cV2 := c.ToV2() var cV2 refs.Checksum
c.WriteToV2(&cV2)
require.Equal(t, refs.SHA256, cV2.GetType()) require.Equal(t, refs.SHA256, cV2.GetType())
require.Equal(t, cSHA256[:], cV2.GetSum()) require.Equal(t, cSHA256[:], cV2.GetSum())
cTZ := [64]byte{} cTZ := [tz.Size]byte{}
_, _ = rand.Read(cSHA256[:]) _, _ = rand.Read(cSHA256[:])
c.SetTillichZemor(cTZ) c.SetTillichZemor(cTZ)
require.Equal(t, TZ, c.Type()) require.Equal(t, TZ, c.Type())
require.Equal(t, cTZ[:], c.Sum()) require.Equal(t, cTZ[:], c.Value())
cV2 = c.ToV2() c.WriteToV2(&cV2)
require.Equal(t, refs.TillichZemor, cV2.GetType()) require.Equal(t, refs.TillichZemor, cV2.GetType())
require.Equal(t, cTZ[:], cV2.GetSum()) require.Equal(t, cTZ[:], cV2.GetSum())
} }
func TestEqualChecksums(t *testing.T) {
require.True(t, Equal(nil, nil))
csSHA := [sha256.Size]byte{}
_, _ = rand.Read(csSHA[:])
cs1 := New()
cs1.SetSHA256(csSHA)
cs2 := New()
cs2.SetSHA256(csSHA)
require.True(t, Equal(cs1, cs2))
csSHA[0]++
cs2.SetSHA256(csSHA)
require.False(t, Equal(cs1, cs2))
}
func TestChecksumEncoding(t *testing.T) {
cs := New()
cs.SetSHA256(randSHA256(t))
t.Run("binary", func(t *testing.T) {
data, err := cs.Marshal()
require.NoError(t, err)
c2 := New()
require.NoError(t, c2.Unmarshal(data))
require.Equal(t, cs, c2)
})
t.Run("json", func(t *testing.T) {
data, err := cs.MarshalJSON()
require.NoError(t, err)
cs2 := New()
require.NoError(t, cs2.UnmarshalJSON(data))
require.Equal(t, cs, cs2)
})
t.Run("string", func(t *testing.T) {
cs2 := New()
require.NoError(t, cs2.Parse(cs.String()))
require.Equal(t, cs, cs2)
})
}
func TestNewChecksumFromV2(t *testing.T) {
t.Run("from nil", func(t *testing.T) {
var x *refs.Checksum
require.Nil(t, NewFromV2(x))
})
}
func TestChecksum_ToV2(t *testing.T) {
t.Run("nil", func(t *testing.T) {
var x *Checksum
require.Nil(t, x.ToV2())
})
}
func TestNewChecksum(t *testing.T) { func TestNewChecksum(t *testing.T) {
t.Run("default values", func(t *testing.T) { t.Run("default values", func(t *testing.T) {
chs := New() var chs Checksum
// check initial values // check initial values
require.Equal(t, Unknown, chs.Type()) require.Equal(t, Unknown, chs.Type())
require.Nil(t, chs.Sum()) require.Nil(t, chs.Value())
// convert to v2 message // convert to v2 message
chsV2 := chs.ToV2() var chsV2 refs.Checksum
chs.WriteToV2(&chsV2)
require.Equal(t, refs.UnknownChecksum, chsV2.GetType()) require.Equal(t, refs.UnknownChecksum, chsV2.GetType())
require.Nil(t, chsV2.GetSum()) require.Nil(t, chsV2.GetSum())
}) })
} }
type enumIface interface { func TestCalculation(t *testing.T) {
FromString(string) bool var c Checksum
String() string payload := []byte{0, 1, 2, 3, 4, 5}
}
type enumStringItem struct { t.Run("SHA256", func(t *testing.T) {
val enumIface orig := sha256.Sum256(payload)
str string
}
func testEnumStrings(t *testing.T, e enumIface, items []enumStringItem) { Calculate(&c, SHA256, payload)
for _, item := range items {
require.Equal(t, item.str, item.val.String())
s := item.val.String() require.Equal(t, orig[:], c.Value())
})
require.True(t, e.FromString(s), s) t.Run("TZ", func(t *testing.T) {
orig := tz.Sum(payload)
require.EqualValues(t, item.val, e, item.val) Calculate(&c, TZ, payload)
}
// incorrect strings require.Equal(t, orig[:], c.Value())
for _, str := range []string{
"some string",
"undefined",
} {
require.False(t, e.FromString(str))
}
}
func TestChecksumType_String(t *testing.T) {
toPtr := func(v Type) *Type {
return &v
}
testEnumStrings(t, new(Type), []enumStringItem{
{val: toPtr(TZ), str: "TZ"},
{val: toPtr(SHA256), str: "SHA256"},
{val: toPtr(Unknown), str: "CHECKSUM_TYPE_UNSPECIFIED"},
}) })
} }

18
checksum/doc.go Normal file
View file

@ -0,0 +1,18 @@
/*
Package checksum provides primitives to work with checksums.
Checksum is a basic type of data checksums.
For example, calculating checksums:
// retrieving any payload for hashing
var sha256Sum Checksum
Calculate(&sha256Sum, SHA256, payload) // sha256Sum contains SHA256 hash of the payload
var tzSum Checksum
Calculate(&tzSum, TZ, payload) // tzSum contains TZ hash of the payload
Using package types in an application is recommended to potentially work with
different protocol versions with which these types are compatible.
*/
package checksum

13
checksum/test/doc.go Normal file
View file

@ -0,0 +1,13 @@
/*
Package checksumtest provides functions for convenient testing of checksum package API.
Note that importing the package into source files is highly discouraged.
Random instance generation functions can be useful when testing expects any value, e.g.:
import checksumtest "github.com/nspcc-dev/neofs-sdk-go/checksum/test"
cs := checksumtest.Checksum()
// test the value
*/
package checksumtest

View file

@ -13,9 +13,9 @@ func Checksum() *checksum.Checksum {
rand.Read(cs[:]) rand.Read(cs[:])
x := checksum.New() var x checksum.Checksum
x.SetSHA256(cs) x.SetSHA256(cs)
return x return &x
} }

1
go.mod
View file

@ -11,6 +11,7 @@ require (
github.com/nspcc-dev/hrw v1.0.9 github.com/nspcc-dev/hrw v1.0.9
github.com/nspcc-dev/neo-go v0.98.0 github.com/nspcc-dev/neo-go v0.98.0
github.com/nspcc-dev/neofs-api-go/v2 v2.12.1 github.com/nspcc-dev/neofs-api-go/v2 v2.12.1
github.com/nspcc-dev/tzhash v1.5.2
github.com/stretchr/testify v1.7.0 github.com/stretchr/testify v1.7.0
go.uber.org/zap v1.18.1 go.uber.org/zap v1.18.1
) )

BIN
go.sum

Binary file not shown.

View file

@ -1,6 +1,7 @@
package object package object
import ( import (
"bytes"
"crypto/ecdsa" "crypto/ecdsa"
"crypto/sha256" "crypto/sha256"
"errors" "errors"
@ -20,10 +21,10 @@ var errIncorrectID = errors.New("incorrect object identifier")
// CalculatePayloadChecksum calculates and returns checksum of // CalculatePayloadChecksum calculates and returns checksum of
// object payload bytes. // object payload bytes.
func CalculatePayloadChecksum(payload []byte) *checksum.Checksum { func CalculatePayloadChecksum(payload []byte) *checksum.Checksum {
res := checksum.New() var res checksum.Checksum
res.SetSHA256(sha256.Sum256(payload)) checksum.Calculate(&res, checksum.SHA256, payload)
return res return &res
} }
// CalculateAndSetPayloadChecksum calculates checksum of current // CalculateAndSetPayloadChecksum calculates checksum of current
@ -38,7 +39,7 @@ func CalculateAndSetPayloadChecksum(obj *Object) {
// corresponds to its payload. // corresponds to its payload.
func VerifyPayloadChecksum(obj *Object) error { func VerifyPayloadChecksum(obj *Object) error {
actual := CalculatePayloadChecksum(obj.Payload()) actual := CalculatePayloadChecksum(obj.Payload())
if !checksum.Equal(obj.PayloadChecksum(), actual) { if !bytes.Equal(obj.PayloadChecksum().Value(), actual.Value()) {
return errCheckSumMismatch return errCheckSumMismatch
} }

View file

@ -201,33 +201,45 @@ func (o *Object) SetCreationEpoch(v uint64) {
// PayloadChecksum returns checksum of the object payload. // PayloadChecksum returns checksum of the object payload.
func (o *Object) PayloadChecksum() *checksum.Checksum { func (o *Object) PayloadChecksum() *checksum.Checksum {
return checksum.NewFromV2( var v checksum.Checksum
(*object.Object)(o). v.ReadFromV2(
*(*object.Object)(o).
GetHeader(). GetHeader().
GetPayloadHash(), GetPayloadHash(),
) )
return &v
} }
// SetPayloadChecksum sets checksum of the object payload. // SetPayloadChecksum sets checksum of the object payload.
func (o *Object) SetPayloadChecksum(v *checksum.Checksum) { func (o *Object) SetPayloadChecksum(v *checksum.Checksum) {
var v2 refs.Checksum
v.WriteToV2(&v2)
o.setHeaderField(func(h *object.Header) { o.setHeaderField(func(h *object.Header) {
h.SetPayloadHash(v.ToV2()) h.SetPayloadHash(&v2)
}) })
} }
// PayloadHomomorphicHash returns homomorphic hash of the object payload. // PayloadHomomorphicHash returns homomorphic hash of the object payload.
func (o *Object) PayloadHomomorphicHash() *checksum.Checksum { func (o *Object) PayloadHomomorphicHash() *checksum.Checksum {
return checksum.NewFromV2( var v checksum.Checksum
(*object.Object)(o). v.ReadFromV2(
*(*object.Object)(o).
GetHeader(). GetHeader().
GetHomomorphicHash(), GetHomomorphicHash(),
) )
return &v
} }
// SetPayloadHomomorphicHash sets homomorphic hash of the object payload. // SetPayloadHomomorphicHash sets homomorphic hash of the object payload.
func (o *Object) SetPayloadHomomorphicHash(v *checksum.Checksum) { func (o *Object) SetPayloadHomomorphicHash(v *checksum.Checksum) {
var v2 refs.Checksum
v.WriteToV2(&v2)
o.setHeaderField(func(h *object.Header) { o.setHeaderField(func(h *object.Header) {
h.SetHomomorphicHash(v.ToV2()) h.SetHomomorphicHash(&v2)
}) })
} }

View file

@ -122,23 +122,23 @@ func TestObject_SetCreationEpoch(t *testing.T) {
func TestObject_SetPayloadChecksum(t *testing.T) { func TestObject_SetPayloadChecksum(t *testing.T) {
obj := New() obj := New()
cs := checksum.New() var cs checksum.Checksum
cs.SetSHA256(randSHA256Checksum(t)) cs.SetSHA256(randSHA256Checksum(t))
obj.SetPayloadChecksum(cs) obj.SetPayloadChecksum(&cs)
require.Equal(t, cs, obj.PayloadChecksum()) require.Equal(t, &cs, obj.PayloadChecksum())
} }
func TestObject_SetPayloadHomomorphicHash(t *testing.T) { func TestObject_SetPayloadHomomorphicHash(t *testing.T) {
obj := New() obj := New()
cs := checksum.New() var cs checksum.Checksum
cs.SetTillichZemor(randTZChecksum(t)) cs.SetTillichZemor(randTZChecksum(t))
obj.SetPayloadHomomorphicHash(cs) obj.SetPayloadHomomorphicHash(&cs)
require.Equal(t, cs, obj.PayloadHomomorphicHash()) require.Equal(t, &cs, obj.PayloadHomomorphicHash())
} }
func TestObject_SetAttributes(t *testing.T) { func TestObject_SetAttributes(t *testing.T) {

View file

@ -43,14 +43,23 @@ func (sg *StorageGroup) SetValidationDataSize(epoch uint64) {
// ValidationDataHash returns homomorphic hash from the // ValidationDataHash returns homomorphic hash from the
// concatenation of the payloads of the storage group members. // concatenation of the payloads of the storage group members.
func (sg *StorageGroup) ValidationDataHash() *checksum.Checksum { func (sg *StorageGroup) ValidationDataHash() *checksum.Checksum {
return checksum.NewFromV2( if v2 := (*storagegroup.StorageGroup)(sg).GetValidationHash(); v2 != nil {
(*storagegroup.StorageGroup)(sg).GetValidationHash()) var v checksum.Checksum
v.ReadFromV2(*v2)
return &v
}
return nil
} }
// SetValidationDataHash sets homomorphic hash from the // SetValidationDataHash sets homomorphic hash from the
// concatenation of the payloads of the storage group members. // concatenation of the payloads of the storage group members.
func (sg *StorageGroup) SetValidationDataHash(hash *checksum.Checksum) { func (sg *StorageGroup) SetValidationDataHash(hash *checksum.Checksum) {
(*storagegroup.StorageGroup)(sg).SetValidationHash(hash.ToV2()) var v2 refs.Checksum
hash.WriteToV2(&v2)
(*storagegroup.StorageGroup)(sg).SetValidationHash(&v2)
} }
// ExpirationEpoch returns last NeoFS epoch number // ExpirationEpoch returns last NeoFS epoch number