diff --git a/accounting/decimal.go b/accounting/decimal.go index a4e560df..639d521b 100644 --- a/accounting/decimal.go +++ b/accounting/decimal.go @@ -3,7 +3,6 @@ package accounting import ( neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" protoaccounting "github.com/nspcc-dev/neofs-sdk-go/proto/accounting" - "google.golang.org/protobuf/proto" ) // Decimal represents decimal number for accounting operations. @@ -75,7 +74,7 @@ func (d *Decimal) SetPrecision(p uint32) { // // See also Unmarshal. func (d Decimal) Marshal() []byte { - return neofsproto.MarshalMessage(d.ProtoMessage()) + return neofsproto.Marshal(d) } // Unmarshal decodes NeoFS API protocol binary format into the Decimal @@ -84,12 +83,5 @@ func (d Decimal) Marshal() []byte { // // See also Marshal. func (d *Decimal) Unmarshal(data []byte) error { - var m protoaccounting.Decimal - - err := proto.Unmarshal(data, &m) - if err != nil { - return err - } - - return d.FromProtoMessage(&m) + return neofsproto.Unmarshal(data, d) } diff --git a/checksum/checksum.go b/checksum/checksum.go index f400acab..d6876b57 100644 --- a/checksum/checksum.go +++ b/checksum/checksum.go @@ -24,7 +24,7 @@ type Checksum struct { // Type represents the enumeration // of checksum types. -type Type uint32 +type Type int32 const ( Unknown Type = iota // Deprecated: use 0 instead. @@ -99,6 +99,9 @@ func NewFromData(typ Type, data []byte) (Checksum, error) { // // See also [Checksum.ProtoMessage]. func (c *Checksum) FromProtoMessage(m *refs.Checksum) error { + if m.Type < 0 { + return fmt.Errorf("negative type %d", m.Type) + } if len(m.Sum) == 0 { return errors.New("missing value") } diff --git a/checksum/checksum_test.go b/checksum/checksum_test.go index 999e10ae..4d7d78d4 100644 --- a/checksum/checksum_test.go +++ b/checksum/checksum_test.go @@ -63,7 +63,7 @@ func TestChecksumDecodingFailures(t *testing.T) { } func TestNew(t *testing.T) { - typ := checksum.Type(rand.Uint32()) + typ := checksum.Type(rand.Int31()) val := make([]byte, 128) //nolint:staticcheck rand.Read(val) @@ -71,7 +71,7 @@ func TestNew(t *testing.T) { require.Equal(t, typ, cs.Type()) require.Equal(t, val, cs.Value()) - otherTyp := checksum.Type(rand.Uint32()) + otherTyp := checksum.Type(rand.Int31()) otherVal := make([]byte, 128) //nolint:staticcheck rand.Read(otherVal) @@ -155,7 +155,7 @@ func TestNewFromHash(t *testing.T) { h.Write([]byte("Hello, world!")) hb := []byte{32, 94, 4, 138} - typ := checksum.Type(rand.Uint32()) + typ := checksum.Type(rand.Int31()) cs := checksum.NewFromHash(typ, h) require.Equal(t, typ, cs.Type()) require.Equal(t, hb, cs.Value()) diff --git a/container/container.go b/container/container.go index 084b128e..e61977e4 100644 --- a/container/container.go +++ b/container/container.go @@ -1,25 +1,32 @@ package container import ( - "bytes" "errors" "fmt" + "slices" "strconv" "strings" "time" "github.com/google/uuid" - "github.com/nspcc-dev/neofs-api-go/v2/container" - v2netmap "github.com/nspcc-dev/neofs-api-go/v2/netmap" - "github.com/nspcc-dev/neofs-api-go/v2/refs" "github.com/nspcc-dev/neofs-sdk-go/container/acl" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" "github.com/nspcc-dev/neofs-sdk-go/netmap" + protocontainer "github.com/nspcc-dev/neofs-sdk-go/proto/container" "github.com/nspcc-dev/neofs-sdk-go/user" "github.com/nspcc-dev/neofs-sdk-go/version" ) +// various attributes. +const ( + sysAttrPrefix = "__NEOFS__" + sysAttrDisableHomohash = sysAttrPrefix + "DISABLE_HOMOMORPHIC_HASHING" + sysAttrDomainName = sysAttrPrefix + "NAME" + sysAttrDomainZone = sysAttrPrefix + "ZONE" +) + // Container represents descriptor of the NeoFS container. Container logically // stores NeoFS objects. Container is one of the basic and at the same time // necessary data storage units in the NeoFS. Container includes data about the @@ -35,10 +42,15 @@ import ( // Instances for existing containers can be initialized using decoding methods // (e.g Unmarshal). // -// Container is mutually compatible with github.com/nspcc-dev/neofs-api-go/v2/container.Container -// message. See ReadFromV2 / WriteToV2 methods. +// Container is mutually compatible with [protocontainer.Container] +// message. See [Container.FromProtoMessage] / [Container.ProtoMessage] methods. type Container struct { - v2 container.Container + version *version.Version + owner user.ID + nonce uuid.UUID + basicACL acl.Basic + attrs [][2]string + policy *netmap.PlacementPolicy } const ( @@ -50,97 +62,93 @@ const ( func (x Container) CopyTo(dst *Container) { dst.SetBasicACL(x.BasicACL()) - if owner := x.v2.GetOwnerID(); owner != nil { - var newOwner refs.OwnerID - newOwner.SetValue(bytes.Clone(owner.GetValue())) + dst.owner = x.owner - dst.v2.SetOwnerID(&newOwner) + if x.version != nil { + dst.version = new(version.Version) + *dst.version = *x.version } else { - dst.v2.SetOwnerID(nil) - } - - if x.v2.GetVersion() != nil { - ver := x.v2.GetVersion() - newVer := *ver - dst.v2.SetVersion(&newVer) - } else { - dst.v2.SetVersion(nil) + dst.version = nil } // do we need to set the different nonce? - dst.v2.SetNonce(bytes.Clone(x.v2.GetNonce())) - - if len(x.v2.GetAttributes()) > 0 { - dst.v2.SetAttributes([]container.Attribute{}) + dst.nonce = x.nonce - attributeIterator := func(key, val string) { - dst.SetAttribute(key, val) - } - - x.IterateAttributes(attributeIterator) + if len(x.attrs) > 0 { + dst.attrs = slices.Clone(x.attrs) } - if x.v2.GetPlacementPolicy() != nil { - var ppCopy netmap.PlacementPolicy - x.PlacementPolicy().CopyTo(&ppCopy) - dst.SetPlacementPolicy(ppCopy) + if x.policy != nil { + dst.policy = new(netmap.PlacementPolicy) + x.policy.CopyTo(dst.policy) } else { - x.v2.SetPlacementPolicy(nil) + dst.policy = nil } } // reads Container from the container.Container message. If checkFieldPresence is set, // returns an error on absence of any protocol-required field. -func (x *Container) readFromV2(m container.Container, checkFieldPresence bool) error { +func (x *Container) fromProtoMessage(m *protocontainer.Container, checkFieldPresence bool) error { var err error - ownerV2 := m.GetOwnerID() - if ownerV2 != nil { - var owner user.ID - - err = owner.ReadFromV2(*ownerV2) + if m.OwnerId != nil { + err = x.owner.FromProtoMessage(m.OwnerId) if err != nil { return fmt.Errorf("invalid owner: %w", err) } } else if checkFieldPresence { return errors.New("missing owner") + } else { + x.owner = user.ID{} } - binNonce := m.GetNonce() - if len(binNonce) > 0 { - var nonce uuid.UUID - - err = nonce.UnmarshalBinary(binNonce) + if len(m.Nonce) > 0 { + err = x.nonce.UnmarshalBinary(m.Nonce) if err != nil { return fmt.Errorf("invalid nonce: %w", err) - } else if ver := nonce.Version(); ver != 4 { + } else if ver := x.nonce.Version(); ver != 4 { return fmt.Errorf("invalid nonce UUID version %d", ver) } } else if checkFieldPresence { return errors.New("missing nonce") + } else { + x.nonce = uuid.Nil } - ver := m.GetVersion() - if checkFieldPresence && ver == nil { + if m.Version != nil { + if x.version == nil { + x.version = new(version.Version) + } + if err = x.version.FromProtoMessage(m.Version); err != nil { + return fmt.Errorf("invalid version: %w", err) + } + } else if checkFieldPresence { return errors.New("missing version") + } else { + x.version = nil } - policyV2 := m.GetPlacementPolicy() - if policyV2 != nil { - var policy netmap.PlacementPolicy - - err = policy.ReadFromV2(*policyV2) + if m.PlacementPolicy != nil { + if x.policy == nil { + x.policy = new(netmap.PlacementPolicy) + } + err = x.policy.FromProtoMessage(m.PlacementPolicy) if err != nil { return fmt.Errorf("invalid placement policy: %w", err) } } else if checkFieldPresence { return errors.New("missing placement policy") + } else { + x.policy = nil } attrs := m.GetAttributes() mAttr := make(map[string]struct{}, len(attrs)) var key, val string var was bool + if attrs != nil { + x.attrs = make([][2]string, len(attrs)) + } for i := range attrs { key = attrs[i].GetKey() @@ -168,27 +176,49 @@ func (x *Container) readFromV2(m container.Container, checkFieldPresence bool) e } mAttr[key] = struct{}{} + x.attrs[i] = [2]string{key, val} } - x.v2 = m + x.basicACL.FromBits(m.BasicAcl) return nil } -// ReadFromV2 reads Container from the container.Container message. Checks if the -// message conforms to NeoFS API V2 protocol. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// x from it. // -// See also WriteToV2. -func (x *Container) ReadFromV2(m container.Container) error { - return x.readFromV2(m, true) +// See also [Container.ProtoMessage]. +func (x *Container) FromProtoMessage(m *protocontainer.Container) error { + return x.fromProtoMessage(m, true) } -// WriteToV2 writes Container into the container.Container message. -// The message MUST NOT be nil. +// ProtoMessage converts x into message to transmit using the NeoFS API +// protocol. // -// See also ReadFromV2. -func (x Container) WriteToV2(m *container.Container) { - *m = x.v2 +// See also [Container.FromProtoMessage]. +func (x Container) ProtoMessage() *protocontainer.Container { + m := &protocontainer.Container{ + BasicAcl: x.basicACL.Bits(), + } + if x.version != nil { + m.Version = x.version.ProtoMessage() + } + if !x.owner.IsZero() { + m.OwnerId = x.owner.ProtoMessage() + } + if x.nonce != uuid.Nil { + m.Nonce = x.nonce[:] + } + if x.policy != nil { + m.PlacementPolicy = x.policy.ProtoMessage() + } + if len(x.attrs) > 0 { + m.Attributes = make([]*protocontainer.Container_Attribute, len(x.attrs)) + for i := range x.attrs { + m.Attributes[i] = &protocontainer.Container_Attribute{Key: x.attrs[i][0], Value: x.attrs[i][1]} + } + } + return m } // Marshal encodes Container into a binary format of the NeoFS API protocol @@ -196,7 +226,7 @@ func (x Container) WriteToV2(m *container.Container) { // // See also Unmarshal. func (x Container) Marshal() []byte { - return x.v2.StableMarshal(nil) + return neofsproto.Marshal(x) } // SignedData returns actual payload to sign. @@ -212,14 +242,7 @@ func (x Container) SignedData() []byte { // // See also Marshal. func (x *Container) Unmarshal(data []byte) error { - var m container.Container - - err := m.Unmarshal(data) - if err != nil { - return err - } - - return x.readFromV2(m, false) + return neofsproto.UnmarshalOptional(data, x, (*Container).fromProtoMessage) } // MarshalJSON encodes Container into a JSON format of the NeoFS API protocol @@ -227,7 +250,7 @@ func (x *Container) Unmarshal(data []byte) error { // // See also UnmarshalJSON. func (x Container) MarshalJSON() ([]byte, error) { - return x.v2.MarshalJSON() + return neofsproto.MarshalMessageJSON(x.ProtoMessage()) } // UnmarshalJSON decodes NeoFS API protocol JSON format into the Container @@ -235,11 +258,7 @@ func (x Container) MarshalJSON() ([]byte, error) { // // See also MarshalJSON. func (x *Container) UnmarshalJSON(data []byte) error { - var m container.Container - if err := m.UnmarshalJSON(data); err != nil { - return err - } - return x.readFromV2(m, false) + return neofsproto.UnmarshalJSONOptional(data, x, (*Container).fromProtoMessage) } // Init initializes all internal data of the Container required by NeoFS API @@ -247,17 +266,13 @@ func (x *Container) UnmarshalJSON(data []byte) error { // be called multiple times. Init SHOULD NOT be called if the Container instance // is used for decoding only. func (x *Container) Init() { - var ver refs.Version - version.Current().WriteToV2(&ver) - - x.v2.SetVersion(&ver) - - nonce, err := uuid.New().MarshalBinary() - if err != nil { - panic(fmt.Sprintf("unexpected error from UUID.MarshalBinary: %v", err)) + ver := version.Current() + x.version = &ver + for { + if x.nonce = uuid.New(); x.nonce != uuid.Nil { + break + } } - - x.v2.SetNonce(nonce) } // SetOwner specifies the owner of the Container. Each Container has exactly @@ -266,26 +281,15 @@ func (x *Container) Init() { // // See also Owner. func (x *Container) SetOwner(owner user.ID) { - var m refs.OwnerID - owner.WriteToV2(&m) - - x.v2.SetOwnerID(&m) + x.owner = owner } // Owner returns owner of the Container set using SetOwner. // // Zero Container has no owner which is incorrect according to NeoFS API // protocol. -func (x Container) Owner() (res user.ID) { - m := x.v2.GetOwnerID() - if m != nil { - err := res.ReadFromV2(*m) - if err != nil { - panic(fmt.Sprintf("unexpected error from user.ID.ReadFromV2: %v", err)) - } - } - - return +func (x Container) Owner() user.ID { + return x.owner } // SetBasicACL specifies basic part of the Container ACL. Basic ACL is used @@ -293,16 +297,15 @@ func (x Container) Owner() (res user.ID) { // // See also BasicACL. func (x *Container) SetBasicACL(basicACL acl.Basic) { - x.v2.SetBasicACL(basicACL.Bits()) + x.basicACL = basicACL } // BasicACL returns basic ACL set using SetBasicACL. // // Zero Container has zero basic ACL which structurally correct but doesn't // make sense since it denies any access to any party. -func (x Container) BasicACL() (res acl.Basic) { - res.FromBits(x.v2.GetBasicACL()) - return +func (x Container) BasicACL() acl.Basic { + return x.basicACL } // SetPlacementPolicy sets placement policy for the objects within the Container. @@ -310,26 +313,18 @@ func (x Container) BasicACL() (res acl.Basic) { // // See also PlacementPolicy. func (x *Container) SetPlacementPolicy(policy netmap.PlacementPolicy) { - var m v2netmap.PlacementPolicy - policy.WriteToV2(&m) - - x.v2.SetPlacementPolicy(&m) + x.policy = &policy } // PlacementPolicy returns placement policy set using SetPlacementPolicy. // // Zero Container has no placement policy which is incorrect according to // NeoFS API protocol. -func (x Container) PlacementPolicy() (res netmap.PlacementPolicy) { - m := x.v2.GetPlacementPolicy() - if m != nil { - err := res.ReadFromV2(*m) - if err != nil { - panic(fmt.Sprintf("unexpected error from PlacementPolicy.ReadFromV2: %v", err)) - } +func (x Container) PlacementPolicy() netmap.PlacementPolicy { + if x.policy != nil { + return *x.policy } - - return + return netmap.PlacementPolicy{} } // SetAttribute sets Container attribute value by key. Both key and value @@ -350,21 +345,14 @@ func (x *Container) SetAttribute(key, value string) { panic("empty attribute value") } - attrs := x.v2.GetAttributes() - ln := len(attrs) - - for i := range ln { - if attrs[i].GetKey() == key { - attrs[i].SetValue(value) + for i := range x.attrs { + if x.attrs[i][0] == key { + x.attrs[i][1] = value return } } - attrs = append(attrs, container.Attribute{}) - attrs[ln].SetKey(key) - attrs[ln].SetValue(value) - - x.v2.SetAttributes(attrs) + x.attrs = append(x.attrs, [2]string{key, value}) } // Attribute reads value of the Container attribute by key. Empty result means @@ -372,10 +360,9 @@ func (x *Container) SetAttribute(key, value string) { // // See also SetAttribute, IterateAttributes. func (x Container) Attribute(key string) string { - attrs := x.v2.GetAttributes() - for i := range attrs { - if attrs[i].GetKey() == key { - return attrs[i].GetValue() + for i := range x.attrs { + if x.attrs[i][0] == key { + return x.attrs[i][1] } } @@ -387,9 +374,8 @@ func (x Container) Attribute(key string) string { // // See also [Container.SetAttribute], [Container.Attribute], [Container.IterateUserAttributes]. func (x Container) IterateAttributes(f func(key, val string)) { - attrs := x.v2.GetAttributes() - for i := range attrs { - f(attrs[i].GetKey(), attrs[i].GetValue()) + for i := range x.attrs { + f(x.attrs[i][0], x.attrs[i][1]) } } @@ -399,7 +385,7 @@ func (x Container) IterateAttributes(f func(key, val string)) { // See also [Container.SetAttribute], [Container.Attribute], [Container.IterateAttributes]. func (x Container) IterateUserAttributes(f func(key, val string)) { x.IterateAttributes(func(key, val string) { - if !strings.HasPrefix(key, container.SysAttributePrefix) { + if !strings.HasPrefix(key, sysAttrPrefix) { f(key, val) } }) @@ -438,7 +424,7 @@ func (x Container) CreatedAt() time.Time { sec, err = strconv.ParseInt(x.Attribute(attributeTimestamp), 10, 64) if err != nil { - panic(fmt.Sprintf("parse container timestamp: %v", err)) + panic(fmt.Sprintf("parse protocontainer timestamp: %v", err)) } } @@ -452,14 +438,14 @@ const attributeHomoHashEnabled = "true" // // See also IsHomomorphicHashingDisabled. func (x *Container) DisableHomomorphicHashing() { - x.SetAttribute(container.SysAttributeHomomorphicHashing, attributeHomoHashEnabled) + x.SetAttribute(sysAttrDisableHomohash, attributeHomoHashEnabled) } // IsHomomorphicHashingDisabled checks if DisableHomomorphicHashing was called. // // Zero Container has enabled hashing. func (x Container) IsHomomorphicHashingDisabled() bool { - return x.Attribute(container.SysAttributeHomomorphicHashing) == attributeHomoHashEnabled + return x.Attribute(sysAttrDisableHomohash) == attributeHomoHashEnabled } // Domain represents information about container domain registered in the NNS @@ -493,22 +479,22 @@ func (x Domain) Zone() string { return x.zone } - return "container" + return "protocontainer" } // WriteDomain writes Domain into the Container. Name MUST NOT be empty. func (x *Container) WriteDomain(domain Domain) { - x.SetAttribute(container.SysAttributeName, domain.Name()) - x.SetAttribute(container.SysAttributeZone, domain.Zone()) + x.SetAttribute(sysAttrDomainName, domain.Name()) + x.SetAttribute(sysAttrDomainZone, domain.Zone()) } // ReadDomain reads Domain from the Container. Returns value with empty name // if domain is not specified. func (x Container) ReadDomain() (res Domain) { - name := x.Attribute(container.SysAttributeName) + name := x.Attribute(sysAttrDomainName) if name != "" { res.SetName(name) - res.SetZone(x.Attribute(container.SysAttributeZone)) + res.SetZone(x.Attribute(sysAttrDomainZone)) } return @@ -560,9 +546,8 @@ func (x Container) AssertID(id cid.ID) bool { // Version returns the NeoFS API version this container was created with. func (x Container) Version() version.Version { - var v version.Version - if m := x.v2.GetVersion(); m != nil { - _ = v.ReadFromV2(*m) + if x.version != nil { + return *x.version } - return v + return version.Version{} } diff --git a/container/container_internal_test.go b/container/container_internal_test.go index 0404a7c1..b85db235 100644 --- a/container/container_internal_test.go +++ b/container/container_internal_test.go @@ -66,7 +66,7 @@ func TestContainer_CopyTo(t *testing.T) { require.True(t, container.Owner() == dst.Owner()) newOwner := usertest.ID() - dst.v2.GetOwnerID().SetValue(newOwner[:]) + copy(dst.owner[:], newOwner[:]) require.False(t, container.Owner() == dst.Owner()) }) @@ -87,48 +87,42 @@ func TestContainer_CopyTo(t *testing.T) { var dst Container container.CopyTo(&dst) - require.True(t, bytes.Equal(container.v2.GetNonce(), dst.v2.GetNonce())) - dst.v2.SetNonce([]byte{1, 2, 3}) - require.False(t, bytes.Equal(container.v2.GetNonce(), dst.v2.GetNonce())) + require.True(t, container.nonce == dst.nonce) + copy(dst.nonce[:], []byte{1, 2, 3}) + require.False(t, container.nonce == dst.nonce) }) t.Run("overwrite nonce", func(t *testing.T) { var local Container - require.Empty(t, local.v2.GetNonce()) + require.Zero(t, local.nonce) var dst Container - dst.v2.SetNonce([]byte{1, 2, 3}) - require.NotEmpty(t, dst.v2.GetNonce()) + copy(dst.nonce[:], []byte{1, 2, 3}) + require.NotZero(t, dst.nonce) local.CopyTo(&dst) require.True(t, bytes.Equal(local.Marshal(), dst.Marshal())) - require.Empty(t, local.v2.GetNonce()) - require.Empty(t, dst.v2.GetNonce()) + require.Zero(t, local.nonce) + require.Zero(t, dst.nonce) - require.True(t, bytes.Equal(local.v2.GetNonce(), dst.v2.GetNonce())) - dst.v2.SetNonce([]byte{1, 2, 3}) - require.False(t, bytes.Equal(local.v2.GetNonce(), dst.v2.GetNonce())) + require.True(t, local.nonce == dst.nonce) + copy(dst.nonce[:], []byte{1, 2, 3}) + require.False(t, local.nonce == dst.nonce) }) t.Run("change version", func(t *testing.T) { var dst Container container.CopyTo(&dst) - oldVer := container.v2.GetVersion() - require.NotNil(t, oldVer) + require.Equal(t, container.Version(), dst.Version()) - newVer := dst.v2.GetVersion() - require.NotNil(t, newVer) + cp := container.Version() + dst.version.SetMajor(rand.Uint32()) + dst.version.SetMinor(rand.Uint32()) - require.Equal(t, oldVer.GetMajor(), newVer.GetMajor()) - require.Equal(t, oldVer.GetMinor(), newVer.GetMinor()) - - newVer.SetMajor(rand.Uint32()) - newVer.SetMinor(rand.Uint32()) - - require.NotEqual(t, oldVer.GetMajor(), newVer.GetMajor()) - require.NotEqual(t, oldVer.GetMinor(), newVer.GetMinor()) + require.NotEqual(t, cp, dst.Version()) + require.Equal(t, cp, container.Version()) }) t.Run("change attributes", func(t *testing.T) { diff --git a/container/container_test.go b/container/container_test.go index fddda709..94e109e4 100644 --- a/container/container_test.go +++ b/container/container_test.go @@ -12,9 +12,6 @@ import ( "time" "github.com/google/uuid" - v2container "github.com/nspcc-dev/neofs-api-go/v2/container" - apinetmap "github.com/nspcc-dev/neofs-api-go/v2/netmap" - "github.com/nspcc-dev/neofs-api-go/v2/refs" "github.com/nspcc-dev/neofs-sdk-go/container" "github.com/nspcc-dev/neofs-sdk-go/container/acl" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" @@ -23,6 +20,9 @@ import ( neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" neofsecdsa "github.com/nspcc-dev/neofs-sdk-go/crypto/ecdsa" "github.com/nspcc-dev/neofs-sdk-go/netmap" + protocontainer "github.com/nspcc-dev/neofs-sdk-go/proto/container" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" "github.com/nspcc-dev/neofs-sdk-go/user" usertest "github.com/nspcc-dev/neofs-sdk-go/user/test" "github.com/nspcc-dev/neofs-sdk-go/version" @@ -107,10 +107,9 @@ func init() { validContainer.DisableHomomorphicHashing() // init sets random nonce, we need a fixed one. There is no setter for it - var m v2container.Container - validContainer.WriteToV2(&m) - m.SetNonce(anyValidNonce[:]) - if err := validContainer.ReadFromV2(m); err != nil { + m := validContainer.ProtoMessage() + m.Nonce = anyValidNonce[:] + if err := validContainer.FromProtoMessage(m); err != nil { panic(fmt.Errorf("unexpected encode-decode failure: %w", err)) } } @@ -301,8 +300,7 @@ var validJSONContainer = ` // type does not provide getter, this is a helper. func extractNonce(t testing.TB, c container.Container) uuid.UUID { - var m v2container.Container - c.WriteToV2(&m) + m := c.ProtoMessage() var nonce uuid.UUID if b := m.GetNonce(); len(b) > 0 { require.NoError(t, nonce.UnmarshalBinary(b)) @@ -310,103 +308,58 @@ func extractNonce(t testing.TB, c container.Container) uuid.UUID { return nonce } -func protoUserFromBytes(b []byte) *refs.OwnerID { - var m refs.OwnerID - m.SetValue(b) - return &m -} - -func setContainerAttributes(m *v2container.Container, els ...string) { +func setContainerAttributes(m *protocontainer.Container, els ...string) { if len(els)%2 != 0 { panic("must be even") } - mas := make([]v2container.Attribute, len(els)/2) + m.Attributes = make([]*protocontainer.Container_Attribute, len(els)/2) for i := range len(els) / 2 { - mas[i].SetKey(els[2*i]) - mas[i].SetValue(els[2*i+1]) + m.Attributes[i] = &protocontainer.Container_Attribute{Key: els[2*i], Value: els[2*i+1]} } - m.SetAttributes(mas) } -func TestContainer_ReadFromV2(t *testing.T) { - var mv refs.Version - mv.SetMajor(2526956385) - mv.SetMinor(95168785) - - var mas []v2container.Attribute - newAttr := func(k, v string) v2container.Attribute { - var a v2container.Attribute - a.SetKey(k) - a.SetValue(v) - return a - } - addAttr := func(k, v string) { mas = append(mas, newAttr(k, v)) } - addAttr("k1", "v1") - addAttr("k2", "v2") - addAttr("Name", anyValidName) - addAttr("Timestamp", "1727681164") - addAttr("__NEOFS__NAME", anyValidDomainName) - addAttr("__NEOFS__ZONE", anyValidDomainZone) - addAttr("__NEOFS__DISABLE_HOMOMORPHIC_HASHING", "true") - - mrs := make([]apinetmap.Replica, 2) - mrs[0].SetSelector("selector_0") - mrs[0].SetCount(2583748530) - mrs[1].SetSelector("selector_1") - mrs[1].SetCount(358755354) - - mss := make([]apinetmap.Selector, 2) - mss[0].SetName("selector_0") - mss[0].SetCount(1814781076) - mss[0].SetClause(apinetmap.Same) - mss[0].SetFilter("filter_0") - mss[0].SetAttribute("attribute_0") - mss[1].SetName("selector_1") - mss[1].SetCount(1814781076) - mss[1].SetClause(apinetmap.Distinct) - mss[1].SetFilter("filter_1") - mss[1].SetAttribute("attribute_1") - - msubs := make([]apinetmap.Filter, 0, 2) - addSub := func(name, key string, op apinetmap.Operation, val string) { - var f apinetmap.Filter - f.SetName(name) - f.SetKey(key) - f.SetOp(op) - f.SetValue(val) - msubs = append(msubs, f) +func TestContainer_FromProtoMessage(t *testing.T) { + m := &protocontainer.Container{ + Version: &refs.Version{Major: 2526956385, Minor: 95168785}, + OwnerId: &refs.OwnerID{Value: anyValidOwner[:]}, + Nonce: anyValidNonce[:], + BasicAcl: anyValidBasicACL.Bits(), + Attributes: []*protocontainer.Container_Attribute{ + {Key: "k1", Value: "v1"}, + {Key: "k2", Value: "v2"}, + {Key: "Name", Value: anyValidName}, + {Key: "Timestamp", Value: "1727681164"}, + {Key: "__NEOFS__NAME", Value: anyValidDomainName}, + {Key: "__NEOFS__ZONE", Value: anyValidDomainZone}, + {Key: "__NEOFS__DISABLE_HOMOMORPHIC_HASHING", Value: "true"}, + }, + PlacementPolicy: &protonetmap.PlacementPolicy{ + Replicas: []*protonetmap.Replica{ + {Count: 2583748530, Selector: "selector_0"}, + {Count: 358755354, Selector: "selector_1"}, + }, + ContainerBackupFactor: anyValidBackupFactor, + Selectors: []*protonetmap.Selector{ + {Name: "selector_0", Count: 1814781076, Clause: protonetmap.Clause_SAME, Attribute: "attribute_0", Filter: "filter_0"}, + {Name: "selector_1", Count: 1814781076, Clause: protonetmap.Clause_DISTINCT, Attribute: "attribute_1", Filter: "filter_1"}, + }, + Filters: []*protonetmap.Filter{ + {Name: "filter_0", Op: protonetmap.Operation_AND, Filters: []*protonetmap.Filter{ + {Name: "filter_0_0", Key: "key_0_0", Op: protonetmap.Operation_EQ, Value: "val_0_0"}, + {Name: "filter_0_1", Key: "key_0_1", Op: protonetmap.Operation_NE, Value: "val_0_1"}, + }}, + {Name: "filter_1", Op: protonetmap.Operation_OR, Filters: []*protonetmap.Filter{ + {Name: "filter_1_0", Key: "key_1_0", Op: protonetmap.Operation_GT, Value: "1889407708985023116"}, + {Name: "filter_1_1", Key: "key_1_1", Op: protonetmap.Operation_GE, Value: "1429243097315344888"}, + {Name: "filter_1_2", Key: "key_1_2", Op: protonetmap.Operation_LT, Value: "3722656060317482335"}, + {Name: "filter_1_3", Key: "key_1_3", Op: protonetmap.Operation_LE, Value: "1950504987705284805"}, + }}, + }, + }, } - addSub("filter_0_0", "key_0_0", apinetmap.EQ, "val_0_0") - addSub("filter_0_1", "key_0_1", apinetmap.NE, "val_0_1") - mfs := make([]apinetmap.Filter, 2) - mfs[0].SetName("filter_0") - mfs[0].SetOp(apinetmap.AND) - mfs[0].SetFilters(msubs) - msubs = make([]apinetmap.Filter, 0, 4) - addSub("filter_1_0", "key_1_0", apinetmap.GT, "1889407708985023116") - addSub("filter_1_1", "key_1_1", apinetmap.GE, "1429243097315344888") - addSub("filter_1_2", "key_1_2", apinetmap.LT, "3722656060317482335") - addSub("filter_1_3", "key_1_3", apinetmap.LE, "1950504987705284805") - mfs[1].SetName("filter_1") - mfs[1].SetOp(apinetmap.OR) - mfs[1].SetFilters(msubs) - - var mp apinetmap.PlacementPolicy - mp.SetContainerBackupFactor(anyValidBackupFactor) - mp.SetReplicas(mrs) - mp.SetSelectors(mss) - mp.SetFilters(mfs) - - var m v2container.Container - m.SetVersion(&mv) - m.SetOwnerID(protoUserFromBytes(anyValidOwner[:])) - m.SetNonce(anyValidNonce[:]) - m.SetBasicACL(anyValidBasicACL.Bits()) - m.SetPlacementPolicy(&mp) - m.SetAttributes(mas) var val container.Container - require.NoError(t, val.ReadFromV2(m)) + require.NoError(t, val.FromProtoMessage(m)) ver := val.Version() require.EqualValues(t, 2526956385, ver.Major()) require.EqualValues(t, 95168785, ver.Minor()) @@ -484,108 +437,95 @@ func TestContainer_ReadFromV2(t *testing.T) { require.Empty(t, subs[3].SubFilters()) // reset optional fields - m.SetBasicACL(0) - m.SetAttributes([]v2container.Attribute{newAttr("__NEOFS__DISABLE_HOMOMORPHIC_HASHING", "anything not true")}) + m.BasicAcl = 0 + m.Attributes = []*protocontainer.Container_Attribute{{Key: "__NEOFS__DISABLE_HOMOMORPHIC_HASHING", Value: "anything not true"}} val2 := val - require.NoError(t, val2.ReadFromV2(m)) + require.NoError(t, val2.FromProtoMessage(m)) require.Zero(t, val2.BasicACL()) require.False(t, val2.IsHomomorphicHashingDisabled()) t.Run("invalid", func(t *testing.T) { for _, tc := range []struct { name, err string - corrupt func(*v2container.Container) + corrupt func(*protocontainer.Container) }{ {name: "version/missing", err: "missing version", - corrupt: func(m *v2container.Container) { m.SetVersion(nil) }}, + corrupt: func(m *protocontainer.Container) { m.Version = nil }}, {name: "owner/missing", err: "missing owner", - corrupt: func(m *v2container.Container) { m.SetOwnerID(nil) }}, + corrupt: func(m *protocontainer.Container) { m.OwnerId = nil }}, {name: "owner/value/nil", err: "invalid owner: invalid length 0, expected 25", - corrupt: func(m *v2container.Container) { m.SetOwnerID(protoUserFromBytes(nil)) }}, + corrupt: func(m *protocontainer.Container) { m.OwnerId.Value = nil }}, {name: "owner/value/empty", err: "invalid owner: invalid length 0, expected 25", - corrupt: func(m *v2container.Container) { m.SetOwnerID(protoUserFromBytes([]byte{})) }}, + corrupt: func(m *protocontainer.Container) { m.OwnerId.Value = []byte{} }}, {name: "owner/value/undersize", err: "invalid owner: invalid length 24, expected 25", - corrupt: func(m *v2container.Container) { m.SetOwnerID(protoUserFromBytes(make([]byte, 24))) }}, + corrupt: func(m *protocontainer.Container) { m.OwnerId.Value = make([]byte, 24) }}, {name: "owner/value/oversize", err: "invalid owner: invalid length 26, expected 25", - corrupt: func(m *v2container.Container) { m.SetOwnerID(protoUserFromBytes(make([]byte, 26))) }}, + corrupt: func(m *protocontainer.Container) { m.OwnerId.Value = make([]byte, 26) }}, {name: "owner/value/wrong prefix", err: "invalid owner: invalid prefix byte 0x42, expected 0x35", - corrupt: func(m *v2container.Container) { - b := bytes.Clone(anyValidOwner[:]) - b[0] = 0x42 - m.SetOwnerID(protoUserFromBytes(b)) + corrupt: func(m *protocontainer.Container) { + m.OwnerId.Value = bytes.Clone(anyValidOwner[:]) + m.OwnerId.Value[0] = 0x42 }}, {name: "owner/value/checksum mismatch", err: "invalid owner: checksum mismatch", - corrupt: func(m *v2container.Container) { - b := bytes.Clone(anyValidOwner[:]) - b[len(b)-1]++ - m.SetOwnerID(protoUserFromBytes(b)) + corrupt: func(m *protocontainer.Container) { + m.OwnerId.Value = bytes.Clone(anyValidOwner[:]) + m.OwnerId.Value[24]++ }}, {name: "nonce/nil", err: "missing nonce", - corrupt: func(m *v2container.Container) { m.SetNonce(nil) }}, + corrupt: func(m *protocontainer.Container) { m.Nonce = nil }}, {name: "nonce/empty", err: "missing nonce", - corrupt: func(m *v2container.Container) { m.SetNonce([]byte{}) }}, + corrupt: func(m *protocontainer.Container) { m.Nonce = []byte{} }}, {name: "nonce/undersize", err: "invalid nonce: invalid UUID (got 15 bytes)", - corrupt: func(m *v2container.Container) { m.SetNonce(anyValidNonce[:15]) }}, + corrupt: func(m *protocontainer.Container) { m.Nonce = anyValidNonce[:15] }}, {name: "nonce/oversize", err: "invalid nonce: invalid UUID (got 17 bytes)", - corrupt: func(m *v2container.Container) { m.SetNonce(append(anyValidNonce[:], 1)) }}, + corrupt: func(m *protocontainer.Container) { m.Nonce = append(anyValidNonce[:], 1) }}, {name: "nonce/wrong version", err: "invalid nonce UUID version 3", - corrupt: func(m *v2container.Container) { - b := bytes.Clone(anyValidNonce[:]) - b[6] = 3 << 4 - m.SetNonce(b) + corrupt: func(m *protocontainer.Container) { + m.Nonce = bytes.Clone(anyValidNonce[:]) + m.Nonce[6] = 3 << 4 }}, {name: "policy/replicas/nil", err: "invalid placement policy: missing replicas", - corrupt: func(m *v2container.Container) { - var mp apinetmap.PlacementPolicy - mp.SetReplicas(nil) - m.SetPlacementPolicy(&mp) - }}, + corrupt: func(m *protocontainer.Container) { m.PlacementPolicy.Replicas = nil }}, {name: "policy/replicas/empty", err: "invalid placement policy: missing replicas", - corrupt: func(m *v2container.Container) { - var mp apinetmap.PlacementPolicy - mp.SetReplicas([]apinetmap.Replica{}) - m.SetPlacementPolicy(&mp) - }}, + corrupt: func(m *protocontainer.Container) { m.PlacementPolicy.Replicas = []*protonetmap.Replica{} }}, {name: "attributes/no key", err: "empty attribute key", - corrupt: func(m *v2container.Container) { setContainerAttributes(m, "k1", "v1", "", "v2") }}, + corrupt: func(m *protocontainer.Container) { setContainerAttributes(m, "k1", "v1", "", "v2") }}, {name: "attributes/no value", err: "empty attribute value k2", - corrupt: func(m *v2container.Container) { setContainerAttributes(m, "k1", "v1", "k2", "") }}, + corrupt: func(m *protocontainer.Container) { setContainerAttributes(m, "k1", "v1", "k2", "") }}, {name: "attributes/duplicated", err: "duplicated attribute k1", - corrupt: func(m *v2container.Container) { setContainerAttributes(m, "k1", "v1", "k2", "v2", "k1", "v3") }}, + corrupt: func(m *protocontainer.Container) { setContainerAttributes(m, "k1", "v1", "k2", "v2", "k1", "v3") }}, {name: "attributes/timestamp", err: "invalid attribute value Timestamp: foo (strconv.ParseInt: parsing \"foo\": invalid syntax)", - corrupt: func(m *v2container.Container) { setContainerAttributes(m, "Timestamp", "foo") }}, + corrupt: func(m *protocontainer.Container) { setContainerAttributes(m, "Timestamp", "foo") }}, } { t.Run(tc.name, func(t *testing.T) { val2 := val - var m v2container.Container - val2.WriteToV2(&m) - tc.corrupt(&m) - require.EqualError(t, new(container.Container).ReadFromV2(m), tc.err) + m := val2.ProtoMessage() + tc.corrupt(m) + require.EqualError(t, new(container.Container).FromProtoMessage(m), tc.err) }) } }) } -func TestContainer_WriteToV2(t *testing.T) { +func TestContainer_ProtoMessage(t *testing.T) { var val container.Container - var m v2container.Container // zero - val.WriteToV2(&m) + m := val.ProtoMessage() require.Zero(t, m.GetVersion()) - require.Zero(t, m.GetOwnerID()) + require.Zero(t, m.GetOwnerId()) require.Zero(t, m.GetNonce()) - require.Zero(t, m.GetBasicACL()) + require.Zero(t, m.GetBasicAcl()) require.Zero(t, m.GetPlacementPolicy()) require.Zero(t, m.GetAttributes()) // filled - validContainer.WriteToV2(&m) + m = validContainer.ProtoMessage() require.EqualValues(t, 2, m.GetVersion().GetMajor()) require.EqualValues(t, 16, m.GetVersion().GetMinor()) require.Len(t, m.GetNonce(), 16) require.EqualValues(t, 4, m.GetNonce()[6]>>4) - require.EqualValues(t, 1043832770, m.GetBasicACL()) + require.EqualValues(t, 1043832770, m.GetBasicAcl()) mas := m.GetAttributes() require.Len(t, mas, 7) for i, pair := range [][2]string{ @@ -615,12 +555,12 @@ func TestContainer_WriteToV2(t *testing.T) { require.Len(t, mss, 2) require.Equal(t, "selector_0", mss[0].GetName()) require.EqualValues(t, 1814781076, mss[0].GetCount()) - require.Equal(t, apinetmap.Same, mss[0].GetClause()) + require.Equal(t, protonetmap.Clause_SAME, mss[0].GetClause()) require.Equal(t, "filter_0", mss[0].GetFilter()) require.Equal(t, "attribute_0", mss[0].GetAttribute()) require.Equal(t, "selector_1", mss[1].GetName()) require.EqualValues(t, 1505136737, mss[1].GetCount()) - require.Equal(t, apinetmap.Distinct, mss[1].GetClause()) + require.Equal(t, protonetmap.Clause_DISTINCT, mss[1].GetClause()) require.Equal(t, "filter_1", mss[1].GetFilter()) require.Equal(t, "attribute_1", mss[1].GetAttribute()) @@ -629,51 +569,51 @@ func TestContainer_WriteToV2(t *testing.T) { // filter#0 require.Equal(t, "filter_0", mfs[0].GetName()) require.Zero(t, mfs[0].GetKey()) - require.Equal(t, apinetmap.AND, mfs[0].GetOp()) + require.Equal(t, protonetmap.Operation_AND, mfs[0].GetOp()) require.Zero(t, mfs[0].GetValue()) msubs := mfs[0].GetFilters() require.Len(t, msubs, 2) // sub#0 require.Equal(t, "filter_0_0", msubs[0].GetName()) require.Equal(t, "key_0_0", msubs[0].GetKey()) - require.Equal(t, apinetmap.EQ, msubs[0].GetOp()) + require.Equal(t, protonetmap.Operation_EQ, msubs[0].GetOp()) require.Equal(t, "val_0_0", msubs[0].GetValue()) require.Zero(t, msubs[0].GetFilters()) // sub#1 require.Equal(t, "filter_0_1", msubs[1].GetName()) require.Equal(t, "key_0_1", msubs[1].GetKey()) - require.Equal(t, apinetmap.NE, msubs[1].GetOp()) + require.Equal(t, protonetmap.Operation_NE, msubs[1].GetOp()) require.Equal(t, "val_0_1", msubs[1].GetValue()) require.Zero(t, msubs[1].GetFilters()) // filter#1 require.Equal(t, "filter_1", mfs[1].GetName()) require.Zero(t, mfs[1].GetKey()) - require.Equal(t, apinetmap.OR, mfs[1].GetOp()) + require.Equal(t, protonetmap.Operation_OR, mfs[1].GetOp()) require.Zero(t, mfs[1].GetValue()) msubs = mfs[1].GetFilters() require.Len(t, msubs, 4) // sub#0 require.Equal(t, "filter_1_0", msubs[0].GetName()) require.Equal(t, "key_1_0", msubs[0].GetKey()) - require.Equal(t, apinetmap.GT, msubs[0].GetOp()) + require.Equal(t, protonetmap.Operation_GT, msubs[0].GetOp()) require.Equal(t, "1889407708985023116", msubs[0].GetValue()) require.Zero(t, msubs[0].GetFilters()) // sub#1 require.Equal(t, "filter_1_1", msubs[1].GetName()) require.Equal(t, "key_1_1", msubs[1].GetKey()) - require.Equal(t, apinetmap.GE, msubs[1].GetOp()) + require.Equal(t, protonetmap.Operation_GE, msubs[1].GetOp()) require.Equal(t, "1429243097315344888", msubs[1].GetValue()) require.Zero(t, msubs[1].GetFilters()) // sub#2 require.Equal(t, "filter_1_2", msubs[2].GetName()) require.Equal(t, "key_1_2", msubs[2].GetKey()) - require.Equal(t, apinetmap.LT, msubs[2].GetOp()) + require.Equal(t, protonetmap.Operation_LT, msubs[2].GetOp()) require.Equal(t, "3722656060317482335", msubs[2].GetValue()) require.Zero(t, msubs[2].GetFilters()) // sub#3 require.Equal(t, "filter_1_3", msubs[3].GetName()) require.Equal(t, "key_1_3", msubs[3].GetKey()) - require.Equal(t, apinetmap.LE, msubs[3].GetOp()) + require.Equal(t, protonetmap.Operation_LE, msubs[3].GetOp()) require.Equal(t, "1950504987705284805", msubs[3].GetValue()) require.Zero(t, msubs[3].GetFilters()) } @@ -956,16 +896,8 @@ func TestCalculateID(t *testing.T) { var id cid.ID val.CalculateID(&id) - var msg refs.ContainerID - id.WriteToV2(&msg) - h := sha256.Sum256(val.Marshal()) - require.Equal(t, h[:], msg.GetValue()) - - var id2 cid.ID - require.NoError(t, id2.ReadFromV2(msg)) - - require.True(t, val.AssertID(id2)) + require.EqualValues(t, h, id) } func TestContainer_CalculateSignature(t *testing.T) { diff --git a/container/example_test.go b/container/example_test.go index 0046bb3b..d6cc26d5 100644 --- a/container/example_test.go +++ b/container/example_test.go @@ -3,7 +3,6 @@ package container_test import ( "time" - apiGoContainer "github.com/nspcc-dev/neofs-api-go/v2/container" "github.com/nspcc-dev/neofs-sdk-go/container" "github.com/nspcc-dev/neofs-sdk-go/container/acl" "github.com/nspcc-dev/neofs-sdk-go/netmap" @@ -43,16 +42,14 @@ func ExampleContainer_Init() { // Instances can be also used to process NeoFS API V2 protocol messages with [https://github.com/nspcc-dev/neofs-api] package. func ExampleContainer_marshalling() { - // import apiGoContainer "github.com/nspcc-dev/neofs-api-go/v2/container" // On the client side. var cnr container.Container - var msg apiGoContainer.Container - cnr.WriteToV2(&msg) + msg := cnr.ProtoMessage() // *send message* // On the server side. - _ = cnr.ReadFromV2(msg) + _ = cnr.FromProtoMessage(msg) } diff --git a/container/size.go b/container/size.go index 2402a5f9..31223141 100644 --- a/container/size.go +++ b/container/size.go @@ -4,62 +4,68 @@ import ( "errors" "fmt" - "github.com/nspcc-dev/neofs-api-go/v2/container" - "github.com/nspcc-dev/neofs-api-go/v2/refs" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" + protocontainer "github.com/nspcc-dev/neofs-sdk-go/proto/container" ) // SizeEstimation groups information about estimation of the size of the data // stored in the NeoFS container. // -// SizeEstimation is mutually compatible with github.com/nspcc-dev/neofs-api-go/v2/container.UsedSpaceAnnouncement -// message. See ReadFromV2 / WriteToV2 methods. +// SizeEstimation is mutually compatible with [container.UsedSpaceAnnouncement] +// message. See [Container.FromProtoMessage] / [Container.ProtoMessage] methods. type SizeEstimation struct { - m container.UsedSpaceAnnouncement + epoch uint64 + cnr cid.ID + val uint64 } -// ReadFromV2 reads SizeEstimation from the container.UsedSpaceAnnouncement message. -// Checks if the message conforms to NeoFS API V2 protocol. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// x from it. // -// See also WriteToV2. -func (x *SizeEstimation) ReadFromV2(m container.UsedSpaceAnnouncement) error { - cnrV2 := m.GetContainerID() - if cnrV2 == nil { +// See also [SizeEstimation.ProtoMessage]. +func (x *SizeEstimation) FromProtoMessage(m *protocontainer.AnnounceUsedSpaceRequest_Body_Announcement) error { + if m.ContainerId == nil { return errors.New("missing container") } - var cnr cid.ID - - err := cnr.ReadFromV2(*cnrV2) + err := x.cnr.FromProtoMessage(m.ContainerId) if err != nil { return fmt.Errorf("invalid container: %w", err) } - x.m = m + x.epoch = m.Epoch + x.val = m.UsedSpace return nil } -// WriteToV2 writes SizeEstimation into the container.UsedSpaceAnnouncement message. -// The message MUST NOT be nil. +// ProtoMessage converts x into message to transmit using the NeoFS API +// protocol. // -// See also ReadFromV2. -func (x SizeEstimation) WriteToV2(m *container.UsedSpaceAnnouncement) { - *m = x.m +// See also [Container.FromProtoMessage]. +func (x SizeEstimation) ProtoMessage() *protocontainer.AnnounceUsedSpaceRequest_Body_Announcement { + m := &protocontainer.AnnounceUsedSpaceRequest_Body_Announcement{ + Epoch: x.epoch, + UsedSpace: x.val, + } + if !x.cnr.IsZero() { + m.ContainerId = x.cnr.ProtoMessage() + } + return m } // SetEpoch sets epoch when estimation of the container data size was calculated. // // See also Epoch. func (x *SizeEstimation) SetEpoch(epoch uint64) { - x.m.SetEpoch(epoch) + x.epoch = epoch } // Epoch return epoch set using SetEpoch. // // Zero SizeEstimation represents estimation in zero epoch. func (x SizeEstimation) Epoch() uint64 { - return x.m.GetEpoch() + return x.epoch } // SetContainer specifies the container for which the amount of data is estimated. @@ -67,10 +73,7 @@ func (x SizeEstimation) Epoch() uint64 { // // See also Container. func (x *SizeEstimation) SetContainer(cnr cid.ID) { - var cidV2 refs.ContainerID - cnr.WriteToV2(&cidV2) - - x.m.SetContainerID(&cidV2) + x.cnr = cnr } // Container returns container set using SetContainer. @@ -78,27 +81,19 @@ func (x *SizeEstimation) SetContainer(cnr cid.ID) { // Zero SizeEstimation is not bound to any container (returns zero) which is // incorrect according to NeoFS API protocol. func (x SizeEstimation) Container() (res cid.ID) { - m := x.m.GetContainerID() - if m != nil { - err := res.ReadFromV2(*m) - if err != nil { - panic(fmt.Errorf("unexpected error from cid.ID.ReadFromV2: %w", err)) - } - } - - return + return x.cnr } // SetValue sets estimated amount of data (in bytes) in the specified container. // // See also Value. func (x *SizeEstimation) SetValue(value uint64) { - x.m.SetUsedSpace(value) + x.val = value } // Value returns data size estimation set using SetValue. // // Zero SizeEstimation has zero value. func (x SizeEstimation) Value() uint64 { - return x.m.GetUsedSpace() + return x.val } diff --git a/container/size_test.go b/container/size_test.go index f6572cc3..e1f64751 100644 --- a/container/size_test.go +++ b/container/size_test.go @@ -3,11 +3,11 @@ package container_test import ( "testing" - v2container "github.com/nspcc-dev/neofs-api-go/v2/container" - "github.com/nspcc-dev/neofs-api-go/v2/refs" "github.com/nspcc-dev/neofs-sdk-go/container" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" cidtest "github.com/nspcc-dev/neofs-sdk-go/container/id/test" + protocontainer "github.com/nspcc-dev/neofs-sdk-go/proto/container" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" "github.com/stretchr/testify/require" ) @@ -63,79 +63,75 @@ func TestSizeEstimation_Value(t *testing.T) { require.EqualValues(t, anyValidVolume+1, val.Value()) } -func protoIDFromBytes(b []byte) *refs.ContainerID { - var m refs.ContainerID - m.SetValue(b) - return &m -} - -func TestSizeEstimation_ReadFromV2(t *testing.T) { - var m v2container.UsedSpaceAnnouncement - m.SetEpoch(anyValidEpoch) - m.SetContainerID(protoIDFromBytes(anyValidID[:])) - m.SetUsedSpace(anyValidVolume) +func TestSizeEstimation_FromProtoMessage(t *testing.T) { + m := &protocontainer.AnnounceUsedSpaceRequest_Body_Announcement{ + Epoch: anyValidEpoch, + ContainerId: &refs.ContainerID{Value: anyValidID[:]}, + UsedSpace: anyValidVolume, + } var val container.SizeEstimation - require.NoError(t, val.ReadFromV2(m)) + require.NoError(t, val.FromProtoMessage(m)) require.EqualValues(t, anyValidEpoch, val.Epoch()) require.Equal(t, anyValidID, val.Container()) require.EqualValues(t, anyValidVolume, val.Value()) // reset optional fields - m.SetEpoch(0) - m.SetUsedSpace(0) + m.Epoch = 0 + m.UsedSpace = 0 val2 := val - require.NoError(t, val2.ReadFromV2(m)) + require.NoError(t, val2.FromProtoMessage(m)) require.Zero(t, val2.Epoch()) require.Zero(t, val2.Value()) t.Run("invalid", func(t *testing.T) { for _, tc := range []struct { name, err string - corrupt func(announcement *v2container.UsedSpaceAnnouncement) + corrupt func(announcement *protocontainer.AnnounceUsedSpaceRequest_Body_Announcement) }{ {name: "container/missing", err: "missing container", - corrupt: func(m *v2container.UsedSpaceAnnouncement) { m.SetContainerID(nil) }}, + corrupt: func(m *protocontainer.AnnounceUsedSpaceRequest_Body_Announcement) { m.ContainerId = nil }}, {name: "container/nil", err: "invalid container: invalid length 0", - corrupt: func(m *v2container.UsedSpaceAnnouncement) { m.SetContainerID(protoIDFromBytes(nil)) }}, + corrupt: func(m *protocontainer.AnnounceUsedSpaceRequest_Body_Announcement) { m.ContainerId.Value = nil }}, {name: "container/empty", err: "invalid container: invalid length 0", - corrupt: func(m *v2container.UsedSpaceAnnouncement) { m.SetContainerID(protoIDFromBytes([]byte{})) }}, + corrupt: func(m *protocontainer.AnnounceUsedSpaceRequest_Body_Announcement) { m.ContainerId.Value = []byte{} }}, {name: "container/undersize", err: "invalid container: invalid length 31", - corrupt: func(m *v2container.UsedSpaceAnnouncement) { m.SetContainerID(protoIDFromBytes(anyValidID[:31])) }}, + corrupt: func(m *protocontainer.AnnounceUsedSpaceRequest_Body_Announcement) { + m.ContainerId.Value = anyValidID[:31] + }}, {name: "container/oversize", err: "invalid container: invalid length 33", - corrupt: func(m *v2container.UsedSpaceAnnouncement) { - m.SetContainerID(protoIDFromBytes(append(anyValidID[:], 1))) + corrupt: func(m *protocontainer.AnnounceUsedSpaceRequest_Body_Announcement) { + m.ContainerId.Value = append(anyValidID[:], 1) }}, } { t.Run(tc.name, func(t *testing.T) { val2 := val - var m v2container.UsedSpaceAnnouncement - val2.WriteToV2(&m) - tc.corrupt(&m) - require.EqualError(t, new(container.SizeEstimation).ReadFromV2(m), tc.err) + m := val2.ProtoMessage() + tc.corrupt(m) + require.EqualError(t, new(container.SizeEstimation).FromProtoMessage(m), tc.err) }) } t.Run("container/zero", func(t *testing.T) { - var m v2container.UsedSpaceAnnouncement - m.SetContainerID(protoIDFromBytes(make([]byte, cid.Size))) - require.ErrorIs(t, val2.ReadFromV2(m), cid.ErrZero) + m := &protocontainer.AnnounceUsedSpaceRequest_Body_Announcement{ + ContainerId: &refs.ContainerID{Value: make([]byte, cid.Size)}, + } + require.ErrorIs(t, val2.FromProtoMessage(m), cid.ErrZero) }) }) } -func TestSizeEstimation_WriteToV2(t *testing.T) { +func TestSizeEstimation_ProtoMessage(t *testing.T) { var val container.SizeEstimation - var m v2container.UsedSpaceAnnouncement // zero - val.WriteToV2(&m) + m := val.ProtoMessage() require.Zero(t, val.Epoch()) require.Zero(t, val.Container()) require.Zero(t, val.Value()) // filled - validSizeEstimation.WriteToV2(&m) + m = validSizeEstimation.ProtoMessage() require.EqualValues(t, anyValidEpoch, m.GetEpoch()) - require.Equal(t, anyValidID[:], m.GetContainerID().GetValue()) + require.Equal(t, anyValidID[:], m.GetContainerId().GetValue()) require.EqualValues(t, anyValidVolume, m.GetUsedSpace()) } diff --git a/eacl/common_test.go b/eacl/common_test.go index 3fd1c096..2a762d0f 100644 --- a/eacl/common_test.go +++ b/eacl/common_test.go @@ -7,12 +7,12 @@ import ( "math/big" "testing" - protoacl "github.com/nspcc-dev/neofs-api-go/v2/acl" "github.com/nspcc-dev/neofs-sdk-go/checksum" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" "github.com/nspcc-dev/neofs-sdk-go/eacl" "github.com/nspcc-dev/neofs-sdk-go/object" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + protoacl "github.com/nspcc-dev/neofs-sdk-go/proto/acl" "github.com/nspcc-dev/neofs-sdk-go/user" "github.com/nspcc-dev/neofs-sdk-go/version" "github.com/nspcc-dev/tzhash/tz" @@ -337,7 +337,7 @@ func init() { } } -func assertProtoTargetsEqual(t testing.TB, ts []eacl.Target, ms []protoacl.Target) { +func assertProtoTargetsEqual(t testing.TB, ts []eacl.Target, ms []*protoacl.EACLRecord_Target) { require.Len(t, ms, len(ts)) for i := range ts { require.EqualValues(t, ts[i].Role(), ms[i].GetRole(), i) @@ -345,7 +345,7 @@ func assertProtoTargetsEqual(t testing.TB, ts []eacl.Target, ms []protoacl.Targe } } -func assertProtoFiltersEqual(t testing.TB, fs []eacl.Filter, ms []protoacl.HeaderFilter) { +func assertProtoFiltersEqual(t testing.TB, fs []eacl.Filter, ms []*protoacl.EACLRecord_Filter) { require.Len(t, ms, len(fs)) for i := range fs { require.EqualValues(t, fs[i].From(), ms[i].GetHeaderType(), i) @@ -355,7 +355,7 @@ func assertProtoFiltersEqual(t testing.TB, fs []eacl.Filter, ms []protoacl.Heade } } -func assertProtoRecordsEqual(t testing.TB, rs []eacl.Record, ms []protoacl.Record) { +func assertProtoRecordsEqual(t testing.TB, rs []eacl.Record, ms []*protoacl.EACLRecord) { require.Len(t, ms, len(rs)) for i := range rs { require.EqualValues(t, rs[i].Action(), ms[i].GetAction(), i) diff --git a/eacl/enums.go b/eacl/enums.go index c9083092..ccfc5e16 100644 --- a/eacl/enums.go +++ b/eacl/enums.go @@ -2,13 +2,11 @@ package eacl import ( "strconv" - - v2acl "github.com/nspcc-dev/neofs-api-go/v2/acl" ) // Action enumerates actions that may be applied within NeoFS access management. // What and how specific Action affects depends on the specific context. -type Action uint32 +type Action int32 const ( ActionUnspecified Action = iota // undefined (zero) @@ -21,7 +19,7 @@ const ( const ActionUnknown = ActionUnspecified // Operation enumerates operations on NeoFS resources under access control. -type Operation uint32 +type Operation int32 const ( OperationUnspecified Operation = iota // undefined (zero) @@ -39,7 +37,7 @@ const ( const OperationUnknown = OperationUnspecified // Role enumerates groups of subjects requesting access to NeoFS resources. -type Role uint32 +type Role int32 const ( RoleUnspecified Role = iota // undefined (zero) @@ -54,7 +52,7 @@ const RoleUnknown = RoleUnspecified // Match enumerates operators to check attribute value compliance. What and how // specific Match affects depends on the specific context. -type Match uint32 +type Match int32 const ( MatchUnspecified Match = iota // undefined (zero) @@ -73,7 +71,7 @@ const MatchUnknown = MatchUnspecified // FilterHeaderType enumerates the classes of resource attributes processed // within NeoFS access management. -type FilterHeaderType uint32 +type FilterHeaderType int32 const ( HeaderTypeUnspecified FilterHeaderType = iota // undefined (zero) @@ -86,14 +84,6 @@ const ( // Deprecated: use HeaderTypeUnspecified instead. const HeaderTypeUnknown = HeaderTypeUnspecified -// ToV2 converts Action to v2 Action enum value. -// Deprecated: do not use it. -func (a Action) ToV2() v2acl.Action { return v2acl.Action(a) } - -// ActionFromV2 converts v2 Action enum value to Action. -// Deprecated: do not use it. -func ActionFromV2(action v2acl.Action) Action { return Action(action) } - const ( actionStringZero = "ACTION_UNSPECIFIED" actionStringAllow = "ALLOW" @@ -124,7 +114,7 @@ func (a Action) EncodeToString() string { return a.String() } func (a Action) String() string { switch a { default: - return strconv.FormatUint(uint64(a), 10) + return strconv.FormatInt(int64(a), 10) case 0: return actionStringZero case ActionAllow: @@ -156,14 +146,6 @@ func (a *Action) DecodeString(s string) bool { return true } -// ToV2 converts Operation to v2 Operation enum value. -// Deprecated: do not use it. -func (o Operation) ToV2() v2acl.Operation { return v2acl.Operation(o) } - -// OperationFromV2 converts v2 Operation enum value to Operation. -// Deprecated: do not use it. -func OperationFromV2(operation v2acl.Operation) Operation { return Operation(operation) } - const ( opStringZero = "OPERATION_UNSPECIFIED" opStringGet = "GET" @@ -209,7 +191,7 @@ func (o Operation) EncodeToString() string { return o.String() } func (o Operation) String() string { switch o { default: - return strconv.FormatUint(uint64(o), 10) + return strconv.FormatInt(int64(o), 10) case 0: return opStringZero case OperationGet: @@ -261,14 +243,6 @@ func (o *Operation) DecodeString(s string) bool { return true } -// ToV2 converts Role to v2 Role enum value. -// Deprecated: do not use it. -func (r Role) ToV2() v2acl.Role { return v2acl.Role(r) } - -// RoleFromV2 converts v2 Role enum value to Role. -// Deprecated: do not use it. -func RoleFromV2(role v2acl.Role) Role { return Role(role) } - const ( roleStringZero = "ROLE_UNSPECIFIED" roleStringUser = "USER" @@ -302,7 +276,7 @@ func (r Role) EncodeToString() string { return r.String() } func (r Role) String() string { switch r { default: - return strconv.FormatUint(uint64(r), 10) + return strconv.FormatInt(int64(r), 10) case 0: return roleStringZero case RoleUser: @@ -338,14 +312,6 @@ func (r *Role) DecodeString(s string) bool { return true } -// ToV2 converts Match to v2 MatchType enum value. -// Deprecated: do not use it. -func (m Match) ToV2() v2acl.MatchType { return v2acl.MatchType(m) } - -// MatchFromV2 converts v2 MatchType enum value to Match. -// Deprecated: do not use it. -func MatchFromV2(match v2acl.MatchType) Match { return Match(match) } - const ( matcherStringZero = "MATCH_TYPE_UNSPECIFIED" matcherStringEqual = "STRING_EQUAL" @@ -391,7 +357,7 @@ func (m Match) EncodeToString() string { return m.String() } func (m Match) String() string { switch m { default: - return strconv.FormatUint(uint64(m), 10) + return strconv.FormatInt(int64(m), 10) case 0: return matcherStringZero case MatchStringEqual: @@ -443,16 +409,6 @@ func (m *Match) DecodeString(s string) bool { return true } -// ToV2 converts FilterHeaderType to v2 HeaderType enum value. -// Deprecated: do not use it. -func (h FilterHeaderType) ToV2() v2acl.HeaderType { return v2acl.HeaderType(h) } - -// FilterHeaderTypeFromV2 converts v2 HeaderType enum value to FilterHeaderType. -// Deprecated: do not use it. -func FilterHeaderTypeFromV2(header v2acl.HeaderType) FilterHeaderType { - return FilterHeaderType(header) -} - const ( headerTypeStringZero = "HEADER_UNSPECIFIED" headerTypeStringRequest = "REQUEST" @@ -485,7 +441,7 @@ func (h FilterHeaderType) EncodeToString() string { return h.String() } func (h FilterHeaderType) String() string { switch h { default: - return strconv.FormatUint(uint64(h), 10) + return strconv.FormatInt(int64(h), 10) case 0: return headerTypeStringZero case HeaderFromRequest: diff --git a/eacl/enums_test.go b/eacl/enums_test.go index 6e47e023..6f5b54e1 100644 --- a/eacl/enums_test.go +++ b/eacl/enums_test.go @@ -4,61 +4,63 @@ import ( "fmt" "testing" - v2acl "github.com/nspcc-dev/neofs-api-go/v2/acl" "github.com/nspcc-dev/neofs-sdk-go/eacl" + protoacl "github.com/nspcc-dev/neofs-sdk-go/proto/acl" "github.com/stretchr/testify/require" ) var ( - eqV2Actions = map[eacl.Action]v2acl.Action{ - eacl.ActionUnspecified: v2acl.ActionUnknown, - eacl.ActionAllow: v2acl.ActionAllow, - eacl.ActionDeny: v2acl.ActionDeny, + protoActions = map[eacl.Action]protoacl.Action{ + eacl.ActionUnspecified: protoacl.Action_ACTION_UNSPECIFIED, + eacl.ActionAllow: protoacl.Action_ALLOW, + eacl.ActionDeny: protoacl.Action_DENY, } - eqV2Operations = map[eacl.Operation]v2acl.Operation{ - eacl.OperationUnspecified: v2acl.OperationUnknown, - eacl.OperationGet: v2acl.OperationGet, - eacl.OperationHead: v2acl.OperationHead, - eacl.OperationPut: v2acl.OperationPut, - eacl.OperationDelete: v2acl.OperationDelete, - eacl.OperationSearch: v2acl.OperationSearch, - eacl.OperationRange: v2acl.OperationRange, - eacl.OperationRangeHash: v2acl.OperationRangeHash, + protoOperations = map[eacl.Operation]protoacl.Operation{ + eacl.OperationUnspecified: protoacl.Operation_OPERATION_UNSPECIFIED, + eacl.OperationGet: protoacl.Operation_GET, + eacl.OperationHead: protoacl.Operation_HEAD, + eacl.OperationPut: protoacl.Operation_PUT, + eacl.OperationDelete: protoacl.Operation_DELETE, + eacl.OperationSearch: protoacl.Operation_SEARCH, + eacl.OperationRange: protoacl.Operation_GETRANGE, + eacl.OperationRangeHash: protoacl.Operation_GETRANGEHASH, } - eqV2Roles = map[eacl.Role]v2acl.Role{ - eacl.RoleUnspecified: v2acl.RoleUnknown, - eacl.RoleUser: v2acl.RoleUser, - eacl.RoleSystem: v2acl.RoleSystem, - eacl.RoleOthers: v2acl.RoleOthers, + protoRoles = map[eacl.Role]protoacl.Role{ + eacl.RoleUnspecified: protoacl.Role_ROLE_UNSPECIFIED, + eacl.RoleUser: protoacl.Role_USER, + eacl.RoleSystem: protoacl.Role_SYSTEM, + eacl.RoleOthers: protoacl.Role_OTHERS, } - eqV2Matches = map[eacl.Match]v2acl.MatchType{ - eacl.MatchUnspecified: v2acl.MatchTypeUnknown, - eacl.MatchStringEqual: v2acl.MatchTypeStringEqual, - eacl.MatchStringNotEqual: v2acl.MatchTypeStringNotEqual, - eacl.MatchNotPresent: v2acl.MatchTypeNotPresent, - eacl.MatchNumGT: v2acl.MatchTypeNumGT, - eacl.MatchNumGE: v2acl.MatchTypeNumGE, - eacl.MatchNumLT: v2acl.MatchTypeNumLT, - eacl.MatchNumLE: v2acl.MatchTypeNumLE, + protoMatches = map[eacl.Match]protoacl.MatchType{ + eacl.MatchUnspecified: protoacl.MatchType_MATCH_TYPE_UNSPECIFIED, + eacl.MatchStringEqual: protoacl.MatchType_STRING_EQUAL, + eacl.MatchStringNotEqual: protoacl.MatchType_STRING_NOT_EQUAL, + eacl.MatchNotPresent: protoacl.MatchType_NOT_PRESENT, + eacl.MatchNumGT: protoacl.MatchType_NUM_GT, + eacl.MatchNumGE: protoacl.MatchType_NUM_GE, + eacl.MatchNumLT: protoacl.MatchType_NUM_LT, + eacl.MatchNumLE: protoacl.MatchType_NUM_LE, } - eqV2HeaderTypes = map[eacl.FilterHeaderType]v2acl.HeaderType{ - eacl.HeaderTypeUnspecified: v2acl.HeaderTypeUnknown, - eacl.HeaderFromRequest: v2acl.HeaderTypeRequest, - eacl.HeaderFromObject: v2acl.HeaderTypeObject, - eacl.HeaderFromService: v2acl.HeaderTypeService, + protoHeaderTypes = map[eacl.FilterHeaderType]protoacl.HeaderType{ + eacl.HeaderTypeUnspecified: protoacl.HeaderType_HEADER_UNSPECIFIED, + eacl.HeaderFromRequest: protoacl.HeaderType_REQUEST, + eacl.HeaderFromObject: protoacl.HeaderType_OBJECT, + eacl.HeaderFromService: protoacl.HeaderType_SERVICE, } actionStrings = map[eacl.Action]string{ + -1: "-1", 0: "ACTION_UNSPECIFIED", eacl.ActionAllow: "ALLOW", eacl.ActionDeny: "DENY", 3: "3", } roleStrings = map[eacl.Role]string{ + -1: "-1", 0: "ROLE_UNSPECIFIED", eacl.RoleUser: "USER", eacl.RoleSystem: "SYSTEM", @@ -66,6 +68,7 @@ var ( 4: "4", } opStrings = map[eacl.Operation]string{ + -1: "-1", 0: "OPERATION_UNSPECIFIED", eacl.OperationGet: "GET", eacl.OperationHead: "HEAD", @@ -77,6 +80,7 @@ var ( 8: "8", } matcherStrings = map[eacl.Match]string{ + -1: "-1", 0: "MATCH_TYPE_UNSPECIFIED", eacl.MatchStringEqual: "STRING_EQUAL", eacl.MatchStringNotEqual: "STRING_NOT_EQUAL", @@ -88,6 +92,7 @@ var ( 8: "8", } headerTypeStrings = map[eacl.FilterHeaderType]string{ + -1: "-1", 0: "HEADER_UNSPECIFIED", eacl.HeaderFromRequest: "REQUEST", eacl.HeaderFromObject: "OBJECT", @@ -96,80 +101,6 @@ var ( } ) -func TestAction(t *testing.T) { - require.Equal(t, eacl.ActionUnspecified, eacl.ActionUnknown) - t.Run("known actions", func(t *testing.T) { - for i := eacl.ActionUnspecified; i <= eacl.ActionDeny; i++ { - require.Equal(t, eqV2Actions[i], i.ToV2()) - require.Equal(t, eacl.ActionFromV2(i.ToV2()), i) - } - }) - - t.Run("unknown actions", func(t *testing.T) { - require.EqualValues(t, 1000, eacl.Action(1000).ToV2()) - require.EqualValues(t, 1000, eacl.ActionFromV2(1000)) - }) -} - -func TestOperation(t *testing.T) { - require.Equal(t, eacl.OperationUnspecified, eacl.OperationUnknown) - t.Run("known operations", func(t *testing.T) { - for i := eacl.OperationUnspecified; i <= eacl.OperationRangeHash; i++ { - require.Equal(t, eqV2Operations[i], i.ToV2()) - require.Equal(t, eacl.OperationFromV2(i.ToV2()), i) - } - }) - - t.Run("unknown operations", func(t *testing.T) { - require.EqualValues(t, 1000, eacl.Operation(1000).ToV2()) - require.EqualValues(t, 1000, eacl.OperationFromV2(1000)) - }) -} - -func TestRole(t *testing.T) { - t.Run("known roles", func(t *testing.T) { - for i := eacl.RoleUnspecified; i <= eacl.RoleOthers; i++ { - require.Equal(t, eqV2Roles[i], i.ToV2()) - require.Equal(t, eacl.RoleFromV2(i.ToV2()), i) - } - }) - - t.Run("unknown roles", func(t *testing.T) { - require.EqualValues(t, 1000, eacl.Operation(1000).ToV2()) - require.EqualValues(t, 1000, eacl.RoleFromV2(1000)) - }) -} - -func TestMatch(t *testing.T) { - require.Equal(t, eacl.MatchUnspecified, eacl.MatchUnknown) - t.Run("known matches", func(t *testing.T) { - for i := eacl.MatchUnspecified; i <= eacl.MatchStringNotEqual; i++ { - require.Equal(t, eqV2Matches[i], i.ToV2()) - require.Equal(t, eacl.MatchFromV2(i.ToV2()), i) - } - }) - - t.Run("unknown matches", func(t *testing.T) { - require.EqualValues(t, 1000, eacl.Match(1000).ToV2()) - require.EqualValues(t, 1000, eacl.MatchFromV2(1000)) - }) -} - -func TestFilterHeaderType(t *testing.T) { - require.Equal(t, eacl.HeaderTypeUnspecified, eacl.HeaderTypeUnknown) - t.Run("known header types", func(t *testing.T) { - for i := eacl.HeaderTypeUnspecified; i <= eacl.HeaderFromService; i++ { - require.Equal(t, eqV2HeaderTypes[i], i.ToV2()) - require.Equal(t, eacl.FilterHeaderTypeFromV2(i.ToV2()), i) - } - }) - - t.Run("unknown header types", func(t *testing.T) { - require.EqualValues(t, 1000, eacl.FilterHeaderType(1000).ToV2()) - require.EqualValues(t, 1000, eacl.FilterHeaderTypeFromV2(1000)) - }) -} - type enumIface interface { DecodeString(string) bool EncodeToString() string @@ -201,11 +132,7 @@ func testEnumStrings(t *testing.T, e enumIface, items []enumStringItem) { } func TestActionProto(t *testing.T) { - for x, y := range map[v2acl.Action]eacl.Action{ - v2acl.ActionUnknown: eacl.ActionUnspecified, - v2acl.ActionAllow: eacl.ActionAllow, - v2acl.ActionDeny: eacl.ActionDeny, - } { + for x, y := range protoActions { require.EqualValues(t, x, y) } } @@ -227,7 +154,7 @@ func TestAction_String(t *testing.T) { } type enum interface { - ~uint32 + ~int32 fmt.Stringer } @@ -260,12 +187,7 @@ func TestActionFromString(t *testing.T) { } func TestRoleProto(t *testing.T) { - for x, y := range map[v2acl.Role]eacl.Role{ - v2acl.RoleUnknown: eacl.RoleUnspecified, - v2acl.RoleUser: eacl.RoleUser, - v2acl.RoleSystem: eacl.RoleSystem, - v2acl.RoleOthers: eacl.RoleOthers, - } { + for x, y := range protoRoles { require.EqualValues(t, x, y) } } @@ -296,16 +218,7 @@ func TestRoleFromString(t *testing.T) { } func TestOperationProto(t *testing.T) { - for x, y := range map[v2acl.Operation]eacl.Operation{ - v2acl.OperationUnknown: eacl.OperationUnspecified, - v2acl.OperationGet: eacl.OperationGet, - v2acl.OperationHead: eacl.OperationHead, - v2acl.OperationPut: eacl.OperationPut, - v2acl.OperationDelete: eacl.OperationDelete, - v2acl.OperationSearch: eacl.OperationSearch, - v2acl.OperationRange: eacl.OperationRange, - v2acl.OperationRangeHash: eacl.OperationRangeHash, - } { + for x, y := range protoOperations { require.EqualValues(t, x, y) } } @@ -340,16 +253,7 @@ func TestOperationFromString(t *testing.T) { } func TestMatchProto(t *testing.T) { - for x, y := range map[v2acl.MatchType]eacl.Match{ - v2acl.MatchTypeUnknown: eacl.MatchUnspecified, - v2acl.MatchTypeStringEqual: eacl.MatchStringEqual, - v2acl.MatchTypeStringNotEqual: eacl.MatchStringNotEqual, - v2acl.MatchTypeNotPresent: eacl.MatchNotPresent, - v2acl.MatchTypeNumGT: eacl.MatchNumGT, - v2acl.MatchTypeNumGE: eacl.MatchNumGE, - v2acl.MatchTypeNumLT: eacl.MatchNumLT, - v2acl.MatchTypeNumLE: eacl.MatchNumLE, - } { + for x, y := range protoMatches { require.EqualValues(t, x, y) } } @@ -384,12 +288,7 @@ func TestMatcherFromString(t *testing.T) { } func TestFilterHeaderTypeProto(t *testing.T) { - for x, y := range map[v2acl.HeaderType]eacl.FilterHeaderType{ - v2acl.HeaderTypeUnknown: eacl.FilterHeaderType(0), - v2acl.HeaderTypeRequest: eacl.HeaderFromRequest, - v2acl.HeaderTypeObject: eacl.HeaderFromObject, - v2acl.HeaderTypeService: eacl.HeaderFromService, - } { + for x, y := range protoHeaderTypes { require.EqualValues(t, x, y) } } diff --git a/eacl/filter.go b/eacl/filter.go index 8816b84a..ed7a4042 100644 --- a/eacl/filter.go +++ b/eacl/filter.go @@ -1,11 +1,13 @@ package eacl import ( + "fmt" "strconv" - v2acl "github.com/nspcc-dev/neofs-api-go/v2/acl" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + protoacl "github.com/nspcc-dev/neofs-sdk-go/proto/acl" "github.com/nspcc-dev/neofs-sdk-go/user" ) @@ -121,31 +123,26 @@ func (f Filter) From() FilterHeaderType { return f.from } -// ToV2 converts Filter to v2 acl.EACLRecord.Filter message. -// -// Nil Filter converts to nil. -// Deprecated: do not use it. -func (f *Filter) ToV2() *v2acl.HeaderFilter { - if f != nil { - return f.toProtoMessage() +func (f Filter) protoMessage() *protoacl.EACLRecord_Filter { + return &protoacl.EACLRecord_Filter{ + HeaderType: protoacl.HeaderType(f.from), + MatchType: protoacl.MatchType(f.matcher), + Key: f.key, + Value: f.value.EncodeToString(), } - return nil -} - -func (f Filter) toProtoMessage() *v2acl.HeaderFilter { - filter := new(v2acl.HeaderFilter) - filter.SetValue(f.value.EncodeToString()) - filter.SetKey(f.key) - filter.SetMatchType(v2acl.MatchType(f.matcher)) - filter.SetHeaderType(v2acl.HeaderType(f.from)) - return filter } -func (f *Filter) fromProtoMessage(m *v2acl.HeaderFilter) error { - f.from = FilterHeaderType(m.GetHeaderType()) - f.matcher = Match(m.GetMatchType()) - f.key = m.GetKey() - f.value = staticStringer(m.GetValue()) +func (f *Filter) fromProtoMessage(m *protoacl.EACLRecord_Filter) error { + if m.HeaderType < 0 { + return fmt.Errorf("negative header type %d", m.HeaderType) + } + if m.MatchType < 0 { + return fmt.Errorf("negative match type %d", m.MatchType) + } + f.from = FilterHeaderType(m.HeaderType) + f.matcher = Match(m.MatchType) + f.key = m.Key + f.value = staticStringer(m.Value) return nil } @@ -163,28 +160,15 @@ func NewFilter() *Filter { return &f } -// NewFilterFromV2 converts v2 acl.EACLRecord.Filter message to Filter. -// Deprecated: do not use it. -func NewFilterFromV2(filter *v2acl.HeaderFilter) *Filter { - f := new(Filter) - - if filter == nil { - return f - } - - _ = f.fromProtoMessage(filter) - return f -} - // Marshal marshals Filter into a protobuf binary form. func (f Filter) Marshal() []byte { - return f.toProtoMessage().StableMarshal(nil) + return neofsproto.MarshalMessage(f.protoMessage()) } // Unmarshal unmarshals protobuf binary representation of Filter. func (f *Filter) Unmarshal(data []byte) error { - m := new(v2acl.HeaderFilter) - if err := m.Unmarshal(data); err != nil { + m := new(protoacl.EACLRecord_Filter) + if err := neofsproto.UnmarshalMessage(data, m); err != nil { return err } return f.fromProtoMessage(m) @@ -192,13 +176,13 @@ func (f *Filter) Unmarshal(data []byte) error { // MarshalJSON encodes Filter to protobuf JSON format. func (f Filter) MarshalJSON() ([]byte, error) { - return f.toProtoMessage().MarshalJSON() + return neofsproto.MarshalMessageJSON(f.protoMessage()) } // UnmarshalJSON decodes Filter from protobuf JSON format. func (f *Filter) UnmarshalJSON(data []byte) error { - m := new(v2acl.HeaderFilter) - if err := m.UnmarshalJSON(data); err != nil { + m := new(protoacl.EACLRecord_Filter) + if err := neofsproto.UnmarshalMessageJSON(data, m); err != nil { return err } return f.fromProtoMessage(m) diff --git a/eacl/filter_test.go b/eacl/filter_test.go index 7ef745cb..b87651f7 100644 --- a/eacl/filter_test.go +++ b/eacl/filter_test.go @@ -2,11 +2,8 @@ package eacl_test import ( "encoding/json" - "math/rand/v2" - "strconv" "testing" - protoacl "github.com/nspcc-dev/neofs-api-go/v2/acl" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" "github.com/nspcc-dev/neofs-sdk-go/eacl" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" @@ -14,58 +11,6 @@ import ( "github.com/stretchr/testify/require" ) -func TestFilter_ToV2(t *testing.T) { - require.Nil(t, (*eacl.Filter)(nil).ToV2()) - key := "key_" + strconv.Itoa(rand.Int()) - val := "val_" + strconv.Itoa(rand.Int()) - r := eacl.ConstructFilter(anyValidHeaderType, key, anyValidMatcher, val) - m := r.ToV2() - require.EqualValues(t, anyValidHeaderType, m.GetHeaderType()) - require.Equal(t, key, m.GetKey()) - require.EqualValues(t, anyValidMatcher, m.GetMatchType()) - require.Equal(t, val, m.GetValue()) - - t.Run("default values", func(t *testing.T) { - filter := eacl.NewFilter() - - // check initial values - require.Empty(t, filter.Key()) - require.Empty(t, filter.Value()) - require.Zero(t, filter.From()) - require.Zero(t, filter.Matcher()) - - // convert to v2 message - filterV2 := filter.ToV2() - - require.Empty(t, filterV2.GetKey()) - require.Empty(t, filterV2.GetValue()) - require.Zero(t, filterV2.GetHeaderType()) - require.Zero(t, filterV2.GetMatchType()) - }) -} - -func TestNewFilterFromV2(t *testing.T) { - typ := protoacl.HeaderType(rand.Uint32()) - key := "key_" + strconv.Itoa(rand.Int()) - op := protoacl.MatchType(rand.Uint32()) - val := "val_" + strconv.Itoa(rand.Int()) - var m protoacl.HeaderFilter - m.SetHeaderType(typ) - m.SetKey(key) - m.SetMatchType(op) - m.SetValue(val) - - f := eacl.NewFilterFromV2(&m) - require.EqualValues(t, typ, f.From()) - require.Equal(t, key, f.Key()) - require.EqualValues(t, op, f.Matcher()) - require.Equal(t, val, f.Value()) - - t.Run("nil", func(t *testing.T) { - require.Equal(t, new(eacl.Filter), eacl.NewFilterFromV2(nil)) - }) -} - func TestFilter_Marshal(t *testing.T) { for i := range anyValidFilters { require.Equal(t, anyValidBinFilters[i], anyValidFilters[i].Marshal(), i) diff --git a/eacl/record.go b/eacl/record.go index a0de31bc..9c04b366 100644 --- a/eacl/record.go +++ b/eacl/record.go @@ -5,11 +5,12 @@ import ( "fmt" "slices" - v2acl "github.com/nspcc-dev/neofs-api-go/v2/acl" "github.com/nspcc-dev/neofs-sdk-go/checksum" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" "github.com/nspcc-dev/neofs-sdk-go/object" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + protoacl "github.com/nspcc-dev/neofs-sdk-go/proto/acl" "github.com/nspcc-dev/neofs-sdk-go/user" "github.com/nspcc-dev/neofs-sdk-go/version" ) @@ -213,63 +214,61 @@ func (r *Record) AddObjectHomomorphicHashFilter(m Match, h checksum.Checksum) { r.SetFilters(append(r.Filters(), NewObjectPropertyFilter(FilterObjectPayloadHomomorphicChecksum, m, h.String()))) } -// ToV2 converts Record to v2 acl.EACLRecord message. -// -// Nil Record converts to nil. -// Deprecated: do not use it. -func (r *Record) ToV2() *v2acl.Record { - if r != nil { - return r.toProtoMessage() +func (r Record) toProtoMessage() *protoacl.EACLRecord { + m := &protoacl.EACLRecord{ + Operation: protoacl.Operation(r.operation), + Action: protoacl.Action(r.action), } - return nil -} - -func (r Record) toProtoMessage() *v2acl.Record { - v2 := new(v2acl.Record) if r.targets != nil { - targets := make([]v2acl.Target, len(r.targets)) + m.Targets = make([]*protoacl.EACLRecord_Target, len(r.targets)) for i := range r.targets { - targets[i] = *r.targets[i].toProtoMessage() + m.Targets[i] = r.targets[i].protoMessage() } - - v2.SetTargets(targets) } if r.filters != nil { - filters := make([]v2acl.HeaderFilter, len(r.filters)) + m.Filters = make([]*protoacl.EACLRecord_Filter, len(r.filters)) for i := range r.filters { - filters[i] = *r.filters[i].toProtoMessage() + m.Filters[i] = r.filters[i].protoMessage() } - - v2.SetFilters(filters) } - v2.SetAction(v2acl.Action(r.action)) - v2.SetOperation(v2acl.Operation(r.operation)) - - return v2 + return m } -func (r *Record) fromProtoMessage(m *v2acl.Record) error { - mt := m.GetTargets() +func (r *Record) fromProtoMessage(m *protoacl.EACLRecord) error { + if m.Action < 0 { + return fmt.Errorf("negative action %d", m.Action) + } + if m.Operation < 0 { + return fmt.Errorf("negative op %d", m.Operation) + } + + mt := m.Targets r.targets = make([]Target, len(mt)) for i := range mt { - if err := r.targets[i].fromProtoMessage(&mt[i]); err != nil { + if mt[i] == nil { + return fmt.Errorf("nil target #%d", i) + } + if err := r.targets[i].fromProtoMessage(mt[i]); err != nil { return fmt.Errorf("invalid subject descriptor #%d: %w", i, err) } } - mf := m.GetFilters() + mf := m.Filters r.filters = make([]Filter, len(mf)) for i := range mf { - if err := r.filters[i].fromProtoMessage(&mf[i]); err != nil { + if mf[i] == nil { + return fmt.Errorf("nil filter #%d", i) + } + if err := r.filters[i].fromProtoMessage(mf[i]); err != nil { return fmt.Errorf("invalid filter #%d: %w", i, err) } } - r.action = Action(m.GetAction()) - r.operation = Operation(m.GetOperation()) + r.action = Action(m.Action) + r.operation = Operation(m.Operation) return nil } @@ -299,28 +298,15 @@ func CreateRecord(action Action, operation Operation) *Record { return r } -// NewRecordFromV2 converts v2 acl.EACLRecord message to Record. -// Deprecated: do not use it. -func NewRecordFromV2(record *v2acl.Record) *Record { - r := NewRecord() - - if record == nil { - return r - } - - _ = r.fromProtoMessage(record) - return r -} - // Marshal marshals Record into a protobuf binary form. func (r Record) Marshal() []byte { - return r.toProtoMessage().StableMarshal(nil) + return neofsproto.MarshalMessage(r.toProtoMessage()) } // Unmarshal unmarshals protobuf binary representation of Record. func (r *Record) Unmarshal(data []byte) error { - m := new(v2acl.Record) - if err := m.Unmarshal(data); err != nil { + m := new(protoacl.EACLRecord) + if err := neofsproto.UnmarshalMessage(data, m); err != nil { return err } return r.fromProtoMessage(m) @@ -328,13 +314,13 @@ func (r *Record) Unmarshal(data []byte) error { // MarshalJSON encodes Record to protobuf JSON format. func (r Record) MarshalJSON() ([]byte, error) { - return r.toProtoMessage().MarshalJSON() + return neofsproto.MarshalMessageJSON(r.toProtoMessage()) } // UnmarshalJSON decodes Record from protobuf JSON format. func (r *Record) UnmarshalJSON(data []byte) error { - m := new(v2acl.Record) - if err := m.UnmarshalJSON(data); err != nil { + m := new(protoacl.EACLRecord) + if err := neofsproto.UnmarshalMessageJSON(data, m); err != nil { return err } return r.fromProtoMessage(m) diff --git a/eacl/record_test.go b/eacl/record_test.go index 748092e4..f1e75846 100644 --- a/eacl/record_test.go +++ b/eacl/record_test.go @@ -3,10 +3,8 @@ package eacl_test import ( "encoding/json" "math/rand/v2" - "strconv" "testing" - protoacl "github.com/nspcc-dev/neofs-api-go/v2/acl" "github.com/nspcc-dev/neofs-sdk-go/eacl" "github.com/stretchr/testify/require" ) @@ -20,7 +18,7 @@ func TestAddFormedTarget(t *testing.T) { require.Zero(t, r.Targets()[0].Role()) require.Equal(t, anyValidECDSABinPublicKeys, r.Targets()[0].BinaryKeys()) - role := eacl.Role(rand.Uint32()) + role := eacl.Role(rand.Int32()) eacl.AddFormedTarget(&r, role) require.Len(t, r.Targets(), 2) require.Equal(t, role, r.Targets()[1].Role()) @@ -36,66 +34,6 @@ func TestRecord_AddFilter(t *testing.T) { require.Equal(t, anyValidFilters, r.Filters()) } -func TestRecord_ToV2(t *testing.T) { - require.Nil(t, (*eacl.Record)(nil).ToV2()) - r := eacl.ConstructRecord(anyValidAction, anyValidOp, anyValidTargets, anyValidFilters...) - m := r.ToV2() - require.EqualValues(t, anyValidAction, m.GetAction()) - require.EqualValues(t, anyValidOp, m.GetOperation()) - assertProtoTargetsEqual(t, anyValidTargets, m.GetTargets()) - assertProtoFiltersEqual(t, anyValidFilters, m.GetFilters()) - - t.Run("default values", func(t *testing.T) { - record := eacl.NewRecord() - - // check initial values - require.Zero(t, record.Operation()) - require.Zero(t, record.Action()) - require.Nil(t, record.Targets()) - require.Nil(t, record.Filters()) - - // convert to v2 message - recordV2 := record.ToV2() - - require.Zero(t, recordV2.GetOperation()) - require.Zero(t, recordV2.GetAction()) - require.Nil(t, recordV2.GetTargets()) - require.Nil(t, recordV2.GetFilters()) - }) -} - -func TestNewRecordFromV2(t *testing.T) { - a := protoacl.Action(rand.Uint32()) - op := protoacl.Operation(rand.Uint32()) - ts := make([]protoacl.Target, 2) - for i := range ts { - ts[i].SetRole(protoacl.Role(rand.Uint32())) - ts[i].SetKeys(anyValidBinPublicKeys) - } - fs := make([]protoacl.HeaderFilter, 2) - for i := range fs { - fs[i].SetHeaderType(protoacl.HeaderType(rand.Uint32())) - fs[i].SetKey("key_" + strconv.Itoa(rand.Int())) - fs[i].SetMatchType(protoacl.MatchType(rand.Uint32())) - fs[i].SetValue("val_" + strconv.Itoa(rand.Int())) - } - var m protoacl.Record - m.SetAction(a) - m.SetOperation(op) - m.SetTargets(ts) - m.SetFilters(fs) - - r := eacl.NewRecordFromV2(&m) - require.EqualValues(t, a, r.Action()) - require.EqualValues(t, op, r.Operation()) - assertProtoTargetsEqual(t, r.Targets(), ts) - assertProtoFiltersEqual(t, r.Filters(), fs) - - t.Run("nil", func(t *testing.T) { - require.Equal(t, new(eacl.Record), eacl.NewRecordFromV2(nil)) - }) -} - func TestRecord_Marshal(t *testing.T) { for i := range anyValidRecords { require.Equal(t, anyValidBinRecords[i], anyValidRecords[i].Marshal(), i) diff --git a/eacl/table.go b/eacl/table.go index 8cc8f737..6ff849ab 100644 --- a/eacl/table.go +++ b/eacl/table.go @@ -4,19 +4,19 @@ import ( "bytes" "fmt" - v2acl "github.com/nspcc-dev/neofs-api-go/v2/acl" - "github.com/nspcc-dev/neofs-api-go/v2/refs" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" + protoacl "github.com/nspcc-dev/neofs-sdk-go/proto/acl" "github.com/nspcc-dev/neofs-sdk-go/version" ) // Table is a group of ContainerEACL records for single container. // -// Table is compatible with v2 acl.EACLTable message. +// Table is compatible with v2 [protoacl.EACLTable] message. // // Table should be created using one of the constructors. type Table struct { - version version.Version + version *version.Version cid cid.ID records []Record } @@ -25,7 +25,8 @@ type Table struct { // [NewTableForContainer] to limit the NeoFS container. The rs must not be // empty. func ConstructTable(rs []Record) Table { - return Table{version: version.Current(), records: rs} + ver := version.Current() + return Table{version: &ver, records: rs} } // NewTableForContainer constructs new Table with given records which apply only @@ -76,12 +77,15 @@ func (t *Table) SetCID(cid cid.ID) { // Version returns version of eACL format. func (t Table) Version() version.Version { - return t.version + if t.version != nil { + return *t.version + } + return version.Version{} } // SetVersion sets version of eACL format. func (t *Table) SetVersion(version version.Version) { - t.version = version + t.version = &version } // Records returns list of extended ACL rules. @@ -108,18 +112,14 @@ func (t *Table) AddRecord(r *Record) { } } -// ReadFromV2 reads Table from the [v2acl.Table] message. Returns an error if -// the message is malformed according to the NeoFS API V2 protocol. The message -// must not be nil. -// -// ReadFromV2 is intended to be used by the NeoFS API V2 client/server -// implementation only and is not expected to be directly used by applications. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// t from it. // -// See also [Table.ToV2]. -func (t *Table) ReadFromV2(m v2acl.Table) error { +// See also [Table.ProtoMessage]. +func (t *Table) FromProtoMessage(m *protoacl.EACLTable) error { // set container id - if id := m.GetContainerID(); id != nil { - if err := t.cid.ReadFromV2(*id); err != nil { + if m.ContainerId != nil { + if err := t.cid.FromProtoMessage(m.ContainerId); err != nil { return fmt.Errorf("invalid container ID: %w", err) } } else { @@ -127,20 +127,21 @@ func (t *Table) ReadFromV2(m v2acl.Table) error { } // set version - if v := m.GetVersion(); v != nil { - ver := version.Version{} - ver.SetMajor(v.GetMajor()) - ver.SetMinor(v.GetMinor()) - - t.SetVersion(ver) + if m.Version != nil { + if err := t.version.FromProtoMessage(m.Version); err != nil { + return fmt.Errorf("invalid version: %w", err) + } } // set eacl records - v2records := m.GetRecords() - t.records = make([]Record, len(v2records)) + rs := m.Records + t.records = make([]Record, len(rs)) - for i := range v2records { - if err := t.records[i].fromProtoMessage(&v2records[i]); err != nil { + for i := range rs { + if rs[i] == nil { + return fmt.Errorf("nil record #%d", i) + } + if err := t.records[i].fromProtoMessage(rs[i]); err != nil { return fmt.Errorf("invalid record #%d: %w", i, err) } } @@ -148,34 +149,28 @@ func (t *Table) ReadFromV2(m v2acl.Table) error { return nil } -// ToV2 converts Table to v2 acl.EACLTable message. -// -// Nil Table converts to nil. +// ProtoMessage converts t into message to transmit using the NeoFS API +// protocol. // -// See also [Table.ReadFromV2]. -func (t Table) ToV2() *v2acl.Table { - v2 := new(v2acl.Table) - var cidV2 refs.ContainerID - +// See also [Table.FromProtoMessage]. +func (t Table) ProtoMessage() *protoacl.EACLTable { + m := new(protoacl.EACLTable) if !t.cid.IsZero() { - t.cid.WriteToV2(&cidV2) - v2.SetContainerID(&cidV2) + m.ContainerId = t.cid.ProtoMessage() } if t.records != nil { - records := make([]v2acl.Record, len(t.records)) + m.Records = make([]*protoacl.EACLRecord, len(t.records)) for i := range t.records { - records[i] = *t.records[i].toProtoMessage() + m.Records[i] = t.records[i].toProtoMessage() } - - v2.SetRecords(records) } - var verV2 refs.Version - t.version.WriteToV2(&verV2) - v2.SetVersion(&verV2) + if t.version != nil { + m.Version = t.version.ProtoMessage() + } - return v2 + return m } // NewTable creates, initializes and returns blank Table instance. @@ -198,45 +193,9 @@ func CreateTable(cid cid.ID) *Table { return &t } -// NewTableFromV2 converts v2 acl.EACLTable message to Table. -// -// Deprecated: BUG: container ID length is not checked. Use [Table.ReadFromV2] -// instead. -func NewTableFromV2(table *v2acl.Table) *Table { - t := new(Table) - - if table == nil { - return t - } - - // set version - if v := table.GetVersion(); v != nil { - ver := version.Version{} - ver.SetMajor(v.GetMajor()) - ver.SetMinor(v.GetMinor()) - - t.SetVersion(ver) - } - - // set container id - if id := table.GetContainerID(); id != nil { - copy(t.cid[:], id.GetValue()) - } - - // set eacl records - v2records := table.GetRecords() - t.records = make([]Record, len(v2records)) - - for i := range v2records { - _ = t.records[i].fromProtoMessage(&v2records[i]) - } - - return t -} - // Marshal marshals Table into a protobuf binary form. func (t Table) Marshal() []byte { - return t.ToV2().StableMarshal(nil) + return neofsproto.Marshal(t) } // SignedData returns actual payload to sign. @@ -249,26 +208,18 @@ func (t Table) SignedData() []byte { // Unmarshal unmarshals protobuf binary representation of Table. Use [Unmarshal] // to decode data into a new Table. func (t *Table) Unmarshal(data []byte) error { - var m v2acl.Table - if err := m.Unmarshal(data); err != nil { - return err - } - return t.ReadFromV2(m) + return neofsproto.Unmarshal(data, t) } // MarshalJSON encodes Table to protobuf JSON format. func (t Table) MarshalJSON() ([]byte, error) { - return t.ToV2().MarshalJSON() + return neofsproto.MarshalJSON(t) } // UnmarshalJSON decodes Table from protobuf JSON format. Use [UnmarshalJSON] to // decode data into a new Table. func (t *Table) UnmarshalJSON(data []byte) error { - var m v2acl.Table - if err := m.UnmarshalJSON(data); err != nil { - return err - } - return t.ReadFromV2(m) + return neofsproto.UnmarshalJSON(data, t) } // EqualTables compares Table with each other. @@ -278,5 +229,5 @@ func EqualTables(t1, t2 Table) bool { return bytes.Equal(t1.Marshal(), t2.Marsha // IsZero checks whether all fields of the table are zero/empty. The property // can be used as a marker of unset eACL. func (t Table) IsZero() bool { - return t.cid.IsZero() && len(t.records) == 0 && t.version == version.Version{} + return t.cid.IsZero() && len(t.records) == 0 && t.version == nil } diff --git a/eacl/table_test.go b/eacl/table_test.go index 63f56683..adf7356d 100644 --- a/eacl/table_test.go +++ b/eacl/table_test.go @@ -6,36 +6,59 @@ import ( "strconv" "testing" - protoacl "github.com/nspcc-dev/neofs-api-go/v2/acl" - "github.com/nspcc-dev/neofs-api-go/v2/refs" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" cidtest "github.com/nspcc-dev/neofs-sdk-go/container/id/test" "github.com/nspcc-dev/neofs-sdk-go/eacl" - "github.com/nspcc-dev/neofs-sdk-go/version" + protoacl "github.com/nspcc-dev/neofs-sdk-go/proto/acl" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" "github.com/stretchr/testify/require" ) var invalidProtoEACLTestcases = []struct { name string err string - corrupt func(*protoacl.Table) + corrupt func(*protoacl.EACLTable) }{ - {name: "invalid container/nil value", err: "invalid container ID: invalid length 0", - corrupt: func(m *protoacl.Table) { m.SetContainerID(new(refs.ContainerID)) }}, - {name: "invalid container/empty value", err: "invalid container ID: invalid length 0", corrupt: func(m *protoacl.Table) { - var mc refs.ContainerID - mc.SetValue([]byte{}) - m.SetContainerID(&mc) + {name: "container/nil value", err: "invalid container ID: invalid length 0", + corrupt: func(m *protoacl.EACLTable) { m.ContainerId = new(refs.ContainerID) }}, + {name: "container/value/empty", err: "invalid container ID: invalid length 0", corrupt: func(m *protoacl.EACLTable) { + m.ContainerId = &refs.ContainerID{Value: []byte{}} }}, - {name: "invalid container/undersized value", err: "invalid container ID: invalid length 31", corrupt: func(m *protoacl.Table) { - var mc refs.ContainerID - mc.SetValue(make([]byte, 31)) - m.SetContainerID(&mc) + {name: "container/undersized value", err: "invalid container ID: invalid length 31", corrupt: func(m *protoacl.EACLTable) { + m.ContainerId = &refs.ContainerID{Value: make([]byte, 31)} }}, - {name: "invalid container/oversized value", err: "invalid container ID: invalid length 33", corrupt: func(m *protoacl.Table) { - var mc refs.ContainerID - mc.SetValue(make([]byte, 33)) - m.SetContainerID(&mc) + {name: "container/oversized value", err: "invalid container ID: invalid length 33", corrupt: func(m *protoacl.EACLTable) { + m.ContainerId = &refs.ContainerID{Value: make([]byte, 33)} + }}, + {name: "container/zero", err: "invalid container ID: zero container ID", corrupt: func(m *protoacl.EACLTable) { + m.ContainerId = &refs.ContainerID{Value: make([]byte, 32)} + }}, + {name: "record/zero", err: "invalid container ID: zero container ID", corrupt: func(m *protoacl.EACLTable) { + m.ContainerId = &refs.ContainerID{Value: make([]byte, 32)} + }}, + {name: "records/nil element", err: "nil record #1", corrupt: func(m *protoacl.EACLTable) { + m.Records[1] = nil + }}, + {name: "records/negative action", err: "negative action -1", corrupt: func(m *protoacl.EACLTable) { + m.Records[1].Action = -1 + }}, + {name: "records/negative op", err: "negative op -1", corrupt: func(m *protoacl.EACLTable) { + m.Records[1].Operation = -1 + }}, + {name: "records/filters/nil element", err: "invalid record #1: nil filter #1", corrupt: func(m *protoacl.EACLTable) { + m.Records[1].Filters[1] = nil + }}, + {name: "records/filters/negative header type", err: "invalid record #1: invalid filter #1: negative header type -1", corrupt: func(m *protoacl.EACLTable) { + m.Records[1].Filters[1].HeaderType = -1 + }}, + {name: "records/filters/negative match type", err: "invalid record #1: invalid filter #1: negative match type -1", corrupt: func(m *protoacl.EACLTable) { + m.Records[1].Filters[1].MatchType = -1 + }}, + {name: "records/targets/nil element", err: "invalid record #1: nil target #1", corrupt: func(m *protoacl.EACLTable) { + m.Records[1].Targets[1] = nil + }}, + {name: "records/targets/negative role", err: "invalid record #1: invalid target #1: negative role -1", corrupt: func(m *protoacl.EACLTable) { + m.Records[1].Targets[1].Role = -1 }}, } @@ -48,44 +71,6 @@ func TestTable_AddRecord(t *testing.T) { } } -func TestTable_ToV2(t *testing.T) { - assert := func(t testing.TB, tbl eacl.Table, m *protoacl.Table) { - require.EqualValues(t, 2, tbl.Version().Major()) - require.EqualValues(t, 16, tbl.Version().Minor()) - require.Len(t, m.GetRecords(), len(tbl.Records())) - assertProtoRecordsEqual(t, tbl.Records(), m.GetRecords()) - } - - tbl := eacl.ConstructTable(anyValidRecords) - assert(t, tbl, tbl.ToV2()) - - t.Run("with container", func(t *testing.T) { - cnr := cidtest.ID() - tbl = eacl.NewTableForContainer(cnr, anyValidRecords) - m := tbl.ToV2() - assert(t, tbl, m) - require.Equal(t, cnr[:], m.GetContainerID().GetValue()) - }) - t.Run("default values", func(t *testing.T) { - table := eacl.NewTable() - - // check initial values - require.Equal(t, version.Current(), table.Version()) - require.Nil(t, table.Records()) - _, set := table.CID() - require.False(t, set) - - // convert to v2 message - tableV2 := table.ToV2() - - var verV2 refs.Version - version.Current().WriteToV2(&verV2) - require.Equal(t, verV2, *tableV2.GetVersion()) - require.Nil(t, tableV2.GetRecords()) - require.Nil(t, tableV2.GetContainerID()) - }) -} - func TestTable_LimitToContainer(t *testing.T) { cnr := cidtest.ID() var tbl eacl.Table @@ -112,109 +97,57 @@ func TestTable_SetCID(t *testing.T) { require.Equal(t, cnr, tbl.GetCID()) } -func TestNewTableFromV2(t *testing.T) { - var ver refs.Version - ver.SetMajor(rand.Uint32()) - ver.SetMinor(rand.Uint32()) - - bCnr := make([]byte, cid.Size) - //nolint:staticcheck - rand.Read(bCnr) - var cnr refs.ContainerID - cnr.SetValue(bCnr) - - rs := make([]protoacl.Record, 2) - for i := range rs { - ts := make([]protoacl.Target, 2) - for i := range ts { - ts[i].SetRole(protoacl.Role(rand.Uint32())) - ts[i].SetKeys(anyValidBinPublicKeys) - } - fs := make([]protoacl.HeaderFilter, 2) - for i := range fs { - fs[i].SetHeaderType(protoacl.HeaderType(rand.Uint32())) - fs[i].SetKey("key_" + strconv.Itoa(rand.Int())) - fs[i].SetMatchType(protoacl.MatchType(rand.Uint32())) - fs[i].SetValue("val_" + strconv.Itoa(rand.Int())) - } - rs[i].SetAction(protoacl.Action(rand.Uint32())) - rs[i].SetOperation(protoacl.Operation(rand.Uint32())) - rs[i].SetTargets(ts) - rs[i].SetFilters(fs) - } - - var m protoacl.Table - m.SetVersion(&ver) - m.SetContainerID(&cnr) - m.SetRecords(rs) - - tbl := eacl.NewTableFromV2(&m) - require.EqualValues(t, ver.GetMajor(), tbl.Version().Major()) - require.EqualValues(t, ver.GetMinor(), tbl.Version().Minor()) - resCnr, ok := tbl.CID() - require.True(t, ok) - require.Equal(t, bCnr, resCnr[:]) - assertProtoRecordsEqual(t, tbl.Records(), rs) - - t.Run("nil", func(t *testing.T) { - require.Equal(t, new(eacl.Table), eacl.NewTableFromV2(nil)) - }) -} - func TestTable_ReadFromV2(t *testing.T) { - var ver refs.Version - ver.SetMajor(rand.Uint32()) - ver.SetMinor(rand.Uint32()) - - bCnr := make([]byte, cid.Size) + m := &protoacl.EACLTable{ + Version: &refs.Version{ + Major: rand.Uint32(), + Minor: rand.Uint32(), + }, + ContainerId: &refs.ContainerID{Value: make([]byte, cid.Size)}, + Records: []*protoacl.EACLRecord{{}, {}}, + } //nolint:staticcheck - rand.Read(bCnr) - var cnr refs.ContainerID - cnr.SetValue(bCnr) - - rs := make([]protoacl.Record, 2) - for i := range rs { - ts := make([]protoacl.Target, 2) - for i := range ts { - ts[i].SetRole(protoacl.Role(rand.Uint32())) - ts[i].SetKeys(anyValidBinPublicKeys) + rand.Read(m.ContainerId.Value) + + for _, r := range m.Records { + r.Targets = make([]*protoacl.EACLRecord_Target, 2) + for i := range r.Targets { + r.Targets[i] = &protoacl.EACLRecord_Target{ + Role: protoacl.Role(rand.Int31()), + Keys: anyValidBinPublicKeys, + } } - fs := make([]protoacl.HeaderFilter, 2) - for i := range fs { - fs[i].SetHeaderType(protoacl.HeaderType(rand.Uint32())) - fs[i].SetKey("key_" + strconv.Itoa(rand.Int())) - fs[i].SetMatchType(protoacl.MatchType(rand.Uint32())) - fs[i].SetValue("val_" + strconv.Itoa(rand.Int())) + r.Filters = make([]*protoacl.EACLRecord_Filter, 2) + for i := range r.Filters { + r.Filters[i] = &protoacl.EACLRecord_Filter{ + HeaderType: protoacl.HeaderType(rand.Int31()), + MatchType: protoacl.MatchType(rand.Int31()), + Key: "key_" + strconv.Itoa(rand.Int()), + Value: "val_" + strconv.Itoa(rand.Int()), + } } - rs[i].SetAction(protoacl.Action(rand.Uint32())) - rs[i].SetOperation(protoacl.Operation(rand.Uint32())) - rs[i].SetTargets(ts) - rs[i].SetFilters(fs) + r.Action = protoacl.Action(rand.Int31()) + r.Operation = protoacl.Operation(rand.Int31()) } - var m protoacl.Table - m.SetVersion(&ver) - m.SetContainerID(&cnr) - m.SetRecords(rs) - var tbl eacl.Table - require.NoError(t, tbl.ReadFromV2(m)) - require.EqualValues(t, ver.GetMajor(), tbl.Version().Major()) - require.EqualValues(t, ver.GetMinor(), tbl.Version().Minor()) - require.EqualValues(t, bCnr, tbl.GetCID()) - assertProtoRecordsEqual(t, tbl.Records(), rs) - - m.SetContainerID(nil) - require.NoError(t, tbl.ReadFromV2(m)) + require.NoError(t, tbl.FromProtoMessage(m)) + require.EqualValues(t, m.Version.GetMajor(), tbl.Version().Major()) + require.EqualValues(t, m.Version.GetMinor(), tbl.Version().Minor()) + require.EqualValues(t, m.ContainerId.Value, tbl.GetCID()) + assertProtoRecordsEqual(t, tbl.Records(), m.Records) + + m.ContainerId = nil + require.NoError(t, tbl.FromProtoMessage(m)) require.Zero(t, tbl.GetCID()) t.Run("invalid", func(t *testing.T) { for _, tc := range invalidProtoEACLTestcases { t.Run(tc.name, func(t *testing.T) { - m := anyValidEACL.ToV2() + m := anyValidEACL.ProtoMessage() require.NotNil(t, m) tc.corrupt(m) - require.EqualError(t, new(eacl.Table).ReadFromV2(*m), tc.err) + require.EqualError(t, new(eacl.Table).FromProtoMessage(m), tc.err) }) } }) diff --git a/eacl/target.go b/eacl/target.go index a029e935..52ca27f0 100644 --- a/eacl/target.go +++ b/eacl/target.go @@ -3,10 +3,12 @@ package eacl import ( "bytes" "crypto/ecdsa" + "fmt" "github.com/nspcc-dev/neo-go/pkg/crypto/keys" "github.com/nspcc-dev/neo-go/pkg/util" - v2acl "github.com/nspcc-dev/neofs-api-go/v2/acl" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" + protoacl "github.com/nspcc-dev/neofs-sdk-go/proto/acl" "github.com/nspcc-dev/neofs-sdk-go/user" ) @@ -200,28 +202,19 @@ func (t Target) Role() Role { return t.role } -// ToV2 converts Target to v2 acl.EACLRecord.Target message. -// -// Nil Target converts to nil. -// Deprecated: do not use it. -func (t *Target) ToV2() *v2acl.Target { - if t != nil { - return t.toProtoMessage() +func (t Target) protoMessage() *protoacl.EACLRecord_Target { + return &protoacl.EACLRecord_Target{ + Role: protoacl.Role(t.role), + Keys: t.subjs, } - return nil -} - -func (t Target) toProtoMessage() *v2acl.Target { - target := new(v2acl.Target) - target.SetRole(v2acl.Role(t.role)) - target.SetKeys(t.subjs) - - return target } -func (t *Target) fromProtoMessage(m *v2acl.Target) error { - t.role = Role(m.GetRole()) - t.subjs = m.GetKeys() +func (t *Target) fromProtoMessage(m *protoacl.EACLRecord_Target) error { + if m.Role < 0 { + return fmt.Errorf("negative role %d", m.Role) + } + t.role = Role(m.Role) + t.subjs = m.Keys return nil } @@ -234,23 +227,15 @@ func (t *Target) fromProtoMessage(m *v2acl.Target) error { // Deprecated: use [NewTargetByRole] or [TargetByPublicKeys] instead. func NewTarget() *Target { return new(Target) } -// NewTargetFromV2 converts v2 acl.EACLRecord.Target message to Target. -// Deprecated: do not use it. -func NewTargetFromV2(target *v2acl.Target) *Target { - t := new(Target) - _ = t.fromProtoMessage(target) - return t -} - // Marshal marshals Target into a protobuf binary form. func (t Target) Marshal() []byte { - return t.toProtoMessage().StableMarshal(nil) + return neofsproto.MarshalMessage(t.protoMessage()) } // Unmarshal unmarshals protobuf binary representation of Target. func (t *Target) Unmarshal(data []byte) error { - m := new(v2acl.Target) - if err := m.Unmarshal(data); err != nil { + m := new(protoacl.EACLRecord_Target) + if err := neofsproto.UnmarshalMessage(data, m); err != nil { return err } return t.fromProtoMessage(m) @@ -258,13 +243,13 @@ func (t *Target) Unmarshal(data []byte) error { // MarshalJSON encodes Target to protobuf JSON format. func (t Target) MarshalJSON() ([]byte, error) { - return t.toProtoMessage().MarshalJSON() + return neofsproto.MarshalMessageJSON(t.protoMessage()) } // UnmarshalJSON decodes Target from protobuf JSON format. func (t *Target) UnmarshalJSON(data []byte) error { - m := new(v2acl.Target) - if err := m.UnmarshalJSON(data); err != nil { + m := new(protoacl.EACLRecord_Target) + if err := neofsproto.UnmarshalMessageJSON(data, m); err != nil { return err } return t.fromProtoMessage(m) diff --git a/eacl/target_test.go b/eacl/target_test.go index e6632d19..8db84ffd 100644 --- a/eacl/target_test.go +++ b/eacl/target_test.go @@ -7,57 +7,12 @@ import ( "github.com/nspcc-dev/neo-go/pkg/crypto/hash" "github.com/nspcc-dev/neo-go/pkg/util" - protoacl "github.com/nspcc-dev/neofs-api-go/v2/acl" "github.com/nspcc-dev/neofs-sdk-go/eacl" "github.com/nspcc-dev/neofs-sdk-go/user" usertest "github.com/nspcc-dev/neofs-sdk-go/user/test" "github.com/stretchr/testify/require" ) -func TestTarget_ToV2(t *testing.T) { - r := eacl.NewTargetByRole(anyValidRole) - subjs := [][]byte{ - anyValidECDSABinPublicKeys[0], - anyUserSet[0][:], - anyValidECDSABinPublicKeys[1], - anyUserSet[1][:], - anyUserSet[2][:], - } - r.SetRawSubjects(subjs) - m := r.ToV2() - require.EqualValues(t, anyValidRole, m.GetRole()) - require.Equal(t, subjs, m.GetKeys()) - - t.Run("default values", func(t *testing.T) { - target := eacl.NewTarget() - - // check initial values - require.Zero(t, target.Role()) - require.Nil(t, target.BinaryKeys()) - - // convert to v2 message - targetV2 := target.ToV2() - - require.Equal(t, protoacl.RoleUnknown, targetV2.GetRole()) - require.Nil(t, targetV2.GetKeys()) - }) -} - -func TestNewTargetFromV2(t *testing.T) { - role := protoacl.Role(rand.Uint32()) - var m protoacl.Target - m.SetRole(role) - m.SetKeys(anyValidBinPublicKeys) - - r := eacl.NewTargetFromV2(&m) - require.EqualValues(t, role, r.Role()) - require.Equal(t, anyValidBinPublicKeys, m.GetKeys()) - - t.Run("nil", func(t *testing.T) { - require.Equal(t, new(eacl.Target), eacl.NewTargetFromV2(nil)) - }) -} - func TestTarget_Marshal(t *testing.T) { for i := range anyValidTargets { require.Equal(t, anyValidBinTargets[i], anyValidTargets[i].Marshal()) diff --git a/eacl/test/generate.go b/eacl/test/generate.go index 93cbc8f0..a012a051 100644 --- a/eacl/test/generate.go +++ b/eacl/test/generate.go @@ -12,7 +12,7 @@ import ( // Target returns random eacl.Target. func Target() eacl.Target { if rand.Int()%2 == 0 { - return eacl.NewTargetByRole(eacl.Role(rand.Uint32())) + return eacl.NewTargetByRole(eacl.Role(rand.Int32())) } return eacl.NewTargetByAccounts(usertest.IDs(1 + rand.N(10))) } @@ -29,7 +29,7 @@ func Targets(n int) []eacl.Target { // Filter returns random eacl.Filter. func Filter() eacl.Filter { si := strconv.Itoa(rand.Int()) - return eacl.ConstructFilter(eacl.FilterHeaderType(rand.Uint32()), "key_"+si, eacl.Match(rand.Uint32()), "val_"+si) + return eacl.ConstructFilter(eacl.FilterHeaderType(rand.Int32()), "key_"+si, eacl.Match(rand.Int32()), "val_"+si) } // Filters returns n random eacl.Filter instances. @@ -45,7 +45,7 @@ func Filters(n int) []eacl.Filter { func Record() eacl.Record { tn := 1 + rand.N(10) fn := 1 + rand.N(10) - return eacl.ConstructRecord(eacl.Action(rand.Uint32()), eacl.Operation(rand.Uint32()), Targets(tn), Filters(fn)...) + return eacl.ConstructRecord(eacl.Action(rand.Int32()), eacl.Operation(rand.Int32()), Targets(tn), Filters(fn)...) } // Records returns n random eacl.Record instances. diff --git a/eacl/types_test.go b/eacl/types_test.go index 53d9a5de..538e5ed1 100644 --- a/eacl/types_test.go +++ b/eacl/types_test.go @@ -15,13 +15,13 @@ func TestValidationUnit_WithContainerID(t *testing.T) { } func TestValidationUnit_WithRole(t *testing.T) { - role := Role(rand.Uint32()) + role := Role(rand.Int32()) u := new(ValidationUnit).WithRole(role) require.Equal(t, role, u.role) } func TestValidationUnit_WithOperation(t *testing.T) { - op := Operation(rand.Uint32()) + op := Operation(rand.Int32()) u := new(ValidationUnit).WithOperation(op) require.Equal(t, op, u.op) } @@ -40,8 +40,8 @@ func TestValidationUnit_WithSenderKey(t *testing.T) { func TestValidationUnit_WithEACLTable(t *testing.T) { eACL := NewTableForContainer(cidtest.ID(), []Record{ - ConstructRecord(Action(rand.Uint32()), Operation(rand.Uint32()), []Target{ - NewTargetByRole(Role(rand.Uint32())), + ConstructRecord(Action(rand.Int32()), Operation(rand.Int32()), []Target{ + NewTargetByRole(Role(rand.Int32())), }), }) u := new(ValidationUnit).WithEACLTable(&eACL) diff --git a/internal/proto/encoding.go b/internal/proto/encoding.go index b8121a7f..05508652 100644 --- a/internal/proto/encoding.go +++ b/internal/proto/encoding.go @@ -23,19 +23,90 @@ type Message interface { MarshalStable(b []byte) } -// MarshalMessage returns m encoded to dynamically allocated buffer. +// MarshalMessage encodes m into a dynamically allocated buffer. func MarshalMessage(m Message) []byte { b := make([]byte, m.MarshaledSize()) m.MarshalStable(b) return b } +// Marshal encodes v transmitted via NeoFS API protocol into a dynamically +// allocated buffer. +func Marshal[M Message, T interface{ ProtoMessage() M }](v T) []byte { + return MarshalMessage(v.ProtoMessage()) +} + +// UnmarshalMessage decodes m from b. +func UnmarshalMessage(b []byte, m proto.Message) error { return proto.Unmarshal(b, m) } + +// Unmarshal decodes v transmitted via NeoFS API protocol from b. +func Unmarshal[M_ any, M interface { + *M_ + proto.Message + Message +}, T interface{ FromProtoMessage(M) error }](b []byte, v T) error { + m := M(new(M_)) + if err := UnmarshalMessage(b, m); err != nil { + return err + } + return v.FromProtoMessage(m) +} + +// UnmarshalOptional decodes v from [protojson] with ignored missing required +// fields. +func UnmarshalOptional[M_ any, M interface { + *M_ + proto.Message + Message +}, T interface{ FromProtoMessage(M) error }](b []byte, v T, dec func(T, M, bool) error) error { + m := M(new(M_)) + if err := UnmarshalMessage(b, m); err != nil { + return err + } + return dec(v, m, false) +} + var jOpts = protojson.MarshalOptions{EmitUnpopulated: true} -// MarshalMessageJSON encodes m into [protojson] with unpopulated fields +// MarshalMessageJSON encodes m into [protojson] with unpopulated fields' // emission. func MarshalMessageJSON(m proto.Message) ([]byte, error) { return jOpts.Marshal(m) } +// MarshalJSON encodes v into [protojson] with unpopulated fields' emission. +func MarshalJSON[M proto.Message, T interface{ ProtoMessage() M }](v T) ([]byte, error) { + return MarshalMessageJSON(v.ProtoMessage()) +} + +// UnmarshalMessageJSON decodes m from [protojson]. +func UnmarshalMessageJSON(b []byte, m proto.Message) error { return protojson.Unmarshal(b, m) } + +// UnmarshalJSON decodes v from [protojson]. +func UnmarshalJSON[M_ any, M interface { + *M_ + proto.Message + Message +}, T interface{ FromProtoMessage(M) error }](b []byte, v T) error { + m := M(new(M_)) + if err := UnmarshalMessageJSON(b, m); err != nil { + return err + } + return v.FromProtoMessage(m) +} + +// UnmarshalJSONOptional decodes v from [protojson] with ignored missing +// required fields. +func UnmarshalJSONOptional[M_ any, M interface { + *M_ + proto.Message + Message +}, T interface{ FromProtoMessage(M) error }](b []byte, v T, dec func(T, M, bool) error) error { + m := M(new(M_)) + if err := UnmarshalMessageJSON(b, m); err != nil { + return err + } + return dec(v, m, false) +} + // Bytes is a type parameter constraint for any byte arrays. type Bytes interface{ ~[]byte | ~string } diff --git a/netmap/context.go b/netmap/context.go index bb280b1f..7a0e04d0 100644 --- a/netmap/context.go +++ b/netmap/context.go @@ -4,7 +4,6 @@ import ( "errors" "github.com/nspcc-dev/hrw/v2" - "github.com/nspcc-dev/neofs-api-go/v2/netmap" ) // context of a placement build process. @@ -13,10 +12,10 @@ type context struct { netMap NetMap // cache of processed filters - processedFilters map[string]*netmap.Filter + processedFilters map[string]*Filter // cache of processed selectors - processedSelectors map[string]*netmap.Selector + processedSelectors map[string]*Selector // stores results of selector processing selections map[string][]nodes @@ -55,8 +54,8 @@ var ( func newContext(nm NetMap) *context { return &context{ netMap: nm, - processedFilters: make(map[string]*netmap.Filter), - processedSelectors: make(map[string]*netmap.Selector), + processedFilters: make(map[string]*Filter), + processedSelectors: make(map[string]*Selector), selections: make(map[string][]nodes), numCache: make(map[string]uint64), diff --git a/netmap/example_test.go b/netmap/example_test.go index bb87378f..6648b0a6 100644 --- a/netmap/example_test.go +++ b/netmap/example_test.go @@ -3,24 +3,21 @@ package netmap_test import ( "fmt" - apiGoNetmap "github.com/nspcc-dev/neofs-api-go/v2/netmap" "github.com/nspcc-dev/neofs-sdk-go/netmap" ) // Instances can be also used to process NeoFS API V2 protocol messages with [https://github.com/nspcc-dev/neofs-api] package. func ExampleNodeInfo_marshalling() { - // import apiGoNetmap "github.com/nspcc-dev/neofs-api-go/v2/netmap" // On the client side. var info netmap.NodeInfo - var msg apiGoNetmap.NodeInfo - info.WriteToV2(&msg) + msg := info.ProtoMessage() // *send message* // On the server side. - _ = info.ReadFromV2(msg) + _ = info.FromProtoMessage(msg) } // When forming information about storage node to be registered the NeoFS diff --git a/netmap/filter.go b/netmap/filter.go index bd229cad..126d743f 100644 --- a/netmap/filter.go +++ b/netmap/filter.go @@ -3,8 +3,6 @@ package netmap import ( "fmt" "strconv" - - "github.com/nspcc-dev/neofs-api-go/v2/netmap" ) // mainFilterName is a name of the filter @@ -15,15 +13,15 @@ const mainFilterName = "*" func (c *context) processFilters(p PlacementPolicy) error { for i := range p.filters { if err := c.processFilter(p.filters[i], true); err != nil { - return fmt.Errorf("process filter #%d (%s): %w", i, p.filters[i].GetName(), err) + return fmt.Errorf("process filter #%d (%s): %w", i, p.filters[i].Name(), err) } } return nil } -func (c *context) processFilter(f netmap.Filter, top bool) error { - fName := f.GetName() +func (c *context) processFilter(f Filter, top bool) error { + fName := f.Name() if fName == mainFilterName { return fmt.Errorf("%w: '%s' is reserved", errInvalidFilterName, mainFilterName) } @@ -36,10 +34,10 @@ func (c *context) processFilter(f netmap.Filter, top bool) error { return errFilterNotFound } - inner := f.GetFilters() + inner := f.SubFilters() - switch op := f.GetOp(); op { - case netmap.AND, netmap.OR: + switch op := f.Op(); op { + case FilterOpAND, FilterOpOR: for i := range inner { if err := c.processFilter(inner[i], false); err != nil { return fmt.Errorf("process inner filter #%d: %w", i, err) @@ -53,12 +51,12 @@ func (c *context) processFilter(f netmap.Filter, top bool) error { } switch op { - case netmap.EQ, netmap.NE: - case netmap.GT, netmap.GE, netmap.LT, netmap.LE: - val := f.GetValue() + case FilterOpEQ, FilterOpNE: + case FilterOpGT, FilterOpGE, FilterOpLT, FilterOpLE: + val := f.Value() n, err := strconv.ParseUint(val, 10, 64) if err != nil { - return fmt.Errorf("%w: '%s'", errInvalidNumber, f.GetValue()) + return fmt.Errorf("%w: '%s'", errInvalidNumber, val) } c.numCache[val] = n @@ -77,38 +75,41 @@ func (c *context) processFilter(f netmap.Filter, top bool) error { // match matches f against b. It returns no errors because // filter should have been parsed during context creation // and missing node properties are considered as a regular fail. -func (c *context) match(f *netmap.Filter, b NodeInfo) bool { - switch f.GetOp() { - case netmap.AND, netmap.OR: - inner := f.GetFilters() +func (c *context) match(f *Filter, b NodeInfo) bool { + if f == nil { + return false + } + switch f.Op() { + case FilterOpAND, FilterOpOR: + inner := f.SubFilters() for i := range inner { fSub := &inner[i] - if name := inner[i].GetName(); name != "" { + if name := inner[i].Name(); name != "" { fSub = c.processedFilters[name] } ok := c.match(fSub, b) - if ok == (f.GetOp() == netmap.OR) { + if ok == (f.Op() == FilterOpOR) { return ok } } - return f.GetOp() == netmap.AND + return f.Op() == FilterOpAND default: - return c.matchKeyValue(f, b) + return c.matchKeyValue(*f, b) } } -func (c *context) matchKeyValue(f *netmap.Filter, b NodeInfo) bool { - switch op := f.GetOp(); op { - case netmap.EQ: - return b.Attribute(f.GetKey()) == f.GetValue() - case netmap.NE: - return b.Attribute(f.GetKey()) != f.GetValue() +func (c *context) matchKeyValue(f Filter, b NodeInfo) bool { + switch op := f.Op(); op { + case FilterOpEQ: + return b.Attribute(f.Key()) == f.Value() + case FilterOpNE: + return b.Attribute(f.Key()) != f.Value() default: var attr uint64 - switch f.GetKey() { + switch f.Key() { case attrPrice: attr = b.Price() case attrCapacity: @@ -116,7 +117,7 @@ func (c *context) matchKeyValue(f *netmap.Filter, b NodeInfo) bool { default: var err error - attr, err = strconv.ParseUint(b.Attribute(f.GetKey()), 10, 64) + attr, err = strconv.ParseUint(b.Attribute(f.Key()), 10, 64) if err != nil { // Note: because filters are somewhat independent from nodes attributes, // We don't report an error here, and fail filter instead. @@ -125,14 +126,14 @@ func (c *context) matchKeyValue(f *netmap.Filter, b NodeInfo) bool { } switch op { - case netmap.GT: - return attr > c.numCache[f.GetValue()] - case netmap.GE: - return attr >= c.numCache[f.GetValue()] - case netmap.LT: - return attr < c.numCache[f.GetValue()] - case netmap.LE: - return attr <= c.numCache[f.GetValue()] + case FilterOpGT: + return attr > c.numCache[f.Value()] + case FilterOpGE: + return attr >= c.numCache[f.Value()] + case FilterOpLT: + return attr < c.numCache[f.Value()] + case FilterOpLE: + return attr <= c.numCache[f.Value()] default: // do nothing and return false } diff --git a/netmap/filter_test.go b/netmap/filter_test.go index fc300737..bb057d08 100644 --- a/netmap/filter_test.go +++ b/netmap/filter_test.go @@ -4,17 +4,16 @@ import ( "errors" "testing" - "github.com/nspcc-dev/neofs-api-go/v2/netmap" "github.com/stretchr/testify/require" ) func TestContext_ProcessFilters(t *testing.T) { fs := []Filter{ - newFilter("StorageSSD", "Storage", "SSD", netmap.EQ), - newFilter("GoodRating", "Rating", "4", netmap.GE), - newFilter("Main", "", "", netmap.AND, + newFilter("StorageSSD", "Storage", "SSD", FilterOpEQ), + newFilter("GoodRating", "Rating", "4", FilterOpGE), + newFilter("Main", "", "", FilterOpAND, newFilter("StorageSSD", "", "", 0), - newFilter("", "IntField", "123", netmap.LT), + newFilter("", "IntField", "123", FilterOpLT), newFilter("GoodRating", "", "", 0)), } @@ -23,11 +22,11 @@ func TestContext_ProcessFilters(t *testing.T) { require.NoError(t, c.processFilters(p)) require.Equal(t, 3, len(c.processedFilters)) for _, f := range fs { - require.Equal(t, f.m, *c.processedFilters[f.m.GetName()]) + require.Equal(t, f, *c.processedFilters[f.Name()]) } - require.Equal(t, uint64(4), c.numCache[fs[1].m.GetValue()]) - require.Equal(t, uint64(123), c.numCache[fs[2].m.GetFilters()[1].GetValue()]) + require.Equal(t, uint64(4), c.numCache[fs[1].Value()]) + require.Equal(t, uint64(123), c.numCache[fs[2].SubFilters()[1].Value()]) } func TestContext_ProcessFiltersInvalid(t *testing.T) { @@ -38,24 +37,24 @@ func TestContext_ProcessFiltersInvalid(t *testing.T) { }{ { "UnnamedTop", - newFilter("", "Storage", "SSD", netmap.EQ), + newFilter("", "Storage", "SSD", FilterOpEQ), errUnnamedTopFilter, }, { "InvalidReference", - newFilter("Main", "", "", netmap.AND, + newFilter("Main", "", "", FilterOpAND, newFilter("StorageSSD", "", "", 0)), errFilterNotFound, }, { "NonEmptyKeyed", - newFilter("Main", "Storage", "SSD", netmap.EQ, + newFilter("Main", "Storage", "SSD", FilterOpEQ, newFilter("StorageSSD", "", "", 0)), errNonEmptyFilters, }, { "InvalidNumber", - newFilter("Main", "Rating", "three", netmap.GE), + newFilter("Main", "Rating", "three", FilterOpGE), errInvalidNumber, }, { @@ -65,7 +64,7 @@ func TestContext_ProcessFiltersInvalid(t *testing.T) { }, { "InvalidName", - newFilter("*", "Rating", "3", netmap.GE), + newFilter("*", "Rating", "3", FilterOpGE), errInvalidFilterName, }, } @@ -84,12 +83,12 @@ func TestFilter_MatchSimple_InvalidOp(t *testing.T) { b.SetAttribute("Rating", "4") b.SetAttribute("Country", "Germany") - f := newFilter("Main", "Rating", "5", netmap.EQ) + f := newFilter("Main", "Rating", "5", FilterOpEQ) c := newContext(NetMap{}) p := newPlacementPolicy(1, nil, nil, []Filter{f}) require.NoError(t, c.processFilters(p)) // just for the coverage - f.m.SetOp(0) - require.False(t, c.match(&f.m, b)) + f.op = 0 + require.False(t, c.match(&f, b)) } diff --git a/netmap/helper_test.go b/netmap/helper_test.go index 0eff22af..e84ff3db 100644 --- a/netmap/helper_test.go +++ b/netmap/helper_test.go @@ -1,19 +1,11 @@ package netmap -import ( - "github.com/nspcc-dev/neofs-api-go/v2/netmap" -) - -func newFilter(name string, k, v string, op netmap.Operation, fs ...Filter) (f Filter) { +func newFilter(name string, k, v string, op FilterOp, fs ...Filter) (f Filter) { f.SetName(name) - f.m.SetKey(k) - f.m.SetOp(op) - f.m.SetValue(v) - inner := make([]netmap.Filter, len(fs)) - for i := range fs { - inner[i] = fs[i].m - } - f.m.SetFilters(inner) + f.key = k + f.op = op + f.val = v + f.subs = fs return f } diff --git a/netmap/netmap.go b/netmap/netmap.go index 0f3249a2..b69080a7 100644 --- a/netmap/netmap.go +++ b/netmap/netmap.go @@ -5,16 +5,16 @@ import ( "slices" "github.com/nspcc-dev/hrw/v2" - "github.com/nspcc-dev/neofs-api-go/v2/netmap" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" ) // NetMap represents NeoFS network map. It includes information about all // storage nodes registered in NeoFS the network. // -// NetMap is mutually compatible with github.com/nspcc-dev/neofs-api-go/v2/netmap.NetMap -// message. See ReadFromV2 / WriteToV2 methods. +// NetMap is mutually compatible with [protonetmap.Netmap] message. See +// [NetMap.FromProtoMessage] / [NetMap.ProtoMessage] methods. // // Instances can be created using built-in var declaration. type NetMap struct { @@ -23,50 +23,51 @@ type NetMap struct { nodes []NodeInfo } -// ReadFromV2 reads NetMap from the netmap.NetMap message. Checks if the -// message conforms to NeoFS API V2 protocol. +// FromProtoMessage validates msg according to the NeoFS API protocol and +// restores m from it. // -// See also WriteToV2. -func (m *NetMap) ReadFromV2(msg netmap.NetMap) error { +// See also [NetMap.ProtoMessage]. +func (m *NetMap) FromProtoMessage(msg *protonetmap.Netmap) error { var err error - nodes := msg.Nodes() - if nodes == nil { + if msg.Nodes == nil { m.nodes = nil } else { - m.nodes = make([]NodeInfo, len(nodes)) + m.nodes = make([]NodeInfo, len(msg.Nodes)) - for i := range nodes { - err = m.nodes[i].ReadFromV2(nodes[i]) + for i := range msg.Nodes { + if msg.Nodes[i] == nil { + return fmt.Errorf("nil node info #%d", i) + } + err = m.nodes[i].FromProtoMessage(msg.Nodes[i]) if err != nil { return fmt.Errorf("invalid node info: %w", err) } } } - m.epoch = msg.Epoch() + m.epoch = msg.Epoch return nil } -// WriteToV2 writes NetMap to the netmap.NetMap message. The message -// MUST NOT be nil. +// ProtoMessage converts m into message to transmit using the NeoFS API +// protocol. // -// See also ReadFromV2. -func (m NetMap) WriteToV2(msg *netmap.NetMap) { - var nodes []netmap.NodeInfo - +// See also [NetMap.FromProtoMessage]. +func (m NetMap) ProtoMessage() *protonetmap.Netmap { + msg := &protonetmap.Netmap{ + Epoch: m.epoch, + } if m.nodes != nil { - nodes = make([]netmap.NodeInfo, len(m.nodes)) + msg.Nodes = make([]*protonetmap.NodeInfo, len(m.nodes)) for i := range m.nodes { - m.nodes[i].WriteToV2(&nodes[i]) + msg.Nodes[i] = m.nodes[i].ProtoMessage() } - - msg.SetNodes(nodes) } - msg.SetEpoch(m.epoch) + return msg } // SetNodes sets information list about all storage nodes from the NeoFS network. @@ -196,12 +197,12 @@ func (m NetMap) ContainerNodes(p PlacementPolicy, containerID cid.ID) ([][]NodeI result := make([][]NodeInfo, len(p.replicas)) for i := range p.replicas { - sName := p.replicas[i].GetSelector() + sName := p.replicas[i].SelectorName() if sName == "" { if len(p.selectors) == 0 { - var s netmap.Selector - s.SetCount(p.replicas[i].GetCount()) - s.SetFilter(mainFilterName) + var s Selector + s.SetNumberOfNodes(p.replicas[i].NumberOfObjects()) + s.SetFilterName(mainFilterName) nodes, err := c.getSelection(p, s) if err != nil { @@ -212,7 +213,7 @@ func (m NetMap) ContainerNodes(p PlacementPolicy, containerID cid.ID) ([][]NodeI } for i := range p.selectors { - result[i] = append(result[i], flattenNodes(c.selections[p.selectors[i].GetName()])...) + result[i] = append(result[i], flattenNodes(c.selections[p.selectors[i].Name()])...) } continue diff --git a/netmap/netmap_test.go b/netmap/netmap_test.go index 9513d6e7..623bc8fe 100644 --- a/netmap/netmap_test.go +++ b/netmap/netmap_test.go @@ -3,8 +3,8 @@ package netmap_test import ( "testing" - apinetmap "github.com/nspcc-dev/neofs-api-go/v2/netmap" "github.com/nspcc-dev/neofs-sdk-go/netmap" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" "github.com/stretchr/testify/require" ) @@ -56,32 +56,33 @@ func init() { validNetmap.SetNodes(anyValidNodes) } -func TestNetMap_ReadFromV2(t *testing.T) { - mns := make([]apinetmap.NodeInfo, 2) - mns[0].SetPublicKey([]byte("public_key_0")) - mns[1].SetPublicKey([]byte("public_key_1")) - mns[0].SetAddresses("endpoint_0_0", "endpoint_0_1") - mns[1].SetAddresses("endpoint_1_0", "endpoint_1_1") - mns[0].SetState(apinetmap.Offline) - mns[1].SetState(apinetmap.Maintenance) - - addAttr := func(m *apinetmap.NodeInfo, k, v string) { - var a apinetmap.Attribute - a.SetKey(k) - a.SetValue(v) - m.SetAttributes(append(m.GetAttributes(), a)) +func TestNetMap_FromProtoMessage(t *testing.T) { + m := &protonetmap.Netmap{ + Epoch: anyValidCurrentEpoch, + Nodes: []*protonetmap.NodeInfo{ + { + PublicKey: []byte("public_key_0"), + Addresses: []string{"endpoint_0_0", "endpoint_0_1"}, + Attributes: []*protonetmap.NodeInfo_Attribute{ + {Key: "k_0_0", Value: "v_0_0"}, + {Key: "k_0_1", Value: "v_0_1"}, + }, + State: protonetmap.NodeInfo_OFFLINE, + }, + { + PublicKey: []byte("public_key_1"), + Addresses: []string{"endpoint_1_0", "endpoint_1_1"}, + Attributes: []*protonetmap.NodeInfo_Attribute{ + {Key: "k_1_0", Value: "v_1_0"}, + {Key: "k_1_1", Value: "v_1_1"}, + }, + State: protonetmap.NodeInfo_MAINTENANCE, + }, + }, } - addAttr(&mns[0], "k_0_0", "v_0_0") - addAttr(&mns[0], "k_0_1", "v_0_1") - addAttr(&mns[1], "k_1_0", "v_1_0") - addAttr(&mns[1], "k_1_1", "v_1_1") - - var m apinetmap.NetMap - m.SetEpoch(anyValidCurrentEpoch) - m.SetNodes(mns) var val netmap.NetMap - require.NoError(t, val.ReadFromV2(m)) + require.NoError(t, val.FromProtoMessage(m)) require.EqualValues(t, anyValidCurrentEpoch, val.Epoch()) ns := val.Nodes() @@ -123,59 +124,59 @@ func TestNetMap_ReadFromV2(t *testing.T) { }, collectedAttrs) // reset optional fields - m.SetEpoch(0) - m.SetNodes(nil) + m.Epoch = 0 + m.Nodes = nil val2 := val - require.NoError(t, val2.ReadFromV2(m)) + require.NoError(t, val2.FromProtoMessage(m)) require.Zero(t, val2.Epoch()) require.Zero(t, val2.Nodes()) t.Run("invalid", func(t *testing.T) { for _, tc := range []struct { name, err string - corrupt func(netMap *apinetmap.NetMap) + corrupt func(netMap *protonetmap.Netmap) }{ + {name: "nodes/nil", err: "nil node info #1", + corrupt: func(m *protonetmap.Netmap) { m.Nodes[1] = nil }}, {name: "nodes/public key/nil", err: "invalid node info: missing public key", - corrupt: func(m *apinetmap.NetMap) { m.Nodes()[1].SetPublicKey(nil) }}, + corrupt: func(m *protonetmap.Netmap) { m.Nodes[1].PublicKey = nil }}, {name: "nodes/public key/empty", err: "invalid node info: missing public key", - corrupt: func(m *apinetmap.NetMap) { m.Nodes()[1].SetPublicKey([]byte{}) }}, + corrupt: func(m *protonetmap.Netmap) { m.Nodes[1].PublicKey = []byte{} }}, {name: "nodes/endpoints/empty", err: "invalid node info: missing network endpoints", - corrupt: func(m *apinetmap.NetMap) { m.Nodes()[1].SetAddresses() }}, + corrupt: func(m *protonetmap.Netmap) { m.Nodes[1].Addresses = nil }}, {name: "nodes/attributes/no key", err: "invalid node info: empty key of the attribute #1", - corrupt: func(m *apinetmap.NetMap) { setNodeAttributes(&m.Nodes()[1], "k1", "v1", "", "v2") }}, + corrupt: func(m *protonetmap.Netmap) { setNodeAttributes(m.Nodes[1], "k1", "v1", "", "v2") }}, {name: "nodes/attributes/no value", err: "invalid node info: empty value of the attribute k2", - corrupt: func(m *apinetmap.NetMap) { setNodeAttributes(&m.Nodes()[1], "k1", "v1", "k2", "") }}, + corrupt: func(m *protonetmap.Netmap) { setNodeAttributes(m.Nodes[1], "k1", "v1", "k2", "") }}, {name: "nodes/attributes/duplicated", err: "invalid node info: duplicated attribute k1", - corrupt: func(m *apinetmap.NetMap) { setNodeAttributes(&m.Nodes()[1], "k1", "v1", "k2", "v2", "k1", "v3") }}, + corrupt: func(m *protonetmap.Netmap) { setNodeAttributes(m.Nodes[1], "k1", "v1", "k2", "v2", "k1", "v3") }}, {name: "nodes/attributes/capacity", err: "invalid node info: invalid Capacity attribute: strconv.ParseUint: parsing \"foo\": invalid syntax", - corrupt: func(m *apinetmap.NetMap) { setNodeAttributes(&m.Nodes()[1], "Capacity", "foo") }}, + corrupt: func(m *protonetmap.Netmap) { setNodeAttributes(m.Nodes[1], "Capacity", "foo") }}, {name: "nodes/attributes/price", err: "invalid node info: invalid Price attribute: strconv.ParseUint: parsing \"foo\": invalid syntax", - corrupt: func(m *apinetmap.NetMap) { setNodeAttributes(&m.Nodes()[1], "Price", "foo") }}, + corrupt: func(m *protonetmap.Netmap) { setNodeAttributes(m.Nodes[1], "Price", "foo") }}, } { t.Run(tc.name, func(t *testing.T) { st := val - var m apinetmap.NetMap - st.WriteToV2(&m) - tc.corrupt(&m) - require.EqualError(t, new(netmap.NetMap).ReadFromV2(m), tc.err) + m := st.ProtoMessage() + tc.corrupt(m) + require.EqualError(t, new(netmap.NetMap).FromProtoMessage(m), tc.err) }) } }) } -func TestNetMap_WriteToV2(t *testing.T) { +func TestNetMap_ProtoMessage(t *testing.T) { var val netmap.NetMap - var m apinetmap.NetMap // zero - val.WriteToV2(&m) - require.Zero(t, m.Epoch()) - require.Zero(t, m.Nodes()) + m := val.ProtoMessage() + require.Zero(t, m.GetEpoch()) + require.Zero(t, m.GetNodes()) // filled - validNetmap.WriteToV2(&m) - require.EqualValues(t, anyValidCurrentEpoch, m.Epoch()) - ns := m.Nodes() + m = validNetmap.ProtoMessage() + require.EqualValues(t, anyValidCurrentEpoch, m.GetEpoch()) + ns := m.GetNodes() require.Len(t, ns, 2) require.EqualValues(t, "public_key_0", ns[0].GetPublicKey()) require.EqualValues(t, "public_key_1", ns[1].GetPublicKey()) diff --git a/netmap/network_info.go b/netmap/network_info.go index 762e1c36..e1faf15f 100644 --- a/netmap/network_info.go +++ b/netmap/network_info.go @@ -1,50 +1,63 @@ package netmap import ( - "bytes" "encoding/binary" "errors" "fmt" "math" "github.com/nspcc-dev/neo-go/pkg/vm/stackitem" - "github.com/nspcc-dev/neofs-api-go/v2/netmap" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" ) // NetworkInfo groups information about the NeoFS network state. Mainly used to // describe the current state of the network. // -// NetworkInfo is mutually compatible with github.com/nspcc-dev/neofs-api-go/v2/netmap.NetworkInfo -// message. See ReadFromV2 / WriteToV2 methods. +// NetworkInfo is mutually compatible with [protonetmap.NetworkInfo] +// message. See [NetworkInfo.FromProtoMessage] / [NetworkInfo.ProtoMessage] methods. // // Instances can be created using built-in var declaration. type NetworkInfo struct { - m netmap.NetworkInfo + curEpoch uint64 + netMagic uint64 + msPerBlock int64 + prms [][2][]byte } // reads NetworkInfo from netmap.NetworkInfo message. If checkFieldPresence is set, // returns an error on absence of any protocol-required field. Verifies format of any // presented field according to NeoFS API V2 protocol. -func (x *NetworkInfo) readFromV2(m netmap.NetworkInfo, checkFieldPresence bool) error { +func (x *NetworkInfo) fromProtoMessage(m *protonetmap.NetworkInfo, checkFieldPresence bool) error { c := m.GetNetworkConfig() if checkFieldPresence && c == nil { return errors.New("missing network config") } - if checkFieldPresence && c.NumberOfParameters() <= 0 { + if checkFieldPresence && len(c.Parameters) == 0 { return errors.New("missing network parameters") } + if c == nil { + x.curEpoch = m.CurrentEpoch + x.netMagic = m.MagicNumber + x.msPerBlock = m.MsPerBlock + return nil + } + var err error - mNames := make(map[string]struct{}, c.NumberOfParameters()) + mNames := make(map[string]struct{}, len(c.Parameters)) + prms := make([][2][]byte, len(c.Parameters)) - c.IterateParameters(func(prm *netmap.NetworkParameter) bool { + for i, prm := range c.Parameters { + if prm == nil { + return fmt.Errorf("nil parameter #%d", i) + } name := string(prm.GetKey()) _, was := mNames[name] if was { - err = fmt.Errorf("duplicated parameter name: %s", name) - return true + return fmt.Errorf("duplicated parameter name: %s", name) } mNames[name] = struct{}{} @@ -52,8 +65,7 @@ func (x *NetworkInfo) readFromV2(m netmap.NetworkInfo, checkFieldPresence bool) switch name { default: if len(prm.GetValue()) == 0 { - err = fmt.Errorf("empty attribute value %s", name) - return true + return fmt.Errorf("empty attribute value %s", name) } case configEigenTrustAlpha: var num uint64 @@ -81,35 +93,47 @@ func (x *NetworkInfo) readFromV2(m netmap.NetworkInfo, checkFieldPresence bool) } if err != nil { - err = fmt.Errorf("invalid %s parameter: %w", name, err) + return fmt.Errorf("invalid %s parameter: %w", name, err) } - return err != nil - }) - - if err != nil { - return err + prms[i] = [2][]byte{prm.Key, prm.Value} } - x.m = m + x.curEpoch = m.CurrentEpoch + x.netMagic = m.MagicNumber + x.msPerBlock = m.MsPerBlock + x.prms = prms return nil } -// ReadFromV2 reads NetworkInfo from the netmap.NetworkInfo message. Checks if the -// message conforms to NeoFS API V2 protocol. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// x from it. // -// See also WriteToV2. -func (x *NetworkInfo) ReadFromV2(m netmap.NetworkInfo) error { - return x.readFromV2(m, true) +// See also [NetworkInfo.ProtoMessage]. +func (x *NetworkInfo) FromProtoMessage(m *protonetmap.NetworkInfo) error { + return x.fromProtoMessage(m, true) } -// WriteToV2 writes NetworkInfo to the netmap.NetworkInfo message. The message -// MUST NOT be nil. +// ProtoMessage converts x into message to transmit using the NeoFS API +// protocol. // -// See also ReadFromV2. -func (x NetworkInfo) WriteToV2(m *netmap.NetworkInfo) { - *m = x.m +// See also [NetworkInfo.FromProtoMessage]. +func (x NetworkInfo) ProtoMessage() *protonetmap.NetworkInfo { + m := &protonetmap.NetworkInfo{ + CurrentEpoch: x.curEpoch, + MagicNumber: x.netMagic, + MsPerBlock: x.msPerBlock, + } + if len(x.prms) > 0 { + m.NetworkConfig = &protonetmap.NetworkConfig{ + Parameters: make([]*protonetmap.NetworkConfig_Parameter, len(x.prms)), + } + for i := range x.prms { + m.NetworkConfig.Parameters[i] = &protonetmap.NetworkConfig_Parameter{Key: x.prms[i][0], Value: x.prms[i][1]} + } + } + return m } // Marshal encodes NetworkInfo into a binary format of the NeoFS API protocol @@ -117,10 +141,7 @@ func (x NetworkInfo) WriteToV2(m *netmap.NetworkInfo) { // // See also Unmarshal. func (x NetworkInfo) Marshal() []byte { - var m netmap.NetworkInfo - x.WriteToV2(&m) - - return m.StableMarshal(nil) + return neofsproto.Marshal(x) } // Unmarshal decodes NeoFS API protocol binary format into the NetworkInfo @@ -129,105 +150,65 @@ func (x NetworkInfo) Marshal() []byte { // // See also Marshal. func (x *NetworkInfo) Unmarshal(data []byte) error { - var m netmap.NetworkInfo - - err := m.Unmarshal(data) - if err != nil { - return err - } - - return x.readFromV2(m, false) + return neofsproto.UnmarshalOptional(data, x, (*NetworkInfo).fromProtoMessage) } // CurrentEpoch returns epoch set using SetCurrentEpoch. // // Zero NetworkInfo has zero current epoch. func (x NetworkInfo) CurrentEpoch() uint64 { - return x.m.GetCurrentEpoch() + return x.curEpoch } // SetCurrentEpoch sets current epoch of the NeoFS network. func (x *NetworkInfo) SetCurrentEpoch(epoch uint64) { - x.m.SetCurrentEpoch(epoch) + x.curEpoch = epoch } // MagicNumber returns magic number set using SetMagicNumber. // // Zero NetworkInfo has zero magic. func (x NetworkInfo) MagicNumber() uint64 { - return x.m.GetMagicNumber() + return x.netMagic } // SetMagicNumber sets magic number of the NeoFS Sidechain. // // See also MagicNumber. func (x *NetworkInfo) SetMagicNumber(epoch uint64) { - x.m.SetMagicNumber(epoch) + x.netMagic = epoch } // MsPerBlock returns network parameter set using SetMsPerBlock. func (x NetworkInfo) MsPerBlock() int64 { - return x.m.GetMsPerBlock() + return x.msPerBlock } // SetMsPerBlock sets MillisecondsPerBlock network parameter of the NeoFS Sidechain. // // See also MsPerBlock. func (x *NetworkInfo) SetMsPerBlock(v int64) { - x.m.SetMsPerBlock(v) + x.msPerBlock = v } func (x *NetworkInfo) setConfig(name string, val []byte) { - c := x.m.GetNetworkConfig() - if c == nil { - c = new(netmap.NetworkConfig) - - var prm netmap.NetworkParameter - prm.SetKey([]byte(name)) - prm.SetValue(val) - - c.SetParameters(prm) - - x.m.SetNetworkConfig(c) - - return - } - - found := false - prms := make([]netmap.NetworkParameter, 0, c.NumberOfParameters()) - - c.IterateParameters(func(prm *netmap.NetworkParameter) bool { - found = bytes.Equal(prm.GetKey(), []byte(name)) - if found { - prm.SetValue(val) - } else { - prms = append(prms, *prm) + for i := range x.prms { + if string(x.prms[i][0]) == name { + x.prms[i][1] = val + return } - - return found - }) - - if !found { - prms = append(prms, netmap.NetworkParameter{}) - prms[len(prms)-1].SetKey([]byte(name)) - prms[len(prms)-1].SetValue(val) - - c.SetParameters(prms...) } -} -func (x NetworkInfo) configValue(name string) (res []byte) { - x.m.GetNetworkConfig().IterateParameters(func(prm *netmap.NetworkParameter) bool { - if string(prm.GetKey()) == name { - res = prm.GetValue() + x.prms = append(x.prms, [2][]byte{[]byte(name), val}) +} - return true +func (x NetworkInfo) configValue(name string) []byte { + for i := range x.prms { + if string(x.prms[i][0]) == name { + return x.prms[i][1] } - - return false - }) - - return + } + return nil } // SetRawNetworkParameter sets named NeoFS network parameter whose value is @@ -258,13 +239,11 @@ func (x *NetworkInfo) RawNetworkParameter(name string) []byte { // // Zero NetworkInfo has no network parameters. func (x *NetworkInfo) IterateRawNetworkParameters(f func(name string, value []byte)) { - c := x.m.GetNetworkConfig() - - c.IterateParameters(func(prm *netmap.NetworkParameter) bool { - name := string(prm.GetKey()) + for i := range x.prms { + name := string(x.prms[i][0]) switch name { default: - f(name, prm.GetValue()) + f(name, x.prms[i][1]) case configEigenTrustAlpha, configAuditFee, @@ -279,9 +258,7 @@ func (x *NetworkInfo) IterateRawNetworkParameters(f func(name string, value []by configHomomorphicHashingDisabled, configMaintenanceModeAllowed: } - - return false - }) + } } func (x *NetworkInfo) setConfigUint64(name string, num uint64) { @@ -332,7 +309,7 @@ func (x NetworkInfo) configUint64(name string) uint64 { res, err := decodeConfigValueUint64(val) if err != nil { // potential panic is OK since value MUST be correct since it is - // verified in ReadFromV2 or set by provided method. + // verified in FromProtoMessage or set by provided method. panic(err) } @@ -348,7 +325,7 @@ func (x NetworkInfo) configBool(name string) bool { res, err := decodeConfigValueBool(val) if err != nil { // potential panic is OK since value MUST be correct since it is - // verified in ReadFromV2 or set by provided method. + // verified in FromProtoMessage or set by provided method. panic(err) } diff --git a/netmap/network_info_test.go b/netmap/network_info_test.go index ceb97298..aedeff89 100644 --- a/netmap/network_info_test.go +++ b/netmap/network_info_test.go @@ -3,8 +3,8 @@ package netmap_test import ( "testing" - apinetmap "github.com/nspcc-dev/neofs-api-go/v2/netmap" "github.com/nspcc-dev/neofs-sdk-go/netmap" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" "github.com/stretchr/testify/require" ) @@ -213,52 +213,46 @@ func TestNetworkInfo_AllowMaintenanceMode(t *testing.T) { require.True(t, x.MaintenanceModeAllowed()) } -func setNetworkPrms[T string | []byte](ni *apinetmap.NetworkInfo, els ...T) { +func setNetworkPrms[T string | []byte](ni *protonetmap.NetworkInfo, els ...T) { if len(els)%2 != 0 { panic("must be even") } - mps := make([]apinetmap.NetworkParameter, len(els)/2) + ni.NetworkConfig.Parameters = make([]*protonetmap.NetworkConfig_Parameter, len(els)/2) for i := range len(els) / 2 { - mps[i].SetKey([]byte(els[2*i])) - mps[i].SetValue([]byte(els[2*i+1])) + ni.NetworkConfig.Parameters[i] = &protonetmap.NetworkConfig_Parameter{ + Key: []byte(els[2*i]), + Value: []byte(els[2*i+1]), + } } - var mc apinetmap.NetworkConfig - mc.SetParameters(mps...) - ni.SetNetworkConfig(&mc) } -func TestNetworkInfo_ReadFromV2(t *testing.T) { - var mps []apinetmap.NetworkParameter - addP := func(k string, v []byte) { - var m apinetmap.NetworkParameter - m.SetKey([]byte(k)) - m.SetValue(v) - mps = append(mps, m) +func TestNetworkInfo_FromProtoMessage(t *testing.T) { + m := &protonetmap.NetworkInfo{ + CurrentEpoch: anyValidCurrentEpoch, + MagicNumber: anyValidMagicNumber, + MsPerBlock: anyValidMSPerBlock, + NetworkConfig: &protonetmap.NetworkConfig{ + Parameters: []*protonetmap.NetworkConfig_Parameter{ + {Key: []byte("k1"), Value: []byte("v1")}, + {Key: []byte("k2"), Value: []byte("v2")}, + {Key: []byte("AuditFee"), Value: anyValidBinAuditFee}, + {Key: []byte("BasicIncomeRate"), Value: anyValidBinStoragePrice}, + {Key: []byte("ContainerAliasFee"), Value: anyValidBinNamedContainerFee}, + {Key: []byte("ContainerFee"), Value: anyValidBinContainerFee}, + {Key: []byte("EigenTrustAlpha"), Value: anyValidBinEigenTrustAlpha}, + {Key: []byte("EigenTrustIterations"), Value: anyValidBinEigenTrustIterations}, + {Key: []byte("EpochDuration"), Value: anyValidBinEpochDuration}, + {Key: []byte("HomomorphicHashingDisabled"), Value: anyValidBinHomoHashDisabled}, + {Key: []byte("InnerRingCandidateFee"), Value: anyValidBinIRCandidateFee}, + {Key: []byte("MaintenanceModeAllowed"), Value: anyValidBinMaintenanceModeAllowed}, + {Key: []byte("MaxObjectSize"), Value: anyValidBinMaxObjectSize}, + {Key: []byte("WithdrawFee"), Value: anyValidBinWithdrawalFee}, + }, + }, } - addP("k1", []byte("v1")) - addP("k2", []byte("v2")) - addP("AuditFee", anyValidBinAuditFee) - addP("BasicIncomeRate", anyValidBinStoragePrice) - addP("ContainerAliasFee", anyValidBinNamedContainerFee) - addP("ContainerFee", anyValidBinContainerFee) - addP("EigenTrustAlpha", anyValidBinEigenTrustAlpha) - addP("EigenTrustIterations", anyValidBinEigenTrustIterations) - addP("EpochDuration", anyValidBinEpochDuration) - addP("HomomorphicHashingDisabled", anyValidBinHomoHashDisabled) - addP("InnerRingCandidateFee", anyValidBinIRCandidateFee) - addP("MaintenanceModeAllowed", anyValidBinMaintenanceModeAllowed) - addP("MaxObjectSize", anyValidBinMaxObjectSize) - addP("WithdrawFee", anyValidBinWithdrawalFee) - var mc apinetmap.NetworkConfig - mc.SetParameters(mps...) - var m apinetmap.NetworkInfo - m.SetCurrentEpoch(anyValidCurrentEpoch) - m.SetMagicNumber(anyValidMagicNumber) - m.SetMsPerBlock(anyValidMSPerBlock) - m.SetNetworkConfig(&mc) var val netmap.NetworkInfo - require.NoError(t, val.ReadFromV2(m)) + require.NoError(t, val.FromProtoMessage(m)) require.EqualValues(t, "v1", val.RawNetworkParameter("k1")) require.EqualValues(t, "v2", val.RawNetworkParameter("k2")) require.Equal(t, anyValidCurrentEpoch, val.CurrentEpoch()) @@ -278,12 +272,12 @@ func TestNetworkInfo_ReadFromV2(t *testing.T) { require.Equal(t, anyValidWithdrawalFee, val.WithdrawalFee()) // reset optional fields - mc.SetParameters(mps[0]) - m.SetCurrentEpoch(0) - m.SetMagicNumber(0) - m.SetMsPerBlock(0) + m.NetworkConfig.Parameters = m.NetworkConfig.Parameters[:1] + m.CurrentEpoch = 0 + m.MagicNumber = 0 + m.MsPerBlock = 0 val2 := val - require.NoError(t, val2.ReadFromV2(m)) + require.NoError(t, val2.FromProtoMessage(m)) require.EqualValues(t, "v1", val.RawNetworkParameter("k1")) require.Zero(t, val2.RawNetworkParameter("k2")) require.Zero(t, val2.CurrentEpoch()) @@ -305,89 +299,85 @@ func TestNetworkInfo_ReadFromV2(t *testing.T) { t.Run("invalid", func(t *testing.T) { for _, tc := range []struct { name, err string - corrupt func(*apinetmap.NetworkInfo) + corrupt func(*protonetmap.NetworkInfo) }{ {name: "netconfig/missing", err: "missing network config", - corrupt: func(m *apinetmap.NetworkInfo) { m.SetNetworkConfig(nil) }}, + corrupt: func(m *protonetmap.NetworkInfo) { m.NetworkConfig = nil }}, {name: "netconfig/prms/missing", err: "missing network parameters", - corrupt: func(m *apinetmap.NetworkInfo) { m.SetNetworkConfig(new(apinetmap.NetworkConfig)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { m.NetworkConfig = new(protonetmap.NetworkConfig) }}, + {name: "netconfig/prms/nil", err: "nil parameter #1", + corrupt: func(m *protonetmap.NetworkInfo) { + m.NetworkConfig.Parameters[1] = nil + }}, {name: "netconfig/prms/no value", err: "empty attribute value k1", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, "k1", "") }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, "k1", "") }}, {name: "netconfig/prms/duplicated", err: "duplicated parameter name: k1", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, "k1", "v1", "k2", "v2", "k1", "v3") }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, "k1", "v1", "k2", "v2", "k1", "v3") }}, {name: "netconfig/prms/eigen trust alpha/overflow", err: "invalid EigenTrustAlpha parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, "EigenTrustAlpha", "123456789") }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, "EigenTrustAlpha", "123456789") }}, {name: "netconfig/prms/eigen trust alpha/negative", err: "invalid EigenTrustAlpha parameter: EigenTrust alpha value -0.50 is out of range [0, 1]", - corrupt: func(m *apinetmap.NetworkInfo) { + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("EigenTrustAlpha"), []byte{0, 0, 0, 0, 0, 0, 224, 191}) }}, {name: "netconfig/prms/eigen trust alpha/too big", err: "invalid EigenTrustAlpha parameter: EigenTrust alpha value 1.50 is out of range [0, 1]", - corrupt: func(m *apinetmap.NetworkInfo) { + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("EigenTrustAlpha"), []byte{0, 0, 0, 0, 0, 0, 248, 63}) }}, {name: "netconfig/prms/homo hash disabled/overflow", err: "invalid HomomorphicHashingDisabled parameter: invalid bool parameter contract format too big: integer", - corrupt: func(m *apinetmap.NetworkInfo) { + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("HomomorphicHashingDisabled"), make([]byte, 33)) }}, {name: "netconfig/prms/maintenance allowed/overflow", err: "invalid MaintenanceModeAllowed parameter: invalid bool parameter contract format too big: integer", - corrupt: func(m *apinetmap.NetworkInfo) { + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("MaintenanceModeAllowed"), make([]byte, 33)) }}, {name: "netconfig/prms/audit fee/overflow", err: "invalid AuditFee parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, []byte("AuditFee"), make([]byte, 9)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("AuditFee"), make([]byte, 9)) }}, {name: "netconfig/prms/storage price/overflow", err: "invalid BasicIncomeRate parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, []byte("BasicIncomeRate"), make([]byte, 9)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("BasicIncomeRate"), make([]byte, 9)) }}, {name: "netconfig/prms/container fee/overflow", err: "invalid ContainerFee parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, []byte("ContainerFee"), make([]byte, 9)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("ContainerFee"), make([]byte, 9)) }}, {name: "netconfig/prms/named container fee/overflow", err: "invalid ContainerAliasFee parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, []byte("ContainerAliasFee"), make([]byte, 9)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("ContainerAliasFee"), make([]byte, 9)) }}, {name: "netconfig/prms/eigen trust iterations/overflow", err: "invalid EigenTrustIterations parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, []byte("EigenTrustIterations"), make([]byte, 9)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("EigenTrustIterations"), make([]byte, 9)) }}, {name: "netconfig/prms/epoch duration/overflow", err: "invalid EpochDuration parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, []byte("EpochDuration"), make([]byte, 9)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("EpochDuration"), make([]byte, 9)) }}, {name: "netconfig/prms/ir candidate fee/overflow", err: "invalid InnerRingCandidateFee parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, []byte("InnerRingCandidateFee"), make([]byte, 9)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("InnerRingCandidateFee"), make([]byte, 9)) }}, {name: "netconfig/prms/max object size/overflow", err: "invalid MaxObjectSize parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, []byte("MaxObjectSize"), make([]byte, 9)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("MaxObjectSize"), make([]byte, 9)) }}, {name: "netconfig/prms/withdrawal fee/overflow", err: "invalid WithdrawFee parameter: invalid uint64 parameter length 9", - corrupt: func(m *apinetmap.NetworkInfo) { setNetworkPrms(m, []byte("WithdrawFee"), make([]byte, 9)) }}, + corrupt: func(m *protonetmap.NetworkInfo) { setNetworkPrms(m, []byte("WithdrawFee"), make([]byte, 9)) }}, } { t.Run(tc.name, func(t *testing.T) { st := val - var m apinetmap.NetworkInfo - st.WriteToV2(&m) - tc.corrupt(&m) - require.EqualError(t, new(netmap.NetworkInfo).ReadFromV2(m), tc.err) + m := st.ProtoMessage() + tc.corrupt(m) + require.EqualError(t, new(netmap.NetworkInfo).FromProtoMessage(m), tc.err) }) } }) } -func TestNetworkInfo_WriteToV2(t *testing.T) { +func TestNetworkInfo_ProtoMessage(t *testing.T) { var val netmap.NetworkInfo - var m apinetmap.NetworkInfo // zero - val.WriteToV2(&m) + m := val.ProtoMessage() require.Zero(t, m.GetCurrentEpoch()) require.Zero(t, m.GetMagicNumber()) require.Zero(t, m.GetMsPerBlock()) require.Zero(t, m.GetNetworkConfig()) // filled - validNetworkInfo.WriteToV2(&m) + m = validNetworkInfo.ProtoMessage() require.Equal(t, anyValidCurrentEpoch, m.GetCurrentEpoch()) require.Equal(t, anyValidMagicNumber, m.GetMagicNumber()) require.Equal(t, anyValidMSPerBlock, m.GetMsPerBlock()) mc := m.GetNetworkConfig() require.NotNil(t, mc) - require.EqualValues(t, 14, mc.NumberOfParameters()) - var collected [][2][]byte - mc.IterateParameters(func(p *apinetmap.NetworkParameter) bool { - collected = append(collected, [2][]byte{p.GetKey(), p.GetValue()}) - return false - }) - require.Len(t, collected, 14) + require.Len(t, mc.Parameters, 14) for i, pair := range [][2]any{ {"k1", "v1"}, {"k2", "v2"}, @@ -404,8 +394,8 @@ func TestNetworkInfo_WriteToV2(t *testing.T) { {"MaxObjectSize", anyValidBinMaxObjectSize}, {"WithdrawFee", anyValidBinWithdrawalFee}, } { - require.EqualValues(t, pair[0], collected[i][0]) - require.EqualValues(t, pair[1], collected[i][1]) + require.EqualValues(t, pair[0], mc.Parameters[i].Key) + require.EqualValues(t, pair[1], mc.Parameters[i].Value) } } diff --git a/netmap/node_info.go b/netmap/node_info.go index d852cb88..c5bb2fa3 100644 --- a/netmap/node_info.go +++ b/netmap/node_info.go @@ -8,8 +8,9 @@ import ( "strings" "github.com/nspcc-dev/hrw/v2" - "github.com/nspcc-dev/neofs-api-go/v2/netmap" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" ) // NodeInfo groups information about NeoFS storage node which is reflected @@ -18,18 +19,24 @@ import ( // about the nodes is available to all network participants to work with the network // map (mainly to comply with container storage policies). // -// NodeInfo is mutually compatible with github.com/nspcc-dev/neofs-api-go/v2/netmap.NodeInfo -// message. See ReadFromV2 / WriteToV2 methods. +// NodeInfo is mutually compatible with [protonetmap.NodeInfo] message. See +// [NodeInfo.FromProtoMessage] / [NodeInfo.ProtoMessage] methods. // // Instances can be created using built-in var declaration. type NodeInfo struct { - m netmap.NodeInfo + state protonetmap.NodeInfo_State + pub []byte + addrs []string + attrs [][2]string } // reads NodeInfo from netmap.NodeInfo message. If checkFieldPresence is set, // returns an error on absence of any protocol-required field. Verifies format of any // presented field according to NeoFS API V2 protocol. -func (x *NodeInfo) readFromV2(m netmap.NodeInfo, checkFieldPresence bool) error { +func (x *NodeInfo) fromProtoMessage(m *protonetmap.NodeInfo, checkFieldPresence bool) error { + if m.State < 0 { + return fmt.Errorf("negative state %d", m.State) + } var err error binPublicKey := m.GetPublicKey() @@ -37,13 +44,17 @@ func (x *NodeInfo) readFromV2(m netmap.NodeInfo, checkFieldPresence bool) error return errors.New("missing public key") } - if checkFieldPresence && m.NumberOfAddresses() <= 0 { + if checkFieldPresence && len(m.Addresses) == 0 { return errors.New("missing network endpoints") } attributes := m.GetAttributes() mAttr := make(map[string]struct{}, len(attributes)) + attrs := make([][2]string, len(attributes)) for i := range attributes { + if attributes[i] == nil { + return fmt.Errorf("nil attribute #%d", i) + } key := attributes[i].GetKey() if key == "" { return fmt.Errorf("empty key of the attribute #%d", i) @@ -51,46 +62,62 @@ func (x *NodeInfo) readFromV2(m netmap.NodeInfo, checkFieldPresence bool) error return fmt.Errorf("duplicated attribute %s", key) } + val := attributes[i].GetValue() switch { case key == attrCapacity: - _, err = strconv.ParseUint(attributes[i].GetValue(), 10, 64) + _, err = strconv.ParseUint(val, 10, 64) if err != nil { return fmt.Errorf("invalid %s attribute: %w", attrCapacity, err) } case key == attrPrice: var err error - _, err = strconv.ParseUint(attributes[i].GetValue(), 10, 64) + _, err = strconv.ParseUint(val, 10, 64) if err != nil { return fmt.Errorf("invalid %s attribute: %w", attrPrice, err) } default: - if attributes[i].GetValue() == "" { + if val == "" { return fmt.Errorf("empty value of the attribute %s", key) } } mAttr[key] = struct{}{} + attrs[i][0], attrs[i][1] = key, val } - x.m = m + x.state = m.State + x.pub = m.PublicKey + x.addrs = m.Addresses + x.attrs = attrs return nil } -// ReadFromV2 reads NodeInfo from the netmap.NodeInfo message. Checks if the -// message conforms to NeoFS API V2 protocol. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// x from it. // -// See also WriteToV2. -func (x *NodeInfo) ReadFromV2(m netmap.NodeInfo) error { - return x.readFromV2(m, true) +// See also [NodeInfo.ProtoMessage]. +func (x *NodeInfo) FromProtoMessage(m *protonetmap.NodeInfo) error { + return x.fromProtoMessage(m, true) } -// WriteToV2 writes NodeInfo to the netmap.NodeInfo message. The message MUST NOT -// be nil. +// ProtoMessage converts x into message to transmit using the NeoFS API +// protocol. // -// See also ReadFromV2. -func (x NodeInfo) WriteToV2(m *netmap.NodeInfo) { - *m = x.m +// See also [NodeInfo.FromProtoMessage]. +func (x NodeInfo) ProtoMessage() *protonetmap.NodeInfo { + m := &protonetmap.NodeInfo{ + PublicKey: x.pub, + Addresses: x.addrs, + State: x.state, + } + if len(x.attrs) > 0 { + m.Attributes = make([]*protonetmap.NodeInfo_Attribute, len(x.attrs)) + for i := range x.attrs { + m.Attributes[i] = &protonetmap.NodeInfo_Attribute{Key: x.attrs[i][0], Value: x.attrs[i][1]} + } + } + return m } // Marshal encodes NodeInfo into a binary format of the NeoFS API protocol @@ -98,10 +125,7 @@ func (x NodeInfo) WriteToV2(m *netmap.NodeInfo) { // // See also Unmarshal. func (x NodeInfo) Marshal() []byte { - var m netmap.NodeInfo - x.WriteToV2(&m) - - return m.StableMarshal(nil) + return neofsproto.Marshal(x) } // Unmarshal decodes NeoFS API protocol binary format into the NodeInfo @@ -110,14 +134,7 @@ func (x NodeInfo) Marshal() []byte { // // See also Marshal. func (x *NodeInfo) Unmarshal(data []byte) error { - var m netmap.NodeInfo - - err := m.Unmarshal(data) - if err != nil { - return err - } - - return x.readFromV2(m, false) + return neofsproto.UnmarshalOptional(data, x, (*NodeInfo).fromProtoMessage) } // MarshalJSON encodes NodeInfo into a JSON format of the NeoFS API protocol @@ -125,10 +142,7 @@ func (x *NodeInfo) Unmarshal(data []byte) error { // // See also UnmarshalJSON. func (x NodeInfo) MarshalJSON() ([]byte, error) { - var m netmap.NodeInfo - x.WriteToV2(&m) - - return m.MarshalJSON() + return neofsproto.MarshalJSON(x) } // UnmarshalJSON decodes NeoFS API protocol JSON format into the NodeInfo @@ -136,14 +150,7 @@ func (x NodeInfo) MarshalJSON() ([]byte, error) { // // See also MarshalJSON. func (x *NodeInfo) UnmarshalJSON(data []byte) error { - var m netmap.NodeInfo - - err := m.UnmarshalJSON(data) - if err != nil { - return err - } - - return x.readFromV2(m, false) + return neofsproto.UnmarshalJSONOptional(data, x, (*NodeInfo).fromProtoMessage) } // SetPublicKey sets binary-encoded public key bound to the node. The key @@ -155,7 +162,7 @@ func (x *NodeInfo) UnmarshalJSON(data []byte) error { // // See also [NodeInfo.PublicKey]. func (x *NodeInfo) SetPublicKey(key []byte) { - x.m.SetPublicKey(key) + x.pub = key } // PublicKey returns value set using [NodeInfo.SetPublicKey]. @@ -169,7 +176,7 @@ func (x *NodeInfo) SetPublicKey(key []byte) { // The value returned shares memory with the structure itself, so changing it can lead to data corruption. // Make a copy if you need to change it. func (x NodeInfo) PublicKey() []byte { - return x.m.GetPublicKey() + return x.pub } // StringifyPublicKey returns HEX representation of PublicKey. @@ -187,14 +194,14 @@ func StringifyPublicKey(node NodeInfo) string { // // See also IterateNetworkEndpoints. func (x *NodeInfo) SetNetworkEndpoints(v ...string) { - x.m.SetAddresses(v...) + x.addrs = v } // NumberOfNetworkEndpoints returns number of network endpoints announced by the node. // // See also SetNetworkEndpoints. func (x NodeInfo) NumberOfNetworkEndpoints() int { - return x.m.NumberOfAddresses() + return len(x.addrs) } // IterateNetworkEndpoints iterates over network endpoints announced by the @@ -206,7 +213,11 @@ func (x NodeInfo) NumberOfNetworkEndpoints() int { // // See also SetNetworkEndpoints. func (x NodeInfo) IterateNetworkEndpoints(f func(string) bool) { - x.m.IterateAddresses(f) + for i := range x.addrs { + if f(x.addrs[i]) { + return + } + } } // IterateNetworkEndpoints is an extra-sugared function over IterateNetworkEndpoints @@ -226,7 +237,7 @@ var _ hrw.Hashable = NodeInfo{} // Hash is needed to support weighted HRW therefore sort function sorts nodes // based on their public key. Hash isn't expected to be used directly. func (x NodeInfo) Hash() uint64 { - return hrw.Hash(x.m.GetPublicKey()) + return hrw.Hash(x.PublicKey()) } // less declares "less than" comparison between two NodeInfo instances: @@ -448,15 +459,14 @@ func (x NodeInfo) ExternalAddresses() []string { // // See also SetAttribute. func (x NodeInfo) NumberOfAttributes() int { - return len(x.m.GetAttributes()) + return len(x.attrs) } // IterateAttributes iterates over all node attributes and passes the into f. // Handler MUST NOT be nil. func (x NodeInfo) IterateAttributes(f func(key, value string)) { - a := x.m.GetAttributes() - for i := range a { - f(a[i].GetKey(), a[i].GetValue()) + for i := range x.attrs { + f(x.attrs[i][0], x.attrs[i][1]) } } @@ -465,11 +475,7 @@ func (x NodeInfo) IterateAttributes(f func(key, value string)) { // // See also Attribute, IterateAttributes. func (x NodeInfo) GetAttributes() [][2]string { - attrs := make([][2]string, len(x.m.GetAttributes())) - for i, attr := range x.m.GetAttributes() { - attrs[i] = [2]string{attr.GetKey(), attr.GetValue()} - } - return attrs + return x.attrs } // SetAttributes sets list of node attributes. @@ -478,7 +484,6 @@ func (x NodeInfo) GetAttributes() [][2]string { // // See also SetAttribute. func (x *NodeInfo) SetAttributes(attrs [][2]string) { - netmapAttrs := make([]netmap.Attribute, 0, len(attrs)) for _, attr := range attrs { if attr[0] == "" { panic("empty key in SetAttributes") @@ -486,13 +491,9 @@ func (x *NodeInfo) SetAttributes(attrs [][2]string) { if attr[1] == "" { panic(fmt.Errorf("empty value in SetAttributes for key: %s", attr[0])) } - - netmapAttrs = append(netmapAttrs, netmap.Attribute{}) - netmapAttrs[len(netmapAttrs)-1].SetKey(attr[0]) - netmapAttrs[len(netmapAttrs)-1].SetValue(attr[1]) } - x.m.SetAttributes(netmapAttrs) + x.attrs = attrs } // SetAttribute sets value of the node attribute value by the given key. @@ -504,28 +505,22 @@ func (x *NodeInfo) SetAttribute(key, value string) { panic("empty value in SetAttribute") } - a := x.m.GetAttributes() - for i := range a { - if a[i].GetKey() == key { - a[i].SetValue(value) + for i := range x.attrs { + if x.attrs[i][0] == key { + x.attrs[i][1] = value return } } - a = append(a, netmap.Attribute{}) - a[len(a)-1].SetKey(key) - a[len(a)-1].SetValue(value) - - x.m.SetAttributes(a) + x.attrs = append(x.attrs, [2]string{key, value}) } // Attribute returns value of the node attribute set using SetAttribute by the // given key. Returns empty string if attribute is missing. func (x NodeInfo) Attribute(key string) string { - a := x.m.GetAttributes() - for i := range a { - if a[i].GetKey() == key { - return a[i].GetValue() + for i := range x.attrs { + if x.attrs[i][0] == key { + return x.attrs[i][1] } } @@ -535,30 +530,27 @@ func (x NodeInfo) Attribute(key string) string { // SortAttributes sorts node attributes set using SetAttribute lexicographically. // The method is only needed to make NodeInfo consistent, e.g. for signing. func (x *NodeInfo) SortAttributes() { - as := x.m.GetAttributes() - if len(as) == 0 { + if len(x.attrs) == 0 { return } - sort.Slice(as, func(i, j int) bool { - switch strings.Compare(as[i].GetKey(), as[j].GetKey()) { + sort.Slice(x.attrs, func(i, j int) bool { + switch strings.Compare(x.attrs[i][0], x.attrs[j][0]) { case -1: return true case 1: return false default: - return as[i].GetValue() < as[j].GetValue() + return x.attrs[i][1] < x.attrs[j][1] } }) - - x.m.SetAttributes(as) } // SetOffline sets the state of the node to "offline". When a node updates // information about itself in the network map, this action is interpreted as // an intention to leave the network. func (x *NodeInfo) SetOffline() { - x.m.SetState(netmap.Offline) + x.state = protonetmap.NodeInfo_OFFLINE } // IsOffline checks if the node is in the "offline" state. @@ -568,7 +560,7 @@ func (x *NodeInfo) SetOffline() { // // See also SetOffline. func (x NodeInfo) IsOffline() bool { - return x.m.GetState() == netmap.Offline + return x.state == protonetmap.NodeInfo_OFFLINE } // SetOnline sets the state of the node to "online". When a node updates @@ -577,7 +569,7 @@ func (x NodeInfo) IsOffline() bool { // // See also IsOnline. func (x *NodeInfo) SetOnline() { - x.m.SetState(netmap.Online) + x.state = protonetmap.NodeInfo_ONLINE } // IsOnline checks if the node is in the "online" state. @@ -587,7 +579,7 @@ func (x *NodeInfo) SetOnline() { // // See also SetOnline. func (x NodeInfo) IsOnline() bool { - return x.m.GetState() == netmap.Online + return x.state == protonetmap.NodeInfo_ONLINE } // SetMaintenance sets the state of the node to "maintenance". When a node updates @@ -596,7 +588,7 @@ func (x NodeInfo) IsOnline() bool { // // See also IsMaintenance. func (x *NodeInfo) SetMaintenance() { - x.m.SetState(netmap.Maintenance) + x.state = protonetmap.NodeInfo_MAINTENANCE } // IsMaintenance checks if the node is in the "maintenance" state. @@ -605,7 +597,7 @@ func (x *NodeInfo) SetMaintenance() { // // See also SetMaintenance. func (x NodeInfo) IsMaintenance() bool { - return x.m.GetState() == netmap.Maintenance + return x.state == protonetmap.NodeInfo_MAINTENANCE } const attrVerifiedNodesDomain = "VerifiedNodesDomain" diff --git a/netmap/node_info_test.go b/netmap/node_info_test.go index 39c70e1f..9218742f 100644 --- a/netmap/node_info_test.go +++ b/netmap/node_info_test.go @@ -5,8 +5,8 @@ import ( "strings" "testing" - apinetmap "github.com/nspcc-dev/neofs-api-go/v2/netmap" "github.com/nspcc-dev/neofs-sdk-go/netmap" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" "github.com/stretchr/testify/require" ) @@ -488,48 +488,41 @@ func TestNodeInfo_SetVerifiedNodesDomain(t *testing.T) { require.Equal(t, anyValidVerifiedNodesDomain, n.VerifiedNodesDomain()) } -func setNodeAttributes(ni *apinetmap.NodeInfo, els ...string) { +func setNodeAttributes(ni *protonetmap.NodeInfo, els ...string) { if len(els)%2 != 0 { panic("must be even") } - mas := make([]apinetmap.Attribute, len(els)/2) + ni.Attributes = make([]*protonetmap.NodeInfo_Attribute, len(els)/2) for i := range len(els) / 2 { - mas[i].SetKey(els[2*i]) - mas[i].SetValue(els[2*i+1]) + ni.Attributes[i] = &protonetmap.NodeInfo_Attribute{Key: els[2*i], Value: els[2*i+1]} } - ni.SetAttributes(mas) } -func TestNodeInfo_ReadFromV2(t *testing.T) { - var mas []apinetmap.Attribute - addAttr := func(k, v string) { - var a apinetmap.Attribute - a.SetKey(k) - a.SetValue(v) - mas = append(mas, a) +func TestNodeInfo_FromProtoMessage(t *testing.T) { + m := &protonetmap.NodeInfo{ + PublicKey: anyValidPublicKey, + Addresses: anyValidNetworkEndpoints, + Attributes: []*protonetmap.NodeInfo_Attribute{ + {Key: "k1", Value: "v1"}, + {Key: "k2", Value: "v2"}, + {Key: "Capacity", Value: "9010937245406684209"}, + {Key: "Price", Value: "10993309018040354285"}, + {Key: "UN-LOCODE", Value: anyValidLOCODE}, + {Key: "CountryCode", Value: anyValidCountryCode}, + {Key: "Country", Value: anyValidCountryName}, + {Key: "Location", Value: anyValidLocationName}, + {Key: "SubDivCode", Value: anyValidSubdivCode}, + {Key: "SubDiv", Value: anyValidSubdivName}, + {Key: "SubDivName", Value: anyValidSubdivName}, + {Key: "Continent", Value: anyValidContinentName}, + {Key: "ExternalAddr", Value: strings.Join(anyValidExternalNetworkEndpoints, ",")}, + {Key: "Version", Value: anyValidNodeVersion}, + {Key: "VerifiedNodesDomain", Value: anyValidVerifiedNodesDomain}, + }, } - addAttr("k1", "v1") - addAttr("k2", "v2") - addAttr("Capacity", "9010937245406684209") - addAttr("Price", "10993309018040354285") - addAttr("UN-LOCODE", anyValidLOCODE) - addAttr("CountryCode", anyValidCountryCode) - addAttr("Country", anyValidCountryName) - addAttr("Location", anyValidLocationName) - addAttr("SubDivCode", anyValidSubdivCode) - addAttr("SubDiv", anyValidSubdivName) - addAttr("SubDivName", anyValidSubdivName) - addAttr("Continent", anyValidContinentName) - addAttr("ExternalAddr", strings.Join(anyValidExternalNetworkEndpoints, ",")) - addAttr("Version", anyValidNodeVersion) - addAttr("VerifiedNodesDomain", anyValidVerifiedNodesDomain) - var m apinetmap.NodeInfo - m.SetPublicKey(anyValidPublicKey) - m.SetAddresses(anyValidNetworkEndpoints...) - m.SetAttributes(mas) var val netmap.NodeInfo - require.NoError(t, val.ReadFromV2(m)) + require.NoError(t, val.FromProtoMessage(m)) require.Equal(t, anyValidPublicKey, val.PublicKey()) var i int val.IterateNetworkEndpoints(func(el string) bool { @@ -562,23 +555,23 @@ func TestNodeInfo_ReadFromV2(t *testing.T) { require.Equal(t, anyValidVerifiedNodesDomain, val.VerifiedNodesDomain()) for _, tc := range []struct { - st apinetmap.NodeState + st protonetmap.NodeInfo_State check func(netmap.NodeInfo) bool }{ - {st: apinetmap.Online, check: netmap.NodeInfo.IsOnline}, - {st: apinetmap.Offline, check: netmap.NodeInfo.IsOffline}, - {st: apinetmap.Maintenance, check: netmap.NodeInfo.IsMaintenance}, + {st: protonetmap.NodeInfo_ONLINE, check: netmap.NodeInfo.IsOnline}, + {st: protonetmap.NodeInfo_OFFLINE, check: netmap.NodeInfo.IsOffline}, + {st: protonetmap.NodeInfo_MAINTENANCE, check: netmap.NodeInfo.IsMaintenance}, } { - m.SetState(tc.st) - require.NoError(t, val.ReadFromV2(m), tc.st) + m.State = tc.st + require.NoError(t, val.FromProtoMessage(m), tc.st) require.True(t, tc.check(val)) } // reset optional fields - m.SetAttributes(nil) - m.SetState(0) + m.Attributes = nil + m.State = 0 val2 := val - require.NoError(t, val2.ReadFromV2(m)) + require.NoError(t, val2.FromProtoMessage(m)) require.Zero(t, val2.NumberOfAttributes()) val2.IterateAttributes(func(string, string) { t.Fatal("handler must not be called") @@ -594,57 +587,56 @@ func TestNodeInfo_ReadFromV2(t *testing.T) { t.Run("invalid", func(t *testing.T) { for _, tc := range []struct { name, err string - corrupt func(info *apinetmap.NodeInfo) + corrupt func(info *protonetmap.NodeInfo) }{ {name: "public key/nil", err: "missing public key", - corrupt: func(m *apinetmap.NodeInfo) { m.SetPublicKey(nil) }}, + corrupt: func(m *protonetmap.NodeInfo) { m.PublicKey = nil }}, {name: "public key/empty", err: "missing public key", - corrupt: func(m *apinetmap.NodeInfo) { m.SetPublicKey([]byte{}) }}, + corrupt: func(m *protonetmap.NodeInfo) { m.PublicKey = []byte{} }}, + {name: "endpoints/nil", err: "missing network endpoints", + corrupt: func(m *protonetmap.NodeInfo) { m.Addresses = nil }}, {name: "endpoints/empty", err: "missing network endpoints", - corrupt: func(m *apinetmap.NodeInfo) { m.SetAddresses() }}, + corrupt: func(m *protonetmap.NodeInfo) { m.Addresses = []string{} }}, + {name: "attributes/nil", err: "nil attribute #1", + corrupt: func(m *protonetmap.NodeInfo) { m.Attributes[1] = nil }}, {name: "attributes/no key", err: "empty key of the attribute #1", - corrupt: func(m *apinetmap.NodeInfo) { setNodeAttributes(m, "k1", "v1", "", "v2") }}, + corrupt: func(m *protonetmap.NodeInfo) { setNodeAttributes(m, "k1", "v1", "", "v2") }}, {name: "attributes/no value", err: "empty value of the attribute k2", - corrupt: func(m *apinetmap.NodeInfo) { setNodeAttributes(m, "k1", "v1", "k2", "") }}, + corrupt: func(m *protonetmap.NodeInfo) { setNodeAttributes(m, "k1", "v1", "k2", "") }}, {name: "attributes/duplicated", err: "duplicated attribute k1", - corrupt: func(m *apinetmap.NodeInfo) { setNodeAttributes(m, "k1", "v1", "k2", "v2", "k1", "v3") }}, + corrupt: func(m *protonetmap.NodeInfo) { setNodeAttributes(m, "k1", "v1", "k2", "v2", "k1", "v3") }}, {name: "attributes/capacity", err: "invalid Capacity attribute: strconv.ParseUint: parsing \"foo\": invalid syntax", - corrupt: func(m *apinetmap.NodeInfo) { setNodeAttributes(m, "Capacity", "foo") }}, + corrupt: func(m *protonetmap.NodeInfo) { setNodeAttributes(m, "Capacity", "foo") }}, {name: "attributes/price", err: "invalid Price attribute: strconv.ParseUint: parsing \"foo\": invalid syntax", - corrupt: func(m *apinetmap.NodeInfo) { setNodeAttributes(m, "Price", "foo") }}, + corrupt: func(m *protonetmap.NodeInfo) { setNodeAttributes(m, "Price", "foo") }}, + {name: "state/negative", err: "negative state -1", + corrupt: func(m *protonetmap.NodeInfo) { m.State = -1 }}, } { t.Run(tc.name, func(t *testing.T) { st := val - var m apinetmap.NodeInfo - st.WriteToV2(&m) - tc.corrupt(&m) - require.EqualError(t, new(netmap.NodeInfo).ReadFromV2(m), tc.err) + m := st.ProtoMessage() + tc.corrupt(m) + require.EqualError(t, new(netmap.NodeInfo).FromProtoMessage(m), tc.err) }) } }) } -func TestNodeInfo_WriteToV2(t *testing.T) { +func TestNodeInfo_ProtoMessage(t *testing.T) { var val netmap.NodeInfo - var m apinetmap.NodeInfo // zero - val.WriteToV2(&m) + m := val.ProtoMessage() require.Zero(t, m.GetPublicKey()) - require.Zero(t, m.NumberOfAddresses()) - m.IterateAddresses(func(string) bool { t.Fatal("handler must not be called"); return false }) + require.Zero(t, m.GetAddresses()) require.Zero(t, m.GetAttributes()) require.Zero(t, m.GetState()) // filled - validNodeInfo.WriteToV2(&m) + m = validNodeInfo.ProtoMessage() require.Equal(t, anyValidPublicKey, m.GetPublicKey()) - require.EqualValues(t, 3, m.NumberOfAddresses()) - var collected []string - m.IterateAddresses(func(el string) bool { - collected = append(collected, el) - return false - }) + require.Equal(t, anyValidNetworkEndpoints, m.GetAddresses()) + mas := m.GetAttributes() require.Len(t, mas, 14) for i, pair := range [][2]string{ @@ -670,15 +662,15 @@ func TestNodeInfo_WriteToV2(t *testing.T) { for _, tc := range []struct { setState func(*netmap.NodeInfo) - exp apinetmap.NodeState + exp protonetmap.NodeInfo_State }{ - {setState: (*netmap.NodeInfo).SetOnline, exp: apinetmap.Online}, - {setState: (*netmap.NodeInfo).SetOffline, exp: apinetmap.Offline}, - {setState: (*netmap.NodeInfo).SetMaintenance, exp: apinetmap.Maintenance}, + {setState: (*netmap.NodeInfo).SetOnline, exp: protonetmap.NodeInfo_ONLINE}, + {setState: (*netmap.NodeInfo).SetOffline, exp: protonetmap.NodeInfo_OFFLINE}, + {setState: (*netmap.NodeInfo).SetMaintenance, exp: protonetmap.NodeInfo_MAINTENANCE}, } { val2 := validNodeInfo tc.setState(&val2) - val2.WriteToV2(&m) + m := val2.ProtoMessage() require.Equal(t, tc.exp, m.GetState(), tc.exp) } } diff --git a/netmap/policy.go b/netmap/policy.go index a2e6b773..d39e8fe5 100644 --- a/netmap/policy.go +++ b/netmap/policy.go @@ -6,33 +6,33 @@ import ( "io" "slices" "strconv" - "strings" "github.com/antlr4-go/antlr/v4" - "github.com/nspcc-dev/neofs-api-go/v2/netmap" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" "github.com/nspcc-dev/neofs-sdk-go/netmap/parser" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" ) // PlacementPolicy declares policy to store objects in the NeoFS container. // Within itself, PlacementPolicy represents a set of rules to select a subset // of nodes from NeoFS network map - node-candidates for object storage. // -// PlacementPolicy is mutually compatible with github.com/nspcc-dev/neofs-api-go/v2/netmap.PlacementPolicy -// message. See ReadFromV2 / WriteToV2 methods. +// PlacementPolicy is mutually compatible with [protonetmap.PlacementPolicy] +// message. See [PlacementPolicy.FromProtoMessage] / [PlacementPolicy.ProtoMessage] methods. // // Instances can be created using built-in var declaration. type PlacementPolicy struct { backupFactor uint32 - filters []netmap.Filter + filters []Filter - selectors []netmap.Selector + selectors []Selector - replicas []netmap.Replica + replicas []ReplicaDescriptor } // FilterOp defines the matching property. -type FilterOp uint32 +type FilterOp int32 // Supported FilterOp values. const ( @@ -71,24 +71,15 @@ func (x FilterOp) String() string { } } -func copyFilter(f netmap.Filter) netmap.Filter { - var filter netmap.Filter +func copyFilter(f Filter) Filter { + filter := f - filter.SetName(f.GetName()) - filter.SetKey(f.GetKey()) - filter.SetOp(f.GetOp()) - filter.SetValue(f.GetValue()) + if len(f.subs) > 0 { + filter.subs = make([]Filter, len(f.subs)) - if f.GetFilters() != nil { - filters := make([]netmap.Filter, len(f.GetFilters())) - - for i, internalFilter := range f.GetFilters() { - filters[i] = copyFilter(internalFilter) + for i := range f.subs { + filter.subs[i] = copyFilter(f.subs[i]) } - - filter.SetFilters(filters) - } else { - filter.SetFilters(nil) } return filter @@ -98,27 +89,52 @@ func copyFilter(f netmap.Filter) netmap.Filter { func (p PlacementPolicy) CopyTo(dst *PlacementPolicy) { dst.SetContainerBackupFactor(p.backupFactor) - dst.filters = make([]netmap.Filter, len(p.filters)) - for i, f := range p.filters { - dst.filters[i] = copyFilter(f) + dst.filters = make([]Filter, len(p.filters)) + for i := range p.filters { + dst.filters[i] = copyFilter(p.filters[i]) } - // netmap.Selector is a struct with simple types, no links inside. Just create a new slice and copy all items inside. + // protonetmap.Selector is a struct with simple types, no links inside. Just create a new slice and copy all items inside. dst.selectors = slices.Clone(p.selectors) - // netmap.Replica is a struct with simple types, no links inside. Just create a new slice and copy all items inside. + // protonetmap.Replica is a struct with simple types, no links inside. Just create a new slice and copy all items inside. dst.replicas = slices.Clone(p.replicas) } -func (p *PlacementPolicy) readFromV2(m netmap.PlacementPolicy, checkFieldPresence bool) error { - p.replicas = m.GetReplicas() - if checkFieldPresence && len(p.replicas) == 0 { +func (p *PlacementPolicy) fromProtoMessage(m *protonetmap.PlacementPolicy, checkFieldPresence bool) error { + if checkFieldPresence && len(m.Replicas) == 0 { return errors.New("missing replicas") } + p.replicas = make([]ReplicaDescriptor, len(m.Replicas)) + for i, r := range m.Replicas { + if r == nil { + return fmt.Errorf("nil replica #%d", i) + } + p.replicas[i].fromProtoMessage(r) + } + + p.selectors = make([]Selector, len(m.Selectors)) + for i, s := range m.Selectors { + if s == nil { + return fmt.Errorf("nil selector #%d", i) + } + if err := p.selectors[i].fromProtoMessage(s); err != nil { + return fmt.Errorf("invalid selector #%d: %w", i, err) + } + } + + p.filters = make([]Filter, len(m.Filters)) + for i, f := range m.Filters { + if f == nil { + return fmt.Errorf("nil filter #%d", i) + } + if err := p.filters[i].fromProtoMessage(f); err != nil { + return fmt.Errorf("invalid filter #%d: %w", i, err) + } + } + p.backupFactor = m.GetContainerBackupFactor() - p.selectors = m.GetSelectors() - p.filters = m.GetFilters() return nil } @@ -128,10 +144,7 @@ func (p *PlacementPolicy) readFromV2(m netmap.PlacementPolicy, checkFieldPresenc // // See also Unmarshal. func (p PlacementPolicy) Marshal() []byte { - var m netmap.PlacementPolicy - p.WriteToV2(&m) - - return m.StableMarshal(nil) + return neofsproto.Marshal(p) } // Unmarshal decodes NeoFS API protocol binary format into the PlacementPolicy @@ -140,14 +153,7 @@ func (p PlacementPolicy) Marshal() []byte { // // See also Marshal. func (p *PlacementPolicy) Unmarshal(data []byte) error { - var m netmap.PlacementPolicy - - err := m.Unmarshal(data) - if err != nil { - return err - } - - return p.readFromV2(m, false) + return neofsproto.UnmarshalOptional(data, p, (*PlacementPolicy).fromProtoMessage) } // MarshalJSON encodes PlacementPolicy into a JSON format of the NeoFS API @@ -155,10 +161,7 @@ func (p *PlacementPolicy) Unmarshal(data []byte) error { // // See also UnmarshalJSON. func (p PlacementPolicy) MarshalJSON() ([]byte, error) { - var m netmap.PlacementPolicy - p.WriteToV2(&m) - - return m.MarshalJSON() + return neofsproto.MarshalMessageJSON(p.ProtoMessage()) } // UnmarshalJSON decodes NeoFS API protocol JSON format into the PlacementPolicy @@ -166,51 +169,79 @@ func (p PlacementPolicy) MarshalJSON() ([]byte, error) { // // See also MarshalJSON. func (p *PlacementPolicy) UnmarshalJSON(data []byte) error { - var m netmap.PlacementPolicy - - err := m.UnmarshalJSON(data) - if err != nil { - return err - } - - return p.readFromV2(m, false) + return neofsproto.UnmarshalJSONOptional(data, p, (*PlacementPolicy).fromProtoMessage) } -// ReadFromV2 reads PlacementPolicy from the netmap.PlacementPolicy message. -// Checks if the message conforms to NeoFS API V2 protocol. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// p from it. // -// See also WriteToV2. -func (p *PlacementPolicy) ReadFromV2(m netmap.PlacementPolicy) error { - return p.readFromV2(m, true) +// See also [PlacementPolicy.ProtoMessage]. +func (p *PlacementPolicy) FromProtoMessage(m *protonetmap.PlacementPolicy) error { + return p.fromProtoMessage(m, true) } -// WriteToV2 writes PlacementPolicy to the netmap.PlacementPolicy message. -// The message must not be nil. +// ProtoMessage converts p into message to transmit using the NeoFS API +// protocol. // -// See also ReadFromV2. -func (p PlacementPolicy) WriteToV2(m *netmap.PlacementPolicy) { - m.SetContainerBackupFactor(p.backupFactor) - m.SetFilters(p.filters) - m.SetSelectors(p.selectors) - m.SetReplicas(p.replicas) +// See also [PlacementPolicy.FromProtoMessage]. +func (p PlacementPolicy) ProtoMessage() *protonetmap.PlacementPolicy { + m := &protonetmap.PlacementPolicy{ + ContainerBackupFactor: p.backupFactor, + } + if len(p.replicas) > 0 { + m.Replicas = make([]*protonetmap.Replica, len(p.replicas)) + for i := range p.replicas { + m.Replicas[i] = p.replicas[i].protoMessage() + } + } + if len(p.selectors) > 0 { + m.Selectors = make([]*protonetmap.Selector, len(p.selectors)) + for i := range p.selectors { + m.Selectors[i] = p.selectors[i].protoMessage() + } + } + if len(p.filters) > 0 { + m.Filters = make([]*protonetmap.Filter, len(p.filters)) + for i := range p.filters { + m.Filters[i] = p.filters[i].protoMessage() + } + } + return m } // ReplicaDescriptor replica descriptor characterizes replicas of objects from // the subset selected by a particular Selector. type ReplicaDescriptor struct { - m netmap.Replica + count uint32 + selector string +} + +// fromProtoMessage validates m according to the NeoFS API protocol and restores +// r from it. +func (r *ReplicaDescriptor) fromProtoMessage(m *protonetmap.Replica) { + r.count = m.Count + r.selector = m.Selector +} + +// protoMessage converts r into message to transmit using the NeoFS API +// protocol. +func (r ReplicaDescriptor) protoMessage() *protonetmap.Replica { + return &protonetmap.Replica{ + Count: r.count, + Selector: r.selector, + } } // SetNumberOfObjects sets number of object replicas. func (r *ReplicaDescriptor) SetNumberOfObjects(c uint32) { - r.m.SetCount(c) + r.count = c } // NumberOfObjects returns number set using SetNumberOfObjects. // // Zero ReplicaDescriptor has zero number of objects. func (r ReplicaDescriptor) NumberOfObjects() uint32 { - return r.m.GetCount() + return r.count } // SetSelectorName sets name of the related Selector. @@ -220,7 +251,7 @@ func (r ReplicaDescriptor) NumberOfObjects() uint32 { // // See also [ReplicaDescriptor.SelectorName]. func (r *ReplicaDescriptor) SetSelectorName(s string) { - r.m.SetSelector(s) + r.selector = s } // SelectorName returns name of the related Selector. @@ -230,7 +261,7 @@ func (r *ReplicaDescriptor) SetSelectorName(s string) { // // See also [ReplicaDescriptor.SetSelectorName]. func (r ReplicaDescriptor) SelectorName() string { - return r.m.GetSelector() + return r.selector } // SetReplicas sets list of object replica's characteristics. @@ -238,11 +269,7 @@ func (r ReplicaDescriptor) SelectorName() string { // See also [PlacementPolicy.Replicas], [PlacementPolicy.NumberOfReplicas], // [PlacementPolicy.ReplicaNumberByIndex]. func (p *PlacementPolicy) SetReplicas(rs []ReplicaDescriptor) { - p.replicas = make([]netmap.Replica, len(rs)) - - for i := range rs { - p.replicas[i] = rs[i].m - } + p.replicas = rs } // Replicas returns list of object replica characteristics. @@ -250,11 +277,7 @@ func (p *PlacementPolicy) SetReplicas(rs []ReplicaDescriptor) { // See also [PlacementPolicy.SetReplicas], [PlacementPolicy.NumberOfReplicas], // [PlacementPolicy.ReplicaNumberByIndex]. func (p PlacementPolicy) Replicas() []ReplicaDescriptor { - rs := make([]ReplicaDescriptor, len(p.replicas)) - for i := range p.replicas { - rs[i].m = p.replicas[i] - } - return rs + return p.replicas } // NumberOfReplicas returns number of replica descriptors set using SetReplicas. @@ -270,7 +293,7 @@ func (p PlacementPolicy) NumberOfReplicas() int { // // Zero PlacementPolicy has no replicas. func (p PlacementPolicy) ReplicaNumberByIndex(i int) uint32 { - return p.replicas[i].GetCount() + return p.replicas[i].NumberOfObjects() } // SetContainerBackupFactor sets container backup factor: it controls how deep @@ -296,14 +319,44 @@ func (p *PlacementPolicy) ContainerBackupFactor() uint32 { // Selector describes the bucket selection operator: choose a number of nodes // from the bucket taking the nearest nodes to the related container by hash distance. type Selector struct { - m netmap.Selector + name string + count uint32 + clause protonetmap.Clause + attr string + filter string +} + +// fromProtoMessage validates m according to the NeoFS API protocol and restores +// s from it. +func (s *Selector) fromProtoMessage(m *protonetmap.Selector) error { + if m.Clause < 0 { + return fmt.Errorf("negative clause %d", m.Clause) + } + s.name = m.Name + s.count = m.Count + s.clause = m.Clause + s.attr = m.Attribute + s.filter = m.Filter + return nil +} + +// protoMessage converts s into message to transmit using the NeoFS API +// protocol. +func (s Selector) protoMessage() *protonetmap.Selector { + return &protonetmap.Selector{ + Name: s.name, + Count: s.count, + Clause: s.clause, + Attribute: s.attr, + Filter: s.filter, + } } // SetName sets name with which the Selector can be referenced. // // Zero Selector is unnamed. func (s *Selector) SetName(name string) { - s.m.SetName(name) + s.name = name } // Name returns name with which the Selector can be referenced. @@ -312,7 +365,7 @@ func (s *Selector) SetName(name string) { // // See also [Selector.Name]. func (s Selector) Name() string { - return s.m.GetName() + return s.name } // SetNumberOfNodes sets number of nodes to select from the bucket. @@ -321,7 +374,7 @@ func (s Selector) Name() string { // // See also [Selector.NumberOfNodes]. func (s *Selector) SetNumberOfNodes(num uint32) { - s.m.SetCount(num) + s.count = num } // NumberOfNodes returns number of nodes to select from the bucket. @@ -330,7 +383,7 @@ func (s *Selector) SetNumberOfNodes(num uint32) { // // See also [Selector.SetNumberOfNodes]. func (s Selector) NumberOfNodes() uint32 { - return s.m.GetCount() + return s.count } // SelectByBucketAttribute sets attribute of the bucket to select nodes from. @@ -339,7 +392,7 @@ func (s Selector) NumberOfNodes() uint32 { // // See also [Selector.BucketAttribute]. func (s *Selector) SelectByBucketAttribute(bucket string) { - s.m.SetAttribute(bucket) + s.attr = bucket } // BucketAttribute returns attribute of the bucket to select nodes from. @@ -348,7 +401,7 @@ func (s *Selector) SelectByBucketAttribute(bucket string) { // // See also [Selector.SelectByBucketAttribute]. func (s *Selector) BucketAttribute() string { - return s.m.GetAttribute() + return s.attr } // SelectSame makes selection algorithm to select only nodes having the same values @@ -358,7 +411,7 @@ func (s *Selector) BucketAttribute() string { // // See also [Selector.SelectByBucketAttribute], [Selector.IsSame]. func (s *Selector) SelectSame() { - s.m.SetClause(netmap.Same) + s.clause = protonetmap.Clause_SAME } // IsSame checks whether selection algorithm is set to select only nodes having @@ -366,7 +419,7 @@ func (s *Selector) SelectSame() { // // See also [Selector.SelectSame]. func (s *Selector) IsSame() bool { - return s.m.GetClause() == netmap.Same + return s.clause == protonetmap.Clause_SAME } // SelectDistinct makes selection algorithm to select only nodes having the different values @@ -376,7 +429,7 @@ func (s *Selector) IsSame() bool { // // See also [Selector.SelectByBucketAttribute], [Selector.IsDistinct]. func (s *Selector) SelectDistinct() { - s.m.SetClause(netmap.Distinct) + s.clause = protonetmap.Clause_DISTINCT } // IsDistinct checks whether selection algorithm is set to select only nodes @@ -384,7 +437,7 @@ func (s *Selector) SelectDistinct() { // // See also [Selector.SelectByBucketAttribute], [Selector.SelectDistinct]. func (s *Selector) IsDistinct() bool { - return s.m.GetClause() == netmap.Distinct + return s.clause == protonetmap.Clause_DISTINCT } // SetFilterName sets reference to pre-filtering nodes for selection. @@ -393,7 +446,7 @@ func (s *Selector) IsDistinct() bool { // // See also Filter.SetName. func (s *Selector) SetFilterName(f string) { - s.m.SetFilter(f) + s.filter = f } // FilterName returns reference to pre-filtering nodes for selection. @@ -402,7 +455,7 @@ func (s *Selector) SetFilterName(f string) { // // See also [Filter.SetName], [Selector.SetFilterName]. func (s *Selector) FilterName() string { - return s.m.GetFilter() + return s.filter } // SetSelectors sets list of Selector to form the subset of the nodes to store @@ -412,11 +465,7 @@ func (s *Selector) FilterName() string { // // See also [PlacementPolicy.Selectors]. func (p *PlacementPolicy) SetSelectors(ss []Selector) { - p.selectors = make([]netmap.Selector, len(ss)) - - for i := range ss { - p.selectors[i] = ss[i].m - } + p.selectors = ss } // Selectors returns list of Selector to form the subset of the nodes to store @@ -426,16 +475,55 @@ func (p *PlacementPolicy) SetSelectors(ss []Selector) { // // See also [PlacementPolicy.SetSelectors]. func (p PlacementPolicy) Selectors() []Selector { - ss := make([]Selector, len(p.selectors)) - for i := range p.selectors { - ss[i].m = p.selectors[i] - } - return ss + return p.selectors } // Filter contains rules for filtering the node sets. type Filter struct { - m netmap.Filter + name string + key string + op FilterOp + val string + subs []Filter +} + +// fromProtoMessage validates m according to the NeoFS API protocol and restores +// x from it. +func (x *Filter) fromProtoMessage(m *protonetmap.Filter) error { + if m.Op < 0 { + return fmt.Errorf("negative op %d", m.Op) + } + var subs []Filter + if len(m.Filters) > 0 { + subs = make([]Filter, len(m.Filters)) + for i := range m.Filters { + if err := subs[i].fromProtoMessage(m.Filters[i]); err != nil { + return fmt.Errorf("invalid sub-filter #%d: %w", i, err) + } + } + } + x.name = m.Name + x.setAttribute(m.Key, FilterOp(m.Op), m.Value) + x.subs = subs + return nil +} + +// protoMessage converts x into message to transmit using the NeoFS API +// protocol. +func (x Filter) protoMessage() *protonetmap.Filter { + m := &protonetmap.Filter{ + Name: x.name, + Key: x.key, + Op: protonetmap.Operation(x.op), + Value: x.val, + } + if len(x.subs) > 0 { + m.Filters = make([]*protonetmap.Filter, len(x.subs)) + for i := range x.subs { + m.Filters[i] = x.subs[i].protoMessage() + } + } + return m } // SetName sets name with which the Filter can be referenced or, for inner filters, @@ -446,7 +534,7 @@ type Filter struct { // // See also [Filter.Name]. func (x *Filter) SetName(name string) { - x.m.SetName(name) + x.name = name } // Name returns name with which the Filter can be referenced or, for inner @@ -457,57 +545,47 @@ func (x *Filter) SetName(name string) { // // See also [Filter.SetName]. func (x Filter) Name() string { - return x.m.GetName() + return x.name } // Key returns key to the property. func (x Filter) Key() string { - return x.m.GetKey() + return x.key } // Op returns operator to match the property. func (x Filter) Op() FilterOp { - return FilterOp(x.m.GetOp()) + return x.op } // Value returns value to check the property against. func (x Filter) Value() string { - return x.m.GetValue() + return x.val } // SubFilters returns list of sub-filters when Filter is complex. func (x Filter) SubFilters() []Filter { - fsm := x.m.GetFilters() - if len(fsm) == 0 { - return nil - } - - fs := make([]Filter, len(fsm)) - for i := range fsm { - fs[i] = Filter{m: fsm[i]} - } - - return fs + return x.subs } -func (x *Filter) setAttribute(key string, op netmap.Operation, val string) { - x.m.SetKey(key) - x.m.SetOp(op) - x.m.SetValue(val) +func (x *Filter) setAttribute(key string, op FilterOp, val string) { + x.key = key + x.op = op + x.val = val } // Equal applies the rule to accept only nodes with the same attribute value. // // Method SHOULD NOT be called along with other similar methods. func (x *Filter) Equal(key, value string) { - x.setAttribute(key, netmap.EQ, value) + x.setAttribute(key, FilterOpEQ, value) } // NotEqual applies the rule to accept only nodes with the distinct attribute value. // // Method SHOULD NOT be called along with other similar methods. func (x *Filter) NotEqual(key, value string) { - x.setAttribute(key, netmap.NE, value) + x.setAttribute(key, FilterOpNE, value) } // NumericGT applies the rule to accept only nodes with the numeric attribute @@ -515,7 +593,7 @@ func (x *Filter) NotEqual(key, value string) { // // Method SHOULD NOT be called along with other similar methods. func (x *Filter) NumericGT(key string, num int64) { - x.setAttribute(key, netmap.GT, strconv.FormatInt(num, 10)) + x.setAttribute(key, FilterOpGT, strconv.FormatInt(num, 10)) } // NumericGE applies the rule to accept only nodes with the numeric attribute @@ -523,7 +601,7 @@ func (x *Filter) NumericGT(key string, num int64) { // // Method SHOULD NOT be called along with other similar methods. func (x *Filter) NumericGE(key string, num int64) { - x.setAttribute(key, netmap.GE, strconv.FormatInt(num, 10)) + x.setAttribute(key, FilterOpGE, strconv.FormatInt(num, 10)) } // NumericLT applies the rule to accept only nodes with the numeric attribute @@ -531,7 +609,7 @@ func (x *Filter) NumericGE(key string, num int64) { // // Method SHOULD NOT be called along with other similar methods. func (x *Filter) NumericLT(key string, num int64) { - x.setAttribute(key, netmap.LT, strconv.FormatInt(num, 10)) + x.setAttribute(key, FilterOpLT, strconv.FormatInt(num, 10)) } // NumericLE applies the rule to accept only nodes with the numeric attribute @@ -539,22 +617,12 @@ func (x *Filter) NumericLT(key string, num int64) { // // Method SHOULD NOT be called along with other similar methods. func (x *Filter) NumericLE(key string, num int64) { - x.setAttribute(key, netmap.LE, strconv.FormatInt(num, 10)) + x.setAttribute(key, FilterOpLE, strconv.FormatInt(num, 10)) } -func (x *Filter) setInnerFilters(op netmap.Operation, filters []Filter) { +func (x *Filter) setInnerFilters(op FilterOp, filters []Filter) { x.setAttribute("", op, "") - - inner := x.m.GetFilters() - if rem := len(filters) - len(inner); rem > 0 { - inner = append(inner, make([]netmap.Filter, rem)...) - } - - for i := range filters { - inner[i] = filters[i].m - } - - x.m.SetFilters(inner) + x.subs = filters } // LogicalOR applies the rule to accept only nodes which satisfy at least one @@ -562,7 +630,7 @@ func (x *Filter) setInnerFilters(op netmap.Operation, filters []Filter) { // // Method SHOULD NOT be called along with other similar methods. func (x *Filter) LogicalOR(filters ...Filter) { - x.setInnerFilters(netmap.OR, filters) + x.setInnerFilters(FilterOpOR, filters) } // LogicalAND applies the rule to accept only nodes which satisfy all the given @@ -570,7 +638,7 @@ func (x *Filter) LogicalOR(filters ...Filter) { // // Method SHOULD NOT be called along with other similar methods. func (x *Filter) LogicalAND(filters ...Filter) { - x.setInnerFilters(netmap.AND, filters) + x.setInnerFilters(FilterOpAND, filters) } // Filters returns list of Filter that will be applied when selecting nodes. @@ -579,11 +647,7 @@ func (x *Filter) LogicalAND(filters ...Filter) { // // See also [PlacementPolicy.SetFilters]. func (p PlacementPolicy) Filters() []Filter { - fs := make([]Filter, len(p.filters)) - for i := range p.filters { - fs[i] = Filter{m: p.filters[i]} - } - return fs + return p.filters } // SetFilters sets list of Filter that will be applied when selecting nodes. @@ -592,11 +656,7 @@ func (p PlacementPolicy) Filters() []Filter { // // See also [PlacementPolicy.Filters]. func (p *PlacementPolicy) SetFilters(fs []Filter) { - p.filters = make([]netmap.Filter, len(fs)) - - for i := range fs { - p.filters[i] = fs[i].m - } + p.filters = fs } // WriteStringTo encodes PlacementPolicy into human-readably query and writes @@ -623,8 +683,8 @@ func (p PlacementPolicy) WriteStringTo(w io.StringWriter) (err error) { return err } - c := p.replicas[i].GetCount() - s := p.replicas[i].GetSelector() + c := p.replicas[i].NumberOfObjects() + s := p.replicas[i].SelectorName() if s != "" { _, err = w.WriteString(fmt.Sprintf("REP %d IN %s", c, s)) @@ -657,18 +717,18 @@ func (p PlacementPolicy) WriteStringTo(w io.StringWriter) (err error) { return err } - _, err = w.WriteString(fmt.Sprintf("SELECT %d", p.selectors[i].GetCount())) + _, err = w.WriteString(fmt.Sprintf("SELECT %d", p.selectors[i].NumberOfNodes())) if err != nil { return err } - if s = p.selectors[i].GetAttribute(); s != "" { + if s = p.selectors[i].BucketAttribute(); s != "" { var clause string - switch p.selectors[i].GetClause() { - case netmap.Same: + switch p.selectors[i].clause { + case protonetmap.Clause_SAME: clause = "SAME " - case netmap.Distinct: + case protonetmap.Clause_DISTINCT: clause = "DISTINCT " default: clause = "" @@ -680,14 +740,14 @@ func (p PlacementPolicy) WriteStringTo(w io.StringWriter) (err error) { } } - if s = p.selectors[i].GetFilter(); s != "" { + if s = p.selectors[i].FilterName(); s != "" { _, err = w.WriteString(" FROM " + s) if err != nil { return err } } - if s = p.selectors[i].GetName(); s != "" { + if s = p.selectors[i].Name(); s != "" { _, err = w.WriteString(" AS " + s) if err != nil { return err @@ -715,25 +775,25 @@ func (p PlacementPolicy) WriteStringTo(w io.StringWriter) (err error) { return nil } -func writeFilterStringTo(w io.StringWriter, f netmap.Filter) error { +func writeFilterStringTo(w io.StringWriter, f Filter) error { var err error var s string - op := f.GetOp() + op := f.Op() unspecified := op == 0 - if s = f.GetKey(); s != "" { - _, err = w.WriteString(fmt.Sprintf("%s %s %s", s, op, f.GetValue())) + if s = f.Key(); s != "" { + _, err = w.WriteString(fmt.Sprintf("%s %s %s", s, op, f.Value())) if err != nil { return err } - } else if s = f.GetName(); unspecified && s != "" { + } else if s = f.Name(); unspecified && s != "" { _, err = w.WriteString(fmt.Sprintf("@%s", s)) if err != nil { return err } } - inner := f.GetFilters() + inner := f.SubFilters() for i := range inner { if i != 0 { _, err = w.WriteString(" " + op.String() + " ") @@ -748,7 +808,7 @@ func writeFilterStringTo(w io.StringWriter, f netmap.Filter) error { } } - if s = f.GetName(); s != "" && !unspecified { + if s = f.Name(); s != "" && !unspecified { _, err = w.WriteString(" AS " + s) if err != nil { return err @@ -828,15 +888,14 @@ func (p *policyVisitor) VisitPolicy(ctx *parser.PolicyContext) any { pl := new(PlacementPolicy) repStmts := ctx.AllRepStmt() - pl.replicas = make([]netmap.Replica, 0, len(repStmts)) + pl.replicas = make([]ReplicaDescriptor, len(repStmts)) - for _, r := range repStmts { - res, ok := r.Accept(p).(*netmap.Replica) + for i, r := range repStmts { + res, ok := r.Accept(p).(*protonetmap.Replica) if !ok { return nil } - - pl.replicas = append(pl.replicas, *res) + pl.replicas[i].fromProtoMessage(res) } if cbfStmt := ctx.CbfStmt(); cbfStmt != nil { @@ -848,22 +907,29 @@ func (p *policyVisitor) VisitPolicy(ctx *parser.PolicyContext) any { } selStmts := ctx.AllSelectStmt() - pl.selectors = make([]netmap.Selector, 0, len(selStmts)) + pl.selectors = make([]Selector, len(selStmts)) - for _, s := range selStmts { - res, ok := s.Accept(p).(*netmap.Selector) + for i, s := range selStmts { + res, ok := s.Accept(p).(*protonetmap.Selector) if !ok { return nil } - - pl.selectors = append(pl.selectors, *res) + if err := pl.selectors[i].fromProtoMessage(res); err != nil { + return fmt.Errorf("invalid selector #%d: %w", i, err) + } } filtStmts := ctx.AllFilterStmt() - pl.filters = make([]netmap.Filter, 0, len(filtStmts)) + pl.filters = make([]Filter, len(filtStmts)) - for _, f := range filtStmts { - pl.filters = append(pl.filters, *f.Accept(p).(*netmap.Filter)) + for i, f := range filtStmts { + res, ok := f.Accept(p).(*protonetmap.Filter) + if !ok { + return nil + } + if err := pl.filters[i].fromProtoMessage(res); err != nil { + return fmt.Errorf("invalid filter #%d: %w", i, err) + } } return pl @@ -885,11 +951,11 @@ func (p *policyVisitor) VisitRepStmt(ctx *parser.RepStmtContext) any { return p.reportError(errInvalidNumber) } - rs := new(netmap.Replica) - rs.SetCount(uint32(num)) + rs := new(protonetmap.Replica) + rs.Count = uint32(num) if sel := ctx.GetSelector(); sel != nil { - rs.SetSelector(sel.GetText()) + rs.Selector = sel.GetText() } return rs @@ -902,29 +968,29 @@ func (p *policyVisitor) VisitSelectStmt(ctx *parser.SelectStmtContext) any { return p.reportError(errInvalidNumber) } - s := new(netmap.Selector) - s.SetCount(uint32(res)) + s := new(protonetmap.Selector) + s.Count = uint32(res) if clStmt := ctx.Clause(); clStmt != nil { - s.SetClause(clauseFromString(clStmt.GetText())) + s.Clause = clauseFromString(clStmt.GetText()) } if bStmt := ctx.GetBucket(); bStmt != nil { - s.SetAttribute(ctx.GetBucket().GetText()) + s.Attribute = ctx.GetBucket().GetText() } - s.SetFilter(ctx.GetFilter().GetText()) // either ident or wildcard + s.Filter = ctx.GetFilter().GetText() // either ident or wildcard if ctx.AS() != nil { - s.SetName(ctx.GetName().GetText()) + s.Name = ctx.GetName().GetText() } return s } // VisitFilterStmt implements parser.QueryVisitor interface. func (p *policyVisitor) VisitFilterStmt(ctx *parser.FilterStmtContext) any { - f := p.VisitFilterExpr(ctx.GetExpr().(*parser.FilterExprContext)).(*netmap.Filter) - f.SetName(ctx.GetName().GetText()) + f := p.VisitFilterExpr(ctx.GetExpr().(*parser.FilterExprContext)).(*protonetmap.Filter) + f.Name = ctx.GetName().GetText() return f } @@ -937,21 +1003,21 @@ func (p *policyVisitor) VisitFilterExpr(ctx *parser.FilterExprContext) any { return inner.Accept(p) } - f := new(netmap.Filter) + f := new(protonetmap.Filter) op := operationFromString(ctx.GetOp().GetText()) - f.SetOp(op) + f.Op = op - f1 := *ctx.GetF1().Accept(p).(*netmap.Filter) - f2 := *ctx.GetF2().Accept(p).(*netmap.Filter) + f1 := ctx.GetF1().Accept(p).(*protonetmap.Filter) + f2 := ctx.GetF2().Accept(p).(*protonetmap.Filter) // Consider f1=(.. AND ..) AND f2. This can be merged because our AND operation // is of arbitrary arity. ANTLR generates left-associative parse-tree by default. if f1.GetOp() == op { - f.SetFilters(append(f1.GetFilters(), f2)) + f.Filters = append(f1.GetFilters(), f2) return f } - f.SetFilters([]netmap.Filter{f1, f2}) + f.Filters = []*protonetmap.Filter{f1, f2} return f } @@ -981,9 +1047,9 @@ func (p *policyVisitor) VisitFilterValue(ctx *parser.FilterValueContext) any { // VisitExpr implements parser.QueryVisitor interface. func (p *policyVisitor) VisitExpr(ctx *parser.ExprContext) any { - f := new(netmap.Filter) + f := new(protonetmap.Filter) if flt := ctx.GetFilter(); flt != nil { - f.SetName(flt.GetText()) + f.Name = flt.GetText() return f } @@ -991,9 +1057,9 @@ func (p *policyVisitor) VisitExpr(ctx *parser.ExprContext) any { opStr := ctx.SIMPLE_OP().GetText() value := ctx.GetValue().Accept(p) - f.SetKey(key.(string)) - f.SetOp(operationFromString(opStr)) - f.SetValue(value.(string)) + f.Key = key.(string) + f.Op = operationFromString(opStr) + f.Value = value.(string) return f } @@ -1004,21 +1070,21 @@ func validatePolicy(p PlacementPolicy) error { seenFilters := map[string]bool{} for i := range p.filters { - seenFilters[p.filters[i].GetName()] = true + seenFilters[p.filters[i].Name()] = true } seenSelectors := map[string]bool{} for i := range p.selectors { - if flt := p.selectors[i].GetFilter(); flt != mainFilterName && !seenFilters[flt] { + if flt := p.selectors[i].FilterName(); flt != mainFilterName && !seenFilters[flt] { return fmt.Errorf("%w: '%s'", errUnknownFilter, flt) } - seenSelectors[p.selectors[i].GetName()] = true + seenSelectors[p.selectors[i].Name()] = true } for i := range p.replicas { - if sel := p.replicas[i].GetSelector(); sel != "" && !seenSelectors[sel] { + if sel := p.replicas[i].SelectorName(); sel != "" && !seenSelectors[sel] { return fmt.Errorf("%w: '%s'", errUnknownSelector, sel) } } @@ -1026,20 +1092,42 @@ func validatePolicy(p PlacementPolicy) error { return nil } -func clauseFromString(s string) (c netmap.Clause) { - if !c.FromString(strings.ToUpper(s)) { +func clauseFromString(s string) protonetmap.Clause { + switch s { + default: // Such errors should be handled by ANTLR code thus this panic. - panic(fmt.Errorf("BUG: invalid clause: %s", c)) + panic(fmt.Errorf("BUG: invalid clause: %s", s)) + case "CLAUSE_UNSPECIFIED": + return protonetmap.Clause_CLAUSE_UNSPECIFIED + case "SAME": + return protonetmap.Clause_SAME + case "DISTINCT": + return protonetmap.Clause_DISTINCT } - - return } -func operationFromString(s string) (op netmap.Operation) { - if !op.FromString(strings.ToUpper(s)) { +func operationFromString(s string) protonetmap.Operation { + switch s { + default: // Such errors should be handled by ANTLR code thus this panic. - panic(fmt.Errorf("BUG: invalid operation: %s", op)) + panic(fmt.Errorf("BUG: invalid operation: %s", s)) + case "OPERATION_UNSPECIFIED": + return protonetmap.Operation_OPERATION_UNSPECIFIED + case "EQ": + return protonetmap.Operation_EQ + case "NE": + return protonetmap.Operation_NE + case "GT": + return protonetmap.Operation_GT + case "GE": + return protonetmap.Operation_GE + case "LT": + return protonetmap.Operation_LT + case "LE": + return protonetmap.Operation_LE + case "OR": + return protonetmap.Operation_OR + case "AND": + return protonetmap.Operation_AND } - - return } diff --git a/netmap/policy_internal_test.go b/netmap/policy_internal_test.go index 9c48fad9..97aa622a 100644 --- a/netmap/policy_internal_test.go +++ b/netmap/policy_internal_test.go @@ -4,7 +4,6 @@ import ( "bytes" "testing" - "github.com/nspcc-dev/neofs-api-go/v2/netmap" "github.com/stretchr/testify/require" ) @@ -37,15 +36,15 @@ func TestPlacementPolicy_CopyTo(t *testing.T) { var dst PlacementPolicy pp.CopyTo(&dst) - var f2 netmap.Filter + var f2 Filter f2.SetName("filter2") - require.Equal(t, pp.filters[0].GetName(), dst.filters[0].GetName()) + require.Equal(t, pp.filters[0].Name(), dst.filters[0].Name()) dst.filters[0].SetName("f2") - require.NotEqual(t, pp.filters[0].GetName(), dst.filters[0].GetName()) + require.NotEqual(t, pp.filters[0].Name(), dst.filters[0].Name()) dst.filters[0] = f2 - require.NotEqual(t, pp.filters[0].GetName(), dst.filters[0].GetName()) + require.NotEqual(t, pp.filters[0].Name(), dst.filters[0].Name()) }) t.Run("internal filters", func(t *testing.T) { @@ -54,7 +53,7 @@ func TestPlacementPolicy_CopyTo(t *testing.T) { var topFilter Filter topFilter.SetName("topFilter") - topFilter.setInnerFilters(netmap.EQ, []Filter{includedFilter}) + topFilter.setInnerFilters(FilterOpEQ, []Filter{includedFilter}) var policy PlacementPolicy policy.SetFilters([]Filter{topFilter}) @@ -64,13 +63,13 @@ func TestPlacementPolicy_CopyTo(t *testing.T) { require.True(t, bytes.Equal(policy.Marshal(), dst.Marshal())) t.Run("change extra filter", func(t *testing.T) { - require.Equal(t, topFilter.m.GetName(), dst.filters[0].GetName()) - require.Equal(t, topFilter.m.GetFilters()[0].GetName(), dst.filters[0].GetFilters()[0].GetName()) + require.Equal(t, topFilter.Name(), dst.filters[0].Name()) + require.Equal(t, topFilter.SubFilters()[0].Name(), dst.filters[0].SubFilters()[0].Name()) - dst.filters[0].GetFilters()[0].SetName("someInternalFilterName") + dst.filters[0].SubFilters()[0].SetName("someInternalFilterName") - require.Equal(t, topFilter.m.GetName(), dst.filters[0].GetName()) - require.NotEqual(t, topFilter.m.GetFilters()[0].GetName(), dst.filters[0].GetFilters()[0].GetName()) + require.Equal(t, topFilter.Name(), dst.filters[0].Name()) + require.NotEqual(t, topFilter.SubFilters()[0].Name(), dst.filters[0].SubFilters()[0].Name()) }) }) @@ -88,23 +87,23 @@ func TestPlacementPolicy_CopyTo(t *testing.T) { var dst PlacementPolicy pp.CopyTo(&dst) - require.Equal(t, pp.selectors[0].GetName(), dst.selectors[0].GetName()) + require.Equal(t, pp.selectors[0].Name(), dst.selectors[0].Name()) dst.selectors[0].SetName("s2") - require.NotEqual(t, pp.selectors[0].GetName(), dst.selectors[0].GetName()) + require.NotEqual(t, pp.selectors[0].Name(), dst.selectors[0].Name()) - var s2 netmap.Selector + var s2 Selector s2.SetName("selector2") dst.selectors[0] = s2 - require.NotEqual(t, pp.selectors[0].GetName(), dst.selectors[0].GetName()) + require.NotEqual(t, pp.selectors[0].Name(), dst.selectors[0].Name()) }) t.Run("change replica", func(t *testing.T) { var dst PlacementPolicy pp.CopyTo(&dst) - require.Equal(t, pp.replicas[0].GetSelector(), dst.replicas[0].GetSelector()) - dst.replicas[0].SetSelector("s2") - require.NotEqual(t, pp.replicas[0].GetSelector(), dst.replicas[0].GetSelector()) + require.Equal(t, pp.replicas[0].SelectorName(), dst.replicas[0].SelectorName()) + dst.replicas[0].SetSelectorName("s2") + require.NotEqual(t, pp.replicas[0].SelectorName(), dst.replicas[0].SelectorName()) }) } diff --git a/netmap/policy_test.go b/netmap/policy_test.go index 8b67306d..292e3fe3 100644 --- a/netmap/policy_test.go +++ b/netmap/policy_test.go @@ -5,8 +5,8 @@ import ( "strings" "testing" - apinetmap "github.com/nspcc-dev/neofs-api-go/v2/netmap" "github.com/nspcc-dev/neofs-sdk-go/netmap" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" "github.com/stretchr/testify/require" ) @@ -254,54 +254,33 @@ func TestPlacementPolicy_SetFilters(t *testing.T) { require.Equal(t, anyValidFilters, p.Filters()) } -func TestPlacementPolicy_ReadFromV2(t *testing.T) { - var m apinetmap.PlacementPolicy - m.SetContainerBackupFactor(anyValidBackupFactor) - mrs := make([]apinetmap.Replica, 2) - mrs[0].SetSelector("selector_0") - mrs[0].SetCount(2583748530) - mrs[1].SetSelector("selector_1") - mrs[1].SetCount(358755354) - m.SetReplicas(mrs) - mss := make([]apinetmap.Selector, 2) - mss[0].SetName("selector_0") - mss[0].SetCount(1814781076) - mss[0].SetClause(apinetmap.Same) - mss[0].SetFilter("filter_0") - mss[0].SetAttribute("attribute_0") - mss[1].SetName("selector_1") - mss[1].SetCount(1814781076) - mss[1].SetClause(apinetmap.Distinct) - mss[1].SetFilter("filter_1") - mss[1].SetAttribute("attribute_1") - m.SetSelectors(mss) - msubs := make([]apinetmap.Filter, 0, 2) - addSub := func(name, key string, op apinetmap.Operation, val string) { - var f apinetmap.Filter - f.SetName(name) - f.SetKey(key) - f.SetOp(op) - f.SetValue(val) - msubs = append(msubs, f) +func TestPlacementPolicy_FromProtoMessage(t *testing.T) { + m := &protonetmap.PlacementPolicy{ + Replicas: []*protonetmap.Replica{ + {Count: 2583748530, Selector: "selector_0"}, + {Count: 358755354, Selector: "selector_1"}, + }, + ContainerBackupFactor: anyValidBackupFactor, + Selectors: []*protonetmap.Selector{ + {Name: "selector_0", Count: 1814781076, Clause: protonetmap.Clause_SAME, Attribute: "attribute_0", Filter: "filter_0"}, + {Name: "selector_1", Count: 1814781076, Clause: protonetmap.Clause_DISTINCT, Attribute: "attribute_1", Filter: "filter_1"}, + }, + Filters: []*protonetmap.Filter{ + {Name: "filter_0", Op: protonetmap.Operation_AND, Filters: []*protonetmap.Filter{ + {Name: "filter_0_0", Key: "key_0_0", Op: protonetmap.Operation_EQ, Value: "val_0_0"}, + {Name: "filter_0_1", Key: "key_0_1", Op: protonetmap.Operation_NE, Value: "val_0_1"}, + }}, + {Name: "filter_1", Key: "", Op: protonetmap.Operation_OR, Value: "", Filters: []*protonetmap.Filter{ + {Name: "filter_1_0", Key: "key_1_0", Op: protonetmap.Operation_GT, Value: "1889407708985023116"}, + {Name: "filter_1_1", Key: "key_1_1", Op: protonetmap.Operation_GE, Value: "1429243097315344888"}, + {Name: "filter_1_2", Key: "key_1_2", Op: protonetmap.Operation_LT, Value: "3722656060317482335"}, + {Name: "filter_1_3", Key: "key_1_3", Op: protonetmap.Operation_LE, Value: "1950504987705284805"}, + }}, + }, } - addSub("filter_0_0", "key_0_0", apinetmap.EQ, "val_0_0") - addSub("filter_0_1", "key_0_1", apinetmap.NE, "val_0_1") - mfs := make([]apinetmap.Filter, 2) - mfs[0].SetName("filter_0") - mfs[0].SetOp(apinetmap.AND) - mfs[0].SetFilters(msubs) - msubs = make([]apinetmap.Filter, 0, 4) - addSub("filter_1_0", "key_1_0", apinetmap.GT, "1889407708985023116") - addSub("filter_1_1", "key_1_1", apinetmap.GE, "1429243097315344888") - addSub("filter_1_2", "key_1_2", apinetmap.LT, "3722656060317482335") - addSub("filter_1_3", "key_1_3", apinetmap.LE, "1950504987705284805") - mfs[1].SetName("filter_1") - mfs[1].SetOp(apinetmap.OR) - mfs[1].SetFilters(msubs) - m.SetFilters(mfs) var val netmap.PlacementPolicy - require.NoError(t, val.ReadFromV2(m)) + require.NoError(t, val.FromProtoMessage(m)) require.EqualValues(t, anyValidBackupFactor, val.ContainerBackupFactor()) rs := val.Replicas() require.Len(t, rs, 2) @@ -365,48 +344,56 @@ func TestPlacementPolicy_ReadFromV2(t *testing.T) { require.Empty(t, subs[3].SubFilters()) // reset optional fields - m.SetSelectors(nil) - m.SetFilters(nil) + m.Selectors = nil + m.Filters = nil val2 := val - require.NoError(t, val2.ReadFromV2(m)) + require.NoError(t, val2.FromProtoMessage(m)) require.Empty(t, val2.Selectors()) require.Empty(t, val2.Filters()) t.Run("invalid", func(t *testing.T) { for _, tc := range []struct { name, err string - corrupt func(*apinetmap.PlacementPolicy) + corrupt func(*protonetmap.PlacementPolicy) }{ {name: "replicas/nil", err: "missing replicas", - corrupt: func(m *apinetmap.PlacementPolicy) { m.SetReplicas(nil) }}, + corrupt: func(m *protonetmap.PlacementPolicy) { m.Replicas = nil }}, + {name: "replicas/nil element", err: "nil replica #1", + corrupt: func(m *protonetmap.PlacementPolicy) { m.Replicas[1] = nil }}, {name: "replicas/empty", err: "missing replicas", - corrupt: func(m *apinetmap.PlacementPolicy) { m.SetReplicas([]apinetmap.Replica{}) }}, + corrupt: func(m *protonetmap.PlacementPolicy) { m.Replicas = []*protonetmap.Replica{} }}, + {name: "selectors/nil element", err: "nil selector #1", + corrupt: func(m *protonetmap.PlacementPolicy) { m.Selectors[1] = nil }}, + {name: "selectors/negative clause", err: "invalid selector #1: negative clause -1", + corrupt: func(m *protonetmap.PlacementPolicy) { m.Selectors[1].Clause = -1 }}, + {name: "filters/nil element", err: "nil filter #1", + corrupt: func(m *protonetmap.PlacementPolicy) { m.Filters[1] = nil }}, + {name: "filters/negative op", err: "invalid filter #1: negative op -1", + corrupt: func(m *protonetmap.PlacementPolicy) { m.Filters[1].Op = -1 }}, } { t.Run(tc.name, func(t *testing.T) { st := val - var m apinetmap.PlacementPolicy - st.WriteToV2(&m) - tc.corrupt(&m) - require.EqualError(t, new(netmap.PlacementPolicy).ReadFromV2(m), tc.err) + m := st.ProtoMessage() + tc.corrupt(m) + require.EqualError(t, new(netmap.PlacementPolicy).FromProtoMessage(m), tc.err) }) } }) } -func TestPlacementPolicy_WriteToV2(t *testing.T) { +func TestPlacementPolicy_ProtoMessage(t *testing.T) { var val netmap.PlacementPolicy - var m apinetmap.PlacementPolicy // zero - val.WriteToV2(&m) + m := val.ProtoMessage() require.Zero(t, m.GetContainerBackupFactor()) require.Zero(t, m.GetReplicas()) require.Zero(t, m.GetSelectors()) require.Zero(t, m.GetFilters()) - require.Zero(t, m.GetSubnetID()) + require.Zero(t, m.GetSubnetId()) // filled - validPlacementPolicy.WriteToV2(&m) + m = validPlacementPolicy.ProtoMessage() require.EqualValues(t, anyValidBackupFactor, m.GetContainerBackupFactor()) mrs := m.GetReplicas() @@ -420,12 +407,12 @@ func TestPlacementPolicy_WriteToV2(t *testing.T) { require.Len(t, mss, 2) require.Equal(t, "selector_0", mss[0].GetName()) require.EqualValues(t, 1814781076, mss[0].GetCount()) - require.Equal(t, apinetmap.Same, mss[0].GetClause()) + require.Equal(t, protonetmap.Clause_SAME, mss[0].GetClause()) require.Equal(t, "filter_0", mss[0].GetFilter()) require.Equal(t, "attribute_0", mss[0].GetAttribute()) require.Equal(t, "selector_1", mss[1].GetName()) require.EqualValues(t, 1505136737, mss[1].GetCount()) - require.Equal(t, apinetmap.Distinct, mss[1].GetClause()) + require.Equal(t, protonetmap.Clause_DISTINCT, mss[1].GetClause()) require.Equal(t, "filter_1", mss[1].GetFilter()) require.Equal(t, "attribute_1", mss[1].GetAttribute()) @@ -434,51 +421,51 @@ func TestPlacementPolicy_WriteToV2(t *testing.T) { // filter#0 require.Equal(t, "filter_0", mfs[0].GetName()) require.Zero(t, mfs[0].GetKey()) - require.Equal(t, apinetmap.AND, mfs[0].GetOp()) + require.Equal(t, protonetmap.Operation_AND, mfs[0].GetOp()) require.Zero(t, mfs[0].GetValue()) msubs := mfs[0].GetFilters() require.Len(t, msubs, 2) // sub#0 require.Equal(t, "filter_0_0", msubs[0].GetName()) require.Equal(t, "key_0_0", msubs[0].GetKey()) - require.Equal(t, apinetmap.EQ, msubs[0].GetOp()) + require.Equal(t, protonetmap.Operation_EQ, msubs[0].GetOp()) require.Equal(t, "val_0_0", msubs[0].GetValue()) require.Zero(t, msubs[0].GetFilters()) // sub#1 require.Equal(t, "filter_0_1", msubs[1].GetName()) require.Equal(t, "key_0_1", msubs[1].GetKey()) - require.Equal(t, apinetmap.NE, msubs[1].GetOp()) + require.Equal(t, protonetmap.Operation_NE, msubs[1].GetOp()) require.Equal(t, "val_0_1", msubs[1].GetValue()) require.Zero(t, msubs[1].GetFilters()) // filter#1 require.Equal(t, "filter_1", mfs[1].GetName()) require.Zero(t, mfs[1].GetKey()) - require.Equal(t, apinetmap.OR, mfs[1].GetOp()) + require.Equal(t, protonetmap.Operation_OR, mfs[1].GetOp()) require.Zero(t, mfs[1].GetValue()) msubs = mfs[1].GetFilters() require.Len(t, msubs, 4) // sub#0 require.Equal(t, "filter_1_0", msubs[0].GetName()) require.Equal(t, "key_1_0", msubs[0].GetKey()) - require.Equal(t, apinetmap.GT, msubs[0].GetOp()) + require.Equal(t, protonetmap.Operation_GT, msubs[0].GetOp()) require.Equal(t, "1889407708985023116", msubs[0].GetValue()) require.Zero(t, msubs[0].GetFilters()) // sub#1 require.Equal(t, "filter_1_1", msubs[1].GetName()) require.Equal(t, "key_1_1", msubs[1].GetKey()) - require.Equal(t, apinetmap.GE, msubs[1].GetOp()) + require.Equal(t, protonetmap.Operation_GE, msubs[1].GetOp()) require.Equal(t, "1429243097315344888", msubs[1].GetValue()) require.Zero(t, msubs[1].GetFilters()) // sub#2 require.Equal(t, "filter_1_2", msubs[2].GetName()) require.Equal(t, "key_1_2", msubs[2].GetKey()) - require.Equal(t, apinetmap.LT, msubs[2].GetOp()) + require.Equal(t, protonetmap.Operation_LT, msubs[2].GetOp()) require.Equal(t, "3722656060317482335", msubs[2].GetValue()) require.Zero(t, msubs[2].GetFilters()) // sub#3 require.Equal(t, "filter_1_3", msubs[3].GetName()) require.Equal(t, "key_1_3", msubs[3].GetKey()) - require.Equal(t, apinetmap.LE, msubs[3].GetOp()) + require.Equal(t, protonetmap.Operation_LE, msubs[3].GetOp()) require.Equal(t, "1950504987705284805", msubs[3].GetValue()) require.Zero(t, msubs[3].GetFilters()) } diff --git a/netmap/selector.go b/netmap/selector.go index 24bd90d9..13bda6c8 100644 --- a/netmap/selector.go +++ b/netmap/selector.go @@ -5,21 +5,20 @@ import ( "sort" "github.com/nspcc-dev/hrw/v2" - "github.com/nspcc-dev/neofs-api-go/v2/netmap" ) // processSelectors processes selectors and returns error is any of them is invalid. func (c *context) processSelectors(p PlacementPolicy) error { for i := range p.selectors { - fName := p.selectors[i].GetFilter() + fName := p.selectors[i].FilterName() if fName != mainFilterName { - _, ok := c.processedFilters[p.selectors[i].GetFilter()] + _, ok := c.processedFilters[p.selectors[i].FilterName()] if !ok { return fmt.Errorf("%w: SELECT FROM '%s'", errFilterNotFound, fName) } } - sName := p.selectors[i].GetName() + sName := p.selectors[i].Name() c.processedSelectors[sName] = &p.selectors[i] @@ -36,13 +35,12 @@ func (c *context) processSelectors(p PlacementPolicy) error { // calcNodesCount returns number of buckets and minimum number of nodes in every bucket // for the given selector. -func calcNodesCount(s netmap.Selector) (int, int) { - switch s.GetClause() { - case netmap.Same: - return 1, int(s.GetCount()) - default: - return int(s.GetCount()), 1 +func calcNodesCount(s Selector) (int, int) { + n := int(s.NumberOfNodes()) + if s.IsSame() { + return 1, n } + return n, 1 } // calcBucketWeight computes weight for a node bucket. @@ -56,12 +54,12 @@ func calcBucketWeight(ns nodes, a aggregator, wf weightFunc) float64 { // getSelection returns nodes grouped by s.attribute. // Last argument specifies if more buckets can be used to fulfill CBF. -func (c *context) getSelection(_ PlacementPolicy, s netmap.Selector) ([]nodes, error) { +func (c *context) getSelection(_ PlacementPolicy, s Selector) ([]nodes, error) { bucketCount, nodesInBucket := calcNodesCount(s) buckets := c.getSelectionBase(s) if len(buckets) < bucketCount { - return nil, fmt.Errorf("%w: '%s'", ErrNotEnoughNodes, s.GetName()) + return nil, fmt.Errorf("%w: '%s'", ErrNotEnoughNodes, s.Name()) } // We need deterministic output in case there is no pivot. @@ -69,7 +67,7 @@ func (c *context) getSelection(_ PlacementPolicy, s netmap.Selector) ([]nodes, e // However, because initial order influences HRW order for buckets with equal weights, // we also need to have deterministic input to HRW sorting routine. if len(c.hrwSeed) == 0 { - if s.GetAttribute() == "" { + if s.BucketAttribute() == "" { sort.Slice(buckets, func(i, j int) bool { return less(buckets[i].nodes[0], buckets[j].nodes[0]) }) @@ -97,7 +95,7 @@ func (c *context) getSelection(_ PlacementPolicy, s netmap.Selector) ([]nodes, e // Fallback to using minimum allowed backup factor (1). res = append(res, fallback...) if len(res) < bucketCount { - return nil, fmt.Errorf("%w: '%s'", ErrNotEnoughNodes, s.GetName()) + return nil, fmt.Errorf("%w: '%s'", ErrNotEnoughNodes, s.Name()) } } @@ -110,7 +108,7 @@ func (c *context) getSelection(_ PlacementPolicy, s netmap.Selector) ([]nodes, e hrw.SortWeighted(res, weights, c.hrwSeedHash) } - if s.GetAttribute() == "" { + if s.BucketAttribute() == "" { res, fallback = res[:bucketCount], res[bucketCount:] for i := range fallback { index := i % bucketCount @@ -131,13 +129,13 @@ type nodeAttrPair struct { // getSelectionBase returns nodes grouped by selector attribute. // It it guaranteed that each pair will contain at least one node. -func (c *context) getSelectionBase(s netmap.Selector) []nodeAttrPair { - fName := s.GetFilter() +func (c *context) getSelectionBase(s Selector) []nodeAttrPair { + fName := s.FilterName() f := c.processedFilters[fName] isMain := fName == mainFilterName result := []nodeAttrPair{} nodeMap := map[string][]NodeInfo{} - attr := s.GetAttribute() + attr := s.BucketAttribute() for i := range c.netMap.nodes { if isMain || c.match(f, c.netMap.nodes[i]) { diff --git a/netmap/selector_test.go b/netmap/selector_test.go index 4ab9ac40..20781428 100644 --- a/netmap/selector_test.go +++ b/netmap/selector_test.go @@ -7,8 +7,8 @@ import ( "testing" "github.com/nspcc-dev/hrw/v2" - "github.com/nspcc-dev/neofs-api-go/v2/netmap" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" + protonetmap "github.com/nspcc-dev/neofs-sdk-go/proto/netmap" "github.com/stretchr/testify/require" ) @@ -73,8 +73,8 @@ func BenchmarkPolicyHRWType(b *testing.B) { newSelector("loc1", "Location", 1, "loc1", (*Selector).SelectSame), newSelector("loc2", "Location", 1, "loc2", (*Selector).SelectSame)}, []Filter{ - newFilter("loc1", "Location", "Shanghai", netmap.EQ), - newFilter("loc2", "Location", "Shanghai", netmap.NE), + newFilter("loc1", "Location", "Shanghai", FilterOpEQ), + newFilter("loc2", "Location", "Shanghai", FilterOpNE), }) nodes := make([]NodeInfo, netmapSize) @@ -118,8 +118,8 @@ func TestPlacementPolicy_DeterministicOrder(t *testing.T) { newSelector("loc1", "Location", 1, "loc1", (*Selector).SelectSame), newSelector("loc2", "Location", 1, "loc2", (*Selector).SelectSame)}, []Filter{ - newFilter("loc1", "Location", "Shanghai", netmap.EQ), - newFilter("loc2", "Location", "Shanghai", netmap.NE), + newFilter("loc1", "Location", "Shanghai", FilterOpEQ), + newFilter("loc2", "Location", "Shanghai", FilterOpNE), }) nodeList := make([]NodeInfo, netmapSize) @@ -174,8 +174,8 @@ func TestPlacementPolicy_ProcessSelectors(t *testing.T) { newSelector("Main", "Country", 3, "*", (*Selector).SelectDistinct), }, []Filter{ - newFilter("FromRU", "Country", "Russia", netmap.EQ), - newFilter("Good", "Rating", "4", netmap.GE), + newFilter("FromRU", "Country", "Russia", FilterOpEQ), + newFilter("Good", "Rating", "4", FilterOpGE), }) nodes := []NodeInfo{ nodeInfoFromAttributes("Country", "Russia", "Rating", "1", "City", "SPB"), @@ -199,13 +199,13 @@ func TestPlacementPolicy_ProcessSelectors(t *testing.T) { require.NoError(t, c.processSelectors(p)) for _, s := range p.selectors { - sel := c.selections[s.GetName()] - s := c.processedSelectors[s.GetName()] + sel := c.selections[s.Name()] + s := c.processedSelectors[s.Name()] bucketCount, nodesInBucket := calcNodesCount(*s) nodesInBucket *= int(c.cbf) - targ := fmt.Sprintf("selector '%s'", s.GetName()) + targ := fmt.Sprintf("selector '%s'", s.Name()) require.Equal(t, bucketCount, len(sel), targ) - fName := s.GetFilter() + fName := s.FilterName() for _, res := range sel { require.Equal(t, nodesInBucket, len(res), targ) for j := range res { @@ -219,11 +219,9 @@ func TestSelector_SetName(t *testing.T) { const name = "some name" var s Selector - require.Zero(t, s.m.GetName()) require.Zero(t, s.Name()) s.SetName(name) - require.Equal(t, name, s.m.GetName()) require.Equal(t, name, s.Name()) } @@ -231,29 +229,27 @@ func TestSelector_SetNumberOfNodes(t *testing.T) { const num = 3 var s Selector - require.Zero(t, s.m.GetCount()) require.Zero(t, s.NumberOfNodes()) s.SetNumberOfNodes(num) - require.EqualValues(t, num, s.m.GetCount()) require.EqualValues(t, num, s.NumberOfNodes()) } func TestSelectorClauses(t *testing.T) { var s Selector - require.Equal(t, netmap.UnspecifiedClause, s.m.GetClause()) + require.Equal(t, protonetmap.Clause_CLAUSE_UNSPECIFIED, s.clause) require.False(t, s.IsSame()) require.False(t, s.IsDistinct()) s.SelectDistinct() - require.Equal(t, netmap.Distinct, s.m.GetClause()) + require.Equal(t, protonetmap.Clause_DISTINCT, s.clause) require.False(t, s.IsSame()) require.True(t, s.IsDistinct()) s.SelectSame() - require.Equal(t, netmap.Same, s.m.GetClause()) + require.Equal(t, protonetmap.Clause_SAME, s.clause) require.True(t, s.IsSame()) require.False(t, s.IsDistinct()) } @@ -262,18 +258,18 @@ func TestSelector_SelectByBucketAttribute(t *testing.T) { const attr = "some attribute" var s Selector - require.Zero(t, s.m.GetAttribute()) + require.Zero(t, s.BucketAttribute()) s.SelectByBucketAttribute(attr) - require.Equal(t, attr, s.m.GetAttribute()) + require.Equal(t, attr, s.BucketAttribute()) } func TestSelector_SetFilterName(t *testing.T) { const fName = "some filter" var s Selector - require.Zero(t, s.m.GetFilter()) + require.Zero(t, s.FilterName()) s.SetFilterName(fName) - require.Equal(t, fName, s.m.GetFilter()) + require.Equal(t, fName, s.FilterName()) } diff --git a/object/attribute.go b/object/attribute.go index c22d760d..e5994cdc 100644 --- a/object/attribute.go +++ b/object/attribute.go @@ -1,11 +1,16 @@ package object import ( - "github.com/nspcc-dev/neofs-api-go/v2/object" + "fmt" + "strconv" + + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" + protoobject "github.com/nspcc-dev/neofs-sdk-go/proto/object" ) // Various system attributes. const ( + sysAttrPrefix = "__NEOFS__" // AttributeExpirationEpoch is a key to an object attribute that determines // after what epoch the object becomes expired. Objects that do not have this // attribute never expire. @@ -17,82 +22,97 @@ const ( // Note that the value determines exactly the last epoch of the object's // relevance: for example, with the value N, the object is relevant in epoch N // and expired in any epoch starting from N+1. - AttributeExpirationEpoch = object.SysAttributeExpEpoch + AttributeExpirationEpoch = sysAttrPrefix + "EXPIRATION_EPOCH" ) -// Attribute represents v2-compatible object attribute. -type Attribute object.Attribute - -// NewAttributeFromV2 wraps v2 [object.Attribute] message to [Attribute]. -// -// Nil [object.Attribute] converts to nil. -func NewAttributeFromV2(aV2 *object.Attribute) *Attribute { - return (*Attribute)(aV2) -} +// Attribute represents an object attribute. +type Attribute struct{ k, v string } // NewAttribute creates and initializes new [Attribute]. func NewAttribute(key, value string) *Attribute { - attr := new(object.Attribute) - attr.SetKey(key) - attr.SetValue(value) - - return NewAttributeFromV2(attr) + return &Attribute{key, value} } // Key returns key to the object attribute. func (a *Attribute) Key() string { - return (*object.Attribute)(a).GetKey() + return a.k } // SetKey sets key to the object attribute. func (a *Attribute) SetKey(v string) { - (*object.Attribute)(a).SetKey(v) + a.k = v } // Value return value of the object attribute. func (a *Attribute) Value() string { - return (*object.Attribute)(a).GetValue() + return a.v } // SetValue sets value of the object attribute. func (a *Attribute) SetValue(v string) { - (*object.Attribute)(a).SetValue(v) + a.v = v } -// ToV2 converts [Attribute] to v2 [object.Attribute] message. -// -// Nil [Attribute] converts to nil. -// -// The value returned shares memory with the structure itself, so changing it can lead to data corruption. -// Make a copy if you need to change it. -func (a *Attribute) ToV2() *object.Attribute { - return (*object.Attribute)(a) +// fromProtoMessage validates m according to the NeoFS API protocol and restores +// a from it. +func (a *Attribute) fromProtoMessage(m *protoobject.Header_Attribute, checkFieldPresence bool) error { + if checkFieldPresence && m.Key == "" { + return fmt.Errorf("missing key") + } + if checkFieldPresence && m.Value == "" { + return fmt.Errorf("empty %q value", m.Key) + } + switch m.Key { + case AttributeExpirationEpoch: + if _, err := strconv.ParseUint(m.Value, 10, 64); err != nil { + return fmt.Errorf("invalid expiration epoch (must be a uint): %w", err) + } + } + a.k, a.v = m.Key, m.Value + return nil +} + +// protoMessage converts a into message to transmit using the NeoFS API +// protocol. +func (a *Attribute) protoMessage() *protoobject.Header_Attribute { + if a != nil { + return &protoobject.Header_Attribute{Key: a.k, Value: a.v} + } + return nil } // Marshal marshals [Attribute] into a protobuf binary form. // // See also [Attribute.Unmarshal]. func (a *Attribute) Marshal() []byte { - return (*object.Attribute)(a).StableMarshal(nil) + return neofsproto.MarshalMessage(a.protoMessage()) } // Unmarshal unmarshals protobuf binary representation of [Attribute]. // // See also [Attribute.Marshal]. func (a *Attribute) Unmarshal(data []byte) error { - return (*object.Attribute)(a).Unmarshal(data) + m := new(protoobject.Header_Attribute) + if err := neofsproto.UnmarshalMessage(data, m); err != nil { + return err + } + return a.fromProtoMessage(m, false) } // MarshalJSON encodes [Attribute] to protobuf JSON format. // // See also [Attribute.UnmarshalJSON]. func (a *Attribute) MarshalJSON() ([]byte, error) { - return (*object.Attribute)(a).MarshalJSON() + return neofsproto.MarshalMessageJSON(a.protoMessage()) } // UnmarshalJSON decodes [Attribute] from protobuf JSON format. // // See also [Attribute.MarshalJSON]. func (a *Attribute) UnmarshalJSON(data []byte) error { - return (*object.Attribute)(a).UnmarshalJSON(data) + m := new(protoobject.Header_Attribute) + if err := neofsproto.UnmarshalMessageJSON(data, m); err != nil { + return err + } + return a.fromProtoMessage(m, false) } diff --git a/object/attribute_test.go b/object/attribute_test.go index 15338dfb..26733680 100644 --- a/object/attribute_test.go +++ b/object/attribute_test.go @@ -1,88 +1,59 @@ -package object +package object_test import ( "testing" - "github.com/nspcc-dev/neofs-api-go/v2/object" + "github.com/nspcc-dev/neofs-sdk-go/object" "github.com/stretchr/testify/require" ) -func TestAttribute(t *testing.T) { +func TestNewAttribute(t *testing.T) { key, val := "some key", "some value" - a := NewAttribute(key, val) + a := object.NewAttribute(key, val) require.Equal(t, key, a.Key()) require.Equal(t, val, a.Value()) - - aV2 := a.ToV2() - - require.Equal(t, key, aV2.GetKey()) - require.Equal(t, val, aV2.GetValue()) } -func TestAttributeEncoding(t *testing.T) { - a := NewAttribute("key", "value") - - t.Run("binary", func(t *testing.T) { - a2 := NewAttribute("", "") - require.NoError(t, a2.Unmarshal(a.Marshal())) - - require.Equal(t, a, a2) - }) - - t.Run("json", func(t *testing.T) { - data, err := a.MarshalJSON() - require.NoError(t, err) - - a2 := NewAttribute("", "") - require.NoError(t, a2.UnmarshalJSON(data)) - - require.Equal(t, a, a2) - }) +func TestAttribute_Marshal(t *testing.T) { + // TODO } -func TestNewAttributeFromV2(t *testing.T) { - t.Run("from nil", func(t *testing.T) { - var x *object.Attribute - - require.Nil(t, NewAttributeFromV2(x)) - }) +func TestAttribute_Unmarshal(t *testing.T) { + // TODO } -func TestAttribute_ToV2(t *testing.T) { - t.Run("nil", func(t *testing.T) { - var x *Attribute - - require.Nil(t, x.ToV2()) - }) +func TestAttribute_MarshalJSON(t *testing.T) { + // TODO } -func TestNewAttribute(t *testing.T) { - t.Run("default values", func(t *testing.T) { - a := NewAttribute("", "") - - // check initial values - require.Empty(t, a.Key()) - require.Empty(t, a.Value()) +func TestAttribute_UnmarshalJSON(t *testing.T) { + // TODO +} - // convert to v2 message - aV2 := a.ToV2() +func TestAttribute_SetKey(t *testing.T) { + var a object.Attribute + require.Zero(t, a.Key()) - require.Empty(t, aV2.GetKey()) - require.Empty(t, aV2.GetValue()) - }) + const key = "key" + a.SetKey(key) + require.Equal(t, key, a.Key()) - t.Run("pre installed key and value", func(t *testing.T) { - a := NewAttribute("key", "value") + const otherKey = key + "_other" + a.SetKey(otherKey) + require.Equal(t, otherKey, a.Key()) +} - require.NotEmpty(t, a.Key()) - require.NotEmpty(t, a.Value()) +func TestAttribute_SetValue(t *testing.T) { + var a object.Attribute + require.Zero(t, a.Value()) - // convert to v2 message - aV2 := a.ToV2() + const val = "key" + a.SetValue(val) + require.Equal(t, val, a.Value()) - require.NotEmpty(t, aV2.GetKey()) - require.NotEmpty(t, aV2.GetValue()) - }) + const otherVal = val + "_other" + a.SetKey(otherVal) + require.Equal(t, otherVal, a.Key()) } diff --git a/object/fmt.go b/object/fmt.go index 13576676..73758021 100644 --- a/object/fmt.go +++ b/object/fmt.go @@ -6,9 +6,9 @@ import ( "errors" "fmt" - "github.com/nspcc-dev/neofs-api-go/v2/object" "github.com/nspcc-dev/neofs-sdk-go/checksum" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" ) @@ -55,7 +55,7 @@ func (o Object) VerifyPayloadChecksum() error { // CalculateID calculates identifier for the object. func (o Object) CalculateID() (oid.ID, error) { - return sha256.Sum256(o.ToV2().GetHeader().StableMarshal(nil)), nil + return sha256.Sum256(neofsproto.MarshalMessage(o.header.protoMessage())), nil } // CalculateAndSetID calculates identifier for the object @@ -119,21 +119,15 @@ func (o Object) SignedData() []byte { // VerifySignature verifies object ID signature. func (o Object) VerifySignature() bool { - m := (*object.Object)(&o) - - sigV2 := m.GetSignature() - if sigV2 == nil { + if o.sig == nil { return false } - idV2 := m.GetObjectID() - if idV2 == nil { + if o.id.IsZero() { return false } - var sig neofscrypto.Signature - - return sig.ReadFromV2(*sigV2) == nil && sig.Verify(idV2.StableMarshal(nil)) + return o.sig.Verify(neofsproto.Marshal(o.id)) } // SetIDWithSignature sets object identifier and signature. diff --git a/object/id/id.go b/object/id/id.go index b7ebbdac..f1d4e9e9 100644 --- a/object/id/id.go +++ b/object/id/id.go @@ -9,8 +9,6 @@ import ( neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" "github.com/nspcc-dev/neofs-sdk-go/proto/refs" - "google.golang.org/protobuf/encoding/protojson" - "google.golang.org/protobuf/proto" ) // Size is the size of an [ID] in bytes. @@ -152,32 +150,22 @@ func (id ID) CalculateIDSignature(signer neofscrypto.Signer) (neofscrypto.Signat // Marshal marshals ID into a protobuf binary form. func (id ID) Marshal() []byte { - return neofsproto.MarshalMessage(id.ProtoMessage()) + return neofsproto.Marshal(id) } // Unmarshal unmarshals protobuf binary representation of ID. func (id *ID) Unmarshal(data []byte) error { - var m refs.ObjectID - if err := proto.Unmarshal(data, &m); err != nil { - return err - } - - return id.FromProtoMessage(&m) + return neofsproto.Unmarshal(data, id) } // MarshalJSON encodes ID to protobuf JSON format. func (id ID) MarshalJSON() ([]byte, error) { - return neofsproto.MarshalMessageJSON(id.ProtoMessage()) + return neofsproto.MarshalJSON(id) } // UnmarshalJSON decodes ID from protobuf JSON format. func (id *ID) UnmarshalJSON(data []byte) error { - var m refs.ObjectID - if err := protojson.Unmarshal(data, &m); err != nil { - return err - } - - return id.FromProtoMessage(&m) + return neofsproto.UnmarshalJSON(data, id) } // IsZero checks whether ID is zero. diff --git a/object/id/id_test.go b/object/id/id_test.go index 8e9337ed..6249afd2 100644 --- a/object/id/id_test.go +++ b/object/id/id_test.go @@ -54,7 +54,7 @@ func toProtoJSON(b []byte) []byte { } func TestID_FromProtoMessage(t *testing.T) { - m := &refs.ObjectID{Value: validIDProtoBytes[:]} + m := &refs.ObjectID{Value: validIDBytes[:]} var id oid.ID require.NoError(t, id.FromProtoMessage(m)) require.EqualValues(t, validIDBytes, id) diff --git a/object/link.go b/object/link.go index 4c775e6e..fabdfbdb 100644 --- a/object/link.go +++ b/object/link.go @@ -3,9 +3,9 @@ package object import ( "fmt" - v2object "github.com/nspcc-dev/neofs-api-go/v2/object" - "github.com/nspcc-dev/neofs-api-go/v2/refs" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + protolink "github.com/nspcc-dev/neofs-sdk-go/proto/link" ) // Link is a payload of helper objects that contain the full list of the split @@ -13,7 +13,9 @@ import ( // // Link instance can be written to the [Object], see // [Object.WriteLink]/[Object.ReadLink]. -type Link v2object.Link +type Link struct { + children []MeasuredObject +} // WriteLink writes a link to the Object, and sets its type to [TypeLink]. // @@ -36,100 +38,97 @@ func (o Object) ReadLink(l *Link) error { // // See also [Link.Unmarshal]. func (l *Link) Marshal() []byte { - return (*v2object.Link)(l).StableMarshal(nil) + if l == nil || len(l.children) == 0 { + return nil + } + m := &protolink.Link{ + Children: make([]*protolink.Link_MeasuredObject, len(l.children)), + } + for i := range l.children { + m.Children[i] = &protolink.Link_MeasuredObject{ + Id: l.children[i].id.ProtoMessage(), + Size: l.children[i].sz, + } + } + return neofsproto.MarshalMessage(m) } // Unmarshal decodes the [Link] from its NeoFS protocol binary representation. // // See also [Link.Marshal]. func (l *Link) Unmarshal(data []byte) error { - m := (*v2object.Link)(l) - err := m.Unmarshal(data) + m := new(protolink.Link) + err := neofsproto.UnmarshalMessage(data, m) if err != nil { return err } - var id oid.ID - var i int - m.IterateChildren(func(mo v2object.MeasuredObject) { - if err == nil { - if err = id.ReadFromV2(mo.ID); err != nil { - err = fmt.Errorf("invalid member #%d: %w", i, err) - } + if m.Children == nil { + l.children = nil + return nil + } + + l.children = make([]MeasuredObject, len(m.Children)) + for i := range m.Children { + if m.Children[i] == nil { + return fmt.Errorf("nil child #%d", i) + } + if m.Children[i].Id == nil { + return fmt.Errorf("invalid child #%d: nil ID", i) } - i++ - }) - return err + if err = l.children[i].id.FromProtoMessage(m.Children[i].Id); err != nil { + return fmt.Errorf("invalid child #%d: invalid ID: %w", i, err) + } + l.children[i].sz = m.Children[i].Size + } + + return nil } -// MeasuredObject groups object ID and its size length. It is compatible with -// NeoFS API V2 protocol. -type MeasuredObject v2object.MeasuredObject +// MeasuredObject groups object ID and its size length. +type MeasuredObject struct { + id oid.ID + sz uint32 +} // SetObjectID sets object identifier. // // See also [MeasuredObject.ObjectID]. func (m *MeasuredObject) SetObjectID(id oid.ID) { - var idV2 refs.ObjectID - id.WriteToV2(&idV2) - - m.ID = idV2 + m.id = id } // ObjectID returns object identifier. // // See also [MeasuredObject.SetObjectID]. func (m *MeasuredObject) ObjectID() oid.ID { - var id oid.ID - if m.ID.GetValue() != nil { - if err := id.ReadFromV2(m.ID); err != nil { - panic(fmt.Errorf("invalid ID: %w", err)) - } - } - - return id + return m.id } // SetObjectSize sets size of the object. // // See also [MeasuredObject.ObjectSize]. func (m *MeasuredObject) SetObjectSize(s uint32) { - m.Size = s + m.sz = s } // ObjectSize returns size of the object. // // See also [MeasuredObject.SetObjectSize]. func (m *MeasuredObject) ObjectSize() uint32 { - return m.Size + return m.sz } // Objects returns split chain's measured objects. // // See also [Link.SetObjects]. func (l *Link) Objects() []MeasuredObject { - res := make([]MeasuredObject, (*v2object.Link)(l).NumberOfChildren()) - var i int - var id oid.ID - (*v2object.Link)(l).IterateChildren(func(object v2object.MeasuredObject) { - if err := id.ReadFromV2(object.ID); err != nil { - panic(fmt.Errorf("invalid member #%d: %w", i, err)) - } - res[i] = MeasuredObject(object) - i++ - }) - - return res + return l.children } // SetObjects sets split chain's measured objects. // // See also [Link.Objects]. func (l *Link) SetObjects(oo []MeasuredObject) { - v2OO := make([]v2object.MeasuredObject, len(oo)) - for i, o := range oo { - v2OO[i] = v2object.MeasuredObject(o) - } - - (*v2object.Link)(l).SetChildren(v2OO) + l.children = oo } diff --git a/object/lock.go b/object/lock.go index c153adb7..9e013b75 100644 --- a/object/lock.go +++ b/object/lock.go @@ -3,16 +3,19 @@ package object import ( "fmt" - v2object "github.com/nspcc-dev/neofs-api-go/v2/object" - "github.com/nspcc-dev/neofs-api-go/v2/refs" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + protolock "github.com/nspcc-dev/neofs-sdk-go/proto/lock" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" ) // Lock represents record with locked objects. It is compatible with // NeoFS API V2 protocol. // // Lock instance can be written to the [Object], see WriteLock/ReadLock. -type Lock v2object.Lock +type Lock struct { + members []oid.ID +} // WriteLock writes [Lock] to the [Object], and sets its type to [TypeLock]. // @@ -33,66 +36,63 @@ func (o Object) ReadLock(l *Lock) error { // NumberOfMembers returns number of members in lock list. func (x Lock) NumberOfMembers() int { - return (*v2object.Lock)(&x).NumberOfMembers() + return len(x.members) } // ReadMembers reads list of locked members. // // Buffer length must not be less than [Lock.NumberOfMembers]. func (x Lock) ReadMembers(buf []oid.ID) { - var i int - - (*v2object.Lock)(&x).IterateMembers(func(idV2 refs.ObjectID) { - if err := buf[i].ReadFromV2(idV2); err != nil { - panic(fmt.Errorf("invalid member #%d: %w", i, err)) - } - i++ - }) + copy(buf, x.members) } // WriteMembers writes list of locked members. // // See also [Lock.ReadMembers]. func (x *Lock) WriteMembers(ids []oid.ID) { - var members []refs.ObjectID - - if ids != nil { - members = make([]refs.ObjectID, len(ids)) - - for i := range ids { - ids[i].WriteToV2(&members[i]) - } - } - - (*v2object.Lock)(x).SetMembers(members) + x.members = ids } // Marshal encodes the [Lock] into a NeoFS protocol binary format. // // See also [Lock.Unmarshal]. func (x Lock) Marshal() []byte { - return (*v2object.Lock)(&x).StableMarshal(nil) + if len(x.members) == 0 { + return nil + } + m := &protolock.Lock{ + Members: make([]*refs.ObjectID, len(x.members)), + } + for i := range x.members { + m.Members[i] = x.members[i].ProtoMessage() + } + return neofsproto.MarshalMessage(m) } // Unmarshal decodes the [Lock] from its NeoFS protocol binary representation. // // See also [Lock.Marshal]. func (x *Lock) Unmarshal(data []byte) error { - m := (*v2object.Lock)(x) - err := m.Unmarshal(data) + m := new(protolock.Lock) + err := neofsproto.UnmarshalMessage(data, m) if err != nil { return err } - var id oid.ID - var i int - m.IterateMembers(func(mid refs.ObjectID) { - if err == nil { - if err = id.ReadFromV2(mid); err != nil { - err = fmt.Errorf("invalid member #%d: %w", i, err) - } + if m.Members == nil { + x.members = nil + return nil + } + + x.members = make([]oid.ID, len(m.Members)) + for i := range m.Members { + if m.Members[i] == nil { + return fmt.Errorf("nil member #%d", i) + } + if err = x.members[i].FromProtoMessage(m.Members[i]); err != nil { + return fmt.Errorf("invalid member #%d: %w", i, err) } - i++ - }) - return err + } + + return nil } diff --git a/object/object.go b/object/object.go index 5e1e3989..479d1d31 100644 --- a/object/object.go +++ b/object/object.go @@ -3,30 +3,58 @@ package object import ( "bytes" "fmt" - "strconv" "strings" "github.com/google/uuid" - "github.com/nspcc-dev/neofs-api-go/v2/object" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - v2session "github.com/nspcc-dev/neofs-api-go/v2/session" "github.com/nspcc-dev/neofs-sdk-go/checksum" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + protoobject "github.com/nspcc-dev/neofs-sdk-go/proto/object" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" "github.com/nspcc-dev/neofs-sdk-go/session" "github.com/nspcc-dev/neofs-sdk-go/user" "github.com/nspcc-dev/neofs-sdk-go/version" ) +type split struct { + parID oid.ID + prev oid.ID + parSig *neofscrypto.Signature + parHdr *header + children []oid.ID + id []byte + first oid.ID +} + +type header struct { + version *version.Version + owner user.ID + cnr cid.ID + created uint64 + payloadLn uint64 + pldHash *checksum.Checksum + typ Type + pldHomoHash *checksum.Checksum + session *session.Object + attrs []Attribute + split split +} + // Object represents in-memory structure of the NeoFS object. // Type is compatible with NeoFS API V2 protocol. // // Instance can be created depending on scenario: // - [Object.InitCreation] (an object to be placed in container); // - New (blank instance, usually needed for decoding); -// - [Object.ReadFromV2] (when working under NeoFS API V2 protocol). -type Object object.Object +// - [Object.FromProtoMessage] (when working under NeoFS API V2 protocol). +type Object struct { + id oid.ID + sig *neofscrypto.Signature + header header + pld []byte +} // RequiredFields contains the minimum set of object data that must be set // by the NeoFS user at the stage of creation. @@ -44,218 +72,323 @@ func (o *Object) InitCreation(rf RequiredFields) { o.SetOwner(rf.Owner) } -// NewFromV2 wraps v2 [object.Object] message to [Object]. -// -// Deprecated: BUG: fields' format is not checked. Use [Object.ReadFromV2] -// instead. -func NewFromV2(oV2 *object.Object) *Object { - return (*Object)(oV2) -} - // New creates and initializes blank [Object]. func New() *Object { return new(Object) } -func verifySplitHeaderV2(m object.SplitHeader) error { +func (x split) isZero() bool { + return x.parID.IsZero() && x.prev.IsZero() && x.parSig == nil && x.parHdr == nil && len(x.children) == 0 && + len(x.id) == 0 && x.prev.IsZero() +} + +func (x *split) fromProtoMessage(m *protoobject.Header_Split) error { // parent ID - if mID := m.GetParent(); mID != nil { - if err := new(oid.ID).ReadFromV2(*mID); err != nil { + if m.Parent != nil { + if err := x.parID.FromProtoMessage(m.Parent); err != nil { return fmt.Errorf("invalid parent split member ID: %w", err) } + } else { + x.parID = oid.ID{} } // previous - if mID := m.GetPrevious(); mID != nil { - if err := new(oid.ID).ReadFromV2(*mID); err != nil { + if m.Previous != nil { + if err := x.prev.FromProtoMessage(m.Previous); err != nil { return fmt.Errorf("invalid previous split member ID: %w", err) } + } else { + x.prev = oid.ID{} } // first - if mID := m.GetFirst(); mID != nil { - if err := new(oid.ID).ReadFromV2(*mID); err != nil { + if m.First != nil { + if err := x.first.FromProtoMessage(m.First); err != nil { return fmt.Errorf("invalid first split member ID: %w", err) } + } else { + x.first = oid.ID{} } // split ID - if b := m.GetSplitID(); len(b) > 0 { + if x.id = m.SplitId; len(m.SplitId) > 0 { var uid uuid.UUID - if err := uid.UnmarshalBinary(b); err != nil { + if err := uid.UnmarshalBinary(m.SplitId); err != nil { return fmt.Errorf("invalid split UUID: %w", err) } else if ver := uid.Version(); ver != 4 { return fmt.Errorf("invalid split UUID version %d", ver) } } // children - if mc := m.GetChildren(); len(mc) > 0 { - for i := range mc { - if err := new(oid.ID).ReadFromV2(mc[i]); err != nil { + if len(m.Children) > 0 { + x.children = make([]oid.ID, len(m.Children)) + for i := range m.Children { + if m.Children[i] == nil { + return fmt.Errorf("nil child split member #%d", i) + } + if err := x.children[i].FromProtoMessage(m.Children[i]); err != nil { return fmt.Errorf("invalid child split member ID #%d: %w", i, err) } } + } else { + x.children = nil } // parent signature - if ms := m.GetParentSignature(); ms != nil { - if err := new(neofscrypto.Signature).ReadFromV2(*ms); err != nil { + if m.ParentSignature != nil { + if x.parSig == nil { + x.parSig = new(neofscrypto.Signature) + } + if err := x.parSig.FromProtoMessage(m.ParentSignature); err != nil { return fmt.Errorf("invalid parent signature: %w", err) } + } else { + x.parSig = nil } // parent header - if mh := m.GetParentHeader(); mh != nil { - if err := verifyHeaderV2(*mh); err != nil { + if m.ParentHeader != nil { + if x.parHdr == nil { + x.parHdr = new(header) + } + if err := x.parHdr.fromProtoMessage(m.ParentHeader); err != nil { return fmt.Errorf("invalid parent header: %w", err) } + } else { + x.parHdr = nil } return nil } -func verifyHeaderV2(m object.Header) error { +func (x split) protoMessage() *protoobject.Header_Split { + m := &protoobject.Header_Split{ + SplitId: x.id, + } + if !x.parID.IsZero() { + m.Parent = x.parID.ProtoMessage() + } + if !x.prev.IsZero() { + m.Previous = x.prev.ProtoMessage() + } + if x.parSig != nil { + m.ParentSignature = x.parSig.ProtoMessage() + } + if x.parHdr != nil { + m.ParentHeader = x.parHdr.protoMessage() + } + if len(x.children) > 0 { + m.Children = make([]*refs.ObjectID, len(x.children)) + for i := range x.children { + m.Children[i] = x.children[i].ProtoMessage() + } + } + if !x.first.IsZero() { + m.First = x.first.ProtoMessage() + } + return m +} + +func (x header) isZero() bool { + return x.version == nil && x.owner.IsZero() && x.cnr.IsZero() && x.created == 0 && x.payloadLn == 0 && + x.pldHash == nil && x.typ == 0 && x.pldHomoHash == nil && x.session == nil && len(x.attrs) == 0 && x.split.isZero() +} + +func (x *header) fromProtoMessage(m *protoobject.Header) error { // version - if mv := m.GetVersion(); mv != nil { - if err := new(version.Version).ReadFromV2(*mv); err != nil { + if m.Version != nil { + if x.version == nil { + x.version = new(version.Version) + } + if err := x.version.FromProtoMessage(m.Version); err != nil { return fmt.Errorf("invalid version: %w", err) } + } else { + x.version = nil } // owner - if mu := m.GetOwnerID(); mu != nil { - if err := new(user.ID).ReadFromV2(*mu); err != nil { + if m.OwnerId != nil { + if err := x.owner.FromProtoMessage(m.OwnerId); err != nil { return fmt.Errorf("invalid owner: %w", err) } + } else { + x.owner = user.ID{} } // container - if mc := m.GetContainerID(); mc != nil { - if err := new(cid.ID).ReadFromV2(*mc); err != nil { + if m.ContainerId != nil { + if err := x.cnr.FromProtoMessage(m.ContainerId); err != nil { return fmt.Errorf("invalid container: %w", err) } + } else { + x.cnr = cid.ID{} } // payload checksum - if mc := m.GetPayloadHash(); mc != nil { - if err := new(checksum.Checksum).ReadFromV2(*mc); err != nil { + if m.PayloadHash != nil { + if x.pldHash == nil { + x.pldHash = new(checksum.Checksum) + } + if err := x.pldHash.FromProtoMessage(m.PayloadHash); err != nil { return fmt.Errorf("invalid payload checksum: %w", err) } + } else { + x.pldHash = nil + } + // type + if m.ObjectType < 0 { + return fmt.Errorf("negative type %d", m.ObjectType) } // payload homomorphic checksum - if mc := m.GetHomomorphicHash(); mc != nil { - if err := new(checksum.Checksum).ReadFromV2(*mc); err != nil { + if m.HomomorphicHash != nil { + if x.pldHomoHash == nil { + x.pldHomoHash = new(checksum.Checksum) + } + if err := x.pldHomoHash.FromProtoMessage(m.HomomorphicHash); err != nil { return fmt.Errorf("invalid payload homomorphic checksum: %w", err) } + } else { + x.pldHomoHash = nil } // session token - if ms := m.GetSessionToken(); ms != nil { - if err := new(session.Object).ReadFromV2(*ms); err != nil { + if m.SessionToken != nil { + if x.session == nil { + x.session = new(session.Object) + } + if err := x.session.FromProtoMessage(m.SessionToken); err != nil { return fmt.Errorf("invalid session token: %w", err) } + } else { + x.session = nil } // split header - if ms := m.GetSplit(); ms != nil { - if err := verifySplitHeaderV2(*ms); err != nil { + if m.Split != nil { + if err := x.split.fromProtoMessage(m.Split); err != nil { return fmt.Errorf("invalid split header: %w", err) } + } else { + x.split = split{} } // attributes if ma := m.GetAttributes(); len(ma) > 0 { + x.attrs = make([]Attribute, len(ma)) done := make(map[string]struct{}, len(ma)) for i := range ma { - key := ma[i].GetKey() - if key == "" { - return fmt.Errorf("empty key of the attribute #%d", i) + if ma[i] == nil { + return fmt.Errorf("nil attribute #%d", i) } - if _, ok := done[key]; ok { - return fmt.Errorf("duplicated attribute %s", key) + if _, ok := done[ma[i].Key]; ok { + return fmt.Errorf("duplicated attribute %s", ma[i].Key) } - val := ma[i].GetValue() - if val == "" { - return fmt.Errorf("empty value of the attribute #%d (%s)", i, key) + if err := x.attrs[i].fromProtoMessage(ma[i], true); err != nil { + return fmt.Errorf("invalid attribute #%d: %w", i, err) } - switch key { - case AttributeExpirationEpoch: - if _, err := strconv.ParseUint(val, 10, 64); err != nil { - return fmt.Errorf("invalid expiration attribute (must be a uint): %w", err) - } - } - done[key] = struct{}{} + done[ma[i].Key] = struct{}{} } + } else { + x.attrs = nil } + x.created = m.CreationEpoch + x.payloadLn = m.PayloadLength + x.typ = Type(m.ObjectType) return nil } -// ReadFromV2 reads Object from the [object.Object] message. Returns an error if -// the message is malformed according to the NeoFS API V2 protocol. The message -// must not be nil. +func (x header) protoMessage() *protoobject.Header { + m := &protoobject.Header{ + CreationEpoch: x.created, + PayloadLength: x.payloadLn, + ObjectType: protoobject.ObjectType(x.typ), + } + if x.version != nil { + m.Version = x.version.ProtoMessage() + } + if !x.cnr.IsZero() { + m.ContainerId = x.cnr.ProtoMessage() + } + if !x.owner.IsZero() { + m.OwnerId = x.owner.ProtoMessage() + } + if x.pldHash != nil { + m.PayloadHash = x.pldHash.ProtoMessage() + } + if x.pldHomoHash != nil { + m.HomomorphicHash = x.pldHomoHash.ProtoMessage() + } + if x.session != nil { + m.SessionToken = x.session.ProtoMessage() + } + if len(x.attrs) > 0 { + m.Attributes = make([]*protoobject.Header_Attribute, len(x.attrs)) + for i := range x.attrs { + m.Attributes[i] = &protoobject.Header_Attribute{Key: x.attrs[i].Key(), Value: x.attrs[i].Value()} + } + } + if !x.split.isZero() { + m.Split = x.split.protoMessage() + } + return m +} + +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// o from it. // -// ReadFromV2 is intended to be used by the NeoFS API V2 client/server -// implementation only and is not expected to be directly used by applications. -func (o *Object) ReadFromV2(m object.Object) error { +// See also [Table.ProtoMessage]. +func (o *Object) FromProtoMessage(m *protoobject.Object) error { // ID - if mID := m.GetObjectID(); mID != nil { - if err := new(oid.ID).ReadFromV2(*mID); err != nil { + if m.ObjectId != nil { + if err := o.id.FromProtoMessage(m.ObjectId); err != nil { return fmt.Errorf("invalid ID: %w", err) } + } else { + o.ResetID() } // signature - if ms := m.GetSignature(); ms != nil { - if err := new(neofscrypto.Signature).ReadFromV2(*ms); err != nil { + if m.Signature != nil { + if o.sig == nil { + o.sig = new(neofscrypto.Signature) + } + if err := o.sig.FromProtoMessage(m.Signature); err != nil { return fmt.Errorf("invalid signature: %w", err) } + } else { + o.sig = nil } // header - if mh := m.GetHeader(); mh != nil { - if err := verifyHeaderV2(*mh); err != nil { + if m.Header != nil { + if err := o.header.fromProtoMessage(m.Header); err != nil { return fmt.Errorf("invalid header: %w", err) } + } else { + o.header = header{} } - *o = Object(m) + o.pld = m.Payload return nil } -// ToV2 converts [Object] to v2 [object.Object] message. +// ProtoMessage converts o into message to transmit using the NeoFS API +// protocol. // -// The value returned shares memory with the structure itself, so changing it can lead to data corruption. -// Make a copy if you need to change it. -func (o Object) ToV2() *object.Object { - return (*object.Object)(&o) +// See also [Object.FromProtoMessage]. +func (o Object) ProtoMessage() *protoobject.Object { + m := &protoobject.Object{ + Payload: o.pld, + } + if !o.id.IsZero() { + m.ObjectId = o.id.ProtoMessage() + } + if o.sig != nil { + m.Signature = o.sig.ProtoMessage() + } + if !o.header.isZero() { + m.Header = o.header.protoMessage() + } + return m } // CopyTo writes deep copy of the [Object] to dst. func (o Object) CopyTo(dst *Object) { - id := (*object.Object)(&o).GetObjectID() - (*object.Object)(dst).SetObjectID(copyObjectID(id)) - - sig := (*object.Object)(&o).GetSignature() - (*object.Object)(dst).SetSignature(copySignature(sig)) - - header := (*object.Object)(&o).GetHeader() - (*object.Object)(dst).SetHeader(copyHeader(header)) - + o.header.copyTo(&dst.header) + dst.id = o.id + dst.sig = cloneSignature(o.sig) dst.SetPayload(bytes.Clone(o.Payload())) } // MarshalHeaderJSON marshals object's header into JSON format. func (o Object) MarshalHeaderJSON() ([]byte, error) { - return (*object.Object)(&o).GetHeader().MarshalJSON() -} - -func (o *Object) setHeaderField(setter func(*object.Header)) { - obj := (*object.Object)(o) - h := obj.GetHeader() - - if h == nil { - h = new(object.Header) - obj.SetHeader(h) - } - - setter(h) -} - -func (o *Object) setSplitFields(setter func(*object.SplitHeader)) { - o.setHeaderField(func(h *object.Header) { - split := h.GetSplit() - if split == nil { - split = new(object.SplitHeader) - h.SetSplit(split) - } - - setter(split) - }) + return neofsproto.MarshalMessageJSON(o.header.protoMessage()) } // ID returns object identifier. @@ -271,66 +404,35 @@ func (o *Object) ID() (oid.ID, bool) { // // See also [Object.SetID]. func (o Object) GetID() oid.ID { - var res oid.ID - m := (*object.Object)(&o) - if id := m.GetObjectID(); id != nil { - if err := res.ReadFromV2(*id); err != nil { - panic(fmt.Errorf("unexpected ID decoding failure: %w", err)) - } - } - - return res + return o.id } // SetID sets object identifier. // // See also [Object.GetID]. func (o *Object) SetID(v oid.ID) { - var v2 refs.ObjectID - v.WriteToV2(&v2) - - (*object.Object)(o). - SetObjectID(&v2) + o.id = v } // ResetID removes object identifier. // // See also [Object.SetID]. func (o *Object) ResetID() { - (*object.Object)(o). - SetObjectID(nil) + o.id = oid.ID{} } // Signature returns signature of the object identifier. // // See also [Object.SetSignature]. func (o Object) Signature() *neofscrypto.Signature { - sigv2 := (*object.Object)(&o).GetSignature() - if sigv2 == nil { - return nil - } - - var sig neofscrypto.Signature - if err := sig.ReadFromV2(*sigv2); err != nil { - return nil - } - - return &sig + return o.sig } // SetSignature sets signature of the object identifier. // // See also [Object.Signature]. func (o *Object) SetSignature(v *neofscrypto.Signature) { - var sigv2 *refs.Signature - - if v != nil { - sigv2 = new(refs.Signature) - - v.WriteToV2(sigv2) - } - - (*object.Object)(o).SetSignature(sigv2) + o.sig = v } // Payload returns payload bytes. @@ -340,59 +442,42 @@ func (o *Object) SetSignature(v *neofscrypto.Signature) { // // See also [Object.SetPayload]. func (o Object) Payload() []byte { - return (*object.Object)(&o).GetPayload() + return o.pld } // SetPayload sets payload bytes. // // See also [Object.Payload]. func (o *Object) SetPayload(v []byte) { - (*object.Object)(o).SetPayload(v) + o.pld = v } // Version returns version of the object. Returns nil if the version is unset. // // See also [Object.SetVersion]. func (o Object) Version() *version.Version { - verV2 := (*object.Object)(&o).GetHeader().GetVersion() - if verV2 == nil { - return nil - } - var ver version.Version - if err := ver.ReadFromV2(*verV2); err != nil { - return nil - } - return &ver + return o.header.version } // SetVersion sets version of the object. // // See also [Object.Version]. func (o *Object) SetVersion(v *version.Version) { - var verV2 refs.Version - v.WriteToV2(&verV2) - - o.setHeaderField(func(h *object.Header) { - h.SetVersion(&verV2) - }) + o.header.version = v } // PayloadSize returns payload length of the object. // // See also [Object.SetPayloadSize]. func (o Object) PayloadSize() uint64 { - return (*object.Object)(&o). - GetHeader(). - GetPayloadLength() + return o.header.payloadLn } // SetPayloadSize sets payload length of the object. // // See also [Object.PayloadSize]. func (o *Object) SetPayloadSize(v uint64) { - o.setHeaderField(func(h *object.Header) { - h.SetPayloadLength(v) - }) + o.header.payloadLn = v } // ContainerID returns identifier of the related container. @@ -408,12 +493,7 @@ func (o *Object) ContainerID() (v cid.ID, isSet bool) { // // See also [Object.GetContainerID]. func (o *Object) SetContainerID(v cid.ID) { - var cidV2 refs.ContainerID - v.WriteToV2(&cidV2) - - o.setHeaderField(func(h *object.Header) { - h.SetContainerID(&cidV2) - }) + o.header.cnr = v } // GetContainerID returns identifier of the related container. Zero means unset @@ -421,13 +501,7 @@ func (o *Object) SetContainerID(v cid.ID) { // // See also [Object.SetContainerID]. func (o Object) GetContainerID() cid.ID { - var cnr cid.ID - if m := (*object.Object)(&o).GetHeader().GetContainerID(); m != nil { - if err := cnr.ReadFromV2(*m); err != nil { - panic(fmt.Errorf("unexpected container ID decoding failure: %w", err)) - } - } - return cnr + return o.header.cnr } // OwnerID returns identifier of the object owner. @@ -440,22 +514,7 @@ func (o Object) OwnerID() *user.ID { res := o.Owner(); return &res } // // See also [Object.SetOwner]. func (o Object) Owner() user.ID { - var id user.ID - - m := (*object.Object)(&o).GetHeader().GetOwnerID() - if m != nil { - // unlike other IDs, user.ID.ReadFromV2 also expects correct prefix and checksum - // suffix. So, we cannot call it and panic on error here because nothing - // prevents user from setting incorrect owner ID (Object.SetOwnerID accepts it). - // At the same time, we always expect correct length. - b := m.GetValue() - if len(b) != user.IDSize { - panic(fmt.Errorf("unexpected user ID decoding failure: invalid length %d, expected %d", len(b), user.IDSize)) - } - copy(id[:], b) - } - - return id + return o.header.owner } // SetOwnerID sets identifier of the object owner. @@ -468,30 +527,21 @@ func (o *Object) SetOwnerID(v *user.ID) { o.SetOwner(*v) } // // See also [Object.GetOwner]. func (o *Object) SetOwner(v user.ID) { - o.setHeaderField(func(h *object.Header) { - var m refs.OwnerID - v.WriteToV2(&m) - - h.SetOwnerID(&m) - }) + o.header.owner = v } // CreationEpoch returns epoch number in which object was created. // // See also [Object.SetCreationEpoch]. func (o Object) CreationEpoch() uint64 { - return (*object.Object)(&o). - GetHeader(). - GetCreationEpoch() + return o.header.created } // SetCreationEpoch sets epoch number in which object was created. // // See also [Object.CreationEpoch]. func (o *Object) SetCreationEpoch(v uint64) { - o.setHeaderField(func(h *object.Header) { - h.SetCreationEpoch(v) - }) + o.header.created = v } // PayloadChecksum returns checksum of the object payload and @@ -501,27 +551,17 @@ func (o *Object) SetCreationEpoch(v uint64) { // // See also [Object.SetPayloadChecksum]. func (o Object) PayloadChecksum() (checksum.Checksum, bool) { - var v checksum.Checksum - v2 := (*object.Object)(&o) - - if hash := v2.GetHeader().GetPayloadHash(); hash != nil { - err := v.ReadFromV2(*hash) - return v, (err == nil) + if o.header.pldHash != nil { + return *o.header.pldHash, true } - - return v, false + return checksum.Checksum{}, false } // SetPayloadChecksum sets checksum of the object payload. // // See also [Object.PayloadChecksum]. func (o *Object) SetPayloadChecksum(v checksum.Checksum) { - var v2 refs.Checksum - v.WriteToV2(&v2) - - o.setHeaderField(func(h *object.Header) { - h.SetPayloadHash(&v2) - }) + o.header.pldHash = &v } // PayloadHomomorphicHash returns homomorphic hash of the object @@ -531,27 +571,17 @@ func (o *Object) SetPayloadChecksum(v checksum.Checksum) { // // See also [Object.SetPayloadHomomorphicHash]. func (o Object) PayloadHomomorphicHash() (checksum.Checksum, bool) { - var v checksum.Checksum - v2 := (*object.Object)(&o) - - if hash := v2.GetHeader().GetHomomorphicHash(); hash != nil { - err := v.ReadFromV2(*hash) - return v, (err == nil) + if o.header.pldHomoHash != nil { + return *o.header.pldHomoHash, true } - - return v, false + return checksum.Checksum{}, false } // SetPayloadHomomorphicHash sets homomorphic hash of the object payload. // // See also [Object.PayloadHomomorphicHash]. func (o *Object) SetPayloadHomomorphicHash(v checksum.Checksum) { - var v2 refs.Checksum - v.WriteToV2(&v2) - - o.setHeaderField(func(h *object.Header) { - h.SetHomomorphicHash(&v2) - }) + o.header.pldHomoHash = &v } // Attributes returns all object attributes. @@ -561,17 +591,7 @@ func (o *Object) SetPayloadHomomorphicHash(v checksum.Checksum) { // // See also [Object.SetAttributes], [Object.UserAttributes]. func (o Object) Attributes() []Attribute { - attrs := (*object.Object)(&o). - GetHeader(). - GetAttributes() - - res := make([]Attribute, len(attrs)) - - for i := range attrs { - res[i] = *NewAttributeFromV2(&attrs[i]) - } - - return res + return o.header.attrs } // UserAttributes returns user attributes of the Object. @@ -581,15 +601,11 @@ func (o Object) Attributes() []Attribute { // // See also [Object.SetAttributes], [Object.Attributes]. func (o Object) UserAttributes() []Attribute { - attrs := (*object.Object)(&o). - GetHeader(). - GetAttributes() - - res := make([]Attribute, 0, len(attrs)) + res := make([]Attribute, 0, len(o.header.attrs)) - for i := range attrs { - if !strings.HasPrefix(attrs[i].GetKey(), object.SysAttributePrefix) { - res = append(res, *NewAttributeFromV2(&attrs[i])) + for i := range o.header.attrs { + if !strings.HasPrefix(o.header.attrs[i].Key(), sysAttrPrefix) { + res = append(res, o.header.attrs[i]) } } @@ -598,15 +614,7 @@ func (o Object) UserAttributes() []Attribute { // SetAttributes sets object attributes. func (o *Object) SetAttributes(v ...Attribute) { - attrs := make([]object.Attribute, len(v)) - - for i := range v { - attrs[i] = *v[i].ToV2() - } - - o.setHeaderField(func(h *object.Header) { - h.SetAttributes(attrs) - }) + o.header.attrs = v } // PreviousID returns identifier of the previous sibling object. @@ -623,75 +631,35 @@ func (o *Object) PreviousID() (oid.ID, bool) { // // See also [Object.SetPreviousID]. func (o Object) GetPreviousID() oid.ID { - var id oid.ID - if m := (*object.Object)(&o).GetHeader().GetSplit().GetPrevious(); m != nil { - if err := id.ReadFromV2(*m); err != nil { - panic(fmt.Errorf("unexpected ID decoding failure: %w", err)) - } - } - return id + return o.header.split.prev } // ResetPreviousID resets identifier of the previous sibling object. // // See also [Object.SetPreviousID], [Object.GetPreviousID]. func (o *Object) ResetPreviousID() { - o.setSplitFields(func(split *object.SplitHeader) { - split.SetPrevious(nil) - }) + o.header.split.prev = oid.ID{} } // SetPreviousID sets identifier of the previous sibling object. // // See also [Object.GetPreviousID]. func (o *Object) SetPreviousID(v oid.ID) { - var v2 refs.ObjectID - v.WriteToV2(&v2) - - o.setSplitFields(func(split *object.SplitHeader) { - split.SetPrevious(&v2) - }) + o.header.split.prev = v } // Children return list of the identifiers of the child objects. // // See also [Object.SetChildren]. func (o Object) Children() []oid.ID { - v2 := (*object.Object)(&o) - ids := v2.GetHeader().GetSplit().GetChildren() - - var ( - id oid.ID - res = make([]oid.ID, len(ids)) - ) - - for i := range ids { - if err := id.ReadFromV2(ids[i]); err != nil { - panic(fmt.Errorf("unexpected child#%d decoding failure: %w", i, err)) - } - res[i] = id - } - - return res + return o.header.split.children } // SetChildren sets list of the identifiers of the child objects. // // See also [Object.Children]. func (o *Object) SetChildren(v ...oid.ID) { - var ( - v2 refs.ObjectID - ids = make([]refs.ObjectID, len(v)) - ) - - for i := range v { - v[i].WriteToV2(&v2) - ids[i] = v2 - } - - o.setSplitFields(func(split *object.SplitHeader) { - split.SetChildren(ids) - }) + o.header.split.children = v } // SetFirstID sets the first part's ID of the object's @@ -699,12 +667,7 @@ func (o *Object) SetChildren(v ...oid.ID) { // // See also [Object.GetFirstID]. func (o *Object) SetFirstID(id oid.ID) { - var v2 refs.ObjectID - id.WriteToV2(&v2) - - o.setSplitFields(func(split *object.SplitHeader) { - split.SetFirst(&v2) - }) + o.header.split.first = id } // FirstID returns the first part of the object's split chain. @@ -719,34 +682,21 @@ func (o Object) FirstID() (oid.ID, bool) { // // See also [Object.SetFirstID]. func (o Object) GetFirstID() oid.ID { - var id oid.ID - if m := (*object.Object)(&o).GetHeader().GetSplit().GetFirst(); m != nil { - if err := id.ReadFromV2(*m); err != nil { - panic(fmt.Errorf("unexpected ID decoding failure: %w", err)) - } - } - return id + return o.header.split.first } // SplitID return split identity of split object. If object is not split returns nil. // // See also [Object.SetSplitID]. func (o Object) SplitID() *SplitID { - return NewSplitIDFromV2( - (*object.Object)(&o). - GetHeader(). - GetSplit(). - GetSplitID(), - ) + return NewSplitIDFromV2(o.header.split.id) } // SetSplitID sets split identifier for the split object. // // See also [Object.SplitID]. func (o *Object) SetSplitID(id *SplitID) { - o.setSplitFields(func(split *object.SplitHeader) { - split.SetSplitID(id.ToV2()) - }) + o.header.split.id = id.ToV2() } // ParentID returns identifier of the parent object. @@ -763,118 +713,79 @@ func (o *Object) ParentID() (oid.ID, bool) { // // See also [Object.SetParentID]. func (o Object) GetParentID() oid.ID { - var id oid.ID - if m := (*object.Object)(&o).GetHeader().GetSplit().GetParent(); m != nil { - if err := id.ReadFromV2(*m); err != nil { - panic(fmt.Errorf("unexpected ID decoding failure: %w", err)) - } - } - return id + return o.header.split.parID } // SetParentID sets identifier of the parent object. // // See also [Object.GetParentID]. func (o *Object) SetParentID(v oid.ID) { - var v2 refs.ObjectID - v.WriteToV2(&v2) - - o.setSplitFields(func(split *object.SplitHeader) { - split.SetParent(&v2) - }) + o.header.split.parID = v } // ResetParentID removes identifier of the parent object. // // See also [Object.SetParentID]. func (o *Object) ResetParentID() { - o.setSplitFields(func(split *object.SplitHeader) { - split.SetParent(nil) - }) + o.header.split.parID = oid.ID{} } // Parent returns parent object w/o payload. // // See also [Object.SetParent]. func (o Object) Parent() *Object { - h := (*object.Object)(&o). - GetHeader(). - GetSplit() - - parSig := h.GetParentSignature() - parHdr := h.GetParentHeader() - - if parSig == nil && parHdr == nil { + if o.header.split.parSig == nil && o.header.split.parHdr == nil { return nil } - oV2 := new(object.Object) - oV2.SetObjectID(h.GetParent()) - oV2.SetSignature(parSig) - oV2.SetHeader(parHdr) - - return NewFromV2(oV2) + return &Object{ + header: *o.header.split.parHdr, + id: o.header.split.parID, + sig: o.header.split.parSig, + } } // SetParent sets parent object w/o payload. // // See also [Object.Parent]. func (o *Object) SetParent(v *Object) { - o.setSplitFields(func(split *object.SplitHeader) { - split.SetParent((*object.Object)(v).GetObjectID()) - split.SetParentSignature((*object.Object)(v).GetSignature()) - split.SetParentHeader((*object.Object)(v).GetHeader()) - }) + if v != nil { + o.header.split.parHdr = &v.header + o.header.split.parID = v.id + o.header.split.parSig = v.sig + return + } + o.header.split.parHdr = nil + o.header.split.parID = oid.ID{} + o.header.split.parSig = nil } // SessionToken returns token of the session within which object was created. // // See also [Object.SetSessionToken]. func (o Object) SessionToken() *session.Object { - tokv2 := (*object.Object)(&o).GetHeader().GetSessionToken() - if tokv2 == nil { - return nil - } - - var res session.Object - - _ = res.ReadFromV2(*tokv2) - - return &res + return o.header.session } // SetSessionToken sets token of the session within which object was created. // // See also [Object.SessionToken]. func (o *Object) SetSessionToken(v *session.Object) { - o.setHeaderField(func(h *object.Header) { - var tokv2 *v2session.Token - - if v != nil { - tokv2 = new(v2session.Token) - v.WriteToV2(tokv2) - } - - h.SetSessionToken(tokv2) - }) + o.header.session = v } // Type returns type of the object. // // See also [Object.SetType]. func (o Object) Type() Type { - return Type((*object.Object)(&o). - GetHeader(). - GetObjectType()) + return o.header.typ } // SetType sets type of the object. // // See also [Object.Type]. func (o *Object) SetType(v Type) { - o.setHeaderField(func(h *object.Header) { - h.SetObjectType(object.Type(v)) - }) + o.header.typ = v } // CutPayload returns [Object] w/ empty payload. @@ -882,75 +793,55 @@ func (o *Object) SetType(v Type) { // The value returned shares memory with the structure itself, so changing it can lead to data corruption. // Make a copy if you need to change it. func (o *Object) CutPayload() *Object { - ov2 := new(object.Object) - *ov2 = *(*object.Object)(o) - ov2.SetPayload(nil) - - return (*Object)(ov2) + if o == nil { + return nil + } + return &Object{ + header: o.header, + id: o.id, + sig: o.sig, + } } // HasParent checks if parent (split ID) is present. func (o Object) HasParent() bool { - return (*object.Object)(&o). - GetHeader(). - GetSplit() != nil + return !o.header.split.isZero() } // ResetRelations removes all fields of links with other objects. func (o *Object) ResetRelations() { - o.setHeaderField(func(h *object.Header) { - h.SetSplit(nil) - }) -} - -// InitRelations initializes relation field. -func (o *Object) InitRelations() { - o.setHeaderField(func(h *object.Header) { - h.SetSplit(new(object.SplitHeader)) - }) + o.header.split = split{} } // Marshal marshals object into a protobuf binary form. // // See also [Object.Unmarshal]. func (o Object) Marshal() []byte { - return (*object.Object)(&o).StableMarshal(nil) + return neofsproto.Marshal(o) } // Unmarshal unmarshals protobuf binary representation of object. // // See also [Object.Marshal]. func (o *Object) Unmarshal(data []byte) error { - var m object.Object - err := m.Unmarshal(data) - if err != nil { - return err - } - - return o.ReadFromV2(m) + return neofsproto.Unmarshal(data, o) } // MarshalJSON encodes object to protobuf JSON format. // // See also [Object.UnmarshalJSON]. func (o Object) MarshalJSON() ([]byte, error) { - return (*object.Object)(&o).MarshalJSON() + return neofsproto.MarshalJSON(o) } // UnmarshalJSON decodes object from protobuf JSON format. // // See also [Object.MarshalJSON]. func (o *Object) UnmarshalJSON(data []byte) error { - var m object.Object - err := m.UnmarshalJSON(data) - if err != nil { - return err - } - - return o.ReadFromV2(m) + return neofsproto.UnmarshalJSON(data, o) } // HeaderLen returns length of the binary header. func (o Object) HeaderLen() int { - return (*object.Object)(&o).GetHeader().StableSize() + return o.header.protoMessage().MarshaledSize() } diff --git a/object/object_copy.go b/object/object_copy.go index bdf6bde9..f23b1a98 100644 --- a/object/object_copy.go +++ b/object/object_copy.go @@ -4,153 +4,68 @@ import ( "bytes" "slices" - "github.com/nspcc-dev/neofs-api-go/v2/object" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - v2session "github.com/nspcc-dev/neofs-api-go/v2/session" + "github.com/nspcc-dev/neofs-sdk-go/checksum" + neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" + "github.com/nspcc-dev/neofs-sdk-go/session" ) -func copyObjectID(id *refs.ObjectID) *refs.ObjectID { - if id == nil { +// TODO: Support CopyTo by type. +func cloneSignature(src *neofscrypto.Signature) *neofscrypto.Signature { + if src == nil { return nil } - - var newID refs.ObjectID - newID.SetValue(bytes.Clone(id.GetValue())) - - return &newID + s := neofscrypto.NewSignatureFromRawKey(src.Scheme(), bytes.Clone(src.PublicKeyBytes()), src.Value()) + return &s } -func copySignature(sig *refs.Signature) *refs.Signature { - if sig == nil { +// TODO: Support CopyTo by type. +func cloneChecksum(src *checksum.Checksum) *checksum.Checksum { + if src == nil { return nil } - - var newSig refs.Signature - newSig.SetScheme(sig.GetScheme()) - newSig.SetKey(bytes.Clone(sig.GetKey())) - newSig.SetSign(bytes.Clone(sig.GetSign())) - - return &newSig + s := checksum.New(src.Type(), bytes.Clone(src.Value())) + return &s } -func copySession(session *v2session.Token) *v2session.Token { - if session == nil { - return nil - } - - var newSession v2session.Token - if body := session.GetBody(); body != nil { - var newBody v2session.TokenBody - newBody.SetID(bytes.Clone(body.GetID())) - - if ownerID := body.GetOwnerID(); ownerID != nil { - var newOwnerID refs.OwnerID - newOwnerID.SetValue(bytes.Clone(ownerID.GetValue())) - - newBody.SetOwnerID(&newOwnerID) - } else { - newBody.SetOwnerID(nil) +func (x split) copyTo(dst *split) { + dst.parID = x.parID + dst.prev = x.prev + dst.id = x.id + dst.children = slices.Clone(x.children) + dst.first = x.first + dst.parSig = cloneSignature(x.parSig) + if x.parHdr != nil { + if dst.parHdr == nil { + dst.parHdr = new(header) } - - if lifetime := body.GetLifetime(); lifetime != nil { - newLifetime := *lifetime - newBody.SetLifetime(&newLifetime) - } else { - newBody.SetLifetime(nil) - } - - newBody.SetSessionKey(bytes.Clone(body.GetSessionKey())) - - // it is an interface. Both implementations do nothing inside implemented functions. - newBody.SetContext(body.GetContext()) - - newSession.SetBody(&newBody) + x.parHdr.copyTo(dst.parHdr) } else { - newSession.SetBody(nil) + dst.parHdr = nil } - - newSession.SetSignature(copySignature(session.GetSignature())) - - return &newSession -} - -func copySplitHeader(spl *object.SplitHeader) *object.SplitHeader { - if spl == nil { - return nil - } - - var newSpl object.SplitHeader - - newSpl.SetParent(copyObjectID(spl.GetParent())) - newSpl.SetPrevious(copyObjectID(spl.GetPrevious())) - newSpl.SetFirst(copyObjectID(spl.GetFirst())) - newSpl.SetParentSignature(copySignature(spl.GetParentSignature())) - newSpl.SetParentHeader(copyHeader(spl.GetParentHeader())) - newSpl.SetChildren(slices.Clone(spl.GetChildren())) - newSpl.SetSplitID(bytes.Clone(spl.GetSplitID())) - - return &newSpl } -func copyHeader(header *object.Header) *object.Header { - if header == nil { - return nil - } - - var newHeader object.Header - - newHeader.SetCreationEpoch(header.GetCreationEpoch()) - newHeader.SetPayloadLength(header.GetPayloadLength()) - newHeader.SetObjectType(header.GetObjectType()) - - if ver := header.GetVersion(); ver != nil { - newVer := *ver - newHeader.SetVersion(&newVer) - } else { - newHeader.SetVersion(nil) - } - - if containerID := header.GetContainerID(); containerID != nil { - var newContainerID refs.ContainerID - newContainerID.SetValue(bytes.Clone(containerID.GetValue())) - - newHeader.SetContainerID(&newContainerID) - } else { - newHeader.SetContainerID(nil) - } - - if ownerID := header.GetOwnerID(); ownerID != nil { - var newOwnerID refs.OwnerID - newOwnerID.SetValue(bytes.Clone(ownerID.GetValue())) - - newHeader.SetOwnerID(&newOwnerID) - } else { - newHeader.SetOwnerID(nil) - } - - if payloadHash := header.GetPayloadHash(); payloadHash != nil { - var newPayloadHash refs.Checksum - newPayloadHash.SetType(payloadHash.GetType()) - newPayloadHash.SetSum(bytes.Clone(payloadHash.GetSum())) - - newHeader.SetPayloadHash(&newPayloadHash) +func (x header) copyTo(dst *header) { + dst.cnr = x.cnr + dst.owner = x.owner + dst.created = x.created + dst.payloadLn = x.payloadLn + dst.typ = x.typ + dst.attrs = slices.Clone(x.attrs) + dst.pldHash = cloneChecksum(x.pldHash) + dst.pldHomoHash = cloneChecksum(x.pldHomoHash) + x.split.copyTo(&dst.split) + if x.version != nil { + ver := *x.version + dst.version = &ver } else { - newHeader.SetPayloadHash(nil) + dst.version = nil } - - if homoHash := header.GetHomomorphicHash(); homoHash != nil { - var newHomoHash refs.Checksum - newHomoHash.SetType(homoHash.GetType()) - newHomoHash.SetSum(bytes.Clone(homoHash.GetSum())) - - newHeader.SetHomomorphicHash(&newHomoHash) + if x.session != nil { + if dst.session == nil { + dst.session = new(session.Object) + } + x.session.CopyTo(dst.session) } else { - newHeader.SetHomomorphicHash(nil) + dst.session = nil } - - newHeader.SetSessionToken(copySession(header.GetSessionToken())) - newHeader.SetAttributes(slices.Clone(header.GetAttributes())) - newHeader.SetSplit(copySplitHeader(header.GetSplit())) - - return &newHeader } diff --git a/object/object_internal_test.go b/object/object_internal_test.go index 33b54984..eca0cd11 100644 --- a/object/object_internal_test.go +++ b/object/object_internal_test.go @@ -5,11 +5,11 @@ import ( "testing" "github.com/google/uuid" - "github.com/nspcc-dev/neofs-api-go/v2/object" - "github.com/nspcc-dev/neofs-api-go/v2/refs" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" cidtest "github.com/nspcc-dev/neofs-sdk-go/container/id/test" oidtest "github.com/nspcc-dev/neofs-sdk-go/object/id/test" + protoobject "github.com/nspcc-dev/neofs-sdk-go/proto/object" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" "github.com/nspcc-dev/neofs-sdk-go/session" "github.com/nspcc-dev/neofs-sdk-go/user" usertest "github.com/nspcc-dev/neofs-sdk-go/user/test" @@ -81,11 +81,11 @@ func TestObject_CopyTo(t *testing.T) { var dst Object obj.CopyTo(&dst) - dstHeader := dst.ToV2().GetHeader() + dstHeader := dst.ProtoMessage().GetHeader() require.NotNil(t, dstHeader) - dstHeader.SetObjectType(object.TypeTombstone) + dstHeader.ObjectType = protoobject.ObjectType_TOMBSTONE - objHeader := obj.ToV2().GetHeader() + objHeader := obj.ProtoMessage().GetHeader() require.NotEqual(t, dstHeader.GetObjectType(), objHeader.GetObjectType()) }) @@ -145,36 +145,36 @@ func TestObject_CopyTo(t *testing.T) { t.Run("overwrite header", func(t *testing.T) { var local Object - require.Nil(t, local.ToV2().GetHeader()) + require.Nil(t, local.ProtoMessage().GetHeader()) var dst Object dst.SetAttributes(attr) - require.NotNil(t, dst.ToV2().GetHeader()) + require.NotNil(t, dst.ProtoMessage().GetHeader()) local.CopyTo(&dst) checkObjectEquals(t, local, dst) - require.Nil(t, local.ToV2().GetHeader()) - require.Nil(t, dst.ToV2().GetHeader()) + require.Nil(t, local.ProtoMessage().GetHeader()) + require.Nil(t, dst.ProtoMessage().GetHeader()) dst.SetAttributes(attr) - require.NotNil(t, dst.ToV2().GetHeader()) - require.Nil(t, local.ToV2().GetHeader()) + require.NotNil(t, dst.ProtoMessage().GetHeader()) + require.Nil(t, local.ProtoMessage().GetHeader()) }) t.Run("header, rewrite container id to nil", func(t *testing.T) { var local Object - var localHeader object.Header - local.ToV2().SetHeader(&localHeader) + var localHeader protoobject.Header + local.ProtoMessage().Header = &localHeader var dstContID refs.ContainerID - dstContID.SetValue([]byte{1}) + dstContID.Value = []byte{1} - var dstHeader object.Header - dstHeader.SetContainerID(&dstContID) + var dstHeader protoobject.Header + dstHeader.ContainerId = &dstContID var dst Object - dst.ToV2().SetHeader(&dstHeader) + dst.ProtoMessage().Header = &dstHeader local.CopyTo(&dst) checkObjectEquals(t, local, dst) @@ -183,16 +183,13 @@ func TestObject_CopyTo(t *testing.T) { t.Run("header, change container id", func(t *testing.T) { c := cidtest.ID() b := c[:] - var mc refs.ContainerID - mc.SetValue(b) - var mh object.Header - mh.SetContainerID(&mc) - var mo object.Object - mo.SetHeader(&mh) + mo := &protoobject.Object{Header: &protoobject.Header{ + ContainerId: &refs.ContainerID{Value: b}, + }} var local, dst Object - require.NoError(t, local.ReadFromV2(mo)) - require.NoError(t, dst.ReadFromV2(mo)) + require.NoError(t, local.FromProtoMessage(mo)) + require.NoError(t, dst.FromProtoMessage(mo)) require.Equal(t, local.GetContainerID(), dst.GetContainerID()) b[0]++ @@ -208,15 +205,15 @@ func TestObject_CopyTo(t *testing.T) { }) t.Run("header, rewrite payload hash", func(t *testing.T) { - var cs refs.Checksum - cs.SetType(refs.TillichZemor) - cs.SetSum([]byte{1}) - - var localHeader object.Header - localHeader.SetPayloadHash(&cs) + localHeader := protoobject.Header{ + PayloadHash: &refs.Checksum{ + Type: refs.ChecksumType_TZ, + Sum: []byte{1}, + }, + } var local Object - local.ToV2().SetHeader(&localHeader) + local.ProtoMessage().Header = &localHeader var dst Object local.CopyTo(&dst) @@ -225,15 +222,15 @@ func TestObject_CopyTo(t *testing.T) { }) t.Run("header, rewrite homo hash", func(t *testing.T) { - var cs refs.Checksum - cs.SetType(refs.TillichZemor) - cs.SetSum([]byte{1}) - - var localHeader object.Header - localHeader.SetHomomorphicHash(&cs) + localHeader := protoobject.Header{ + HomomorphicHash: &refs.Checksum{ + Type: refs.ChecksumType_TZ, + Sum: []byte{1}, + }, + } var local Object - local.ToV2().SetHeader(&localHeader) + local.ProtoMessage().Header = &localHeader var dst Object local.CopyTo(&dst) @@ -242,13 +239,10 @@ func TestObject_CopyTo(t *testing.T) { }) t.Run("header, rewrite split header", func(t *testing.T) { - var spl object.SplitHeader - - var localHeader object.Header - localHeader.SetSplit(&spl) + localHeader := protoobject.Header{Split: new(protoobject.Header_Split)} var local Object - local.ToV2().SetHeader(&localHeader) + local.ProtoMessage().Header = &localHeader var dst Object dst.SetChildren(oidtest.ID(), oidtest.ID()) @@ -267,8 +261,8 @@ func TestObject_CopyTo(t *testing.T) { var dst Object require.NotEqual(t, - local.ToV2().GetHeader().GetSessionToken().GetBody().GetOwnerID(), - dst.ToV2().GetHeader().GetSessionToken().GetBody().GetOwnerID(), + local.ProtoMessage().GetHeader().GetSessionToken().GetBody().GetOwnerId(), + dst.ProtoMessage().GetHeader().GetSessionToken().GetBody().GetOwnerId(), ) local.CopyTo(&dst) @@ -286,8 +280,8 @@ func TestObject_CopyTo(t *testing.T) { dst.SetSessionToken(sess) require.NotEqual(t, - local.ToV2().GetHeader().GetSessionToken().GetBody().GetOwnerID(), - dst.ToV2().GetHeader().GetSessionToken().GetBody().GetOwnerID(), + local.ProtoMessage().GetHeader().GetSessionToken().GetBody().GetOwnerId(), + dst.ProtoMessage().GetHeader().GetSessionToken().GetBody().GetOwnerId(), ) local.CopyTo(&dst) @@ -304,8 +298,8 @@ func TestObject_CopyTo(t *testing.T) { var dst Object require.NotEqual(t, - local.ToV2().GetHeader().GetSessionToken().GetBody().GetLifetime(), - dst.ToV2().GetHeader().GetSessionToken().GetBody().GetLifetime(), + local.ProtoMessage().GetHeader().GetSessionToken().GetBody().GetLifetime(), + dst.ProtoMessage().GetHeader().GetSessionToken().GetBody().GetLifetime(), ) local.CopyTo(&dst) @@ -317,7 +311,7 @@ func TestObject_CopyTo(t *testing.T) { sessLocal := sessionToken(cnr) local.SetSessionToken(sessLocal) - local.ToV2().GetHeader().GetSessionToken().SetBody(nil) + local.ProtoMessage().GetHeader().GetSessionToken().Body = nil sessDst := sessionToken(cnr) sessDst.SetID(uuid.New()) @@ -326,8 +320,8 @@ func TestObject_CopyTo(t *testing.T) { dst.SetSessionToken(sessDst) require.NotEqual(t, - local.ToV2().GetHeader().GetSessionToken().GetBody(), - dst.ToV2().GetHeader().GetSessionToken().GetBody(), + local.ProtoMessage().GetHeader().GetSessionToken().GetBody(), + dst.ProtoMessage().GetHeader().GetSessionToken().GetBody(), ) local.CopyTo(&dst) diff --git a/object/object_test.go b/object/object_test.go index 11f3e0e1..5aeb3bb8 100644 --- a/object/object_test.go +++ b/object/object_test.go @@ -4,14 +4,10 @@ import ( "bytes" "crypto/elliptic" "encoding/json" - "math" "math/big" "testing" "github.com/google/uuid" - apiobject "github.com/nspcc-dev/neofs-api-go/v2/object" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - apisession "github.com/nspcc-dev/neofs-api-go/v2/session" cidtest "github.com/nspcc-dev/neofs-sdk-go/container/id/test" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" neofsecdsa "github.com/nspcc-dev/neofs-sdk-go/crypto/ecdsa" @@ -19,6 +15,9 @@ import ( oid "github.com/nspcc-dev/neofs-sdk-go/object/id" oidtest "github.com/nspcc-dev/neofs-sdk-go/object/id/test" objecttest "github.com/nspcc-dev/neofs-sdk-go/object/test" + protoobject "github.com/nspcc-dev/neofs-sdk-go/proto/object" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" "github.com/nspcc-dev/neofs-sdk-go/session" sessiontest "github.com/nspcc-dev/neofs-sdk-go/session/test" usertest "github.com/nspcc-dev/neofs-sdk-go/user/test" @@ -691,790 +690,324 @@ func TestObject_ResetRelations(t *testing.T) { assertNoSplitFields(t, obj) } -func TestObject_InitRelations(t *testing.T) { - var obj object.Object - assertNoSplitFields(t, obj) - require.False(t, obj.HasParent()) - - obj.InitRelations() - assertNoSplitFields(t, obj) - require.True(t, obj.HasParent()) -} - -func TestObject_ReadFromV2(t *testing.T) { - var pv refs.Version - pv.SetMajor(88789927) - pv.SetMinor(2018985309) - var pcs refs.Checksum - pcs.SetType(1974315742) - pcs.SetSum([]byte("checksum_1")) - var phcs refs.Checksum - phcs.SetType(1922538608) - phcs.SetSum([]byte("checksum_2")) - pas := make([]apiobject.Attribute, 3) - pas[0].SetKey("par_attr_key1") - pas[0].SetValue("par_attr_val1") - pas[1].SetKey("__NEOFS__EXPIRATION_EPOCH") - pas[1].SetValue("14208497712700580130") - pas[2].SetKey("par_attr_key2") - pas[2].SetValue("par_attr_val2") - var ph apiobject.Header - ph.SetVersion(&pv) - ph.SetContainerID(protoContainerIDFromBytes(anyValidContainers[0][:])) - ph.SetOwnerID(protoUserIDFromBytes(anyValidUsers[0][:])) - ph.SetCreationEpoch(anyValidCreationEpoch) - ph.SetPayloadLength(anyValidPayloadSize) - ph.SetPayloadHash(&pcs) - ph.SetObjectType(apiobject.Type(anyValidType)) - ph.SetHomomorphicHash(&phcs) - ph.SetAttributes(pas) - - var mtl apisession.TokenLifetime - mtl.SetExp(16429376563136800338) - mtl.SetIat(7956510363313998522) - mtl.SetNbf(17237208928641773338) - var mtc apisession.ObjectSessionContext - mtc.SetVerb(1047242055) - mtc.SetTarget(protoContainerIDFromBytes(anyValidContainers[2][:]), - *protoIDFromBytes(anyValidIDs[8][:]), *protoIDFromBytes(anyValidIDs[9][:])) - var mtb apisession.TokenBody - mtb.SetID(anyValidSessionID[:]) - mtb.SetOwnerID(protoUserIDFromBytes(anyValidUsers[2][:])) - mtb.SetLifetime(&mtl) - mtb.SetSessionKey(anySessionIssuerPubKeyBytes) - mtb.SetContext(&mtc) - var mts refs.Signature - mts.SetScheme(1134494890) - mts.SetKey([]byte("session_signer")) - mts.SetSign([]byte("session_signature")) - var mt apisession.Token - mt.SetBody(&mtb) - mt.SetSignature(&mts) - - var mv refs.Version - mv.SetMajor(525747025) - mv.SetMinor(171993162) - var cs refs.Checksum - cs.SetType(126384577) - cs.SetSum([]byte("checksum_3")) - var hcs refs.Checksum - hcs.SetType(1001923429) - hcs.SetSum([]byte("checksum_4")) - as := make([]apiobject.Attribute, 3) - as[0].SetKey("attr_key1") - as[0].SetValue("attr_val1") - as[1].SetKey("__NEOFS__EXPIRATION_EPOCH") - as[1].SetValue("8516691293958955670") - as[2].SetKey("attr_key2") - as[2].SetValue("attr_val2") - var ps refs.Signature - ps.SetKey([]byte("pub_1")) - ps.SetSign([]byte("sig_1")) - ps.SetScheme(1277002296) - var sh apiobject.SplitHeader - sh.SetParent(protoIDFromBytes(anyValidIDs[1][:])) - sh.SetPrevious(protoIDFromBytes(anyValidIDs[2][:])) - sh.SetParentSignature(&ps) - sh.SetParentHeader(&ph) - sh.SetChildren([]refs.ObjectID{ - *protoIDFromBytes(anyValidIDs[3][:]), - *protoIDFromBytes(anyValidIDs[4][:]), - *protoIDFromBytes(anyValidIDs[5][:]), - }) - sh.SetSplitID(anyValidSplitIDBytes) - sh.SetFirst(protoIDFromBytes(anyValidIDs[6][:])) - - var mh apiobject.Header - mh.SetVersion(&mv) - mh.SetContainerID(protoContainerIDFromBytes(anyValidContainers[1][:])) - mh.SetOwnerID(protoUserIDFromBytes(anyValidUsers[1][:])) - mh.SetCreationEpoch(anyValidCreationEpoch + 1) - mh.SetPayloadLength(anyValidPayloadSize + 1) - mh.SetPayloadHash(&cs) - mh.SetObjectType(apiobject.Type(anyValidType) + 1) - mh.SetHomomorphicHash(&hcs) - mh.SetSessionToken(&mt) - mh.SetAttributes(as) - mh.SetSplit(&sh) - - var ms refs.Signature - ms.SetKey([]byte("pub_2")) - ms.SetSign([]byte("sig_2")) - ms.SetScheme(1242896683) - - var m apiobject.Object - m.SetObjectID(protoIDFromBytes(anyValidIDs[0][:])) - m.SetSignature(&ms) - m.SetHeader(&mh) - m.SetPayload(anyValidRegularPayload) +func TestObject_FromProtoMessage(t *testing.T) { + m := &protoobject.Object{ + ObjectId: protoIDFromBytes(anyValidIDs[0][:]), + Signature: &refs.Signature{Key: []byte("pub_2"), Sign: []byte("sig_2"), Scheme: 1242896683}, + Header: &protoobject.Header{ + Version: &refs.Version{Major: 525747025, Minor: 171993162}, + ContainerId: protoContainerIDFromBytes(anyValidContainers[1][:]), + OwnerId: protoUserIDFromBytes(anyValidUsers[1][:]), + CreationEpoch: anyValidCreationEpoch + 1, + PayloadLength: anyValidPayloadSize + 1, + PayloadHash: &refs.Checksum{Type: 126384577, Sum: []byte("checksum_3")}, + ObjectType: protoobject.ObjectType(anyValidType) + 1, + HomomorphicHash: &refs.Checksum{Type: 1001923429, Sum: []byte("checksum_4")}, + SessionToken: &protosession.SessionToken{ + Body: &protosession.SessionToken_Body{ + Id: anyValidSessionID[:], + OwnerId: protoUserIDFromBytes(anyValidUsers[2][:]), + Lifetime: &protosession.SessionToken_Body_TokenLifetime{ + Exp: 16429376563136800338, + Nbf: 17237208928641773338, + Iat: 7956510363313998522, + }, + SessionKey: anySessionIssuerPubKeyBytes, + Context: &protosession.SessionToken_Body_Object{Object: &protosession.ObjectSessionContext{ + Verb: 1047242055, + Target: &protosession.ObjectSessionContext_Target{ + Container: protoContainerIDFromBytes(anyValidContainers[2][:]), + Objects: []*refs.ObjectID{protoIDFromBytes(anyValidIDs[8][:]), protoIDFromBytes(anyValidIDs[9][:])}, + }, + }}, + }, + Signature: &refs.Signature{Key: []byte("session_signer"), Sign: []byte("session_signature"), Scheme: 1134494890}, + }, + Attributes: []*protoobject.Header_Attribute{ + {Key: "attr_key1", Value: "attr_val1"}, + {Key: "__NEOFS__EXPIRATION_EPOCH", Value: "8516691293958955670"}, + {Key: "attr_key2", Value: "attr_val2"}, + }, + Split: &protoobject.Header_Split{ + Parent: protoIDFromBytes(anyValidIDs[1][:]), + Previous: protoIDFromBytes(anyValidIDs[2][:]), + ParentSignature: &refs.Signature{Key: []byte("pub_1"), Sign: []byte("sig_1"), Scheme: 1277002296}, + ParentHeader: &protoobject.Header{ + Version: &refs.Version{Major: 88789927, Minor: 2018985309}, + ContainerId: protoContainerIDFromBytes(anyValidContainers[0][:]), + OwnerId: protoUserIDFromBytes(anyValidUsers[0][:]), + CreationEpoch: anyValidCreationEpoch, + PayloadLength: anyValidPayloadSize, + PayloadHash: &refs.Checksum{Type: 1974315742, Sum: []byte("checksum_1")}, + ObjectType: protoobject.ObjectType(anyValidType), + HomomorphicHash: &refs.Checksum{Type: 1922538608, Sum: []byte("checksum_2")}, + Attributes: []*protoobject.Header_Attribute{ + {Key: "par_attr_key1", Value: "par_attr_val1"}, + {Key: "__NEOFS__EXPIRATION_EPOCH", Value: "14208497712700580130"}, + {Key: "par_attr_key2", Value: "par_attr_val2"}, + }, + }, + Children: []*refs.ObjectID{ + protoIDFromBytes(anyValidIDs[3][:]), + protoIDFromBytes(anyValidIDs[4][:]), + protoIDFromBytes(anyValidIDs[5][:]), + }, + SplitId: anyValidSplitIDBytes, + First: protoIDFromBytes(anyValidIDs[6][:]), + }, + }, + Payload: anyValidRegularPayload, + } var obj object.Object - require.NoError(t, obj.ReadFromV2(m)) + require.NoError(t, obj.FromProtoMessage(m)) require.Equal(t, validObject, obj) // reset optional fields - m.SetObjectID(nil) - m.SetSignature(nil) - m.SetHeader(nil) - m.SetPayload(nil) + m.ObjectId = nil + m.Signature = nil + m.Header = nil + m.Payload = nil obj2 := obj - require.NoError(t, obj2.ReadFromV2(m)) + require.NoError(t, obj2.FromProtoMessage(m)) require.Zero(t, obj2) t.Run("invalid", func(t *testing.T) { for _, tc := range []struct { name, err string - corrupt func(*apiobject.Object) + corrupt func(*protoobject.Object) }{ {name: "id/nil value", err: "invalid ID: invalid length 0", - corrupt: func(m *apiobject.Object) { m.SetObjectID(protoIDFromBytes(nil)) }}, + corrupt: func(m *protoobject.Object) { m.ObjectId.Value = nil }}, {name: "id/empty value", err: "invalid ID: invalid length 0", - corrupt: func(m *apiobject.Object) { m.SetObjectID(protoIDFromBytes([]byte{})) }}, + corrupt: func(m *protoobject.Object) { m.ObjectId.Value = []byte{} }}, {name: "id/undersize", err: "invalid ID: invalid length 31", - corrupt: func(m *apiobject.Object) { m.SetObjectID(protoIDFromBytes(anyValidIDs[0][:31])) }}, + corrupt: func(m *protoobject.Object) { m.ObjectId.Value = anyValidIDs[0][:31] }}, {name: "id/oversize", err: "invalid ID: invalid length 33", - corrupt: func(m *apiobject.Object) { m.SetObjectID(protoIDFromBytes(append(anyValidIDs[0][:], 1))) }}, + corrupt: func(m *protoobject.Object) { m.ObjectId.Value = append(anyValidIDs[0][:], 1) }}, {name: "id/zero", err: "invalid ID: zero object ID", - corrupt: func(m *apiobject.Object) { m.SetObjectID(protoIDFromBytes(make([]byte, 32))) }}, - {name: "signature/scheme/overflow", err: "invalid signature: scheme 2147483648 overflows int32", - corrupt: func(m *apiobject.Object) { - var s refs.Signature - s.SetScheme(math.MaxInt32 + 1) - m.SetSignature(&s) - }}, + corrupt: func(m *protoobject.Object) { m.ObjectId.Value = make([]byte, 32) }}, + {name: "signature/scheme/negative", err: "invalid signature: negative scheme -1", + corrupt: func(m *protoobject.Object) { m.Signature.Scheme = -1 }}, {name: "header/owner/value/nil", err: "invalid header: invalid owner: invalid length 0, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetOwnerID(protoUserIDFromBytes(nil)) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.OwnerId.Value = nil }}, {name: "header/owner/value/empty", err: "invalid header: invalid owner: invalid length 0, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetOwnerID(protoUserIDFromBytes([]byte{})) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.OwnerId.Value = []byte{} }}, {name: "header/owner/value/undersize", err: "invalid header: invalid owner: invalid length 24, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetOwnerID(protoUserIDFromBytes(anyValidUsers[0][:24])) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.OwnerId.Value = anyValidUsers[0][:24] }}, {name: "header/owner/value/oversize", err: "invalid header: invalid owner: invalid length 26, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetOwnerID(protoUserIDFromBytes(append(anyValidUsers[0][:], 1))) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.OwnerId.Value = append(anyValidUsers[0][:], 1) }}, {name: "header/owner/value/wrong prefix", err: "invalid header: invalid owner: invalid prefix byte 0x42, expected 0x35", - corrupt: func(m *apiobject.Object) { - id := anyValidUsers[0] - id[0] = 0x42 - h := *m.GetHeader() - h.SetOwnerID(protoUserIDFromBytes(id[:])) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.OwnerId.Value[0] = 0x42 }}, {name: "header/owner/value/checksum mismatch", err: "invalid header: invalid owner: checksum mismatch", - corrupt: func(m *apiobject.Object) { - id := anyValidUsers[0] - id[len(id)-1]++ - h := *m.GetHeader() - h.SetOwnerID(protoUserIDFromBytes(id[:])) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.OwnerId.Value[24]++ }}, {name: "header/container/nil value", err: "invalid header: invalid container: invalid length 0", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetContainerID(protoContainerIDFromBytes(nil)) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.ContainerId.Value = nil }}, {name: "header/container/empty value", err: "invalid header: invalid container: invalid length 0", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetContainerID(protoContainerIDFromBytes([]byte{})) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.ContainerId.Value = []byte{} }}, {name: "header/container/undersize", err: "invalid header: invalid container: invalid length 31", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetContainerID(protoContainerIDFromBytes(anyValidContainers[0][:31])) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.ContainerId.Value = anyValidContainers[0][:31] }}, {name: "header/container/oversize", err: "invalid header: invalid container: invalid length 33", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetContainerID(protoContainerIDFromBytes(append(anyValidContainers[0][:], 1))) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.ContainerId.Value = append(anyValidContainers[0][:], 1) }}, {name: "header/container/zero", err: "invalid header: invalid container: zero container ID", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetContainerID(protoContainerIDFromBytes(make([]byte, 32))) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.ContainerId.Value = make([]byte, 32) }}, {name: "header/payload checksum/missing value", err: "invalid header: invalid payload checksum: missing value", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetPayloadHash(new(refs.Checksum)) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.PayloadHash.Sum = nil }}, + {name: "header/payload checksum/negative type", err: "invalid header: invalid payload checksum: negative type -1", + corrupt: func(m *protoobject.Object) { m.Header.PayloadHash.Type = -1 }}, {name: "header/payload homomorphic checksum/missing value", err: "invalid header: invalid payload homomorphic checksum: missing value", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - h.SetHomomorphicHash(new(refs.Checksum)) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.HomomorphicHash.Sum = nil }}, + {name: "header/payload checksum/negative type", err: "invalid header: invalid payload homomorphic checksum: negative type -1", + corrupt: func(m *protoobject.Object) { m.Header.HomomorphicHash.Type = -1 }}, {name: "header/session/body/ID/undersize", err: "invalid header: invalid session token: invalid session ID: invalid UUID (got 15 bytes)", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtb.SetID(anyValidSessionID[:15]) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.SessionToken.Body.Id = anyValidSessionID[:15] }}, {name: "header/session/body/ID/oversize", err: "invalid header: invalid session token: invalid session ID: invalid UUID (got 17 bytes)", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtb.SetID(append(anyValidSessionID[:], 1)) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.SessionToken.Body.Id = append(anyValidSessionID[:], 1) }}, {name: "header/session/body/ID/wrong UUID version", err: "invalid header: invalid session token: invalid session UUID version 3", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() + corrupt: func(m *protoobject.Object) { b := bytes.Clone(anyValidSessionID[:]) b[6] = 3 << 4 - mtb.SetID(b) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) + m.Header.SessionToken.Body.Id = b }}, {name: "header/session/body/issuer/value/empty", err: "invalid header: invalid session token: invalid session issuer: invalid length 0, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtb.SetOwnerID(protoUserIDFromBytes(nil)) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.SessionToken.Body.OwnerId.Value = nil }}, {name: "header/session/body/issuer/value/undersize", err: "invalid header: invalid session token: invalid session issuer: invalid length 24, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtb.SetOwnerID(protoUserIDFromBytes(anyValidUsers[0][:24])) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.SessionToken.Body.OwnerId.Value = anyValidUsers[0][:24] }}, {name: "header/session/body/issuer/value/oversize", err: "invalid header: invalid session token: invalid session issuer: invalid length 26, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtb.SetOwnerID(protoUserIDFromBytes(append(anyValidUsers[0][:], 1))) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.SessionToken.Body.OwnerId.Value = append(anyValidUsers[0][:], 1) }}, {name: "header/session/body/issuer/value/wrong prefix", err: "invalid header: invalid session token: invalid session issuer: invalid prefix byte 0x42, expected 0x35", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() + corrupt: func(m *protoobject.Object) { b := bytes.Clone(anyValidUsers[0][:]) b[0] = 0x42 - mtb.SetOwnerID(protoUserIDFromBytes(b)) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) + m.Header.SessionToken.Body.OwnerId.Value = b }}, {name: "header/session/body/issuer/value/checksum mismatch", err: "invalid header: invalid session token: invalid session issuer: checksum mismatch", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() + corrupt: func(m *protoobject.Object) { b := bytes.Clone(anyValidUsers[0][:]) b[len(b)-1]++ - mtb.SetOwnerID(protoUserIDFromBytes(b)) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) + m.Header.SessionToken.Body.OwnerId.Value = b }}, {name: "header/session/body/context/wrong oneof", err: "invalid header: invalid session token: invalid context: invalid context *session.ContainerSessionContext", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtb.SetContext(new(apisession.ContainerSessionContext)) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.SessionToken.Body.Context = new(protosession.SessionToken_Body_Container) }}, {name: "header/session/body/context/container/empty value", err: "invalid header: invalid session token: invalid context: invalid container ID: invalid length 0", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtc := *mtb.GetContext().(*apisession.ObjectSessionContext) - mtc.SetTarget(protoContainerIDFromBytes(nil), mtc.GetObjects()...) - mtb.SetContext(&mtc) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.SessionToken.Body.Context.(*protosession.SessionToken_Body_Object).Object.Target.Container.Value = nil }}, {name: "header/session/body/context/container/undersize", err: "invalid header: invalid session token: invalid context: invalid container ID: invalid length 31", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtc := *mtb.GetContext().(*apisession.ObjectSessionContext) - mtc.SetTarget(protoContainerIDFromBytes(anyValidContainers[0][:31]), mtc.GetObjects()...) - mtb.SetContext(&mtc) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.SessionToken.Body.Context.(*protosession.SessionToken_Body_Object).Object.Target.Container.Value = anyValidContainers[0][:31] }}, {name: "header/session/body/context/container/oversize", err: "invalid header: invalid session token: invalid context: invalid container ID: invalid length 33", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtc := *mtb.GetContext().(*apisession.ObjectSessionContext) - mtc.SetTarget(protoContainerIDFromBytes(append(anyValidContainers[0][:], 1)), mtc.GetObjects()...) - mtb.SetContext(&mtc) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.SessionToken.Body.Context.(*protosession.SessionToken_Body_Object).Object.Target.Container.Value = + append(anyValidContainers[0][:], 1) }}, {name: "header/session/body/context/object/empty value", err: "invalid header: invalid session token: invalid context: invalid target object: invalid length 0", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtc := *mtb.GetContext().(*apisession.ObjectSessionContext) - objs := make([]refs.ObjectID, 2) - anyValidIDs[0].WriteToV2(&objs[0]) - mtc.SetTarget(protoContainerIDFromBytes(anyValidContainers[0][:]), objs...) - mtb.SetContext(&mtc) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.SessionToken.Body.Context.(*protosession.SessionToken_Body_Object).Object.Target.Objects[1].Value = nil }}, {name: "header/session/body/context/object/undersize", err: "invalid header: invalid session token: invalid context: invalid target object: invalid length 31", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtc := *mtb.GetContext().(*apisession.ObjectSessionContext) - objs := make([]refs.ObjectID, 2) - anyValidIDs[0].WriteToV2(&objs[0]) - objs[1].SetValue(anyValidIDs[1][:31]) - mtc.SetTarget(protoContainerIDFromBytes(anyValidContainers[0][:]), objs...) - mtb.SetContext(&mtc) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.SessionToken.Body.Context.(*protosession.SessionToken_Body_Object).Object.Target.Objects[1].Value = anyValidIDs[1][:31] }}, {name: "header/session/body/context/object/oversize", err: "invalid header: invalid session token: invalid context: invalid target object: invalid length 33", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mtb := *mt.GetBody() - mtc := *mtb.GetContext().(*apisession.ObjectSessionContext) - objs := make([]refs.ObjectID, 2) - anyValidIDs[0].WriteToV2(&objs[0]) - objs[1].SetValue(append(anyValidIDs[1][:], 1)) - mtc.SetTarget(protoContainerIDFromBytes(anyValidContainers[0][:]), objs...) - mtb.SetContext(&mtc) - mt.SetBody(&mtb) - h.SetSessionToken(&mt) - m.SetHeader(&h) - }}, - {name: "header/session/signature/scheme/overflow", err: "invalid header: invalid session token: invalid body signature: scheme 2147483648 overflows int32", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - mt := *h.GetSessionToken() - mts := *mt.GetSignature() - mts.SetScheme(math.MaxInt32 + 1) - mt.SetSignature(&mts) - h.SetSessionToken(&mt) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.SessionToken.Body.Context.(*protosession.SessionToken_Body_Object).Object.Target.Objects[1].Value = + append(anyValidIDs[1][:], 1) }}, + {name: "header/session/signature/scheme/negative", err: "invalid header: invalid session token: invalid body signature: negative scheme -1", + corrupt: func(m *protoobject.Object) { m.Header.SessionToken.Signature.Scheme = -1 }}, {name: "attributes/no key", err: "invalid header: empty key of the attribute #1", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - as := make([]apiobject.Attribute, 3) - as[0].SetKey("k1") - as[0].SetValue("v1") - as[1].SetValue("v2") - as[2].SetKey("k3") - as[2].SetValue("v3") - h.SetAttributes(as) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.Attributes = []*protoobject.Header_Attribute{ + {Key: "k1", Value: "v1"}, {Key: "", Value: "v2"}, {Key: "k3", Value: "v3"}, + } }}, {name: "attributes/no value", err: "invalid header: empty value of the attribute #1 (k2)", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - as := make([]apiobject.Attribute, 3) - as[0].SetKey("k1") - as[0].SetValue("v1") - as[1].SetKey("k2") - as[2].SetKey("k3") - as[2].SetValue("v3") - h.SetAttributes(as) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.Attributes = []*protoobject.Header_Attribute{ + {Key: "k1", Value: "v1"}, {Key: "k2", Value: ""}, {Key: "k3", Value: "v3"}, + } }}, {name: "attributes/duplicated", err: "invalid header: duplicated attribute k1", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - as := make([]apiobject.Attribute, 3) - as[0].SetKey("k1") - as[0].SetValue("v1") - as[1].SetKey("k2") - as[1].SetValue("v2") - as[2].SetKey("k1") - as[2].SetValue("v3") - h.SetAttributes(as) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.Attributes = []*protoobject.Header_Attribute{ + {Key: "k1", Value: "v1"}, {Key: "k2", Value: "v2"}, {Key: "k1", Value: "v3"}, + } }}, {name: "attributes/expiration", err: "invalid header: invalid expiration attribute (must be a uint): strconv.ParseUint: parsing \"foo\": invalid syntax", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - as := make([]apiobject.Attribute, 3) - as[0].SetKey("k1") - as[0].SetValue("v1") - as[1].SetKey("__NEOFS__EXPIRATION_EPOCH") - as[1].SetValue("foo") - as[2].SetKey("k3") - as[2].SetValue("v3") - h.SetAttributes(as) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.Attributes = []*protoobject.Header_Attribute{ + {Key: "k1", Value: "v1"}, {Key: "__NEOFS__EXPIRATION_EPOCH", Value: "foo"}, {Key: "k3", Value: "v3"}, + } }}, {name: "header/split/parent ID/empty value", err: "invalid header: invalid split header: invalid parent split member ID: invalid length 0", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetParent(protoIDFromBytes(nil)) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Parent.Value = nil }}, {name: "header/split/parent ID/undersize", err: "invalid header: invalid split header: invalid parent split member ID: invalid length 31", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetParent(protoIDFromBytes(anyValidIDs[0][:31])) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Parent.Value = anyValidIDs[0][:31] }}, {name: "header/split/parent ID/oversize", err: "invalid header: invalid split header: invalid parent split member ID: invalid length 33", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetParent(protoIDFromBytes(append(anyValidIDs[0][:], 1))) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Parent.Value = append(anyValidIDs[0][:], 1) }}, {name: "header/split/parent ID/zero", err: "invalid header: invalid split header: invalid parent split member ID: zero object ID", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetParent(protoIDFromBytes(make([]byte, 32))) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Parent.Value = make([]byte, 32) }}, {name: "header/split/previous/empty value", err: "invalid header: invalid split header: invalid previous split member ID: invalid length 0", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetPrevious(protoIDFromBytes(nil)) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Previous.Value = nil }}, {name: "header/split/previous/undersize", err: "invalid header: invalid split header: invalid previous split member ID: invalid length 31", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetPrevious(protoIDFromBytes(anyValidIDs[0][:31])) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Previous.Value = anyValidIDs[0][:31] }}, {name: "header/split/previous/oversize", err: "invalid header: invalid split header: invalid previous split member ID: invalid length 33", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetPrevious(protoIDFromBytes(append(anyValidIDs[0][:], 1))) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Previous.Value = append(anyValidIDs[0][:], 1) }}, {name: "header/split/previous/zero", err: "invalid header: invalid split header: invalid previous split member ID: zero object ID", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetPrevious(protoIDFromBytes(make([]byte, 32))) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Previous.Value = make([]byte, 32) }}, {name: "header/split/first/empty value", err: "invalid header: invalid split header: invalid first split member ID: invalid length 0", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetFirst(protoIDFromBytes(nil)) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.First.Value = nil }}, {name: "header/split/first/undersize", err: "invalid header: invalid split header: invalid first split member ID: invalid length 31", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetFirst(protoIDFromBytes(anyValidIDs[0][:31])) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.First.Value = anyValidIDs[0][:31] }}, {name: "header/split/first/oversize", err: "invalid header: invalid split header: invalid first split member ID: invalid length 33", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetFirst(protoIDFromBytes(append(anyValidIDs[0][:], 1))) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.First.Value = append(anyValidIDs[0][:], 1) }}, {name: "header/split/first/zero", err: "invalid header: invalid split header: invalid first split member ID: zero object ID", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetFirst(protoIDFromBytes(make([]byte, 32))) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.First.Value = make([]byte, 32) }}, {name: "header/split/ID/undersize", err: "invalid header: invalid split header: invalid split UUID: invalid UUID (got 15 bytes)", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetSplitID(anyValidSplitIDBytes[:15]) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.SplitId = anyValidSplitIDBytes[:15] }}, {name: "header/split/ID/oversize", err: "invalid header: invalid split header: invalid split UUID: invalid UUID (got 17 bytes)", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - sh.SetSplitID(append(anyValidSplitIDBytes, 1)) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.SplitId = append(anyValidSplitIDBytes, 1) }}, {name: "header/split/ID/wrong UUID version", err: "invalid header: invalid split header: invalid split UUID version 3", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() + corrupt: func(m *protoobject.Object) { b := bytes.Clone(anyValidSplitIDBytes) b[6] = 3 << 4 - sh.SetSplitID(b) - h.SetSplit(&sh) - m.SetHeader(&h) + m.Header.Split.SplitId = b }}, {name: "header/split/children/empty value", err: "invalid header: invalid split header: invalid child split member ID #1: invalid length 0", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - c := make([]refs.ObjectID, 3) - anyValidIDs[0].WriteToV2(&c[0]) - anyValidIDs[2].WriteToV2(&c[2]) - sh.SetChildren(c) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Children[1].Value = nil }}, {name: "header/split/children/undersize", err: "invalid header: invalid split header: invalid child split member ID #1: invalid length 31", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - c := make([]refs.ObjectID, 3) - anyValidIDs[0].WriteToV2(&c[0]) - c[1].SetValue(anyValidIDs[1][:31]) - anyValidIDs[2].WriteToV2(&c[2]) - sh.SetChildren(c) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Children[1].Value = anyValidIDs[1][:31] }}, {name: "header/split/children/oversize", err: "invalid header: invalid split header: invalid child split member ID #1: invalid length 33", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - c := make([]refs.ObjectID, 3) - anyValidIDs[0].WriteToV2(&c[0]) - c[1].SetValue(append(anyValidIDs[1][:], 1)) - anyValidIDs[2].WriteToV2(&c[2]) - sh.SetChildren(c) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Children[1].Value = append(anyValidIDs[1][:], 1) }}, {name: "header/split/children/zero", err: "invalid header: invalid split header: invalid child split member ID #1: zero object ID", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - c := make([]refs.ObjectID, 3) - anyValidIDs[0].WriteToV2(&c[0]) - c[1].SetValue(make([]byte, 32)) - anyValidIDs[2].WriteToV2(&c[2]) - sh.SetChildren(c) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, - {name: "header/split/parent signature/scheme/overflow", err: "invalid header: invalid split header: invalid parent signature: scheme 2147483648 overflows int32", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - var s refs.Signature - anyValidSignatures[0].WriteToV2(&s) - s.SetScheme(math.MaxInt32 + 1) - sh.SetParentSignature(&s) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.Children[1].Value = make([]byte, 32) }}, + {name: "header/split/parent signature/scheme/negative", err: "invalid header: invalid split header: invalid parent signature: negative scheme -1", + corrupt: func(m *protoobject.Object) { m.Header.Split.ParentSignature.Scheme = -1 }}, {name: "header/split/parent/owner/value/empty", err: "invalid header: invalid split header: invalid parent header: invalid owner: invalid length 0, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() - ph.SetOwnerID(protoUserIDFromBytes(nil)) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.ParentHeader.OwnerId.Value = nil }}, {name: "header/split/parent/owner/value/undersize", err: "invalid header: invalid split header: invalid parent header: invalid owner: invalid length 24, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() - ph.SetOwnerID(protoUserIDFromBytes(anyValidUsers[0][:24])) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.ParentHeader.OwnerId.Value = anyValidUsers[0][:24] }}, {name: "header/split/parent/owner/value/oversize", err: "invalid header: invalid split header: invalid parent header: invalid owner: invalid length 26, expected 25", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() - ph.SetOwnerID(protoUserIDFromBytes(append(anyValidUsers[0][:], 1))) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.Split.ParentHeader.OwnerId.Value = append(anyValidUsers[0][:], 1) }}, {name: "header/split/parent/owner/value/wrong prefix", err: "invalid header: invalid split header: invalid parent header: invalid owner: invalid prefix byte 0x42, expected 0x35", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() + corrupt: func(m *protoobject.Object) { b := bytes.Clone(anyValidUsers[0][:]) b[0] = 0x42 - ph.SetOwnerID(protoUserIDFromBytes(b)) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) + m.Header.Split.ParentHeader.OwnerId.Value = b }}, {name: "header/split/parent/owner/value/checksum mismatch", err: "invalid header: invalid split header: invalid parent header: invalid owner: checksum mismatch", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() + corrupt: func(m *protoobject.Object) { b := bytes.Clone(anyValidUsers[0][:]) b[len(b)-1]++ - ph.SetOwnerID(protoUserIDFromBytes(b)) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) + m.Header.Split.ParentHeader.OwnerId.Value = b }}, {name: "header/split/parent/container/empty value", err: "invalid header: invalid split header: invalid parent header: invalid container: invalid length 0", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() - ph.SetContainerID(protoContainerIDFromBytes(nil)) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.ParentHeader.ContainerId.Value = nil }}, {name: "header/split/parent/container/undersize", err: "invalid header: invalid split header: invalid parent header: invalid container: invalid length 31", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() - ph.SetContainerID(protoContainerIDFromBytes(anyValidContainers[0][:31])) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.Split.ParentHeader.ContainerId.Value = anyValidContainers[0][:31] }}, {name: "header/split/parent/container/oversize", err: "invalid header: invalid split header: invalid parent header: invalid container: invalid length 33", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() - ph.SetContainerID(protoContainerIDFromBytes(append(anyValidContainers[0][:], 1))) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) + corrupt: func(m *protoobject.Object) { + m.Header.Split.ParentHeader.ContainerId.Value = append(anyValidContainers[0][:], 1) }}, {name: "header/split/parent/container/zero", err: "invalid header: invalid split header: invalid parent header: invalid container: zero container ID", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() - ph.SetContainerID(protoContainerIDFromBytes(make([]byte, 32))) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.ParentHeader.ContainerId.Value = make([]byte, 32) }}, {name: "header/split/parent/payload checksum/missing value", err: "invalid header: invalid split header: invalid parent header: invalid payload checksum: missing value", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() - ph.SetPayloadHash(new(refs.Checksum)) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.ParentHeader.PayloadHash.Sum = nil }}, + {name: "header/split/parent/payload checksum/negative type", err: "invalid header: invalid split header: invalid parent header: invalid payload checksum: negative type -1", + corrupt: func(m *protoobject.Object) { m.Header.Split.ParentHeader.PayloadHash.Type = -1 }}, {name: "header/split/parent/payload homomorphic checksum/missing value", err: "invalid header: invalid split header: invalid parent header: invalid payload homomorphic checksum: missing value", - corrupt: func(m *apiobject.Object) { - h := *m.GetHeader() - sh := *h.GetSplit() - ph := *sh.GetParentHeader() - ph.SetHomomorphicHash(new(refs.Checksum)) - sh.SetParentHeader(&ph) - h.SetSplit(&sh) - m.SetHeader(&h) - }}, + corrupt: func(m *protoobject.Object) { m.Header.Split.ParentHeader.HomomorphicHash.Sum = nil }}, + {name: "header/split/parent/payload homomorphic checksum/negative type", err: "invalid header: invalid split header: invalid parent header: invalid payload homomorphic checksum: negative type -1", + corrupt: func(m *protoobject.Object) { m.Header.Split.ParentHeader.HomomorphicHash.Type = -1 }}, } { t.Run(tc.name, func(t *testing.T) { - m := obj.ToV2() + m := obj.ProtoMessage() tc.corrupt(m) - require.EqualError(t, new(object.Object).ReadFromV2(*m), tc.err) + require.EqualError(t, new(object.Object).FromProtoMessage(m), tc.err) }) } }) } -func TestObject_ToV2(t *testing.T) { +func TestObject_ProtoMessage(t *testing.T) { // zero - m := object.Object{}.ToV2() - require.Zero(t, m.GetObjectID()) + m := object.Object{}.ProtoMessage() + require.Zero(t, m.GetObjectId()) require.Zero(t, m.GetSignature()) require.Zero(t, m.GetHeader()) require.Zero(t, m.GetPayload()) // filled - m = validObject.ToV2() - require.Equal(t, anyValidIDs[0][:], m.GetObjectID().GetValue()) + m = validObject.ProtoMessage() + require.Equal(t, anyValidIDs[0][:], m.GetObjectId().GetValue()) require.Equal(t, anyValidRegularPayload, m.GetPayload()) msig := m.GetSignature() require.Equal(t, anyValidSignatures[1].PublicKeyBytes(), msig.GetKey()) @@ -1484,8 +1017,8 @@ func TestObject_ToV2(t *testing.T) { mh := m.GetHeader() require.EqualValues(t, 525747025, mh.GetVersion().GetMajor()) require.EqualValues(t, 171993162, mh.GetVersion().GetMinor()) - require.Equal(t, anyValidContainers[1][:], mh.GetContainerID().GetValue()) - require.Equal(t, anyValidUsers[1][:], mh.GetOwnerID().GetValue()) + require.Equal(t, anyValidContainers[1][:], mh.GetContainerId().GetValue()) + require.Equal(t, anyValidUsers[1][:], mh.GetOwnerId().GetValue()) require.EqualValues(t, anyValidCreationEpoch+1, mh.GetCreationEpoch()) require.EqualValues(t, anyValidPayloadSize+1, mh.GetPayloadLength()) require.EqualValues(t, 126384577, mh.GetPayloadHash().GetType()) @@ -1496,18 +1029,18 @@ func TestObject_ToV2(t *testing.T) { mt := mh.GetSessionToken() mb := mt.GetBody() - require.Equal(t, anyValidSessionID[:], mb.GetID()) - require.Equal(t, anyValidUsers[2][:], mb.GetOwnerID().GetValue()) + require.Equal(t, anyValidSessionID[:], mb.GetId()) + require.Equal(t, anyValidUsers[2][:], mb.GetOwnerId().GetValue()) require.Equal(t, anySessionIssuerPubKeyBytes, mb.GetSessionKey()) require.EqualValues(t, uint64(16429376563136800338), mb.GetLifetime().GetExp()) require.EqualValues(t, 7956510363313998522, mb.GetLifetime().GetIat()) require.EqualValues(t, uint64(17237208928641773338), mb.GetLifetime().GetNbf()) c := mb.GetContext() - require.IsType(t, new(apisession.ObjectSessionContext), c) - co := c.(*apisession.ObjectSessionContext) + require.IsType(t, new(protosession.SessionToken_Body_Object), c) + co := c.(*protosession.SessionToken_Body_Object).Object require.EqualValues(t, 1047242055, co.GetVerb()) - require.Equal(t, anyValidContainers[2][:], co.GetContainer().GetValue()) - objs := co.GetObjects() + require.Equal(t, anyValidContainers[2][:], co.GetTarget().GetContainer().GetValue()) + objs := co.GetTarget().GetObjects() require.Len(t, objs, 2) require.Equal(t, anyValidIDs[8][:], objs[0].GetValue()) require.Equal(t, anyValidIDs[9][:], objs[1].GetValue()) @@ -1528,7 +1061,7 @@ func TestObject_ToV2(t *testing.T) { sh := mh.GetSplit() require.Equal(t, anyValidIDs[1][:], sh.GetParent().GetValue()) require.Equal(t, anyValidIDs[2][:], sh.GetPrevious().GetValue()) - require.Equal(t, anyValidSplitIDBytes, sh.GetSplitID()) + require.Equal(t, anyValidSplitIDBytes, sh.GetSplitId()) require.Equal(t, anyValidIDs[6][:], sh.GetFirst().GetValue()) ch := sh.GetChildren() require.Len(t, ch, 3) @@ -1546,8 +1079,8 @@ func TestObject_ToV2(t *testing.T) { require.Zero(t, ph.GetSplit()) require.EqualValues(t, 88789927, ph.GetVersion().GetMajor()) require.EqualValues(t, 2018985309, ph.GetVersion().GetMinor()) - require.Equal(t, anyValidContainers[0][:], ph.GetContainerID().GetValue()) - require.Equal(t, anyValidUsers[0][:], ph.GetOwnerID().GetValue()) + require.Equal(t, anyValidContainers[0][:], ph.GetContainerId().GetValue()) + require.Equal(t, anyValidUsers[0][:], ph.GetOwnerId().GetValue()) require.EqualValues(t, anyValidCreationEpoch, ph.GetCreationEpoch()) require.EqualValues(t, anyValidPayloadSize, ph.GetPayloadLength()) require.EqualValues(t, 1974315742, ph.GetPayloadHash().GetType()) diff --git a/object/range.go b/object/range.go index 98eb7710..9c2260b8 100644 --- a/object/range.go +++ b/object/range.go @@ -1,18 +1,7 @@ package object -import ( - "github.com/nspcc-dev/neofs-api-go/v2/object" -) - // Range represents v2 [object.Range] object payload range. -type Range object.Range - -// NewRangeFromV2 wraps v2 [object.Range] message to [Range]. -// -// Nil [object.Range] converts to nil. -func NewRangeFromV2(rV2 *object.Range) *Range { - return (*Range)(rV2) -} +type Range struct{ off, ln uint64 } // NewRange creates and initializes blank [Range]. // @@ -20,43 +9,33 @@ func NewRangeFromV2(rV2 *object.Range) *Range { // - offset: 0; // - length: 0. func NewRange() *Range { - return NewRangeFromV2(new(object.Range)) -} - -// ToV2 converts [Range] to v2 [object.Range] message. -// -// Nil [Range] converts to nil. -// -// The value returned shares memory with the structure itself, so changing it can lead to data corruption. -// Make a copy if you need to change it. -func (r *Range) ToV2() *object.Range { - return (*object.Range)(r) + return new(Range) } // GetLength returns payload range size. // // See also [Range.SetLength]. func (r *Range) GetLength() uint64 { - return (*object.Range)(r).GetLength() + return r.ln } // SetLength sets payload range size. // // See also [Range.GetLength]. func (r *Range) SetLength(v uint64) { - (*object.Range)(r).SetLength(v) + r.ln = v } // GetOffset sets payload range offset from start. // // See also [Range.SetOffset]. func (r *Range) GetOffset() uint64 { - return (*object.Range)(r).GetOffset() + return r.off } // SetOffset gets payload range offset from start. // // See also [Range.GetOffset]. func (r *Range) SetOffset(v uint64) { - (*object.Range)(r).SetOffset(v) + r.off = v } diff --git a/object/range_test.go b/object/range_test.go index e14d3600..22fdf31e 100644 --- a/object/range_test.go +++ b/object/range_test.go @@ -3,7 +3,6 @@ package object import ( "testing" - "github.com/nspcc-dev/neofs-api-go/v2/object" "github.com/stretchr/testify/require" ) @@ -25,22 +24,6 @@ func TestRange_SetLength(t *testing.T) { require.Equal(t, ln, r.GetLength()) } -func TestNewRangeFromV2(t *testing.T) { - t.Run("from nil", func(t *testing.T) { - var x *object.Range - - require.Nil(t, NewRangeFromV2(x)) - }) -} - -func TestRange_ToV2(t *testing.T) { - t.Run("nil", func(t *testing.T) { - var x *Range - - require.Nil(t, x.ToV2()) - }) -} - func TestNewRange(t *testing.T) { t.Run("default values", func(t *testing.T) { r := NewRange() @@ -48,11 +31,5 @@ func TestNewRange(t *testing.T) { // check initial values require.Zero(t, r.GetLength()) require.Zero(t, r.GetOffset()) - - // convert to v2 message - rV2 := r.ToV2() - - require.Zero(t, rV2.GetLength()) - require.Zero(t, rV2.GetOffset()) }) } diff --git a/object/search.go b/object/search.go index 75f45dbe..b704e8ff 100644 --- a/object/search.go +++ b/object/search.go @@ -4,19 +4,20 @@ import ( "crypto/sha256" "encoding/hex" "encoding/json" + "fmt" "strconv" "strings" - v2object "github.com/nspcc-dev/neofs-api-go/v2/object" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + protoobject "github.com/nspcc-dev/neofs-sdk-go/proto/object" "github.com/nspcc-dev/neofs-sdk-go/user" "github.com/nspcc-dev/neofs-sdk-go/version" "github.com/nspcc-dev/tzhash/tz" ) // SearchMatchType indicates match operation on specified header. -type SearchMatchType uint32 +type SearchMatchType int32 // MatchUnknown is an SearchMatchType value used to mark operator as undefined. // Deprecated: use MatchUnspecified instead. @@ -34,18 +35,6 @@ const ( MatchNumLE ) -// ToV2 converts [SearchMatchType] to v2 [v2object.MatchType] enum value. -// Deprecated: cast instead. -func (m SearchMatchType) ToV2() v2object.MatchType { - return v2object.MatchType(m) -} - -// SearchMatchFromV2 converts v2 [v2object.MatchType] to [SearchMatchType] enum value. -// Deprecated: cast instead. -func SearchMatchFromV2(t v2object.MatchType) SearchMatchType { - return SearchMatchType(t) -} - const ( matcherStringZero = "MATCH_TYPE_UNSPECIFIED" matcherStringEqual = "STRING_EQUAL" @@ -163,18 +152,19 @@ type SearchFilters []SearchFilter // Various header filters. const ( - FilterVersion = v2object.FilterHeaderVersion - FilterID = v2object.FilterHeaderObjectID - FilterContainerID = v2object.FilterHeaderContainerID - FilterOwnerID = v2object.FilterHeaderOwnerID - FilterPayloadChecksum = v2object.FilterHeaderPayloadHash - FilterType = v2object.FilterHeaderObjectType - FilterPayloadHomomorphicHash = v2object.FilterHeaderHomomorphicHash - FilterParentID = v2object.FilterHeaderParent - FilterSplitID = v2object.FilterHeaderSplitID - FilterFirstSplitObject = v2object.ReservedFilterPrefix + "split.first" - FilterCreationEpoch = v2object.FilterHeaderCreationEpoch - FilterPayloadSize = v2object.FilterHeaderPayloadLength + reservedFilterPrefix = "$Object:" + FilterVersion = reservedFilterPrefix + "version" + FilterID = reservedFilterPrefix + "objectID" + FilterContainerID = reservedFilterPrefix + "containerID" + FilterOwnerID = reservedFilterPrefix + "ownerID" + FilterPayloadChecksum = reservedFilterPrefix + "payloadHash" + FilterType = reservedFilterPrefix + "objectType" + FilterPayloadHomomorphicHash = reservedFilterPrefix + "homomorphicHash" + FilterParentID = reservedFilterPrefix + "split.parent" + FilterSplitID = reservedFilterPrefix + "split.splitID" + FilterFirstSplitObject = reservedFilterPrefix + "split.first" + FilterCreationEpoch = reservedFilterPrefix + "creationEpoch" + FilterPayloadSize = reservedFilterPrefix + "payloadLength" ) // Various filters to match certain object properties. @@ -183,12 +173,12 @@ const ( // with user data that are not system-specific. In addition to such objects, the // system may contain service objects that do not fall under this property // (like split leaves, tombstones, storage groups, etc.). - FilterRoot = v2object.FilterPropertyRoot + FilterRoot = reservedFilterPrefix + "ROOT" // FilterPhysical filters indivisible objects that are intended to be stored // on the physical devices of the system. In addition to such objects, the // system may contain so-called "virtual" objects that exist in the system in // disassembled form (like "huge" user object sliced into smaller ones). - FilterPhysical = v2object.FilterPropertyPhy + FilterPhysical = reservedFilterPrefix + "PHY" ) // Header returns filter header value. @@ -209,7 +199,7 @@ func (f SearchFilter) Operation() SearchMatchType { // IsNonAttribute checks if SearchFilter is non-attribute: such filter is // related to the particular property of the object instead of its attribute. func (f SearchFilter) IsNonAttribute() bool { - return strings.HasPrefix(f.header, v2object.ReservedFilterPrefix) + return strings.HasPrefix(f.header, reservedFilterPrefix) } // NewSearchFilters constructs empty filter group. @@ -217,21 +207,6 @@ func NewSearchFilters() SearchFilters { return SearchFilters{} } -// NewSearchFiltersFromV2 converts slice of [v2object.SearchFilter] to [SearchFilters]. -func NewSearchFiltersFromV2(v2 []v2object.SearchFilter) SearchFilters { - filters := make(SearchFilters, 0, len(v2)) - - for i := range v2 { - filters.AddFilter( - v2[i].GetKey(), - v2[i].GetValue(), - SearchMatchType(v2[i].GetMatchType()), - ) - } - - return filters -} - func (f *SearchFilters) addFilter(op SearchMatchType, key string, val string) { if *f == nil { *f = make(SearchFilters, 0, 1) @@ -279,17 +254,46 @@ func (f *SearchFilters) AddObjectOwnerIDFilter(m SearchMatchType, id user.ID) { f.addFilter(m, FilterOwnerID, id.EncodeToString()) } -// ToV2 converts [SearchFilters] to [v2object.SearchFilter] slice. -func (f SearchFilters) ToV2() []v2object.SearchFilter { - result := make([]v2object.SearchFilter, len(f)) +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// f from it. +// +// See also [SearchFilters.ProtoMessage]. +func (f *SearchFilters) FromProtoMessage(ms []*protoobject.SearchRequest_Body_Filter) error { + fs := make(SearchFilters, len(ms)) + for i, m := range ms { + if m == nil { + return fmt.Errorf("nil filter #%d", i) + } + if m.MatchType < 0 { + return fmt.Errorf("invalid filter #%d: negative match type %d", i, m.MatchType) + } + if m.Key == "" { + return fmt.Errorf("invalid filter #%d: missing key", i) + } + fs[i] = SearchFilter{ + header: m.Key, + value: m.Value, + op: SearchMatchType(m.MatchType), + } + } + *f = fs + return nil +} +// ProtoMessage converts f into message to transmit using the NeoFS API +// protocol. +// +// See also [SearchFilters.FromProtoMessage]. +func (f SearchFilters) ProtoMessage() []*protoobject.SearchRequest_Body_Filter { + m := make([]*protoobject.SearchRequest_Body_Filter, len(f)) for i := range f { - result[i].SetKey(f[i].header) - result[i].SetValue(f[i].value) - result[i].SetMatchType(v2object.MatchType(f[i].op)) + m[i] = &protoobject.SearchRequest_Body_Filter{ + MatchType: protoobject.MatchType(f[i].op), + Key: f[i].header, + Value: f[i].value, + } } - - return result + return m } // AddRootFilter adds filter by objects that have been created by a user explicitly. @@ -341,22 +345,20 @@ func (f *SearchFilters) AddTypeFilter(m SearchMatchType, typ Type) { // // See also [SearchFilters.UnmarshalJSON]. func (f SearchFilters) MarshalJSON() ([]byte, error) { - return json.Marshal(f.ToV2()) + return json.Marshal(f.ProtoMessage()) } // UnmarshalJSON decodes [SearchFilters] from protobuf JSON format. // // See also [SearchFilters.MarshalJSON]. func (f *SearchFilters) UnmarshalJSON(data []byte) error { - var fsV2 []v2object.SearchFilter + var m []*protoobject.SearchRequest_Body_Filter - if err := json.Unmarshal(data, &fsV2); err != nil { + if err := json.Unmarshal(data, &m); err != nil { return err } - *f = NewSearchFiltersFromV2(fsV2) - - return nil + return f.FromProtoMessage(m) } // AddPayloadHashFilter adds filter by payload hash. diff --git a/object/search_test.go b/object/search_test.go index c113f4b1..3e89049a 100644 --- a/object/search_test.go +++ b/object/search_test.go @@ -5,29 +5,32 @@ import ( "encoding/hex" "encoding/json" "fmt" + "math/rand/v2" + "slices" "testing" - v2object "github.com/nspcc-dev/neofs-api-go/v2/object" "github.com/nspcc-dev/neofs-sdk-go/checksum" "github.com/nspcc-dev/neofs-sdk-go/object" + protoobject "github.com/nspcc-dev/neofs-sdk-go/proto/object" "github.com/nspcc-dev/tzhash/tz" "github.com/stretchr/testify/require" + "google.golang.org/protobuf/proto" ) const ( anyValidSearchMatcher = object.SearchMatchType(1937803447) ) -var eqV2Matches = map[object.SearchMatchType]v2object.MatchType{ - object.MatchUnspecified: v2object.MatchUnknown, - object.MatchStringEqual: v2object.MatchStringEqual, - object.MatchStringNotEqual: v2object.MatchStringNotEqual, - object.MatchNotPresent: v2object.MatchNotPresent, - object.MatchCommonPrefix: v2object.MatchCommonPrefix, - object.MatchNumGT: v2object.MatchNumGT, - object.MatchNumGE: v2object.MatchNumGE, - object.MatchNumLT: v2object.MatchNumLT, - object.MatchNumLE: v2object.MatchNumLE, +var protoMatches = map[object.SearchMatchType]protoobject.MatchType{ + object.MatchUnspecified: protoobject.MatchType_MATCH_TYPE_UNSPECIFIED, + object.MatchStringEqual: protoobject.MatchType_STRING_EQUAL, + object.MatchStringNotEqual: protoobject.MatchType_STRING_NOT_EQUAL, + object.MatchNotPresent: protoobject.MatchType_NOT_PRESENT, + object.MatchCommonPrefix: protoobject.MatchType_COMMON_PREFIX, + object.MatchNumGT: protoobject.MatchType_NUM_GT, + object.MatchNumGE: protoobject.MatchType_NUM_GE, + object.MatchNumLT: protoobject.MatchType_NUM_LT, + object.MatchNumLE: protoobject.MatchType_NUM_LE, } var searchMatchTypeStrings = map[object.SearchMatchType]string{ @@ -73,19 +76,19 @@ func init() { // corresponds to validSearchFilters. var validSearchFiltersProto = []struct { k string - m v2object.MatchType + m protoobject.MatchType v string }{ - {"$Object:PHY", v2object.MatchUnknown, ""}, - {"$Object:ROOT", v2object.MatchUnknown, ""}, - {"k1", v2object.MatchStringEqual, "v1"}, - {"k2", v2object.MatchStringNotEqual, "v2"}, - {"k3", v2object.MatchNotPresent, "v3"}, - {"k4", v2object.MatchCommonPrefix, "v4"}, - {"k5", v2object.MatchNumGT, "v5"}, - {"k6", v2object.MatchNumGE, "v6"}, - {"k7", v2object.MatchNumLT, "v7"}, - {"k8", v2object.MatchNumLE, "v8"}, + {"$Object:PHY", protoobject.MatchType_MATCH_TYPE_UNSPECIFIED, ""}, + {"$Object:ROOT", protoobject.MatchType_MATCH_TYPE_UNSPECIFIED, ""}, + {"k1", protoobject.MatchType_STRING_NOT_EQUAL, "v1"}, + {"k2", protoobject.MatchType_STRING_NOT_EQUAL, "v2"}, + {"k3", protoobject.MatchType_NOT_PRESENT, "v3"}, + {"k4", protoobject.MatchType_COMMON_PREFIX, "v4"}, + {"k5", protoobject.MatchType_NUM_GT, "v5"}, + {"k6", protoobject.MatchType_NUM_GE, "v6"}, + {"k7", protoobject.MatchType_NUM_LT, "v7"}, + {"k8", protoobject.MatchType_NUM_LE, "v8"}, {"$Object:version", 100, "v88789927.2018985309"}, {"$Object:objectID", 101, "CzyDjRYWpwLHxqXVFBXKQGP5XM7ebAR9ndTvBdaSxMMV"}, {"$Object:containerID", 102, "HWpbBkyxCi7nhDnn4W3v5rYt2mDfH2wedknQzRkTwquj"}, @@ -248,23 +251,8 @@ func TestSearchFilters_UnmarshalJSON(t *testing.T) { require.Equal(t, validSearchFilters, fs) } -func TestMatch(t *testing.T) { - require.EqualValues(t, object.MatchUnspecified, v2object.MatchUnknown) - t.Run("known matches", func(t *testing.T) { - for matchType, matchTypeV2 := range eqV2Matches { - require.Equal(t, matchTypeV2, matchType.ToV2()) - require.Equal(t, object.SearchMatchFromV2(matchTypeV2), matchType) - } - }) - - t.Run("unknown matches", func(t *testing.T) { - require.EqualValues(t, 1000, object.SearchMatchType(1000).ToV2()) - require.EqualValues(t, 1000, object.SearchMatchFromV2(1000)) - }) -} - func TestSearchFilters_AddFilter(t *testing.T) { - const k1, m1, v1 = "k1", object.SearchMatchType(2584744206), "v1" + const k1, m1, v1 = "k1", object.SearchMatchType(584744206), "v1" const k2, m2, v2 = "k2", object.SearchMatchType(930572326), "v2" var filters object.SearchFilters @@ -357,17 +345,7 @@ func TestSearchFilters_AddTypeFilter(t *testing.T) { } func TestSearchMatchTypeProto(t *testing.T) { - for x, y := range map[v2object.MatchType]object.SearchMatchType{ - v2object.MatchUnknown: object.MatchUnspecified, - v2object.MatchStringEqual: object.MatchStringEqual, - v2object.MatchStringNotEqual: object.MatchStringNotEqual, - v2object.MatchNotPresent: object.MatchNotPresent, - v2object.MatchCommonPrefix: object.MatchCommonPrefix, - v2object.MatchNumGT: object.MatchNumGT, - v2object.MatchNumGE: object.MatchNumGE, - v2object.MatchNumLT: object.MatchNumLT, - v2object.MatchNumLE: object.MatchNumLE, - } { + for x, y := range protoMatches { require.EqualValues(t, x, y) } } @@ -462,17 +440,17 @@ func TestNewSearchFilters(t *testing.T) { require.Empty(t, object.NewSearchFilters()) } -func TestSearchFilters_ToV2(t *testing.T) { +func TestSearchFilters_ProtoMessage(t *testing.T) { const nFilters = 22 require.Len(t, validSearchFiltersProto, nFilters, "not all applied filters are asserted") var fs object.SearchFilters // zero - m := fs.ToV2() + m := fs.ProtoMessage() require.Empty(t, m) // filled - m = validSearchFilters.ToV2() + m = validSearchFilters.ProtoMessage() require.Len(t, m, nFilters) for i, exp := range validSearchFiltersProto { @@ -483,18 +461,35 @@ func TestSearchFilters_ToV2(t *testing.T) { } } -func TestNewSearchFiltersFromV2(t *testing.T) { - // empty - require.Empty(t, object.NewSearchFiltersFromV2(nil)) - require.Empty(t, object.NewSearchFiltersFromV2([]v2object.SearchFilter{})) +func TestSearchFilters_FromProtoMessage(t *testing.T) { + ms := []*protoobject.SearchRequest_Body_Filter{ + {MatchType: protoobject.MatchType(rand.Int32()), Key: "key_1", Value: "val_1"}, + {MatchType: protoobject.MatchType(rand.Int32()), Key: "key_2", Value: "val_2"}, + } - // filled - m := make([]v2object.SearchFilter, len(validSearchFiltersProto)) - for i, f := range validSearchFiltersProto { - m[i].SetKey(f.k) - m[i].SetMatchType(f.m) - m[i].SetValue(f.v) + var fs object.SearchFilters + require.NoError(t, fs.FromProtoMessage(ms)) + require.Len(t, fs, len(ms)) + for i := range ms { + require.EqualValues(t, ms[i].MatchType, fs[i].Operation()) + require.Equal(t, ms[i].Key, fs[i].Header()) + require.Equal(t, ms[i].Value, fs[i].Value()) } - require.Equal(t, object.NewSearchFiltersFromV2(m), validSearchFilters) + require.NoError(t, fs.FromProtoMessage(nil)) + require.Empty(t, fs) + + t.Run("invalid", func(t *testing.T) { + for _, tc := range []struct { + name, err string + corrupt func([]*protoobject.SearchRequest_Body_Filter) + }{} { + cp := slices.Clone(ms) + for i := range ms { + cp[i] = proto.Clone(ms[i]).(*protoobject.SearchRequest_Body_Filter) + } + tc.corrupt(cp) + require.EqualError(t, fs.FromProtoMessage(ms), tc.err) + } + }) } diff --git a/object/slicer/slicer_test.go b/object/slicer/slicer_test.go index b58dfe7f..9c86f59c 100644 --- a/object/slicer/slicer_test.go +++ b/object/slicer/slicer_test.go @@ -14,7 +14,6 @@ import ( "math/rand" "testing" - netmapv2 "github.com/nspcc-dev/neofs-api-go/v2/netmap" "github.com/nspcc-dev/neofs-sdk-go/checksum" "github.com/nspcc-dev/neofs-sdk-go/client" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" @@ -136,20 +135,7 @@ func benchmarkSliceDataIntoObjects(b *testing.B, size, sizeLimit uint64) { func networkInfoFromOpts(opts slicer.Options) (netmap.NetworkInfo, error) { var ni netmap.NetworkInfo - var v2 netmapv2.NetworkInfo - var netConfig netmapv2.NetworkConfig - var p1 netmapv2.NetworkParameter - - p1.SetKey(randomData(10)) - p1.SetValue(randomData(10)) - - netConfig.SetParameters(p1) - v2.SetNetworkConfig(&netConfig) - - if err := ni.ReadFromV2(v2); err != nil { - return ni, err - } - + ni.SetRawNetworkParameter(string(randomData(10)), randomData(10)) ni.SetCurrentEpoch(opts.CurrentNeoFSEpoch()) ni.SetMaxObjectSize(opts.ObjectPayloadLimit()) if !opts.IsHomomorphicChecksumEnabled() { diff --git a/object/splitinfo.go b/object/splitinfo.go index fd39fd88..6fca367e 100644 --- a/object/splitinfo.go +++ b/object/splitinfo.go @@ -5,21 +5,17 @@ import ( "fmt" "github.com/google/uuid" - "github.com/nspcc-dev/neofs-api-go/v2/object" - "github.com/nspcc-dev/neofs-api-go/v2/refs" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + protoobject "github.com/nspcc-dev/neofs-sdk-go/proto/object" ) // SplitInfo is an SDK representation of [object.SplitInfo]. -type SplitInfo object.SplitInfo - -// NewSplitInfoFromV2 wraps v2 [object.SplitInfo] message to [SplitInfo]. -// -// Nil object.SplitInfo converts to nil. -// Deprecated: BUG: format of fields is not checked. Use [SplitInfo.ReadFromV2] -// instead. -func NewSplitInfoFromV2(v2 *object.SplitInfo) *SplitInfo { - return (*SplitInfo)(v2) +type SplitInfo struct { + splitID []byte + last oid.ID + link oid.ID + first oid.ID } // NewSplitInfo creates and initializes blank [SplitInfo]. @@ -32,14 +28,24 @@ func NewSplitInfo() *SplitInfo { return new(SplitInfo) } -// ToV2 converts [SplitInfo] to v2 [object.SplitInfo] message. -// -// Nil SplitInfo converts to nil. +// ProtoMessage converts s into message to transmit using the NeoFS API +// protocol. // -// The value returned shares memory with the structure itself, so changing it can lead to data corruption. -// Make a copy if you need to change it. -func (s SplitInfo) ToV2() *object.SplitInfo { - return (*object.SplitInfo)(&s) +// See also [SplitInfo.FromProtoMessage]. +func (s SplitInfo) ProtoMessage() *protoobject.SplitInfo { + m := &protoobject.SplitInfo{ + SplitId: s.splitID, + } + if !s.last.IsZero() { + m.LastPart = s.last.ProtoMessage() + } + if !s.first.IsZero() { + m.FirstPart = s.first.ProtoMessage() + } + if !s.link.IsZero() { + m.LastPart = s.link.ProtoMessage() + } + return m } // SplitID returns [SplitID] if it has been set. New objects may miss it, @@ -50,8 +56,7 @@ func (s SplitInfo) ToV2() *object.SplitInfo { // // See also [SplitInfo.SetSplitID]. func (s SplitInfo) SplitID() *SplitID { - return NewSplitIDFromV2( - (*object.SplitInfo)(&s).GetSplitID()) + return NewSplitIDFromV2(s.splitID) } // SetSplitID sets split ID in object ID. It resets split ID if nil passed. @@ -61,7 +66,7 @@ func (s SplitInfo) SplitID() *SplitID { // DEPRECATED.[SplitInfo.SetFirstPart] usage is required for the _new_ split // objects, it serves as chain identification. func (s *SplitInfo) SetSplitID(v *SplitID) { - (*object.SplitInfo)(s).SetSplitID(v.ToV2()) + s.splitID = v.ToV2() } // LastPart returns last object ID, can be used to retrieve original object. @@ -79,22 +84,14 @@ func (s SplitInfo) LastPart() (oid.ID, bool) { // // See also [SplitInfo.SetLastPart]. func (s SplitInfo) GetLastPart() oid.ID { - var id oid.ID - m := (*object.SplitInfo)(&s).GetLastPart() - if m != nil { - _ = id.ReadFromV2(*m) - } - return id + return s.last } // SetLastPart sets the last object ID. // // See also [SplitInfo.GetLastPart]. func (s *SplitInfo) SetLastPart(v oid.ID) { - var idV2 refs.ObjectID - v.WriteToV2(&idV2) - - (*object.SplitInfo)(s).SetLastPart(&idV2) + s.last = v } // Link returns a linker object ID. @@ -111,21 +108,14 @@ func (s SplitInfo) Link() (oid.ID, bool) { // // See also [SplitInfo.SetLink]. func (s SplitInfo) GetLink() oid.ID { - var id oid.ID - if m := (*object.SplitInfo)(&s).GetLink(); m != nil { - _ = id.ReadFromV2(*m) - } - return id + return s.link } // SetLink sets linker object ID. // // See also [SplitInfo.GetLink]. func (s *SplitInfo) SetLink(v oid.ID) { - var idV2 refs.ObjectID - v.WriteToV2(&idV2) - - (*object.SplitInfo)(s).SetLink(&idV2) + s.link = v } // FirstPart returns the first part of the split chain. @@ -141,106 +131,90 @@ func (s SplitInfo) FirstPart() (oid.ID, bool) { // // See also [SplitInfo.SetFirstPart]. func (s SplitInfo) GetFirstPart() oid.ID { - var id oid.ID - if m := (*object.SplitInfo)(&s).GetFirstPart(); m != nil { - _ = id.ReadFromV2(*m) - } - return id + return s.first } // SetFirstPart sets the first part of the split chain. // // See also [SplitInfo.GetFirstPart]. func (s *SplitInfo) SetFirstPart(v oid.ID) { - var idV2 refs.ObjectID - v.WriteToV2(&idV2) - - (*object.SplitInfo)(s).SetFirstPart(&idV2) + s.first = v } // Marshal marshals [SplitInfo] into a protobuf binary form. // // See also [SplitInfo.Unmarshal]. func (s SplitInfo) Marshal() []byte { - return (*object.SplitInfo)(&s).StableMarshal(nil) + return neofsproto.Marshal(s) } // Unmarshal unmarshals protobuf binary representation of [SplitInfo]. // // See also [SplitInfo.Marshal]. func (s *SplitInfo) Unmarshal(data []byte) error { - var m object.SplitInfo - if err := m.Unmarshal(data); err != nil { - return err - } - return s.ReadFromV2(m) + return neofsproto.Unmarshal(data, s) } // MarshalJSON implements json.Marshaler. // // See also [SplitInfo.UnmarshalJSON]. func (s SplitInfo) MarshalJSON() ([]byte, error) { - return (*object.SplitInfo)(&s).MarshalJSON() + return neofsproto.MarshalJSON(s) } // UnmarshalJSON implements json.Unmarshaler. // // See also [SplitInfo.MarshalJSON]. func (s *SplitInfo) UnmarshalJSON(data []byte) error { - var m object.SplitInfo - if err := m.UnmarshalJSON(data); err != nil { - return err - } - return s.ReadFromV2(m) + return neofsproto.UnmarshalJSON(data, s) } var errSplitInfoMissingFields = errors.New("neither link object ID nor last part object ID is set") -// ReadFromV2 reads SplitInfo from the [object.SplitInfo] message. Returns an -// error if the message is malformed according to the NeoFS API V2 protocol. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// s from it. // -// ReadFromV2 is intended to be used by the NeoFS API V2 client/server -// implementation only and is not expected to be directly used by applications. -func (s *SplitInfo) ReadFromV2(m object.SplitInfo) error { - if b := m.GetSplitID(); len(b) > 0 { +// See also [SplitInfo.ProtoMessage]. +func (s *SplitInfo) FromProtoMessage(m *protoobject.SplitInfo) error { + if s.splitID = m.SplitId; len(m.SplitId) > 0 { var uid uuid.UUID - if err := uid.UnmarshalBinary(b); err != nil { + if err := uid.UnmarshalBinary(m.SplitId); err != nil { return fmt.Errorf("invalid split ID: %w", err) } else if v := uid.Version(); v != 4 { return fmt.Errorf("invalid split UUID version %d", v) } } - link := m.GetLink() - lastPart := m.GetLastPart() - if link == nil && lastPart == nil { + if m.Link == nil && m.LastPart == nil { return errSplitInfoMissingFields } - var oID oid.ID - - if link != nil { - err := oID.ReadFromV2(*link) + if m.Link != nil { + err := s.link.FromProtoMessage(m.Link) if err != nil { return fmt.Errorf("could not convert link object ID: %w", err) } + } else { + s.link = oid.ID{} } - if lastPart != nil { - err := oID.ReadFromV2(*lastPart) + if m.LastPart != nil { + err := s.last.FromProtoMessage(m.LastPart) if err != nil { return fmt.Errorf("could not convert last part object ID: %w", err) } + } else { + s.last = oid.ID{} } - firstPart := m.GetFirstPart() - if firstPart != nil { // can be missing for old objects - err := oID.ReadFromV2(*firstPart) + if m.FirstPart != nil { // can be missing for old objects + err := s.first.FromProtoMessage(m.LastPart) if err != nil { return fmt.Errorf("could not convert first part object ID: %w", err) } + } else { + s.first = oid.ID{} } - *s = SplitInfo(m) return nil } diff --git a/object/splitinfo_test.go b/object/splitinfo_test.go index cef1b3f1..7ecfd9f9 100644 --- a/object/splitinfo_test.go +++ b/object/splitinfo_test.go @@ -5,10 +5,10 @@ import ( "encoding/json" "testing" - apiobject "github.com/nspcc-dev/neofs-api-go/v2/object" "github.com/nspcc-dev/neofs-sdk-go/object" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" oidtest "github.com/nspcc-dev/neofs-sdk-go/object/id/test" + protoobject "github.com/nspcc-dev/neofs-sdk-go/proto/object" "github.com/stretchr/testify/require" ) @@ -154,14 +154,14 @@ func TestSplitInfo(t *testing.T) { } func TestSplitInfoMarshal(t *testing.T) { - testToV2 := func(t *testing.T, s *object.SplitInfo) { - v2 := s.ToV2() - newS := object.NewSplitInfoFromV2(v2) + testMessage := func(t *testing.T, s object.SplitInfo) { + var newS object.SplitInfo + require.NoError(t, newS.FromProtoMessage(s.ProtoMessage())) require.Equal(t, s, newS) } - testMarshal := func(t *testing.T, s *object.SplitInfo) { - newS := object.NewSplitInfo() + testMarshal := func(t *testing.T, s object.SplitInfo) { + var newS object.SplitInfo err := newS.Unmarshal(s.Marshal()) require.NoError(t, err) @@ -175,24 +175,24 @@ func TestSplitInfoMarshal(t *testing.T) { s.SetLastPart(oidtest.ID()) s.SetFirstPart(oidtest.ID()) - testToV2(t, s) - testMarshal(t, s) + testMessage(t, *s) + testMarshal(t, *s) }) t.Run("good, only link is set", func(t *testing.T) { s := object.NewSplitInfo() s.SetSplitID(object.NewSplitID()) s.SetLink(oidtest.ID()) - testToV2(t, s) - testMarshal(t, s) + testMessage(t, *s) + testMarshal(t, *s) }) t.Run("good, only last part is set", func(t *testing.T) { s := object.NewSplitInfo() s.SetSplitID(object.NewSplitID()) s.SetLastPart(oidtest.ID()) - testToV2(t, s) - testMarshal(t, s) + testMessage(t, *s) + testMarshal(t, *s) }) t.Run("bad, no fields are set", func(t *testing.T) { s := object.NewSplitInfo() @@ -202,23 +202,16 @@ func TestSplitInfoMarshal(t *testing.T) { }) } -func TestNewSplitInfoFromV2(t *testing.T) { - t.Run("from nil", func(t *testing.T) { - var x *apiobject.SplitInfo - - require.Nil(t, object.NewSplitInfoFromV2(x)) - }) -} - -func TestSplitInfo_ReadFromV2(t *testing.T) { - var m apiobject.SplitInfo - m.SetSplitID(anyValidSplitIDBytes) - m.SetLastPart(protoIDFromBytes(anyValidIDs[0][:])) - m.SetLink(protoIDFromBytes(anyValidIDs[1][:])) - m.SetFirstPart(protoIDFromBytes(anyValidIDs[2][:])) +func TestSplitInfo_FromProtoMessage(t *testing.T) { + m := &protoobject.SplitInfo{ + SplitId: anyValidSplitIDBytes, + LastPart: protoIDFromBytes(anyValidIDs[0][:]), + Link: protoIDFromBytes(anyValidIDs[1][:]), + FirstPart: protoIDFromBytes(anyValidIDs[2][:]), + } var s object.SplitInfo - require.NoError(t, s.ReadFromV2(m)) + require.NoError(t, s.FromProtoMessage(m)) require.Equal(t, anyValidSplitID, s.SplitID()) require.Equal(t, anyValidIDs[0], s.GetLastPart()) id, ok := s.LastPart() @@ -234,11 +227,11 @@ func TestSplitInfo_ReadFromV2(t *testing.T) { require.Equal(t, anyValidIDs[2], id) // reset optional fields - m.SetSplitID(nil) - m.SetFirstPart(nil) - m.SetLink(nil) + m.SplitId = nil + m.FirstPart = nil + m.Link = nil s2 := s - require.NoError(t, s2.ReadFromV2(m)) + require.NoError(t, s2.FromProtoMessage(m)) require.Zero(t, s2.SplitID()) require.True(t, s2.GetFirstPart().IsZero()) @@ -253,9 +246,9 @@ func TestSplitInfo_ReadFromV2(t *testing.T) { require.False(t, ok) // either linking or last part must be set, so lets swap - m.SetLink(protoIDFromBytes(anyValidIDs[1][:])) - m.SetLastPart(nil) - require.NoError(t, s2.ReadFromV2(m)) + m.Link = protoIDFromBytes(anyValidIDs[1][:]) + m.LastPart = nil + require.NoError(t, s2.FromProtoMessage(m)) require.Equal(t, anyValidIDs[1], s2.GetLink()) require.True(t, s2.GetLastPart().IsZero()) @@ -265,66 +258,65 @@ func TestSplitInfo_ReadFromV2(t *testing.T) { t.Run("invalid", func(t *testing.T) { for _, tc := range []struct { name, err string - corrupt func(*apiobject.SplitInfo) + corrupt func(*protoobject.SplitInfo) }{ {name: "split ID/undersize", err: "invalid split ID: invalid UUID (got 15 bytes)", - corrupt: func(m *apiobject.SplitInfo) { m.SetSplitID(anyValidSplitIDBytes[:15]) }}, + corrupt: func(m *protoobject.SplitInfo) { m.SplitId = anyValidSplitIDBytes[:15] }}, {name: "split ID/oversize", err: "invalid split ID: invalid UUID (got 17 bytes)", - corrupt: func(m *apiobject.SplitInfo) { m.SetSplitID(append(anyValidSplitIDBytes[:], 1)) }}, + corrupt: func(m *protoobject.SplitInfo) { m.SplitId = append(anyValidSplitIDBytes[:], 1) }}, {name: "split ID/wrong version", err: "invalid split UUID version 3", - corrupt: func(m *apiobject.SplitInfo) { - b := bytes.Clone(anyValidSplitIDBytes[:]) - b[6] = 3 << 4 - m.SetSplitID(b) + corrupt: func(m *protoobject.SplitInfo) { + m.SplitId = bytes.Clone(anyValidSplitIDBytes[:]) + m.SplitId[6] = 3 << 4 }}, {name: "last part/nil value", err: "could not convert last part object ID: invalid length 0", - corrupt: func(m *apiobject.SplitInfo) { m.SetLastPart(protoIDFromBytes(nil)) }}, + corrupt: func(m *protoobject.SplitInfo) { m.LastPart = protoIDFromBytes(nil) }}, {name: "last part/empty value", err: "could not convert last part object ID: invalid length 0", - corrupt: func(m *apiobject.SplitInfo) { m.SetLastPart(protoIDFromBytes([]byte{})) }}, + corrupt: func(m *protoobject.SplitInfo) { m.LastPart = protoIDFromBytes([]byte{}) }}, {name: "last part/undersize", err: "could not convert last part object ID: invalid length 31", - corrupt: func(m *apiobject.SplitInfo) { m.SetLastPart(protoIDFromBytes(make([]byte, 31))) }}, + corrupt: func(m *protoobject.SplitInfo) { m.LastPart = protoIDFromBytes(make([]byte, 31)) }}, {name: "last part/oversize", err: "could not convert last part object ID: invalid length 33", - corrupt: func(m *apiobject.SplitInfo) { m.SetLastPart(protoIDFromBytes(make([]byte, 33))) }}, + corrupt: func(m *protoobject.SplitInfo) { m.LastPart = protoIDFromBytes(make([]byte, 33)) }}, {name: "link/nil value", err: "could not convert link object ID: invalid length 0", - corrupt: func(m *apiobject.SplitInfo) { m.SetLink(protoIDFromBytes(nil)) }}, + corrupt: func(m *protoobject.SplitInfo) { m.Link = protoIDFromBytes(nil) }}, {name: "link/empty value", err: "could not convert link object ID: invalid length 0", - corrupt: func(m *apiobject.SplitInfo) { m.SetLink(protoIDFromBytes([]byte{})) }}, + corrupt: func(m *protoobject.SplitInfo) { m.Link = protoIDFromBytes([]byte{}) }}, {name: "link/undersize", err: "could not convert link object ID: invalid length 31", - corrupt: func(m *apiobject.SplitInfo) { m.SetLink(protoIDFromBytes(make([]byte, 31))) }}, + corrupt: func(m *protoobject.SplitInfo) { m.Link = protoIDFromBytes(make([]byte, 31)) }}, {name: "link/oversize", err: "could not convert link object ID: invalid length 33", - corrupt: func(m *apiobject.SplitInfo) { m.SetLink(protoIDFromBytes(make([]byte, 33))) }}, + corrupt: func(m *protoobject.SplitInfo) { m.Link = protoIDFromBytes(make([]byte, 33)) }}, {name: "first part/nil value", err: "could not convert first part object ID: invalid length 0", - corrupt: func(m *apiobject.SplitInfo) { m.SetFirstPart(protoIDFromBytes(nil)) }}, + corrupt: func(m *protoobject.SplitInfo) { m.FirstPart = protoIDFromBytes(nil) }}, {name: "first part/empty value", err: "could not convert first part object ID: invalid length 0", - corrupt: func(m *apiobject.SplitInfo) { m.SetFirstPart(protoIDFromBytes([]byte{})) }}, + corrupt: func(m *protoobject.SplitInfo) { m.FirstPart = protoIDFromBytes([]byte{}) }}, {name: "first part/undersize", err: "could not convert first part object ID: invalid length 31", - corrupt: func(m *apiobject.SplitInfo) { m.SetFirstPart(protoIDFromBytes(make([]byte, 31))) }}, + corrupt: func(m *protoobject.SplitInfo) { m.FirstPart = protoIDFromBytes(make([]byte, 31)) }}, {name: "first part/oversize", err: "could not convert first part object ID: invalid length 33", - corrupt: func(m *apiobject.SplitInfo) { m.SetFirstPart(protoIDFromBytes(make([]byte, 33))) }}, + corrupt: func(m *protoobject.SplitInfo) { m.FirstPart = protoIDFromBytes(make([]byte, 33)) }}, } { t.Run(tc.name, func(t *testing.T) { s2 := s - m := s2.ToV2() + m := s2.ProtoMessage() tc.corrupt(m) - require.EqualError(t, new(object.SplitInfo).ReadFromV2(*m), tc.err) + require.EqualError(t, new(object.SplitInfo).FromProtoMessage(m), tc.err) }) } }) } -func TestSplitInfo_ToV2(t *testing.T) { +func TestSplitInfo_ProtoMessage(t *testing.T) { var s object.SplitInfo // zero - m := s.ToV2() - require.Zero(t, m.GetSplitID()) + m := s.ProtoMessage() + require.Zero(t, m.GetSplitId()) require.Zero(t, m.GetFirstPart()) require.Zero(t, m.GetLink()) require.Zero(t, m.GetFirstPart()) // filled - m = validSplitInfo.ToV2() - require.EqualValues(t, anyValidSplitIDBytes, m.GetSplitID()) + m = validSplitInfo.ProtoMessage() + require.EqualValues(t, anyValidSplitIDBytes, m.GetSplitId()) require.Equal(t, anyValidIDs[0][:], m.GetLastPart().GetValue()) require.Equal(t, anyValidIDs[1][:], m.GetLink().GetValue()) require.Equal(t, anyValidIDs[2][:], m.GetFirstPart().GetValue()) @@ -424,9 +416,9 @@ func TestNewSplitInfo(t *testing.T) { require.True(t, si.GetFirstPart().IsZero()) // convert to v2 message - siV2 := si.ToV2() + siV2 := si.ProtoMessage() - require.Nil(t, siV2.GetSplitID()) + require.Nil(t, siV2.GetSplitId()) require.Nil(t, siV2.GetLastPart()) require.Nil(t, siV2.GetLink()) require.Nil(t, siV2.GetFirstPart()) diff --git a/object/test/generate.go b/object/test/generate.go index fb5010df..371d0d9e 100644 --- a/object/test/generate.go +++ b/object/test/generate.go @@ -5,7 +5,6 @@ import ( "strconv" "github.com/google/uuid" - objecttest "github.com/nspcc-dev/neofs-api-go/v2/object/test" checksumtest "github.com/nspcc-dev/neofs-sdk-go/checksum/test" cidtest "github.com/nspcc-dev/neofs-sdk-go/container/id/test" "github.com/nspcc-dev/neofs-sdk-go/object" @@ -134,5 +133,12 @@ func Lock() *object.Lock { // Link returns random object.Link. func Link() *object.Link { - return (*object.Link)(objecttest.GenerateLink(false)) + ms := make([]object.MeasuredObject, rand.Int()%10) + for i := range ms { + ms[i].SetObjectID(oidtest.ID()) + ms[i].SetObjectSize(rand.Uint32()) + } + var l object.Link + l.SetObjects(ms) + return &l } diff --git a/object/test/generate_test.go b/object/test/generate_test.go index b89f26f2..058e8105 100644 --- a/object/test/generate_test.go +++ b/object/test/generate_test.go @@ -17,7 +17,7 @@ func TestObject(t *testing.T) { require.Equal(t, obj, dst1) var dst2 object.Object - require.NoError(t, dst2.ReadFromV2(*obj.ToV2())) + require.NoError(t, dst2.FromProtoMessage(obj.ProtoMessage())) require.Equal(t, obj, dst2) j, err := obj.MarshalJSON() diff --git a/object/tombstone.go b/object/tombstone.go index dc7376b3..81721909 100644 --- a/object/tombstone.go +++ b/object/tombstone.go @@ -4,21 +4,17 @@ import ( "fmt" "github.com/google/uuid" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - "github.com/nspcc-dev/neofs-api-go/v2/tombstone" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" + prototombstone "github.com/nspcc-dev/neofs-sdk-go/proto/tombstone" ) -// Tombstone represents v2-compatible tombstone structure. -type Tombstone tombstone.Tombstone - -// NewTombstoneFromV2 wraps v2 [tombstone.Tombstone] message to [Tombstone]. -// -// Nil [tombstone.Tombstone] converts to nil. -// Deprecated: BUG: members' ID length is not checked. Use -// [Tombstone.ReadFromV2] instead. -func NewTombstoneFromV2(tV2 *tombstone.Tombstone) *Tombstone { - return (*Tombstone)(tV2) +// Tombstone represents object tombstone structure. +type Tombstone struct { + exp uint64 + splitID []byte + members []oid.ID } // NewTombstone creates and initializes blank [Tombstone]. @@ -31,53 +27,64 @@ func NewTombstone() *Tombstone { return new(Tombstone) } -// ReadFromV2 reads Tombstone from the [tombstone.Tombstone] message. Returns an -// error if the message is malformed according to the NeoFS API V2 protocol. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// t from it. // -// ReadFromV2 is intended to be used by the NeoFS API V2 client/server -// implementation only and is not expected to be directly used by applications. -func (t *Tombstone) ReadFromV2(m tombstone.Tombstone) error { - var id oid.ID - ms := m.GetMembers() - for i := range ms { - if err := id.ReadFromV2(ms[i]); err != nil { +// See also [Tombstone.ProtoMessage]. +func (t *Tombstone) FromProtoMessage(m *prototombstone.Tombstone) error { + if m.Members == nil { + t.members = nil + return nil + } + t.members = make([]oid.ID, len(m.Members)) + for i := range m.Members { + if m.Members[i] == nil { + return fmt.Errorf("nil member #%d", i) + } + if err := t.members[i].FromProtoMessage(m.Members[i]); err != nil { return fmt.Errorf("invalid member #%d: %w", i, err) } } - if b := m.GetSplitID(); len(b) > 0 { + if t.splitID = m.SplitId; len(m.SplitId) > 0 { var uid uuid.UUID - if err := uid.UnmarshalBinary(b); err != nil { + if err := uid.UnmarshalBinary(m.SplitId); err != nil { return fmt.Errorf("invalid split ID: %w", err) } else if v := uid.Version(); v != 4 { return fmt.Errorf("invalid split UUID version %d", v) } } - *t = Tombstone(m) + t.exp = m.ExpirationEpoch return nil } -// ToV2 converts [Tombstone] to v2 [tombstone.Tombstone] message. -// -// Nil [Tombstone] converts to nil. +// ProtoMessage converts t into message to transmit using the NeoFS API +// protocol. // -// The value returned shares memory with the structure itself, so changing it can lead to data corruption. -// Make a copy if you need to change it. -func (t Tombstone) ToV2() *tombstone.Tombstone { - return (*tombstone.Tombstone)(&t) +// See also [Tombstone.FromProtoMessage]. +func (t Tombstone) ProtoMessage() *prototombstone.Tombstone { + m := &prototombstone.Tombstone{ + ExpirationEpoch: t.exp, + SplitId: t.splitID, + Members: make([]*refs.ObjectID, len(t.members)), + } + for i := range t.members { + m.Members[i] = t.members[i].ProtoMessage() + } + return m } // ExpirationEpoch returns the last NeoFS epoch number of the tombstone lifetime. // // See also [Tombstone.SetExpirationEpoch]. func (t Tombstone) ExpirationEpoch() uint64 { - return (*tombstone.Tombstone)(&t).GetExpirationEpoch() + return t.exp } // SetExpirationEpoch sets the last NeoFS epoch number of the tombstone lifetime. // // See also [Tombstone.ExpirationEpoch]. func (t *Tombstone) SetExpirationEpoch(v uint64) { - (*tombstone.Tombstone)(t).SetExpirationEpoch(v) + t.exp = v } // SplitID returns identifier of object split hierarchy. @@ -87,96 +94,54 @@ func (t *Tombstone) SetExpirationEpoch(v uint64) { // // See also [Tombstone.SetSplitID]. func (t Tombstone) SplitID() *SplitID { - return NewSplitIDFromV2( - (*tombstone.Tombstone)(&t).GetSplitID()) + return NewSplitIDFromV2(t.splitID) } // SetSplitID sets identifier of object split hierarchy. // // See also [Tombstone.SplitID]. func (t *Tombstone) SetSplitID(v *SplitID) { - (*tombstone.Tombstone)(t).SetSplitID(v.ToV2()) + t.splitID = v.ToV2() } // Members returns list of objects to be deleted. // // See also [Tombstone.SetMembers]. func (t Tombstone) Members() []oid.ID { - msV2 := (*tombstone.Tombstone)(&t).GetMembers() - - if msV2 == nil { - return nil - } - - res := make([]oid.ID, len(msV2)) - for i := range msV2 { - if err := res[i].ReadFromV2(msV2[i]); err != nil { - panic(fmt.Errorf("invalid member #%d: %w", i, err)) - } - } - - return res + return t.members } // SetMembers sets list of objects to be deleted. // // See also [Tombstone.Members]. func (t *Tombstone) SetMembers(v []oid.ID) { - var ms []refs.ObjectID - - if v != nil { - ms = (*tombstone.Tombstone)(t). - GetMembers() - - if ln := len(v); cap(ms) >= ln { - ms = ms[:0] - } else { - ms = make([]refs.ObjectID, 0, ln) - } - - var idV2 refs.ObjectID - - for i := range v { - v[i].WriteToV2(&idV2) - ms = append(ms, idV2) - } - } - - (*tombstone.Tombstone)(t).SetMembers(ms) + t.members = v } // Marshal marshals [Tombstone] into a protobuf binary form. // // See also [Tombstone.Unmarshal]. func (t Tombstone) Marshal() []byte { - return (*tombstone.Tombstone)(&t).StableMarshal(nil) + return neofsproto.Marshal(t) } // Unmarshal unmarshals protobuf binary representation of [Tombstone]. // // See also [Tombstone.Marshal]. func (t *Tombstone) Unmarshal(data []byte) error { - var m tombstone.Tombstone - if err := m.Unmarshal(data); err != nil { - return err - } - return t.ReadFromV2(m) + return neofsproto.Unmarshal(data, t) } // MarshalJSON encodes [Tombstone] to protobuf JSON format. // // See also [Tombstone.UnmarshalJSON]. func (t Tombstone) MarshalJSON() ([]byte, error) { - return (*tombstone.Tombstone)(&t).MarshalJSON() + return neofsproto.MarshalJSON(t) } // UnmarshalJSON decodes [Tombstone] from protobuf JSON format. // // See also [Tombstone.MarshalJSON]. func (t *Tombstone) UnmarshalJSON(data []byte) error { - var m tombstone.Tombstone - if err := m.UnmarshalJSON(data); err != nil { - return err - } - return t.ReadFromV2(m) + return neofsproto.UnmarshalJSON(data, t) } diff --git a/object/tombstone_test.go b/object/tombstone_test.go index ec6224d1..13fb6578 100644 --- a/object/tombstone_test.go +++ b/object/tombstone_test.go @@ -3,14 +3,14 @@ package object_test import ( "bytes" "encoding/json" - "slices" "testing" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - "github.com/nspcc-dev/neofs-api-go/v2/tombstone" "github.com/nspcc-dev/neofs-sdk-go/object" oidtest "github.com/nspcc-dev/neofs-sdk-go/object/id/test" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" + prototombstone "github.com/nspcc-dev/neofs-sdk-go/proto/tombstone" "github.com/stretchr/testify/require" + "google.golang.org/protobuf/proto" ) var validTombstone object.Tombstone // set by init. @@ -50,93 +50,85 @@ const validJSONTombstone = ` } ` -func TestTombstone_ReadFromV2(t *testing.T) { - ms := make([]refs.ObjectID, len(anyValidIDs)) +func TestTombstone_FromProtoMessage(t *testing.T) { + ms := make([]*refs.ObjectID, len(anyValidIDs)) for i := range anyValidIDs { - ms[i].SetValue(anyValidIDs[i][:]) + ms[i] = protoIDFromBytes(anyValidIDs[i][:]) } - var m tombstone.Tombstone - m.SetExpirationEpoch(anyValidExpirationEpoch) - m.SetSplitID(anyValidSplitIDBytes) - m.SetMembers(ms) + m := &prototombstone.Tombstone{ + ExpirationEpoch: anyValidExpirationEpoch, + SplitId: anyValidSplitIDBytes, + Members: ms, + } var ts object.Tombstone - require.NoError(t, ts.ReadFromV2(m)) + require.NoError(t, ts.FromProtoMessage(m)) require.EqualValues(t, anyValidExpirationEpoch, ts.ExpirationEpoch()) require.Equal(t, anyValidSplitID, ts.SplitID()) require.Equal(t, anyValidIDs, ts.Members()) // reset optional fields - m.SetExpirationEpoch(0) - m.SetSplitID(nil) - m.SetMembers(nil) + m.ExpirationEpoch = 0 + m.SplitId = nil + m.Members = nil ts2 := ts - require.NoError(t, ts2.ReadFromV2(m)) + require.NoError(t, ts2.FromProtoMessage(m)) require.Zero(t, ts2) t.Run("invalid", func(t *testing.T) { for _, tc := range []struct { name, err string - corrupt func(*tombstone.Tombstone) + corrupt func(*prototombstone.Tombstone) }{ {name: "members/nil value", err: "invalid member #1: invalid length 0", - corrupt: func(m *tombstone.Tombstone) { - ms := slices.Clone(ms) - ms[1] = *protoIDFromBytes(nil) - m.SetMembers(ms) + corrupt: func(m *prototombstone.Tombstone) { + m.Members[1].Value = nil }}, {name: "members/empty value", err: "invalid member #1: invalid length 0", - corrupt: func(m *tombstone.Tombstone) { - ms := slices.Clone(ms) - ms[1] = *protoIDFromBytes([]byte{}) - m.SetMembers(ms) + corrupt: func(m *prototombstone.Tombstone) { + m.Members[1].Value = []byte{} }}, {name: "members/undersize", err: "invalid member #1: invalid length 31", - corrupt: func(m *tombstone.Tombstone) { - ms := slices.Clone(ms) - ms[1] = *protoIDFromBytes(make([]byte, 31)) - m.SetMembers(ms) + corrupt: func(m *prototombstone.Tombstone) { + m.Members[1].Value = make([]byte, 31) }}, {name: "members/oversize", err: "invalid member #1: invalid length 33", - corrupt: func(m *tombstone.Tombstone) { - ms := slices.Clone(ms) - ms[1] = *protoIDFromBytes(make([]byte, 33)) - m.SetMembers(ms) + corrupt: func(m *prototombstone.Tombstone) { + m.Members[1].Value = make([]byte, 33) }}, {name: "split ID/undersize", err: "invalid split ID: invalid UUID (got 15 bytes)", - corrupt: func(m *tombstone.Tombstone) { m.SetSplitID(anyValidSplitIDBytes[:15]) }}, + corrupt: func(m *prototombstone.Tombstone) { m.SplitId = anyValidSplitIDBytes[:15] }}, {name: "split ID/oversize", err: "invalid split ID: invalid UUID (got 17 bytes)", - corrupt: func(m *tombstone.Tombstone) { m.SetSplitID(append(anyValidSplitIDBytes[:], 1)) }}, + corrupt: func(m *prototombstone.Tombstone) { m.SplitId = append(anyValidSplitIDBytes[:], 1) }}, {name: "split ID/wrong version", err: "invalid split UUID version 3", - corrupt: func(m *tombstone.Tombstone) { - b := bytes.Clone(anyValidSplitIDBytes[:]) - b[6] = 3 << 4 - m.SetSplitID(b) + corrupt: func(m *prototombstone.Tombstone) { + m.SplitId = bytes.Clone(anyValidSplitIDBytes[:]) + m.SplitId[6] = 3 << 4 }}, } { t.Run(tc.name, func(t *testing.T) { - m := ts.ToV2() + m := proto.Clone(ts.ProtoMessage()).(*prototombstone.Tombstone) tc.corrupt(m) - require.EqualError(t, new(object.Tombstone).ReadFromV2(*m), tc.err) + require.EqualError(t, new(object.Tombstone).FromProtoMessage(m), tc.err) }) } }) } -func TestTombstone_WriteToV2(t *testing.T) { +func TestTombstone_ProtoMessage(t *testing.T) { var ts object.Tombstone // zero - m := ts.ToV2() + m := ts.ProtoMessage() require.Zero(t, m.GetExpirationEpoch()) - require.Zero(t, m.GetSplitID()) + require.Zero(t, m.GetSplitId()) require.Zero(t, m.GetMembers()) // filled - m = validTombstone.ToV2() + m = validTombstone.ProtoMessage() require.EqualValues(t, anyValidExpirationEpoch, m.GetExpirationEpoch()) - require.EqualValues(t, anyValidSplitIDBytes, m.GetSplitID()) + require.EqualValues(t, anyValidSplitIDBytes, m.GetSplitId()) ms := m.GetMembers() require.Len(t, ms, 3) for i := range ms { @@ -273,14 +265,6 @@ func TestTombstone_SetMembers(t *testing.T) { require.Equal(t, otherIDs, ts.Members()) } -func TestNewTombstoneFromV2(t *testing.T) { - t.Run("from nil", func(t *testing.T) { - var x *tombstone.Tombstone - - require.Nil(t, object.NewTombstoneFromV2(x)) - }) -} - func TestNewTombstone(t *testing.T) { t.Run("default values", func(t *testing.T) { ts := object.NewTombstone() @@ -291,9 +275,9 @@ func TestNewTombstone(t *testing.T) { require.Zero(t, ts.ExpirationEpoch()) // convert to v2 message - tsV2 := ts.ToV2() + tsV2 := ts.ProtoMessage() - require.Nil(t, tsV2.GetSplitID()) + require.Nil(t, tsV2.GetSplitId()) require.Nil(t, tsV2.GetMembers()) require.Zero(t, tsV2.GetExpirationEpoch()) }) diff --git a/object/type.go b/object/type.go index fc5550e7..3544542e 100644 --- a/object/type.go +++ b/object/type.go @@ -2,12 +2,10 @@ package object import ( "strconv" - - "github.com/nspcc-dev/neofs-api-go/v2/object" ) // Type is an enumerator for possible object types. -type Type object.Type +type Type int32 const ( TypeRegular Type = iota @@ -17,18 +15,6 @@ const ( TypeLink ) -// ToV2 converts [Type] to v2 [object.Type]. -// Deprecated: cast instead. -func (t Type) ToV2() object.Type { - return object.Type(t) -} - -// TypeFromV2 converts v2 [object.Type] to [Type]. -// Deprecated: cast instead. -func TypeFromV2(t object.Type) Type { - return Type(t) -} - const ( typeStringRegular = "REGULAR" typeStringTombstone = "TOMBSTONE" @@ -65,7 +51,7 @@ func (t Type) EncodeToString() string { return t.String() } func (t Type) String() string { switch t { default: - return strconv.FormatUint(uint64(t), 10) + return strconv.FormatInt(int64(t), 10) case TypeRegular: return typeStringRegular case TypeTombstone: diff --git a/object/type_test.go b/object/type_test.go index 40d3b40e..1b57470f 100644 --- a/object/type_test.go +++ b/object/type_test.go @@ -3,8 +3,8 @@ package object_test import ( "testing" - v2object "github.com/nspcc-dev/neofs-api-go/v2/object" "github.com/nspcc-dev/neofs-sdk-go/object" + protoobject "github.com/nspcc-dev/neofs-sdk-go/proto/object" "github.com/stretchr/testify/require" ) @@ -17,49 +17,13 @@ var typeStrings = map[object.Type]string{ 5: "5", } -func TestType_ToV2(t *testing.T) { - typs := []struct { - t object.Type - t2 v2object.Type - }{ - { - t: object.TypeRegular, - t2: v2object.TypeRegular, - }, - { - t: object.TypeTombstone, - t2: v2object.TypeTombstone, - }, - { - t: object.TypeStorageGroup, - t2: v2object.TypeStorageGroup, - }, - { - t: object.TypeLock, - t2: v2object.TypeLock, - }, - { - t: object.TypeLink, - t2: v2object.TypeLink, - }, - } - - for _, item := range typs { - t2 := item.t.ToV2() - - require.Equal(t, item.t2, t2) - - require.Equal(t, item.t, object.TypeFromV2(item.t2)) - } -} - func TestTypeProto(t *testing.T) { - for x, y := range map[v2object.Type]object.Type{ - v2object.TypeRegular: object.TypeRegular, - v2object.TypeTombstone: object.TypeTombstone, - v2object.TypeStorageGroup: object.TypeStorageGroup, - v2object.TypeLock: object.TypeLock, - v2object.TypeLink: object.TypeLink, + for x, y := range map[protoobject.ObjectType]object.Type{ + protoobject.ObjectType_REGULAR: object.TypeRegular, + protoobject.ObjectType_TOMBSTONE: object.TypeTombstone, + protoobject.ObjectType_STORAGE_GROUP: object.TypeStorageGroup, + protoobject.ObjectType_LOCK: object.TypeLock, + protoobject.ObjectType_LINK: object.TypeLink, } { require.EqualValues(t, x, y) } diff --git a/object/util_test.go b/object/util_test.go index 64cf2c39..27b0a63a 100644 --- a/object/util_test.go +++ b/object/util_test.go @@ -3,12 +3,12 @@ package object_test import ( "crypto/sha256" - "github.com/nspcc-dev/neofs-api-go/v2/refs" "github.com/nspcc-dev/neofs-sdk-go/checksum" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" "github.com/nspcc-dev/neofs-sdk-go/object" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" "github.com/nspcc-dev/neofs-sdk-go/user" "github.com/nspcc-dev/neofs-sdk-go/version" "github.com/nspcc-dev/tzhash/tz" @@ -87,19 +87,13 @@ var ( ) func protoIDFromBytes(b []byte) *refs.ObjectID { - var id refs.ObjectID - id.SetValue(b) - return &id + return &refs.ObjectID{Value: b} } func protoUserIDFromBytes(b []byte) *refs.OwnerID { - var id refs.OwnerID - id.SetValue(b) - return &id + return &refs.OwnerID{Value: b} } func protoContainerIDFromBytes(b []byte) *refs.ContainerID { - var id refs.ContainerID - id.SetValue(b) - return &id + return &refs.ContainerID{Value: b} } diff --git a/session/common.go b/session/common.go index f40a4610..cfa775b6 100644 --- a/session/common.go +++ b/session/common.go @@ -6,15 +6,14 @@ import ( "fmt" "github.com/google/uuid" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - "github.com/nspcc-dev/neofs-api-go/v2/session" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" "github.com/nspcc-dev/neofs-sdk-go/user" ) type commonData struct { - idSet bool - id uuid.UUID + id uuid.UUID issuer user.ID @@ -22,14 +21,12 @@ type commonData struct { authKey []byte - sigSet bool - sig neofscrypto.Signature + sig *neofscrypto.Signature } -type contextReader func(session.TokenContext, bool) error +type contextReader func(any, bool) error func (x commonData) copyTo(dst *commonData) { - dst.idSet = x.idSet dst.id = x.id dst.issuer = x.issuer @@ -38,15 +35,19 @@ func (x commonData) copyTo(dst *commonData) { dst.nbf = x.nbf dst.exp = x.exp dst.authKey = bytes.Clone(x.authKey) - dst.sigSet = x.sigSet - dst.sig = neofscrypto.NewSignatureFromRawKey(x.sig.Scheme(), bytes.Clone(x.sig.PublicKeyBytes()), bytes.Clone(x.sig.Value())) + if x.sig != nil { + sig := neofscrypto.NewSignatureFromRawKey(x.sig.Scheme(), bytes.Clone(x.sig.PublicKeyBytes()), bytes.Clone(x.sig.Value())) + dst.sig = &sig + } else { + dst.sig = nil + } } // reads commonData and custom context from the session.Token message. // If checkFieldPresence is set, returns an error on absence of any protocol-required // field. Verifies format of any presented field according to NeoFS API V2 protocol. // Calls contextReader if session context is set. Passes checkFieldPresence into contextReader. -func (x *commonData) readFromV2(m session.Token, checkFieldPresence bool, r contextReader) error { +func (x *commonData) fromProtoMessage(m *protosession.SessionToken, checkFieldPresence bool, r contextReader) error { var err error body := m.GetBody() @@ -54,8 +55,8 @@ func (x *commonData) readFromV2(m session.Token, checkFieldPresence bool, r cont return errors.New("missing token body") } - binID := body.GetID() - if x.idSet = len(binID) > 0; x.idSet { + binID := body.GetId() + if len(binID) > 0 { err = x.id.UnmarshalBinary(binID) if err != nil { return fmt.Errorf("invalid session ID: %w", err) @@ -64,11 +65,13 @@ func (x *commonData) readFromV2(m session.Token, checkFieldPresence bool, r cont } } else if checkFieldPresence { return errors.New("missing session ID") + } else { + x.id = uuid.Nil } - issuer := body.GetOwnerID() + issuer := body.GetOwnerId() if issuer != nil { - err = x.issuer.ReadFromV2(*issuer) + err = x.issuer.FromProtoMessage(issuer) if err != nil { return fmt.Errorf("invalid session issuer: %w", err) } @@ -98,13 +101,17 @@ func (x *commonData) readFromV2(m session.Token, checkFieldPresence bool, r cont return errors.New("missing session context") } - sig := m.GetSignature() - if x.sigSet = sig != nil; sig != nil { - if err = x.sig.ReadFromV2(*sig); err != nil { + if m.Signature != nil { + if x.sig == nil { + x.sig = new(neofscrypto.Signature) + } + if err = x.sig.FromProtoMessage(m.Signature); err != nil { return fmt.Errorf("invalid body signature: %w", err) } } else if checkFieldPresence { return errors.New("missing body signature") + } else { + x.sig = nil } x.iat = lifetime.GetIat() @@ -114,109 +121,90 @@ func (x *commonData) readFromV2(m session.Token, checkFieldPresence bool, r cont return nil } -type contextWriter func() session.TokenContext +type contextWriter func(body *protosession.SessionToken_Body) -func (x commonData) fillBody(w contextWriter) *session.TokenBody { - var body session.TokenBody +func (x commonData) fillBody(w contextWriter) *protosession.SessionToken_Body { + var body protosession.SessionToken_Body - if x.idSet { - binID, err := x.id.MarshalBinary() - if err != nil { - panic(fmt.Sprintf("unexpected error from UUID.MarshalBinary: %v", err)) - } - - body.SetID(binID) + if x.id != uuid.Nil { + body.Id = x.id[:] } if !x.issuer.IsZero() { - var issuer refs.OwnerID - x.issuer.WriteToV2(&issuer) - - body.SetOwnerID(&issuer) + body.OwnerId = x.issuer.ProtoMessage() } if x.iat != 0 || x.nbf != 0 || x.exp != 0 { - var lifetime session.TokenLifetime - lifetime.SetIat(x.iat) - lifetime.SetNbf(x.nbf) - lifetime.SetExp(x.exp) - - body.SetLifetime(&lifetime) + body.Lifetime = &protosession.SessionToken_Body_TokenLifetime{ + Exp: x.exp, + Nbf: x.nbf, + Iat: x.iat, + } } - body.SetSessionKey(x.authKey) + body.SessionKey = x.authKey - body.SetContext(w()) + w(&body) return &body } -func (x commonData) writeToV2(m *session.Token, w contextWriter) { - body := x.fillBody(w) - - m.SetBody(body) - - var sig *refs.Signature +func (x commonData) protoMessage(w contextWriter) *protosession.SessionToken { + m := &protosession.SessionToken{ + Body: x.fillBody(w), + } - if x.sigSet { - sig = new(refs.Signature) - x.sig.WriteToV2(sig) + if x.sig != nil { + m.Signature = x.sig.ProtoMessage() } - m.SetSignature(sig) + return m } func (x commonData) signedData(w contextWriter) []byte { - return x.fillBody(w).StableMarshal(nil) + return neofsproto.MarshalMessage(x.fillBody(w)) } func (x *commonData) sign(signer neofscrypto.Signer, w contextWriter) error { - err := x.sig.Calculate(signer, x.signedData(w)) - if err == nil { - x.sigSet = true + if x.sig == nil { + x.sig = new(neofscrypto.Signature) } - return err + return x.sig.Calculate(signer, x.signedData(w)) } func (x commonData) verifySignature(w contextWriter) bool { // TODO: (#233) check owner<->key relation - return x.sigSet && x.sig.Verify(x.signedData(w)) + return x.sig != nil && x.sig.Verify(x.signedData(w)) } func (x commonData) marshal(w contextWriter) []byte { - var m session.Token - x.writeToV2(&m, w) - - return m.StableMarshal(nil) + return neofsproto.MarshalMessage(x.protoMessage(w)) } func (x *commonData) unmarshal(data []byte, r contextReader) error { - var m session.Token + var m protosession.SessionToken - err := m.Unmarshal(data) + err := neofsproto.UnmarshalMessage(data, &m) if err != nil { return err } - return x.readFromV2(m, false, r) + return x.fromProtoMessage(&m, false, r) } func (x commonData) marshalJSON(w contextWriter) ([]byte, error) { - var m session.Token - x.writeToV2(&m, w) - - return m.MarshalJSON() + return neofsproto.MarshalMessageJSON(x.protoMessage(w)) } func (x *commonData) unmarshalJSON(data []byte, r contextReader) error { - var m session.Token + var m protosession.SessionToken - err := m.UnmarshalJSON(data) + err := neofsproto.UnmarshalMessageJSON(data, &m) if err != nil { return err } - return x.readFromV2(m, false, r) + return x.fromProtoMessage(&m, false, r) } // SetExp sets "exp" (expiration time) claim which identifies the expiration @@ -299,7 +287,6 @@ func (x commonData) InvalidAt(epoch uint64) bool { return !x.ValidAt(epoch) } // See also ID. func (x *commonData) SetID(id uuid.UUID) { x.id = id - x.idSet = true } // ID returns a unique identifier for the session. @@ -309,11 +296,7 @@ func (x *commonData) SetID(id uuid.UUID) { // // See also SetID. func (x commonData) ID() uuid.UUID { - if x.idSet { - return x.id - } - - return uuid.Nil + return x.id } // SetAuthKey public key corresponding to the private key bound to the session. @@ -351,7 +334,7 @@ func (x commonData) Issuer() user.ID { // IssuerPublicKeyBytes returns binary-encoded public key of the session issuer. // -// IssuerPublicKeyBytes MUST NOT be called before ReadFromV2 or Sign methods. +// IssuerPublicKeyBytes MUST NOT be called before ProtoMessage or Sign methods. // Deprecated: use Signature method. func (x *commonData) IssuerPublicKeyBytes() []byte { if sig, ok := x.Signature(); ok { @@ -363,12 +346,15 @@ func (x *commonData) IssuerPublicKeyBytes() []byte { // AttachSignature attaches given signature to the token. Use SignedData method // for calculation. If signature instance itself is not needed, use Sign method. func (x *commonData) AttachSignature(sig neofscrypto.Signature) { - x.sig, x.sigSet = sig, true + x.sig = &sig } // Signature returns token signature. If the signature is missing, false is // returned. Use SignedData method for verification. If signature instance // itself is not needed, use VerifySignature method. func (x commonData) Signature() (neofscrypto.Signature, bool) { - return x.sig, x.sigSet + if x.sig != nil { + return *x.sig, true + } + return neofscrypto.Signature{}, false } diff --git a/session/common_internal_test.go b/session/common_internal_test.go index 4baab9f9..c38afc41 100644 --- a/session/common_internal_test.go +++ b/session/common_internal_test.go @@ -5,8 +5,8 @@ import ( "testing" "github.com/google/uuid" - "github.com/nspcc-dev/neofs-api-go/v2/session" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" usertest "github.com/nspcc-dev/neofs-sdk-go/user/test" "github.com/stretchr/testify/require" ) @@ -17,23 +17,21 @@ func Test_commonData_copyTo(t *testing.T) { usr := usertest.User() data := commonData{ - idSet: true, id: uuid.New(), issuer: usr.UserID(), iat: 1, nbf: 2, exp: 3, authKey: []byte{1, 2, 3, 4}, - sigSet: true, - sig: sig, + sig: &sig, } t.Run("copy", func(t *testing.T) { var dst commonData data.copyTo(&dst) - emptyWriter := func() session.TokenContext { - return &session.ContainerSessionContext{} + emptyWriter := func(body *protosession.SessionToken_Body) { + body.Context = &protosession.SessionToken_Body_Container{} } require.Equal(t, data, dst) @@ -46,42 +44,40 @@ func Test_commonData_copyTo(t *testing.T) { var dst commonData data.copyTo(&dst) - require.Equal(t, data.idSet, dst.idSet) require.Equal(t, data.id.String(), dst.id.String()) dst.SetID(uuid.New()) - require.Equal(t, data.idSet, dst.idSet) require.NotEqual(t, data.id.String(), dst.id.String()) }) t.Run("overwrite id", func(t *testing.T) { // id is not set local := commonData{} - require.False(t, local.idSet) + require.Zero(t, local.id) // id is set var dst commonData dst.SetID(uuid.New()) - require.True(t, dst.idSet) + require.NotZero(t, dst.id) // overwrite ID data local.copyTo(&dst) - emptyWriter := func() session.TokenContext { - return &session.ContainerSessionContext{} + emptyWriter := func(body *protosession.SessionToken_Body) { + body.Context = &protosession.SessionToken_Body_Container{} } require.True(t, bytes.Equal(local.marshal(emptyWriter), dst.marshal(emptyWriter))) - require.False(t, local.idSet) - require.False(t, dst.idSet) + require.Zero(t, local.id) + require.Zero(t, dst.id) // update id dst.SetID(uuid.New()) // check that affects only dst - require.False(t, local.idSet) - require.True(t, dst.idSet) + require.Zero(t, local.id) + require.NotZero(t, dst.id) }) t.Run("change issuer", func(t *testing.T) { @@ -107,8 +103,8 @@ func Test_commonData_copyTo(t *testing.T) { require.Zero(t, local.issuer) require.Zero(t, dst.issuer) - emptyWriter := func() session.TokenContext { - return &session.ContainerSessionContext{} + emptyWriter := func(body *protosession.SessionToken_Body) { + body.Context = &protosession.SessionToken_Body_Container{} } require.True(t, bytes.Equal(local.marshal(emptyWriter), dst.marshal(emptyWriter))) @@ -150,8 +146,8 @@ func Test_commonData_copyTo(t *testing.T) { local.copyTo(&dst) - emptyWriter := func() session.TokenContext { - return &session.ContainerSessionContext{} + emptyWriter := func(body *protosession.SessionToken_Body) { + body.Context = &protosession.SessionToken_Body_Container{} } require.True(t, bytes.Equal(local.marshal(emptyWriter), dst.marshal(emptyWriter))) @@ -175,10 +171,7 @@ func Test_commonData_copyTo(t *testing.T) { var dst commonData data.copyTo(&dst) - require.Equal(t, data.sigSet, dst.sigSet) - require.Equal(t, data.sig.Scheme(), dst.sig.Scheme()) - require.True(t, bytes.Equal(data.sig.PublicKeyBytes(), dst.sig.PublicKeyBytes())) - require.True(t, bytes.Equal(data.sig.Value(), dst.sig.Value())) + require.Equal(t, data.sig, dst.sig) dst.sig.SetPublicKeyBytes([]byte{1, 2, 3}) dst.sig.SetScheme(100) @@ -192,20 +185,20 @@ func Test_commonData_copyTo(t *testing.T) { t.Run("overwrite sig", func(t *testing.T) { var local Container - require.False(t, local.sigSet) + require.Nil(t, local.sig) var dst Container require.NoError(t, dst.Sign(usr)) - require.True(t, dst.sigSet) + require.NotNil(t, dst.sig) local.CopyTo(&dst) - require.False(t, local.sigSet) - require.False(t, dst.sigSet) + require.Nil(t, local.sig) + require.Nil(t, dst.sig) require.True(t, bytes.Equal(local.Marshal(), dst.Marshal())) require.NoError(t, dst.Sign(usr)) - require.False(t, local.sigSet) - require.True(t, dst.sigSet) + require.Nil(t, local.sig) + require.NotNil(t, dst.sig) }) } diff --git a/session/common_test.go b/session/common_test.go index 923ac2f5..2a8116a5 100644 --- a/session/common_test.go +++ b/session/common_test.go @@ -7,17 +7,16 @@ import ( "crypto/sha512" "encoding/base64" "encoding/hex" - "math" "math/big" "testing" "github.com/google/uuid" "github.com/nspcc-dev/neo-go/pkg/io" - apisession "github.com/nspcc-dev/neofs-api-go/v2/session" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" neofsecdsa "github.com/nspcc-dev/neofs-sdk-go/crypto/ecdsa" neofscryptotest "github.com/nspcc-dev/neofs-sdk-go/crypto/test" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" "github.com/nspcc-dev/neofs-sdk-go/session" "github.com/nspcc-dev/neofs-sdk-go/user" usertest "github.com/nspcc-dev/neofs-sdk-go/user/test" @@ -65,67 +64,66 @@ var ( type invalidProtoTokenTestcase struct { name, err string - corrupt func(*apisession.Token) + corrupt func(*protosession.SessionToken) } var invalidProtoTokenCommonTestcases = []invalidProtoTokenTestcase{ - {name: "missing body", err: "missing token body", corrupt: func(st *apisession.Token) { - st.SetBody(nil) + {name: "missing body", err: "missing token body", corrupt: func(st *protosession.SessionToken) { + st.Body = nil }}, - {name: "missing body signature", err: "missing body signature", corrupt: func(st *apisession.Token) { - st.SetSignature(nil) + {name: "missing body signature", err: "missing body signature", corrupt: func(st *protosession.SessionToken) { + st.Signature = nil }}, - {name: "body/ID/nil", err: "missing session ID", corrupt: func(st *apisession.Token) { - st.GetBody().SetID(nil) + {name: "body/ID/nil", err: "missing session ID", corrupt: func(st *protosession.SessionToken) { + st.Body.Id = nil }}, - {name: "body/ID/empty", err: "missing session ID", corrupt: func(st *apisession.Token) { - st.GetBody().SetID([]byte{}) + {name: "body/ID/empty", err: "missing session ID", corrupt: func(st *protosession.SessionToken) { + st.Body.Id = []byte{} }}, - {name: "body/ID/undersize", err: "invalid session ID: invalid UUID (got 15 bytes)", corrupt: func(st *apisession.Token) { - st.GetBody().SetID(make([]byte, 15)) + {name: "body/ID/undersize", err: "invalid session ID: invalid UUID (got 15 bytes)", corrupt: func(st *protosession.SessionToken) { + st.Body.Id = make([]byte, 15) }}, - {name: "body/ID/wrong UUID version", err: "invalid session UUID version 3", corrupt: func(st *apisession.Token) { - st.GetBody().GetID()[6] = 3 << 4 + {name: "body/ID/wrong UUID version", err: "invalid session UUID version 3", corrupt: func(st *protosession.SessionToken) { + st.Body.Id[6] = 3 << 4 }}, - {name: "body/ID/oversize", err: "invalid session ID: invalid UUID (got 17 bytes)", corrupt: func(st *apisession.Token) { - st.GetBody().SetID(make([]byte, 17)) + {name: "body/ID/oversize", err: "invalid session ID: invalid UUID (got 17 bytes)", corrupt: func(st *protosession.SessionToken) { + st.Body.Id = make([]byte, 17) }}, - {name: "body/issuer", err: "missing session issuer", corrupt: func(st *apisession.Token) { - st.GetBody().SetOwnerID(nil) + {name: "body/issuer", err: "missing session issuer", corrupt: func(st *protosession.SessionToken) { + st.Body.OwnerId = nil }}, - {name: "body/issuer/value/nil", err: "invalid session issuer: invalid length 0, expected 25", corrupt: func(st *apisession.Token) { - st.GetBody().GetOwnerID().SetValue(nil) + {name: "body/issuer/value/nil", err: "invalid session issuer: invalid length 0, expected 25", corrupt: func(st *protosession.SessionToken) { + st.Body.OwnerId.Value = nil }}, - {name: "body/issuer/value/empty", err: "invalid session issuer: invalid length 0, expected 25", corrupt: func(st *apisession.Token) { - st.GetBody().GetOwnerID().SetValue([]byte{}) + {name: "body/issuer/value/empty", err: "invalid session issuer: invalid length 0, expected 25", corrupt: func(st *protosession.SessionToken) { + st.Body.OwnerId.Value = []byte{} }}, - {name: "body/issuer/value/undersize", err: "invalid session issuer: invalid length 24, expected 25", corrupt: func(st *apisession.Token) { - st.GetBody().GetOwnerID().SetValue(make([]byte, 24)) + {name: "body/issuer/value/undersize", err: "invalid session issuer: invalid length 24, expected 25", corrupt: func(st *protosession.SessionToken) { + st.Body.OwnerId.Value = make([]byte, 24) }}, - {name: "body/issuer/value/oversize", err: "invalid session issuer: invalid length 26, expected 25", corrupt: func(st *apisession.Token) { - st.GetBody().GetOwnerID().SetValue(make([]byte, 26)) + {name: "body/issuer/value/oversize", err: "invalid session issuer: invalid length 26, expected 25", corrupt: func(st *protosession.SessionToken) { + st.Body.OwnerId.Value = make([]byte, 26) }}, - {name: "body/issuer/value/wrong prefix", err: "invalid session issuer: invalid prefix byte 0x42, expected 0x35", corrupt: func(st *apisession.Token) { - st.GetBody().GetOwnerID().GetValue()[0] = 0x42 + {name: "body/issuer/value/wrong prefix", err: "invalid session issuer: invalid prefix byte 0x42, expected 0x35", corrupt: func(st *protosession.SessionToken) { + st.Body.OwnerId.Value[0] = 0x42 }}, - {name: "body/issuer/value/checksum mismatch", err: "invalid session issuer: checksum mismatch", corrupt: func(st *apisession.Token) { - v := st.GetBody().GetOwnerID().GetValue() - v[len(v)-1]++ + {name: "body/issuer/value/checksum mismatch", err: "invalid session issuer: checksum mismatch", corrupt: func(st *protosession.SessionToken) { + st.Body.OwnerId.Value[24]++ }}, - {name: "body/lifetime", err: "missing token lifetime", corrupt: func(st *apisession.Token) { - st.GetBody().SetLifetime(nil) + {name: "body/lifetime", err: "missing token lifetime", corrupt: func(st *protosession.SessionToken) { + st.Body.Lifetime = nil }}, - {name: "body/session key/nil", err: "missing session public key", corrupt: func(st *apisession.Token) { - st.GetBody().SetSessionKey(nil) + {name: "body/session key/nil", err: "missing session public key", corrupt: func(st *protosession.SessionToken) { + st.Body.SessionKey = nil }}, - {name: "body/session key/empty", err: "missing session public key", corrupt: func(st *apisession.Token) { - st.GetBody().SetSessionKey([]byte{}) + {name: "body/session key/empty", err: "missing session public key", corrupt: func(st *protosession.SessionToken) { + st.Body.SessionKey = []byte{} }}, - {name: "body/context/nil", err: "missing session context", corrupt: func(st *apisession.Token) { - st.GetBody().SetContext(nil) + {name: "body/context/nil", err: "missing session context", corrupt: func(st *protosession.SessionToken) { + st.Body.Context = nil }}, - {name: "signature/invalid scheme", err: "invalid body signature: scheme 2147483648 overflows int32", corrupt: func(st *apisession.Token) { - st.GetSignature().SetScheme(math.MaxInt32 + 1) + {name: "signature/scheme/negative", err: "invalid body signature: negative scheme -1", corrupt: func(st *protosession.SessionToken) { + st.Signature.Scheme = -1 }}, } @@ -155,7 +153,7 @@ var invalidBinTokenCommonTestcases = []invalidBinTokenTestcase{ {name: "body/issuer/value/checksum mismatch", err: "invalid session issuer: checksum mismatch", b: []byte{10, 29, 18, 27, 10, 25, 53, 51, 5, 166, 111, 29, 20, 101, 192, 165, 28, 167, 57, 160, 82, 80, 41, 203, 20, 254, 30, 138, 195, 17, 93}}, - {name: "signature/invalid scheme", err: "invalid body signature: scheme 2147483648 overflows int32", + {name: "signature/scheme/negative", err: "invalid body signature: negative scheme -2147483648", b: []byte{18, 11, 24, 128, 128, 128, 128, 248, 255, 255, 255, 255, 1}}, } @@ -212,7 +210,7 @@ var invalidJSONTokenCommonTestcases = []invalidJSONTokenTestcase{ {name: "body/issuer/value/checksum mismatch", err: "invalid session issuer: checksum mismatch", j: ` {"body":{"ownerID":{"value":"NTMFpm8dFGXApRynOaBSUCnLFP4eisMRXQ=="}}} `}, - {name: "signature/invalid scheme", err: "invalid body signature: scheme 2147483648 overflows int32", j: ` + {name: "signature/scheme/negative", err: "invalid body signature: negative scheme -2147483648", j: ` {"signature":{"scheme":-2147483648}} `}, } diff --git a/session/container.go b/session/container.go index 253b5514..44ebf8c4 100644 --- a/session/container.go +++ b/session/container.go @@ -3,12 +3,11 @@ package session import ( "errors" "fmt" - "math" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - "github.com/nspcc-dev/neofs-api-go/v2/session" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" "github.com/nspcc-dev/neofs-sdk-go/user" ) @@ -18,8 +17,8 @@ import ( // limited validity period, and applies to a strictly defined set of operations. // See methods for details. // -// Container is mutually compatible with github.com/nspcc-dev/neofs-api-go/v2/session.Token -// message. See ReadFromV2 / WriteToV2 methods. +// Container is mutually compatible with [protosession.SessionToken] message. +// See [Container.FromProtoMessage] / [Container.ProtoMessage] methods. // // Instances can be created using built-in var declaration. type Container struct { @@ -39,18 +38,18 @@ func (x Container) CopyTo(dst *Container) { } // readContext is a contextReader needed for commonData methods. -func (x *Container) readContext(c session.TokenContext, checkFieldPresence bool) error { - cCnr, ok := c.(*session.ContainerSessionContext) - if !ok || cCnr == nil { +func (x *Container) readContext(c any, checkFieldPresence bool) error { + cc, ok := c.(*protosession.SessionToken_Body_Container) + if !ok || cc == nil { return fmt.Errorf("invalid context %T", c) } + cCnr := cc.Container - wildcard := cCnr.Wildcard() - cnr := cCnr.ContainerID() + cnr := cCnr.GetContainerId() - if !wildcard { + if !cCnr.Wildcard { if cnr != nil { - err := x.cnr.ReadFromV2(*cnr) + err := x.cnr.FromProtoMessage(cnr) if err != nil { return fmt.Errorf("invalid container ID: %w", err) } @@ -63,49 +62,46 @@ func (x *Container) readContext(c session.TokenContext, checkFieldPresence bool) return errors.New("container conflicts with wildcard flag") } - verb := cCnr.Verb() - if verb > math.MaxInt32 { - return fmt.Errorf("verb %d overflows int32", verb) + verb := cCnr.GetVerb() + if verb < 0 { + return fmt.Errorf("negative verb %d", verb) } - x.verb = ContainerVerb(verb) + x.verb = ContainerVerb(cCnr.Verb) return nil } -func (x *Container) readFromV2(m session.Token, checkFieldPresence bool) error { - return x.commonData.readFromV2(m, checkFieldPresence, x.readContext) +func (x *Container) fromProtoMessage(m *protosession.SessionToken, checkFieldPresence bool) error { + return x.commonData.fromProtoMessage(m, checkFieldPresence, x.readContext) } -// ReadFromV2 reads Container from the session.Token message. Checks if the -// message conforms to NeoFS API V2 protocol. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// x from it. // -// See also WriteToV2. -func (x *Container) ReadFromV2(m session.Token) error { - return x.readFromV2(m, true) +// See also [Container.ProtoMessage]. +func (x *Container) FromProtoMessage(m *protosession.SessionToken) error { + return x.fromProtoMessage(m, true) } -func (x Container) writeContext() session.TokenContext { - wildcard := x.cnr.IsZero() - var c session.ContainerSessionContext - c.SetWildcard(wildcard) - c.SetVerb(session.ContainerSessionVerb(x.verb)) - - if !wildcard { - var cnr refs.ContainerID - x.cnr.WriteToV2(&cnr) +func (x Container) writeContext(m *protosession.SessionToken_Body) { + c := &protosession.ContainerSessionContext{ + Verb: protosession.ContainerSessionContext_Verb(x.verb), + Wildcard: x.cnr.IsZero(), + } - c.SetContainerID(&cnr) + if !c.Wildcard { + c.ContainerId = x.cnr.ProtoMessage() } - return &c + m.Context = &protosession.SessionToken_Body_Container{Container: c} } -// WriteToV2 writes Container to the session.Token message. -// The message must not be nil. +// ProtoMessage converts x into message to transmit using the NeoFS API +// protocol. // -// See also ReadFromV2. -func (x Container) WriteToV2(m *session.Token) { - x.writeToV2(m, x.writeContext) +// See also [Container.FromProtoMessage]. +func (x Container) ProtoMessage() *protosession.SessionToken { + return x.protoMessage(x.writeContext) } // Marshal encodes Container into a binary format of the NeoFS API protocol @@ -175,15 +171,13 @@ func (x *Container) SignedData() []byte { // UnmarshalSignedData is a reverse op to [Container.SignedData]. func (x *Container) UnmarshalSignedData(data []byte) error { - var body session.TokenBody - err := body.Unmarshal(data) + var body protosession.SessionToken_Body + err := neofsproto.UnmarshalMessage(data, &body) if err != nil { return fmt.Errorf("decode body: %w", err) } - var tok session.Token - tok.SetBody(&body) - return x.readFromV2(tok, false) + return x.fromProtoMessage(&protosession.SessionToken{Body: &body}, false) } // VerifySignature checks if Container signature is presented and valid. diff --git a/session/container_internal_test.go b/session/container_internal_test.go index 0d32735f..f9877ae4 100644 --- a/session/container_internal_test.go +++ b/session/container_internal_test.go @@ -4,8 +4,8 @@ import ( "bytes" "testing" - "github.com/nspcc-dev/neofs-api-go/v2/session" cidtest "github.com/nspcc-dev/neofs-sdk-go/container/id/test" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" "github.com/stretchr/testify/require" ) @@ -21,8 +21,8 @@ func TestContainer_CopyTo(t *testing.T) { var dst Container container.CopyTo(&dst) - emptyWriter := func() session.TokenContext { - return &session.ContainerSessionContext{} + emptyWriter := func(body *protosession.SessionToken_Body) { + body.Context = &protosession.SessionToken_Body_Container{} } require.Equal(t, container, dst) @@ -53,8 +53,8 @@ func TestContainer_CopyTo(t *testing.T) { require.NotZero(t, dst.cnr) local.CopyTo(&dst) - emptyWriter := func() session.TokenContext { - return &session.ContainerSessionContext{} + emptyWriter := func(body *protosession.SessionToken_Body) { + body.Context = &protosession.SessionToken_Body_Container{} } require.Equal(t, local, dst) diff --git a/session/container_test.go b/session/container_test.go index 5f0fe64d..f946b76e 100644 --- a/session/container_test.go +++ b/session/container_test.go @@ -5,17 +5,16 @@ import ( "crypto/ecdsa" "crypto/elliptic" "encoding/json" - "math" "math/big" "strconv" "testing" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - apisession "github.com/nspcc-dev/neofs-api-go/v2/session" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" cidtest "github.com/nspcc-dev/neofs-sdk-go/container/id/test" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" neofscryptotest "github.com/nspcc-dev/neofs-sdk-go/crypto/test" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" "github.com/nspcc-dev/neofs-sdk-go/session" "github.com/nspcc-dev/neofs-sdk-go/user" usertest "github.com/nspcc-dev/neofs-sdk-go/user/test" @@ -49,30 +48,30 @@ var validSignedContainerTokens = [][]byte{ 51, 5, 166, 111, 29, 20, 101, 192, 165, 28, 167, 57, 160, 82, 80, 41, 203, 20, 254, 30, 138, 195, 17, 92, 26, 18, 8, 238, 215, 164, 15, 16, 183, 189, 151, 204, 221, 2, 24, 190, 132, 217, 192, 4, 34, 33, 2, 149, 43, 50, 196, 91, 177, 62, 131, 233, 126, 241, 177, 13, 78, 96, 94, 119, 71, 55, 179, - 8, 53, 241, 79, 2, 1, 95, 85, 78, 45, 197, 136, 50, 2, 16, 1}, + 8, 53, 241, 79, 2, 1, 95, 85, 78, 45, 197, 136, 50, 8, 8, 147, 236, 159, 143, 3, 16, 1}, {10, 16, 99, 24, 111, 70, 22, 172, 72, 20, 139, 187, 175, 98, 10, 255, 231, 188, 18, 27, 10, 25, 53, 51, 5, 166, 111, 29, 20, 101, 192, 165, 28, 167, 57, 160, 82, 80, 41, 203, 20, 254, 30, 138, 195, 17, 92, 26, 18, 8, 238, 215, 164, 15, 16, 183, 189, 151, 204, 221, 2, 24, 190, 132, 217, 192, 4, 34, 33, 2, 149, 43, 50, 196, 91, 177, 62, 131, 233, 126, 241, 177, 13, 78, 96, 94, 119, 71, 55, 179, - 8, 53, 241, 79, 2, 1, 95, 85, 78, 45, 197, 136, 50, 36, 26, 34, 10, 32, 243, 245, 75, 198, 48, 107, + 8, 53, 241, 79, 2, 1, 95, 85, 78, 45, 197, 136, 50, 42, 8, 147, 236, 159, 143, 3, 26, 34, 10, 32, 243, 245, 75, 198, 48, 107, 141, 121, 255, 49, 51, 168, 21, 254, 62, 66, 6, 147, 43, 35, 99, 242, 163, 20, 26, 30, 147, 240, 79, 114, 252, 227}, } // corresponds to validContainerTokens. var validBinContainerTokens = [][]byte{ - {10, 106, 10, 16, 99, 24, 111, 70, 22, 172, 72, 20, 139, 187, 175, 98, 10, 255, 231, 188, 18, 27, 10, 25, 53, + {10, 112, 10, 16, 99, 24, 111, 70, 22, 172, 72, 20, 139, 187, 175, 98, 10, 255, 231, 188, 18, 27, 10, 25, 53, 51, 5, 166, 111, 29, 20, 101, 192, 165, 28, 167, 57, 160, 82, 80, 41, 203, 20, 254, 30, 138, 195, 17, 92, 26, 18, 8, 238, 215, 164, 15, 16, 183, 189, 151, 204, 221, 2, 24, 190, 132, 217, 192, 4, 34, 33, 2, 149, 43, 50, 196, 91, 177, 62, 131, 233, 126, 241, 177, 13, 78, 96, 94, 119, 71, 55, 179, 8, 53, 241, 79, 2, - 1, 95, 85, 78, 45, 197, 136, 50, 2, 16, 1, 18, 56, 10, 33, 3, 202, 217, 142, 98, 209, 190, 188, 145, 123, + 1, 95, 85, 78, 45, 197, 136, 50, 8, 8, 147, 236, 159, 143, 3, 16, 1, 18, 56, 10, 33, 3, 202, 217, 142, 98, 209, 190, 188, 145, 123, 174, 21, 173, 239, 239, 245, 67, 148, 205, 119, 58, 223, 219, 209, 220, 113, 215, 134, 228, 101, 249, 34, 218, 18, 13, 97, 110, 121, 95, 115, 105, 103, 110, 97, 116, 117, 114, 101, 24, 236, 236, 175, 192, 4}, - {10, 140, 1, 10, 16, 99, 24, 111, 70, 22, 172, 72, 20, 139, 187, 175, 98, 10, 255, 231, 188, 18, 27, 10, 25, 53, + {10, 146, 1, 10, 16, 99, 24, 111, 70, 22, 172, 72, 20, 139, 187, 175, 98, 10, 255, 231, 188, 18, 27, 10, 25, 53, 51, 5, 166, 111, 29, 20, 101, 192, 165, 28, 167, 57, 160, 82, 80, 41, 203, 20, 254, 30, 138, 195, 17, 92, 26, 18, 8, 238, 215, 164, 15, 16, 183, 189, 151, 204, 221, 2, 24, 190, 132, 217, 192, 4, 34, 33, 2, 149, 43, 50, 196, 91, 177, 62, 131, 233, 126, 241, 177, 13, 78, 96, 94, 119, 71, 55, 179, 8, 53, 241, 79, 2, - 1, 95, 85, 78, 45, 197, 136, 50, 36, 26, 34, 10, 32, 243, 245, 75, 198, 48, 107, 141, 121, 255, 49, 51, 168, + 1, 95, 85, 78, 45, 197, 136, 50, 42, 8, 147, 236, 159, 143, 3, 26, 34, 10, 32, 243, 245, 75, 198, 48, 107, 141, 121, 255, 49, 51, 168, 21, 254, 62, 66, 6, 147, 43, 35, 99, 242, 163, 20, 26, 30, 147, 240, 79, 114, 252, 227, 18, 56, 10, 33, 3, 202, 217, 142, 98, 209, 190, 188, 145, 123, 174, 21, 173, 239, 239, 245, 67, 148, 205, 119, 58, 223, 219, 209, 220, 113, 215, 134, 228, 101, 249, 34, 218, 18, 13, 97, 110, 121, 95, 115, 105, 103, 110, 97, 116, @@ -134,34 +133,27 @@ var validJSONContainerTokens = []string{` } `} -func TestContainer_ReadFromV2(t *testing.T) { - var lt apisession.TokenLifetime - lt.SetExp(anyValidExp) - lt.SetIat(anyValidIat) - lt.SetNbf(anyValidNbf) - var mo refs.OwnerID - mo.SetValue(anyValidUserID[:]) - var mcnr refs.ContainerID - mcnr.SetValue(anyValidContainerID[:]) - var mc apisession.ContainerSessionContext - mc.SetContainerID(&mcnr) - mc.SetVerb(anyValidContainerVerb) - var mb apisession.TokenBody - mb.SetID(anyValidSessionID[:]) - mb.SetOwnerID(&mo) - mb.SetLifetime(<) - mb.SetSessionKey(anyValidSessionKeyBytes) - mb.SetContext(&mc) - var msig refs.Signature - msig.SetKey(anyValidIssuerPublicKeyBytes) - msig.SetScheme(refs.SignatureScheme(anyValidSignatureScheme)) - msig.SetSign(anyValidSignatureBytes) - var m apisession.Token - m.SetBody(&mb) - m.SetSignature(&msig) +func TestContainer_FromProtoMessage(t *testing.T) { + m := &protosession.SessionToken{ + Body: &protosession.SessionToken_Body{ + Id: anyValidSessionID[:], + OwnerId: &refs.OwnerID{Value: anyValidUserID[:]}, + Lifetime: &protosession.SessionToken_Body_TokenLifetime{Exp: anyValidExp, Nbf: anyValidNbf, Iat: anyValidIat}, + SessionKey: anyValidSessionKeyBytes, + Context: &protosession.SessionToken_Body_Container{Container: &protosession.ContainerSessionContext{ + Verb: anyValidContainerVerb, + ContainerId: &refs.ContainerID{Value: anyValidContainerID[:]}, + }}, + }, + Signature: &refs.Signature{ + Key: anyValidIssuerPublicKeyBytes, + Sign: anyValidSignatureBytes, + Scheme: anyValidSignatureScheme, + }, + } var val session.Container - require.NoError(t, val.ReadFromV2(m)) + require.NoError(t, val.FromProtoMessage(m)) require.Equal(t, val.ID(), anyValidSessionID) require.Equal(t, val.Issuer(), anyValidUserID) require.EqualValues(t, anyValidExp, val.Exp()) @@ -179,94 +171,74 @@ func TestContainer_ReadFromV2(t *testing.T) { t.Run("invalid", func(t *testing.T) { for _, tc := range append(invalidProtoTokenCommonTestcases, invalidProtoTokenTestcase{ name: "context/missing", err: "missing session context", - corrupt: func(m *apisession.Token) { m.GetBody().SetContext(nil) }, + corrupt: func(m *protosession.SessionToken) { m.Body.Context = nil }, }, invalidProtoTokenTestcase{ - name: "context/wrong", err: "invalid context: invalid context *session.ObjectSessionContext", - corrupt: func(m *apisession.Token) { m.GetBody().SetContext(new(apisession.ObjectSessionContext)) }, + name: "context/wrong", err: "invalid context: invalid context *session.SessionToken_Body_Object", + corrupt: func(m *protosession.SessionToken) { m.Body.Context = new(protosession.SessionToken_Body_Object) }, }, invalidProtoTokenTestcase{ - name: "context/invalid verb", err: "invalid context: verb 2147483648 overflows int32", - corrupt: func(m *apisession.Token) { - var c apisession.ContainerSessionContext - c.SetWildcard(true) - c.SetVerb(math.MaxInt32 + 1) - m.GetBody().SetContext(&c) + name: "context/invalid verb", err: "invalid context: negative verb -1", + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Container).Container.Verb = -1 }, }, invalidProtoTokenTestcase{ name: "context/neither container nor wildcard", err: "invalid context: missing container or wildcard flag", - corrupt: func(m *apisession.Token) { m.GetBody().SetContext(new(apisession.ContainerSessionContext)) }, + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Container).Container.Reset() + }, }, invalidProtoTokenTestcase{ name: "context/both container and wildcard", err: "invalid context: container conflicts with wildcard flag", - corrupt: func(m *apisession.Token) { - var c apisession.ContainerSessionContext - c.SetContainerID(new(refs.ContainerID)) - c.SetWildcard(true) - m.GetBody().SetContext(&c) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Container).Container.Wildcard = true }, }, invalidProtoTokenTestcase{ name: "context/container/nil value", err: "invalid context: invalid container ID: invalid length 0", - corrupt: func(m *apisession.Token) { - var c apisession.ContainerSessionContext - c.SetContainerID(new(refs.ContainerID)) - m.GetBody().SetContext(&c) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Container).Container.ContainerId.Value = nil }, }, invalidProtoTokenTestcase{ name: "context/container/empty value", err: "invalid context: invalid container ID: invalid length 0", - corrupt: func(m *apisession.Token) { - var id refs.ContainerID - id.SetValue([]byte{}) - var c apisession.ContainerSessionContext - c.SetContainerID(&id) - m.GetBody().SetContext(&c) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Container).Container.ContainerId.Value = []byte{} }, }, invalidProtoTokenTestcase{ name: "context/container/undersize", err: "invalid context: invalid container ID: invalid length 31", - corrupt: func(m *apisession.Token) { - var id refs.ContainerID - id.SetValue(make([]byte, 31)) - var c apisession.ContainerSessionContext - c.SetContainerID(&id) - m.GetBody().SetContext(&c) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Container).Container.ContainerId.Value = make([]byte, 31) }, }, invalidProtoTokenTestcase{ name: "context/container/oversize", err: "invalid context: invalid container ID: invalid length 33", - corrupt: func(m *apisession.Token) { - var id refs.ContainerID - id.SetValue(make([]byte, 33)) - var c apisession.ContainerSessionContext - c.SetContainerID(&id) - m.GetBody().SetContext(&c) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Container).Container.ContainerId.Value = make([]byte, 33) }, }) { t.Run(tc.name, func(t *testing.T) { st := val - var m apisession.Token - st.WriteToV2(&m) - tc.corrupt(&m) - require.EqualError(t, new(session.Container).ReadFromV2(m), tc.err) + m := st.ProtoMessage() + tc.corrupt(m) + require.EqualError(t, new(session.Container).FromProtoMessage(m), tc.err) }) } }) } -func TestContainer_WriteToV2(t *testing.T) { +func TestContainer_ProtoMessage(t *testing.T) { var val session.Container - var m apisession.Token // zero - val.WriteToV2(&m) + m := val.ProtoMessage() require.Zero(t, m.GetSignature()) body := m.GetBody() require.NotNil(t, body) - require.Zero(t, body.GetID()) - require.Zero(t, body.GetOwnerID()) + require.Zero(t, body.GetId()) + require.Zero(t, body.GetOwnerId()) require.Zero(t, body.GetLifetime()) require.Zero(t, body.GetSessionKey()) c := body.GetContext() - require.IsType(t, new(apisession.ContainerSessionContext), c) - cc := c.(*apisession.ContainerSessionContext) - require.Zero(t, cc.Verb()) - require.Zero(t, cc.ContainerID()) - require.True(t, cc.Wildcard()) + require.IsType(t, new(protosession.SessionToken_Body_Container), c) + cc := c.(*protosession.SessionToken_Body_Container).Container + require.Zero(t, cc.GetVerb()) + require.Zero(t, cc.GetContainerId()) + require.True(t, cc.GetWildcard()) // filled val.SetID(anyValidSessionID) @@ -279,11 +251,11 @@ func TestContainer_WriteToV2(t *testing.T) { val.ApplyOnlyTo(anyValidContainerID) val.AttachSignature(anyValidSignature) - val.WriteToV2(&m) + m = val.ProtoMessage() body = m.GetBody() require.NotNil(t, body) - require.Equal(t, anyValidSessionID[:], body.GetID()) - require.Equal(t, anyValidUserID[:], body.GetOwnerID().GetValue()) + require.Equal(t, anyValidSessionID[:], body.GetId()) + require.Equal(t, anyValidUserID[:], body.GetOwnerId().GetValue()) lt := body.GetLifetime() require.EqualValues(t, anyValidExp, lt.GetExp()) require.EqualValues(t, anyValidIat, lt.GetIat()) @@ -295,11 +267,11 @@ func TestContainer_WriteToV2(t *testing.T) { require.Equal(t, anyValidIssuerPublicKeyBytes, sig.GetKey()) require.Equal(t, anyValidSignatureBytes, sig.GetSign()) c = body.GetContext() - require.IsType(t, new(apisession.ContainerSessionContext), c) - cc = c.(*apisession.ContainerSessionContext) - require.EqualValues(t, anyValidContainerVerb, cc.Verb()) - require.Equal(t, anyValidContainerID[:], cc.ContainerID().GetValue()) - require.Zero(t, cc.Wildcard()) + require.IsType(t, new(protosession.SessionToken_Body_Container), c) + cc = c.(*protosession.SessionToken_Body_Container).Container + require.EqualValues(t, anyValidContainerVerb, cc.GetVerb()) + require.Equal(t, anyValidContainerID[:], cc.GetContainerId().GetValue()) + require.Zero(t, cc.GetWildcard()) } func TestContainer_Marshal(t *testing.T) { @@ -316,7 +288,7 @@ func TestContainer_Unmarshal(t *testing.T) { require.ErrorContains(t, err, "cannot parse invalid wire-format data") }) for _, tc := range append(invalidBinTokenCommonTestcases, invalidBinTokenTestcase{ - name: "body/context/wrong oneof", err: "invalid context: invalid context *session.ObjectSessionContext", + name: "body/context/wrong oneof", err: "invalid context: invalid context *session.SessionToken_Body_Object", b: []byte{10, 4, 42, 2, 18, 0}, }, invalidBinTokenTestcase{ name: "body/context/both container and wildcard", err: "invalid context: container conflicts with wildcard flag", @@ -370,17 +342,14 @@ func TestContainer_Unmarshal(t *testing.T) { for i := range validContainerTokens { err := val.Unmarshal(validBinContainerTokens[i]) require.NoError(t, err) - t.Skip("https://github.com/nspcc-dev/neofs-sdk-go/issues/606") require.Equal(t, validContainerTokens[i], val) } } func TestContainer_MarshalJSON(t *testing.T) { for i := range validContainerTokens { - //nolint:staticcheck b, err := json.MarshalIndent(validContainerTokens[i], "", " ") require.NoError(t, err, i) - t.Skip("https://github.com/nspcc-dev/neofs-sdk-go/issues/606") require.JSONEq(t, validJSONContainerTokens[i], string(b)) } } @@ -393,7 +362,7 @@ func TestContainer_UnmarshalJSON(t *testing.T) { require.ErrorContains(t, err, "syntax error") }) for _, tc := range append(invalidJSONTokenCommonTestcases, invalidJSONTokenTestcase{ - name: "body/context/wrong oneof", err: "invalid context: invalid context *session.ObjectSessionContext", j: ` + name: "body/context/wrong oneof", err: "invalid context: invalid context *session.SessionToken_Body_Object", j: ` {"body":{"object":{}}} `}, invalidJSONTokenTestcase{ name: "body/context/both container and wildcard", err: "invalid context: container conflicts with wildcard flag", j: ` @@ -431,7 +400,6 @@ func TestContainer_UnmarshalJSON(t *testing.T) { // filled for i := range validContainerTokens { require.NoError(t, val.UnmarshalJSON([]byte(validJSONContainerTokens[i])), i) - t.Skip("https://github.com/nspcc-dev/neofs-sdk-go/issues/606") require.Equal(t, validContainerTokens[i], val, i) } } @@ -539,12 +507,12 @@ func TestContainer_Sign(t *testing.T) { var c session.Container for i, rfc6979Sig := range [][]byte{ - {7, 252, 130, 23, 167, 44, 8, 109, 123, 206, 34, 95, 110, 184, 195, 141, 43, 84, 35, 138, 93, 216, 168, - 230, 242, 242, 159, 103, 133, 142, 141, 104, 77, 166, 42, 74, 3, 150, 102, 137, 185, 116, 51, 101, - 147, 33, 4, 7, 14, 65, 174, 28, 44, 91, 168, 58, 128, 38, 163, 102, 52, 239, 213, 118}, - {239, 43, 34, 239, 180, 70, 238, 28, 100, 254, 33, 4, 89, 177, 32, 18, 215, 175, 8, 126, 126, 104, 102, - 180, 121, 13, 39, 78, 50, 132, 119, 250, 114, 225, 242, 135, 253, 191, 99, 129, 229, 108, 148, 223, - 24, 240, 44, 229, 102, 141, 124, 151, 121, 196, 250, 63, 116, 107, 113, 75, 109, 169, 249, 11}, + {190, 18, 239, 30, 103, 101, 136, 235, 201, 103, 161, 14, 141, 211, 187, 115, 174, 185, 216, 240, 250, 20, + 104, 255, 159, 187, 123, 153, 69, 51, 114, 161, 38, 249, 83, 23, 227, 242, 14, 169, 163, 96, 174, + 153, 174, 130, 142, 199, 157, 243, 8, 254, 0, 177, 165, 9, 148, 18, 72, 211, 199, 188, 220, 44}, + {5, 89, 114, 211, 237, 183, 201, 129, 24, 221, 131, 188, 255, 135, 221, 55, 49, 206, 184, 128, 44, 66, + 244, 148, 51, 41, 242, 41, 97, 153, 7, 70, 72, 63, 192, 149, 12, 37, 63, 4, 192, 125, 161, 27, 123, + 242, 76, 178, 148, 202, 241, 54, 4, 108, 34, 182, 217, 246, 125, 107, 132, 53, 91, 188}, } { validContainerTokens[i].CopyTo(&c) t.Run("container#"+strconv.Itoa(i), func(t *testing.T) { @@ -576,26 +544,26 @@ func TestContainer_VerifySignature(t *testing.T) { sigs [][]byte // of validContainerTokens }{ {scheme: neofscrypto.ECDSA_SHA512, sigs: [][]byte{ - {4, 31, 195, 240, 176, 85, 91, 249, 98, 82, 96, 126, 76, 27, 6, 181, 195, 193, 197, 62, 209, 78, 170, 109, 31, 169, - 249, 24, 211, 167, 110, 165, 200, 49, 194, 72, 123, 151, 121, 63, 29, 111, 22, 71, 220, 145, 58, 135, 95, - 244, 202, 224, 70, 162, 136, 39, 30, 58, 151, 240, 9, 65, 144, 32, 184}, - {4, 43, 119, 4, 66, 20, 131, 214, 29, 233, 25, 125, 222, 56, 184, 15, 153, 70, 48, 112, 211, 193, 79, 49, 233, - 36, 188, 130, 244, 42, 19, 134, 179, 5, 32, 143, 63, 35, 52, 228, 149, 202, 170, 174, 150, 246, 116, 182, - 44, 89, 25, 91, 172, 56, 163, 22, 33, 103, 8, 245, 245, 140, 212, 146, 186}, + {4, 42, 31, 236, 138, 99, 174, 186, 104, 85, 109, 115, 31, 152, 42, 84, 148, 73, 12, 21, 206, 199, 211, 246, 191, + 185, 143, 181, 125, 99, 149, 43, 26, 49, 26, 152, 186, 161, 95, 12, 157, 144, 212, 203, 158, 233, 148, 226, + 165, 55, 67, 155, 84, 84, 129, 65, 10, 137, 254, 20, 157, 139, 229, 46, 218}, + {4, 9, 87, 55, 182, 1, 68, 11, 29, 1, 146, 125, 72, 110, 146, 231, 62, 138, 245, 54, 16, 161, 248, 28, 7, 201, + 26, 25, 158, 27, 144, 224, 99, 226, 173, 191, 116, 60, 207, 247, 101, 233, 87, 205, 55, 162, 129, 182, + 211, 149, 194, 23, 242, 124, 238, 56, 80, 109, 45, 165, 15, 129, 91, 7, 180}, }}, {scheme: neofscrypto.ECDSA_DETERMINISTIC_SHA256, sigs: [][]byte{ - {184, 33, 118, 69, 74, 185, 216, 122, 57, 209, 165, 34, 215, 252, 81, 171, 91, 211, 169, 223, 107, 78, 246, 20, - 87, 15, 37, 126, 255, 170, 43, 89, 138, 25, 255, 54, 243, 205, 122, 120, 184, 22, 43, 72, 252, 254, 109, - 91, 176, 30, 116, 54, 181, 75, 172, 137, 245, 155, 232, 0, 96, 102, 15, 228}, - {221, 183, 51, 58, 146, 202, 120, 39, 156, 110, 158, 90, 11, 13, 7, 216, 227, 69, 190, 152, 110, 159, 17, 64, 251, - 35, 96, 40, 106, 69, 211, 112, 139, 127, 24, 179, 13, 199, 161, 102, 117, 217, 61, 25, 144, 222, 171, 203, 240, - 247, 50, 152, 151, 244, 69, 69, 69, 21, 221, 232, 12, 131, 163, 87}, + {211, 170, 29, 36, 209, 239, 196, 159, 185, 21, 248, 226, 171, 179, 107, 14, 171, 214, 250, 240, 188, 188, 95, 8, + 217, 230, 5, 85, 176, 231, 159, 77, 23, 181, 10, 140, 183, 169, 166, 218, 181, 21, 216, 53, 5, 39, 29, 89, 189, + 7, 79, 67, 114, 72, 62, 136, 144, 73, 91, 76, 151, 52, 1, 205}, + {129, 98, 32, 222, 16, 112, 71, 181, 155, 28, 175, 176, 189, 243, 132, 130, 112, 157, 244, 105, 218, 22, 28, 27, + 105, 109, 49, 184, 52, 180, 37, 151, 104, 161, 105, 108, 247, 104, 201, 72, 75, 5, 233, 94, 152, 136, 202, 63, + 121, 77, 193, 129, 137, 248, 215, 211, 77, 6, 147, 100, 201, 79, 18, 125}, }}, {scheme: neofscrypto.ECDSA_WALLETCONNECT, sigs: [][]byte{ - {30, 49, 223, 33, 75, 83, 235, 194, 92, 37, 74, 128, 38, 58, 215, 178, 79, 130, 40, 59, 77, 83, 126, 46, 68, 1, - 233, 170, 162, 153, 83, 65, 53, 171, 44, 138, 187, 214, 130, 160, 167, 96, 171, 7, 164, 95, 40, 58, 108, 214, 246, - 192, 239, 15, 36, 194, 179, 189, 192, 117, 166, 80, 176, 247, 117, 104, 6, 229, 191, 221, 25, 152, 75, 103, 187, - 125, 152, 193, 180, 204}, + {105, 91, 121, 219, 8, 156, 202, 92, 24, 217, 154, 168, 237, 8, 93, 138, 226, 111, 165, 72, 22, 245, 197, 64, 14, + 26, 207, 40, 110, 182, 182, 190, 53, 107, 12, 43, 115, 20, 250, 194, 251, 194, 160, 151, 48, 244, 126, 10, 185, + 226, 201, 137, 35, 122, 186, 69, 8, 239, 68, 66, 87, 126, 116, 12, 150, 15, 108, 163, 129, 197, 192, 140, 15, 96, + 16, 38, 160, 81, 110, 250}, }}, } { sig.SetScheme(tc.scheme) @@ -638,7 +606,7 @@ func TestContainer_UnmarshalSignedData(t *testing.T) { require.ErrorContains(t, err, "cannot parse invalid wire-format data") }) for _, tc := range append(invalidSignedTokenCommonTestcases, invalidBinTokenTestcase{ - name: "body/context/wrong oneof", err: "invalid context: invalid context *session.ObjectSessionContext", + name: "body/context/wrong oneof", err: "invalid context: invalid context *session.SessionToken_Body_Object", b: []byte{42, 2, 18, 0}, }, invalidBinTokenTestcase{ name: "body/context/both container and wildcard", err: "invalid context: container conflicts with wildcard flag", @@ -677,8 +645,7 @@ func TestContainer_UnmarshalSignedData(t *testing.T) { for i := range validContainerTokens { err := val.UnmarshalSignedData(validSignedContainerTokens[i]) require.NoError(t, err) - t.Skip("https://github.com/nspcc-dev/neofs-sdk-go/issues/606") - require.Equal(t, validContainerTokens[i], val) + require.Equal(t, validSignedContainerTokens[i], val.SignedData()) } } diff --git a/session/example_test.go b/session/example_test.go index 14c39db1..d87773c2 100644 --- a/session/example_test.go +++ b/session/example_test.go @@ -1,7 +1,6 @@ package session_test import ( - apiGoSession "github.com/nspcc-dev/neofs-api-go/v2/session" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" "github.com/nspcc-dev/neofs-sdk-go/session" @@ -35,17 +34,14 @@ func ExampleContainer() { // Instances can be also used to process NeoFS API V2 protocol messages with [https://github.com/nspcc-dev/neofs-api] package. func ExampleObject_marshalling() { - // import apiGoSession "github.com/nspcc-dev/neofs-api-go/v2/session" - // On the client side. var tok session.Object - var msg apiGoSession.Token - tok.WriteToV2(&msg) + msg := tok.ProtoMessage() // *send message* // On the server side. - _ = tok.ReadFromV2(msg) + _ = tok.FromProtoMessage(msg) } diff --git a/session/object.go b/session/object.go index 9b1cc565..c2ee9e80 100644 --- a/session/object.go +++ b/session/object.go @@ -3,14 +3,14 @@ package session import ( "errors" "fmt" - "math" "slices" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - "github.com/nspcc-dev/neofs-api-go/v2/session" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" + neofsproto "github.com/nspcc-dev/neofs-sdk-go/internal/proto" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" "github.com/nspcc-dev/neofs-sdk-go/user" ) @@ -20,8 +20,8 @@ import ( // limited validity period, and applies to a strictly defined set of operations. // See methods for details. // -// Object is mutually compatible with github.com/nspcc-dev/neofs-api-go/v2/session.Token -// message. See ReadFromV2 / WriteToV2 methods. +// Object is mutually compatible with [protosession.SessionToken] message. See +// [Object.FromProtoMessage] / [Object.ProtoMessage] methods. // // Instances can be created using built-in var declaration. type Object struct { @@ -43,17 +43,18 @@ func (x Object) CopyTo(dst *Object) { dst.objs = slices.Clone(x.objs) } -func (x *Object) readContext(c session.TokenContext, checkFieldPresence bool) error { - cObj, ok := c.(*session.ObjectSessionContext) - if !ok || cObj == nil { +func (x *Object) readContext(c any, checkFieldPresence bool) error { + cc, ok := c.(*protosession.SessionToken_Body_Object) + if !ok || cc == nil { return fmt.Errorf("invalid context %T", c) } + cObj := cc.Object var err error - cnr := cObj.GetContainer() + cnr := cObj.Target.GetContainer() if cnr != nil { - err := x.cnr.ReadFromV2(*cnr) + err := x.cnr.FromProtoMessage(cnr) if err != nil { return fmt.Errorf("invalid container ID: %w", err) } @@ -63,12 +64,15 @@ func (x *Object) readContext(c session.TokenContext, checkFieldPresence bool) er x.cnr = cid.ID{} } - objs := cObj.GetObjects() + objs := cObj.Target.GetObjects() if objs != nil { x.objs = make([]oid.ID, len(objs)) for i := range objs { - err = x.objs[i].ReadFromV2(objs[i]) + if objs[i] == nil { + return fmt.Errorf("nil target object #%d", i) + } + err = x.objs[i].FromProtoMessage(objs[i]) if err != nil { return fmt.Errorf("invalid target object: %w", err) } @@ -78,60 +82,56 @@ func (x *Object) readContext(c session.TokenContext, checkFieldPresence bool) er } verb := cObj.GetVerb() - if verb > math.MaxInt32 { - return fmt.Errorf("verb %d overflows int32", verb) + if verb < 0 { + return fmt.Errorf("negative verb %d", verb) } x.verb = ObjectVerb(verb) return nil } -func (x *Object) readFromV2(m session.Token, checkFieldPresence bool) error { - return x.commonData.readFromV2(m, checkFieldPresence, x.readContext) +func (x *Object) fromProtoMessage(m *protosession.SessionToken, checkFieldPresence bool) error { + return x.commonData.fromProtoMessage(m, checkFieldPresence, x.readContext) } -// ReadFromV2 reads Object from the session.Token message. Checks if the -// message conforms to NeoFS API V2 protocol. +// FromProtoMessage validates m according to the NeoFS API protocol and restores +// x from it. // -// See also WriteToV2. -func (x *Object) ReadFromV2(m session.Token) error { - return x.readFromV2(m, true) +// See also [Object.ProtoMessage]. +func (x *Object) FromProtoMessage(m *protosession.SessionToken) error { + return x.fromProtoMessage(m, true) } -func (x Object) writeContext() session.TokenContext { - var c session.ObjectSessionContext - c.SetVerb(session.ObjectSessionVerb(x.verb)) +func (x Object) writeContext(m *protosession.SessionToken_Body) { + c := &protosession.ObjectSessionContext{ + Verb: protosession.ObjectSessionContext_Verb(x.verb), + } if !x.cnr.IsZero() || len(x.objs) > 0 { - var cnr *refs.ContainerID + c.Target = new(protosession.ObjectSessionContext_Target) if !x.cnr.IsZero() { - cnr = new(refs.ContainerID) - x.cnr.WriteToV2(cnr) + c.Target.Container = x.cnr.ProtoMessage() } - var objs []refs.ObjectID - if x.objs != nil { - objs = make([]refs.ObjectID, len(x.objs)) + c.Target.Objects = make([]*refs.ObjectID, len(x.objs)) for i := range x.objs { - x.objs[i].WriteToV2(&objs[i]) + c.Target.Objects[i] = x.objs[i].ProtoMessage() } } - - c.SetTarget(cnr, objs...) } - return &c + m.Context = &protosession.SessionToken_Body_Object{Object: c} } -// WriteToV2 writes Object to the session.Token message. -// The message must not be nil. +// ProtoMessage converts x into message to transmit using the NeoFS API +// protocol. // -// See also ReadFromV2. -func (x Object) WriteToV2(m *session.Token) { - x.writeToV2(m, x.writeContext) +// See also [Object.FromProtoMessage]. +func (x Object) ProtoMessage() *protosession.SessionToken { + return x.protoMessage(x.writeContext) } // Marshal encodes Object into a binary format of the NeoFS API protocol @@ -139,9 +139,6 @@ func (x Object) WriteToV2(m *session.Token) { // // See also Unmarshal. func (x Object) Marshal() []byte { - var m session.Token - x.WriteToV2(&m) - return x.marshal(x.writeContext) } @@ -202,15 +199,13 @@ func (x *Object) SignedData() []byte { // UnmarshalSignedData is a reverse op to [Object.SignedData]. func (x *Object) UnmarshalSignedData(data []byte) error { - var body session.TokenBody - err := body.Unmarshal(data) + var body protosession.SessionToken_Body + err := neofsproto.UnmarshalMessage(data, &body) if err != nil { return fmt.Errorf("decode body: %w", err) } - var tok session.Token - tok.SetBody(&body) - return x.readFromV2(tok, false) + return x.fromProtoMessage(&protosession.SessionToken{Body: &body}, false) } // VerifySignature checks if Object signature is presented and valid. diff --git a/session/object_internal_test.go b/session/object_internal_test.go index a3a44329..38d6b689 100644 --- a/session/object_internal_test.go +++ b/session/object_internal_test.go @@ -4,9 +4,9 @@ import ( "bytes" "testing" - "github.com/nspcc-dev/neofs-api-go/v2/session" cidtest "github.com/nspcc-dev/neofs-sdk-go/container/id/test" oidtest "github.com/nspcc-dev/neofs-sdk-go/object/id/test" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" "github.com/stretchr/testify/require" ) @@ -23,8 +23,8 @@ func TestObject_CopyTo(t *testing.T) { var dst Object container.CopyTo(&dst) - emptyWriter := func() session.TokenContext { - return &session.ContainerSessionContext{} + emptyWriter := func(body *protosession.SessionToken_Body) { + body.Context = &protosession.SessionToken_Body_Container{} } require.Equal(t, container, dst) @@ -73,8 +73,8 @@ func TestObject_CopyTo(t *testing.T) { require.NotZero(t, dst.cnr) local.CopyTo(&dst) - emptyWriter := func() session.TokenContext { - return &session.ContainerSessionContext{} + emptyWriter := func(body *protosession.SessionToken_Body) { + body.Context = &protosession.SessionToken_Body_Container{} } require.Equal(t, local, dst) diff --git a/session/object_test.go b/session/object_test.go index 31c57edb..3d873e18 100644 --- a/session/object_test.go +++ b/session/object_test.go @@ -5,18 +5,17 @@ import ( "crypto/ecdsa" "crypto/elliptic" "encoding/json" - "math" "math/big" "testing" - "github.com/nspcc-dev/neofs-api-go/v2/refs" - apisession "github.com/nspcc-dev/neofs-api-go/v2/session" cid "github.com/nspcc-dev/neofs-sdk-go/container/id" cidtest "github.com/nspcc-dev/neofs-sdk-go/container/id/test" neofscrypto "github.com/nspcc-dev/neofs-sdk-go/crypto" neofscryptotest "github.com/nspcc-dev/neofs-sdk-go/crypto/test" oid "github.com/nspcc-dev/neofs-sdk-go/object/id" oidtest "github.com/nspcc-dev/neofs-sdk-go/object/id/test" + "github.com/nspcc-dev/neofs-sdk-go/proto/refs" + protosession "github.com/nspcc-dev/neofs-sdk-go/proto/session" "github.com/nspcc-dev/neofs-sdk-go/session" "github.com/nspcc-dev/neofs-sdk-go/user" usertest "github.com/nspcc-dev/neofs-sdk-go/user/test" @@ -92,7 +91,7 @@ var validJSONObjectToken = ` }, "sessionKey": "ApUrMsRbsT6D6X7xsQ1OYF53RzezCDXxTwIBX1VOLcWI", "object": { - "verb": 32905430, + "verb": 837285395, "target": { "container": { "value": "8/VLxjBrjXn/MTOoFf4+QgaTKyNj8qMUGh6T8E9y/OM=" @@ -119,38 +118,34 @@ var validJSONObjectToken = ` } ` -func TestObject_ReadFromV2(t *testing.T) { - var lt apisession.TokenLifetime - lt.SetExp(anyValidExp) - lt.SetIat(anyValidIat) - lt.SetNbf(anyValidNbf) - var mo refs.OwnerID - mo.SetValue(anyValidUserID[:]) - var mcnr refs.ContainerID - mcnr.SetValue(anyValidContainerID[:]) - mobjs := make([]refs.ObjectID, len(anyValidObjectIDs)) +func TestObject_FromProtoMessage(t *testing.T) { + mobjs := make([]*refs.ObjectID, len(anyValidObjectIDs)) for i := range anyValidObjectIDs { - mobjs[i].SetValue(anyValidObjectIDs[i][:]) + mobjs[i] = &refs.ObjectID{Value: anyValidObjectIDs[i][:]} + } + m := &protosession.SessionToken{ + Body: &protosession.SessionToken_Body{ + Id: anyValidSessionID[:], + OwnerId: &refs.OwnerID{Value: anyValidUserID[:]}, + Lifetime: &protosession.SessionToken_Body_TokenLifetime{Exp: anyValidExp, Nbf: anyValidNbf, Iat: anyValidIat}, + SessionKey: anyValidSessionKeyBytes, + Context: &protosession.SessionToken_Body_Object{Object: &protosession.ObjectSessionContext{ + Verb: anyValidObjectVerb, + Target: &protosession.ObjectSessionContext_Target{ + Container: &refs.ContainerID{Value: anyValidContainerID[:]}, + Objects: mobjs, + }, + }}, + }, + Signature: &refs.Signature{ + Key: anyValidIssuerPublicKeyBytes, + Sign: anyValidSignatureBytes, + Scheme: anyValidSignatureScheme, + }, } - var mc apisession.ObjectSessionContext - mc.SetTarget(&mcnr, mobjs...) - mc.SetVerb(anyValidObjectVerb) - var mb apisession.TokenBody - mb.SetID(anyValidSessionID[:]) - mb.SetOwnerID(&mo) - mb.SetLifetime(<) - mb.SetSessionKey(anyValidSessionKeyBytes) - mb.SetContext(&mc) - var msig refs.Signature - msig.SetKey(anyValidIssuerPublicKeyBytes) - msig.SetScheme(refs.SignatureScheme(anyValidSignatureScheme)) - msig.SetSign(anyValidSignatureBytes) - var m apisession.Token - m.SetBody(&mb) - m.SetSignature(&msig) var val session.Object - require.NoError(t, val.ReadFromV2(m)) + require.NoError(t, val.FromProtoMessage(m)) require.Equal(t, val.ID(), anyValidSessionID) require.Equal(t, val.Issuer(), anyValidUserID) require.EqualValues(t, anyValidExp, val.Exp()) @@ -171,109 +166,98 @@ func TestObject_ReadFromV2(t *testing.T) { t.Run("invalid", func(t *testing.T) { for _, tc := range append(invalidProtoTokenCommonTestcases, invalidProtoTokenTestcase{ name: "context/missing", err: "missing session context", - corrupt: func(m *apisession.Token) { m.GetBody().SetContext(nil) }, + corrupt: func(m *protosession.SessionToken) { m.Body.Context = nil }, + }, invalidProtoTokenTestcase{ + name: "context/wrong", err: "invalid context: invalid context *session.SessionToken_Body_Container", + corrupt: func(m *protosession.SessionToken) { m.Body.Context = new(protosession.SessionToken_Body_Container) }, }, invalidProtoTokenTestcase{ - name: "context/wrong", err: "invalid context: invalid context *session.ContainerSessionContext", - corrupt: func(m *apisession.Token) { m.GetBody().SetContext(new(apisession.ContainerSessionContext)) }, + name: "context/invalid verb", err: "invalid context: negative verb -1", + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Verb = -1 + }, }, invalidProtoTokenTestcase{ - name: "context/invalid verb", err: "invalid context: verb 2147483648 overflows int32", - corrupt: func(m *apisession.Token) { - m.GetBody().GetContext().(*apisession.ObjectSessionContext).SetVerb(math.MaxInt32 + 1) + name: "context/invalid verb", err: "invalid context: negative verb -1", + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Verb = -1 }, }, invalidProtoTokenTestcase{ name: "context/missing container", err: "invalid context: missing target container", - corrupt: func(m *apisession.Token) { - m.GetBody().GetContext().(*apisession.ObjectSessionContext).SetTarget(nil) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target = nil }, }, invalidProtoTokenTestcase{ name: "context/container/nil value", err: "invalid context: invalid container ID: invalid length 0", - corrupt: func(m *apisession.Token) { - m.GetBody().GetContext().(*apisession.ObjectSessionContext).SetTarget(new(refs.ContainerID)) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target.Container = new(refs.ContainerID) }, }, invalidProtoTokenTestcase{ name: "context/container/empty value", err: "invalid context: invalid container ID: invalid length 0", - corrupt: func(m *apisession.Token) { - var id refs.ContainerID - id.SetValue([]byte{}) - m.GetBody().GetContext().(*apisession.ObjectSessionContext).SetTarget(&id) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target.Container.Value = []byte{} }, }, invalidProtoTokenTestcase{ name: "context/container/undersize", err: "invalid context: invalid container ID: invalid length 31", - corrupt: func(m *apisession.Token) { - var id refs.ContainerID - id.SetValue(make([]byte, 31)) - m.GetBody().GetContext().(*apisession.ObjectSessionContext).SetTarget(&id) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target.Container.Value = make([]byte, 31) }, }, invalidProtoTokenTestcase{ name: "context/container/oversize", err: "invalid context: invalid container ID: invalid length 33", - corrupt: func(m *apisession.Token) { - var id refs.ContainerID - id.SetValue(make([]byte, 33)) - m.GetBody().GetContext().(*apisession.ObjectSessionContext).SetTarget(&id) + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target.Container.Value = make([]byte, 33) + }, + }, invalidProtoTokenTestcase{ + name: "context/objects/nil element", err: "invalid context: nil target object #1", + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target.Objects[1] = nil }, }, invalidProtoTokenTestcase{ - name: "context/object/nil value", err: "invalid context: invalid target object: invalid length 0", - corrupt: func(m *apisession.Token) { - c := m.GetBody().GetContext().(*apisession.ObjectSessionContext) - mo := c.GetObjects() - mo[1].SetValue(nil) - c.SetTarget(c.GetContainer(), mo...) + name: "context/objects/nil value", err: "invalid context: invalid target object: invalid length 0", + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target.Objects[1].Value = nil }, }, invalidProtoTokenTestcase{ - name: "context/object/empty value", err: "invalid context: invalid target object: invalid length 0", - corrupt: func(m *apisession.Token) { - c := m.GetBody().GetContext().(*apisession.ObjectSessionContext) - mo := c.GetObjects() - mo[1].SetValue([]byte{}) - c.SetTarget(c.GetContainer(), mo...) + name: "context/objects/empty value", err: "invalid context: invalid target object: invalid length 0", + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target.Objects[1].Value = []byte{} }, }, invalidProtoTokenTestcase{ - name: "context/object/undersize", err: "invalid context: invalid target object: invalid length 31", - corrupt: func(m *apisession.Token) { - c := m.GetBody().GetContext().(*apisession.ObjectSessionContext) - mo := c.GetObjects() - mo[1].SetValue(make([]byte, 31)) - c.SetTarget(c.GetContainer(), mo...) + name: "context/objects/undersize", err: "invalid context: invalid target object: invalid length 31", + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target.Objects[1].Value = make([]byte, 31) }, }, invalidProtoTokenTestcase{ - name: "context/object/oversize", err: "invalid context: invalid target object: invalid length 33", - corrupt: func(m *apisession.Token) { - c := m.GetBody().GetContext().(*apisession.ObjectSessionContext) - mo := c.GetObjects() - mo[1].SetValue(make([]byte, 33)) - c.SetTarget(c.GetContainer(), mo...) + name: "context/objects/oversize", err: "invalid context: invalid target object: invalid length 33", + corrupt: func(m *protosession.SessionToken) { + m.GetBody().GetContext().(*protosession.SessionToken_Body_Object).Object.Target.Objects[1].Value = make([]byte, 33) }, }) { t.Run(tc.name, func(t *testing.T) { st := val - var m apisession.Token - st.WriteToV2(&m) - tc.corrupt(&m) - require.EqualError(t, new(session.Object).ReadFromV2(m), tc.err) + m := st.ProtoMessage() + tc.corrupt(m) + require.EqualError(t, new(session.Object).FromProtoMessage(m), tc.err) }) } }) } -func TestObject_WriteToV2(t *testing.T) { +func TestObject_ProtoMessage(t *testing.T) { var val session.Object - var m apisession.Token // zero - val.WriteToV2(&m) + m := val.ProtoMessage() require.Zero(t, m.GetSignature()) body := m.GetBody() require.NotNil(t, body) - require.Zero(t, body.GetID()) - require.Zero(t, body.GetOwnerID()) + require.Zero(t, body.GetId()) + require.Zero(t, body.GetOwnerId()) require.Zero(t, body.GetLifetime()) require.Zero(t, body.GetSessionKey()) c := body.GetContext() - require.IsType(t, new(apisession.ObjectSessionContext), c) - oc := c.(*apisession.ObjectSessionContext) + require.IsType(t, new(protosession.SessionToken_Body_Object), c) + oc := c.(*protosession.SessionToken_Body_Object).Object require.Zero(t, oc.GetVerb()) - require.Zero(t, oc.GetContainer()) - require.Zero(t, oc.GetObjects()) + require.Zero(t, oc.GetTarget()) // filled val.SetID(anyValidSessionID) @@ -287,11 +271,11 @@ func TestObject_WriteToV2(t *testing.T) { val.LimitByObjects(anyValidObjectIDs...) val.AttachSignature(anyValidSignature) - val.WriteToV2(&m) + m = val.ProtoMessage() body = m.GetBody() require.NotNil(t, body) - require.Equal(t, anyValidSessionID[:], body.GetID()) - require.Equal(t, anyValidUserID[:], body.GetOwnerID().GetValue()) + require.Equal(t, anyValidSessionID[:], body.GetId()) + require.Equal(t, anyValidUserID[:], body.GetOwnerId().GetValue()) lt := body.GetLifetime() require.EqualValues(t, anyValidExp, lt.GetExp()) require.EqualValues(t, anyValidIat, lt.GetIat()) @@ -303,11 +287,12 @@ func TestObject_WriteToV2(t *testing.T) { require.Equal(t, anyValidIssuerPublicKeyBytes, sig.GetKey()) require.Equal(t, anyValidSignatureBytes, sig.GetSign()) c = body.GetContext() - require.IsType(t, new(apisession.ObjectSessionContext), c) - oc = c.(*apisession.ObjectSessionContext) + require.IsType(t, new(protosession.SessionToken_Body_Object), c) + oc = c.(*protosession.SessionToken_Body_Object).Object require.EqualValues(t, anyValidObjectVerb, oc.GetVerb()) - require.Equal(t, anyValidContainerID[:], oc.GetContainer().GetValue()) - mo := oc.GetObjects() + require.NotNil(t, oc.Target) + require.Equal(t, anyValidContainerID[:], oc.Target.GetContainer().GetValue()) + mo := oc.Target.GetObjects() require.Len(t, mo, len(anyValidObjectIDs)) for i := range anyValidObjectIDs { require.Equal(t, anyValidObjectIDs[i][:], mo[i].GetValue()) @@ -326,7 +311,7 @@ func TestObject_Unmarshal(t *testing.T) { require.ErrorContains(t, err, "cannot parse invalid wire-format data") }) for _, tc := range append(invalidBinTokenCommonTestcases, invalidBinTokenTestcase{ - name: "body/context/wrong oneof", err: "invalid context: invalid context *session.ContainerSessionContext", + name: "body/context/wrong oneof", err: "invalid context: invalid context *session.SessionToken_Body_Container", b: []byte{10, 2, 50, 0}, }, invalidBinTokenTestcase{ name: "body/context/container/empty value", err: "invalid context: invalid container ID: invalid length 0", @@ -406,7 +391,6 @@ func TestObject_Unmarshal(t *testing.T) { // filled err := val.Unmarshal(validBinObjectToken) require.NoError(t, err) - t.Skip("https://github.com/nspcc-dev/neofs-sdk-go/issues/606") require.Equal(t, validObjectToken, val) } @@ -414,7 +398,6 @@ func TestObject_MarshalJSON(t *testing.T) { //nolint:staticcheck b, err := json.MarshalIndent(validObjectToken, "", " ") require.NoError(t, err) - t.Skip("https://github.com/nspcc-dev/neofs-sdk-go/issues/606") require.JSONEq(t, validJSONObjectToken, string(b)) } @@ -426,7 +409,7 @@ func TestObject_UnmarshalJSON(t *testing.T) { require.ErrorContains(t, err, "syntax error") }) for _, tc := range append(invalidJSONTokenCommonTestcases, invalidJSONTokenTestcase{ - name: "body/context/wrong oneof", err: "invalid context: invalid context *session.ContainerSessionContext", j: ` + name: "body/context/wrong oneof", err: "invalid context: invalid context *session.SessionToken_Body_Container", j: ` {"body":{"container":{}}} `}, invalidJSONTokenTestcase{ name: "body/context/container/empty value", err: "invalid context: invalid container ID: invalid length 0", j: ` @@ -471,7 +454,6 @@ func TestObject_UnmarshalJSON(t *testing.T) { // filled require.NoError(t, val.UnmarshalJSON([]byte(validJSONObjectToken))) - t.Skip("https://github.com/nspcc-dev/neofs-sdk-go/issues/606") require.Equal(t, validObjectToken, val) } @@ -664,7 +646,7 @@ func TestObject_UnmarshalSignedData(t *testing.T) { require.ErrorContains(t, err, "cannot parse invalid wire-format data") }) for _, tc := range append(invalidSignedTokenCommonTestcases, invalidBinTokenTestcase{ - name: "body/context/wrong oneof", err: "invalid context: invalid context *session.ContainerSessionContext", + name: "body/context/wrong oneof", err: "invalid context: invalid context *session.SessionToken_Body_Container", b: []byte{50, 0}, }, invalidBinTokenTestcase{ name: "body/context/container/empty value", err: "invalid context: invalid container ID: invalid length 0", @@ -721,8 +703,7 @@ func TestObject_UnmarshalSignedData(t *testing.T) { // filled err := val.UnmarshalSignedData(validSignedObjectToken) require.NoError(t, err) - t.Skip("https://github.com/nspcc-dev/neofs-sdk-go/issues/606") - require.Equal(t, validObjectToken, val) + require.Equal(t, validSignedObjectToken, val.SignedData()) } func TestObject_SetExp(t *testing.T) {