forked from TrueCloudLab/frostfs-api-go
[#371] Support recent changes in NeoFS API protocol
Support: * new status codes (object, container, session); * object `Lock` message; * different signature schemes. Signed-off-by: Leonard Lyubich <leonard@nspcc.ru>
This commit is contained in:
parent
a4349f6692
commit
99370889d1
41 changed files with 476 additions and 0 deletions
BIN
accounting/grpc/service.pb.go
generated
BIN
accounting/grpc/service.pb.go
generated
Binary file not shown.
BIN
accounting/grpc/service_grpc.pb.go
generated
BIN
accounting/grpc/service_grpc.pb.go
generated
Binary file not shown.
BIN
accounting/grpc/types.pb.go
generated
BIN
accounting/grpc/types.pb.go
generated
Binary file not shown.
BIN
acl/grpc/types.pb.go
generated
BIN
acl/grpc/types.pb.go
generated
Binary file not shown.
BIN
audit/grpc/types.pb.go
generated
BIN
audit/grpc/types.pb.go
generated
Binary file not shown.
BIN
container/grpc/service.pb.go
generated
BIN
container/grpc/service.pb.go
generated
Binary file not shown.
BIN
container/grpc/service_grpc.pb.go
generated
BIN
container/grpc/service_grpc.pb.go
generated
Binary file not shown.
BIN
container/grpc/types.pb.go
generated
BIN
container/grpc/types.pb.go
generated
Binary file not shown.
28
container/status.go
Normal file
28
container/status.go
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
package container
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/status"
|
||||||
|
statusgrpc "github.com/nspcc-dev/neofs-api-go/v2/status/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
// LocalizeFailStatus checks if passed global status.Code is related to container failure and:
|
||||||
|
// then localizes the code and returns true,
|
||||||
|
// else leaves the code unchanged and returns false.
|
||||||
|
//
|
||||||
|
// Arg must not be nil.
|
||||||
|
func LocalizeFailStatus(c *status.Code) bool {
|
||||||
|
return status.LocalizeIfInSection(c, uint32(statusgrpc.Section_SECTION_CONTAINER))
|
||||||
|
}
|
||||||
|
|
||||||
|
// GlobalizeFail globalizes local code of container failure.
|
||||||
|
//
|
||||||
|
// Arg must not be nil.
|
||||||
|
func GlobalizeFail(c *status.Code) {
|
||||||
|
c.GlobalizeSection(uint32(statusgrpc.Section_SECTION_CONTAINER))
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
// StatusNotFound is a local status.Code value for
|
||||||
|
// CONTAINER_NOT_FOUND container failure.
|
||||||
|
StatusNotFound status.Code = iota
|
||||||
|
)
|
14
container/status_test.go
Normal file
14
container/status_test.go
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
package container_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/container"
|
||||||
|
statustest "github.com/nspcc-dev/neofs-api-go/v2/status/test"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestStatusCodes(t *testing.T) {
|
||||||
|
statustest.TestCodes(t, container.LocalizeFailStatus, container.GlobalizeFail,
|
||||||
|
container.StatusNotFound, 3072,
|
||||||
|
)
|
||||||
|
}
|
8
lock/grpc/types.go
Normal file
8
lock/grpc/types.go
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
package lock
|
||||||
|
|
||||||
|
import refs "github.com/nspcc-dev/neofs-api-go/v2/refs/grpc"
|
||||||
|
|
||||||
|
// SetMembers sets `members` field.
|
||||||
|
func (x *Lock) SetMembers(ids []*refs.ObjectID) {
|
||||||
|
x.Members = ids
|
||||||
|
}
|
BIN
lock/grpc/types.pb.go
generated
Normal file
BIN
lock/grpc/types.pb.go
generated
Normal file
Binary file not shown.
BIN
netmap/grpc/service.pb.go
generated
BIN
netmap/grpc/service.pb.go
generated
Binary file not shown.
BIN
netmap/grpc/service_grpc.pb.go
generated
BIN
netmap/grpc/service_grpc.pb.go
generated
Binary file not shown.
BIN
netmap/grpc/types.pb.go
generated
BIN
netmap/grpc/types.pb.go
generated
Binary file not shown.
BIN
object/grpc/service.pb.go
generated
BIN
object/grpc/service.pb.go
generated
Binary file not shown.
BIN
object/grpc/service_grpc.pb.go
generated
BIN
object/grpc/service_grpc.pb.go
generated
Binary file not shown.
BIN
object/grpc/types.pb.go
generated
BIN
object/grpc/types.pb.go
generated
Binary file not shown.
170
object/lock.go
Normal file
170
object/lock.go
Normal file
|
@ -0,0 +1,170 @@
|
||||||
|
package object
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
lock "github.com/nspcc-dev/neofs-api-go/v2/lock/grpc"
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/refs"
|
||||||
|
refsGRPC "github.com/nspcc-dev/neofs-api-go/v2/refs/grpc"
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/rpc/grpc"
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/rpc/message"
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/util/proto"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Lock represents object Lock message from NeoFS API V2 protocol.
|
||||||
|
type Lock struct {
|
||||||
|
members []refs.ObjectID
|
||||||
|
}
|
||||||
|
|
||||||
|
// NumberOfMembers returns length of lock list.
|
||||||
|
func (x *Lock) NumberOfMembers() int {
|
||||||
|
if x != nil {
|
||||||
|
return len(x.members)
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// IterateMembers passes members of the lock list to f.
|
||||||
|
func (x *Lock) IterateMembers(f func(refs.ObjectID)) {
|
||||||
|
if x != nil {
|
||||||
|
for i := range x.members {
|
||||||
|
f(x.members[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetMembers sets list of locked members.
|
||||||
|
// Arg must not be mutated for the duration of the Lock.
|
||||||
|
func (x *Lock) SetMembers(ids []refs.ObjectID) {
|
||||||
|
x.members = ids
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
_ = iota
|
||||||
|
fNumLockMembers
|
||||||
|
)
|
||||||
|
|
||||||
|
// StableMarshal encodes the Lock into Protocol Buffers binary format
|
||||||
|
// with direct field order.
|
||||||
|
func (x *Lock) StableMarshal(buf []byte) ([]byte, error) {
|
||||||
|
if x == nil || len(x.members) == 0 {
|
||||||
|
return []byte{}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if buf == nil {
|
||||||
|
buf = make([]byte, x.StableSize())
|
||||||
|
}
|
||||||
|
|
||||||
|
var offset, n int
|
||||||
|
var err error
|
||||||
|
|
||||||
|
for i := range x.members {
|
||||||
|
n, err = proto.NestedStructureMarshal(fNumLockMembers, buf[offset:], &x.members[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
offset += n
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// StableSize size of the buffer required to write the Lock in Protocol Buffers
|
||||||
|
// binary format.
|
||||||
|
func (x *Lock) StableSize() (sz int) {
|
||||||
|
if x != nil {
|
||||||
|
for i := range x.members {
|
||||||
|
sz += proto.NestedStructureSize(fNumLockMembers, &x.members[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unmarshal decodes the Lock from its Protocol Buffers binary format.
|
||||||
|
func (x *Lock) Unmarshal(data []byte) error {
|
||||||
|
return message.Unmarshal(x, data, new(lock.Lock))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *Lock) ToGRPCMessage() grpc.Message {
|
||||||
|
var m *lock.Lock
|
||||||
|
|
||||||
|
if x != nil {
|
||||||
|
m = new(lock.Lock)
|
||||||
|
|
||||||
|
var members []*refsGRPC.ObjectID
|
||||||
|
|
||||||
|
if x.members != nil {
|
||||||
|
members = make([]*refsGRPC.ObjectID, len(x.members))
|
||||||
|
|
||||||
|
for i := range x.members {
|
||||||
|
members[i] = x.members[i].ToGRPCMessage().(*refsGRPC.ObjectID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
m.SetMembers(members)
|
||||||
|
}
|
||||||
|
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *Lock) FromGRPCMessage(m grpc.Message) error {
|
||||||
|
v, ok := m.(*lock.Lock)
|
||||||
|
if !ok {
|
||||||
|
return message.NewUnexpectedMessageType(m, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
members := v.GetMembers()
|
||||||
|
if members == nil {
|
||||||
|
x.members = nil
|
||||||
|
} else {
|
||||||
|
x.members = make([]refs.ObjectID, len(members))
|
||||||
|
var err error
|
||||||
|
|
||||||
|
for i := range x.members {
|
||||||
|
err = x.members[i].FromGRPCMessage(members[i])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// WriteLock writes Lock to the Object as a payload content.
|
||||||
|
// The object must not be nil.
|
||||||
|
func WriteLock(obj *Object, lock Lock) {
|
||||||
|
hdr := obj.GetHeader()
|
||||||
|
if hdr == nil {
|
||||||
|
hdr = new(Header)
|
||||||
|
obj.SetHeader(hdr)
|
||||||
|
}
|
||||||
|
|
||||||
|
hdr.SetObjectType(TypeLock)
|
||||||
|
|
||||||
|
payload, err := lock.StableMarshal(nil)
|
||||||
|
if err != nil {
|
||||||
|
panic(fmt.Sprintf("encode lock content: %v", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
obj.SetPayload(payload)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReadLock reads Lock from the Object payload content.
|
||||||
|
func ReadLock(lock *Lock, obj Object) error {
|
||||||
|
payload := obj.GetPayload()
|
||||||
|
if len(payload) == 0 {
|
||||||
|
return errors.New("empty payload")
|
||||||
|
}
|
||||||
|
|
||||||
|
err := lock.Unmarshal(payload)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("decode lock content from payload: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
26
object/lock_test.go
Normal file
26
object/lock_test.go
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
package object_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/object"
|
||||||
|
objecttest "github.com/nspcc-dev/neofs-api-go/v2/object/test"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestLockRW(t *testing.T) {
|
||||||
|
var l object.Lock
|
||||||
|
var obj object.Object
|
||||||
|
|
||||||
|
require.Error(t, object.ReadLock(&l, obj))
|
||||||
|
|
||||||
|
l = *objecttest.GenerateLock(false)
|
||||||
|
|
||||||
|
object.WriteLock(&obj, l)
|
||||||
|
|
||||||
|
var l2 object.Lock
|
||||||
|
|
||||||
|
require.NoError(t, object.ReadLock(&l2, obj))
|
||||||
|
|
||||||
|
require.Equal(t, l, l2)
|
||||||
|
}
|
|
@ -50,5 +50,6 @@ func TestMessageConvert(t *testing.T) {
|
||||||
func(empty bool) message.Message { return objecttest.GenerateGetRangeHashRequest(empty) },
|
func(empty bool) message.Message { return objecttest.GenerateGetRangeHashRequest(empty) },
|
||||||
func(empty bool) message.Message { return objecttest.GenerateGetRangeHashResponseBody(empty) },
|
func(empty bool) message.Message { return objecttest.GenerateGetRangeHashResponseBody(empty) },
|
||||||
func(empty bool) message.Message { return objecttest.GenerateGetRangeHashResponse(empty) },
|
func(empty bool) message.Message { return objecttest.GenerateGetRangeHashResponse(empty) },
|
||||||
|
func(empty bool) message.Message { return objecttest.GenerateLock(empty) },
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
87
object/status.go
Normal file
87
object/status.go
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
package object
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/status"
|
||||||
|
statusgrpc "github.com/nspcc-dev/neofs-api-go/v2/status/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
// LocalizeFailStatus checks if passed global status.Code is related to object failure and:
|
||||||
|
// then localizes the code and returns true,
|
||||||
|
// else leaves the code unchanged and returns false.
|
||||||
|
//
|
||||||
|
// Arg must not be nil.
|
||||||
|
func LocalizeFailStatus(c *status.Code) bool {
|
||||||
|
return status.LocalizeIfInSection(c, uint32(statusgrpc.Section_SECTION_OBJECT))
|
||||||
|
}
|
||||||
|
|
||||||
|
// GlobalizeFail globalizes local code of object failure.
|
||||||
|
//
|
||||||
|
// Arg must not be nil.
|
||||||
|
func GlobalizeFail(c *status.Code) {
|
||||||
|
c.GlobalizeSection(uint32(statusgrpc.Section_SECTION_OBJECT))
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
// StatusAccessDenied is a local status.Code value for
|
||||||
|
// ACCESS_DENIED object failure.
|
||||||
|
StatusAccessDenied status.Code = iota
|
||||||
|
// StatusNotFound is a local status.Code value for
|
||||||
|
// OBJECT_NOT_FOUND object failure.
|
||||||
|
StatusNotFound
|
||||||
|
// StatusLocked is a local status.Code value for
|
||||||
|
// LOCKED object failure.
|
||||||
|
StatusLocked
|
||||||
|
// StatusLockNonRegularObject is a local status.Code value for
|
||||||
|
// LOCK_NON_REGULAR_OBJECT object failure.
|
||||||
|
StatusLockNonRegularObject
|
||||||
|
// StatusAlreadyRemoved is a local status.Code value for
|
||||||
|
// OBJECT_ALREADY_REMOVED object failure.
|
||||||
|
StatusAlreadyRemoved
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// detailAccessDeniedDesc is a StatusAccessDenied detail ID for
|
||||||
|
// human-readable description.
|
||||||
|
detailAccessDeniedDesc = iota
|
||||||
|
)
|
||||||
|
|
||||||
|
// WriteAccessDeniedDesc writes human-readable description of StatusAccessDenied
|
||||||
|
// into status.Status as a detail. The status must not be nil.
|
||||||
|
//
|
||||||
|
// Existing details are expected to be ID-unique, otherwise undefined behavior.
|
||||||
|
func WriteAccessDeniedDesc(st *status.Status, desc string) {
|
||||||
|
var found bool
|
||||||
|
|
||||||
|
st.IterateDetails(func(d *status.Detail) bool {
|
||||||
|
if d.ID() == detailAccessDeniedDesc {
|
||||||
|
found = true
|
||||||
|
d.SetValue([]byte(desc))
|
||||||
|
}
|
||||||
|
|
||||||
|
return found
|
||||||
|
})
|
||||||
|
|
||||||
|
if !found {
|
||||||
|
var d status.Detail
|
||||||
|
|
||||||
|
d.SetID(detailAccessDeniedDesc)
|
||||||
|
d.SetValue([]byte(desc))
|
||||||
|
|
||||||
|
st.AppendDetails(&d)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReadAccessDeniedDesc looks up for status detail with human-readable description
|
||||||
|
// of StatusAccessDenied. Returns empty string if detail is missing.
|
||||||
|
func ReadAccessDeniedDesc(st status.Status) (desc string) {
|
||||||
|
st.IterateDetails(func(d *status.Detail) bool {
|
||||||
|
if d.ID() == detailAccessDeniedDesc {
|
||||||
|
desc = string(d.Value())
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
34
object/status_test.go
Normal file
34
object/status_test.go
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
package object_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/object"
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/status"
|
||||||
|
statustest "github.com/nspcc-dev/neofs-api-go/v2/status/test"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestStatusCodes(t *testing.T) {
|
||||||
|
statustest.TestCodes(t, object.LocalizeFailStatus, object.GlobalizeFail,
|
||||||
|
object.StatusAccessDenied, 2048,
|
||||||
|
object.StatusNotFound, 2049,
|
||||||
|
object.StatusLocked, 2050,
|
||||||
|
object.StatusLockNonRegularObject, 2051,
|
||||||
|
object.StatusAlreadyRemoved, 2052,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccessDeniedDesc(t *testing.T) {
|
||||||
|
var st status.Status
|
||||||
|
|
||||||
|
require.Empty(t, object.ReadAccessDeniedDesc(st))
|
||||||
|
|
||||||
|
const desc = "some description"
|
||||||
|
|
||||||
|
object.WriteAccessDeniedDesc(&st, desc)
|
||||||
|
require.Equal(t, desc, object.ReadAccessDeniedDesc(st))
|
||||||
|
|
||||||
|
object.WriteAccessDeniedDesc(&st, desc+"1")
|
||||||
|
require.Equal(t, desc+"1", object.ReadAccessDeniedDesc(st))
|
||||||
|
}
|
|
@ -2,6 +2,7 @@ package objecttest
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/nspcc-dev/neofs-api-go/v2/object"
|
"github.com/nspcc-dev/neofs-api-go/v2/object"
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/refs"
|
||||||
refstest "github.com/nspcc-dev/neofs-api-go/v2/refs/test"
|
refstest "github.com/nspcc-dev/neofs-api-go/v2/refs/test"
|
||||||
sessiontest "github.com/nspcc-dev/neofs-api-go/v2/session/test"
|
sessiontest "github.com/nspcc-dev/neofs-api-go/v2/session/test"
|
||||||
)
|
)
|
||||||
|
@ -571,3 +572,16 @@ func GenerateGetRangeHashResponse(empty bool) *object.GetRangeHashResponse {
|
||||||
|
|
||||||
return m
|
return m
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func GenerateLock(empty bool) *object.Lock {
|
||||||
|
m := new(object.Lock)
|
||||||
|
|
||||||
|
if !empty {
|
||||||
|
m.SetMembers([]refs.ObjectID{
|
||||||
|
*refstest.GenerateObjectID(false),
|
||||||
|
*refstest.GenerateObjectID(false),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
|
|
@ -305,6 +305,7 @@ const (
|
||||||
TypeRegular Type = iota
|
TypeRegular Type = iota
|
||||||
TypeTombstone
|
TypeTombstone
|
||||||
TypeStorageGroup
|
TypeStorageGroup
|
||||||
|
TypeLock
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
|
BIN
reputation/grpc/service.pb.go
generated
BIN
reputation/grpc/service.pb.go
generated
Binary file not shown.
BIN
reputation/grpc/service_grpc.pb.go
generated
BIN
reputation/grpc/service_grpc.pb.go
generated
Binary file not shown.
BIN
reputation/grpc/types.pb.go
generated
BIN
reputation/grpc/types.pb.go
generated
Binary file not shown.
BIN
session/grpc/service.pb.go
generated
BIN
session/grpc/service.pb.go
generated
Binary file not shown.
BIN
session/grpc/service_grpc.pb.go
generated
BIN
session/grpc/service_grpc.pb.go
generated
Binary file not shown.
BIN
session/grpc/types.pb.go
generated
BIN
session/grpc/types.pb.go
generated
Binary file not shown.
31
session/status.go
Normal file
31
session/status.go
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
package session
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/status"
|
||||||
|
statusgrpc "github.com/nspcc-dev/neofs-api-go/v2/status/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
// LocalizeFailStatus checks if passed global status.Code is related to session failure and:
|
||||||
|
// then localizes the code and returns true,
|
||||||
|
// else leaves the code unchanged and returns false.
|
||||||
|
//
|
||||||
|
// Arg must not be nil.
|
||||||
|
func LocalizeFailStatus(c *status.Code) bool {
|
||||||
|
return status.LocalizeIfInSection(c, uint32(statusgrpc.Section_SECTION_SESSION))
|
||||||
|
}
|
||||||
|
|
||||||
|
// GlobalizeFail globalizes local code of session failure.
|
||||||
|
//
|
||||||
|
// Arg must not be nil.
|
||||||
|
func GlobalizeFail(c *status.Code) {
|
||||||
|
c.GlobalizeSection(uint32(statusgrpc.Section_SECTION_SESSION))
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
// StatusTokenNotFound is a local status.Code value for
|
||||||
|
// TOKEN_NOT_FOUND session failure.
|
||||||
|
StatusTokenNotFound status.Code = iota
|
||||||
|
// StatusTokenExpired is a local status.Code value for
|
||||||
|
// TOKEN_EXPIRED session failure.
|
||||||
|
StatusTokenExpired
|
||||||
|
)
|
15
session/status_test.go
Normal file
15
session/status_test.go
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
package session_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/session"
|
||||||
|
statustest "github.com/nspcc-dev/neofs-api-go/v2/status/test"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestStatusCodes(t *testing.T) {
|
||||||
|
statustest.TestCodes(t, session.LocalizeFailStatus, session.GlobalizeFail,
|
||||||
|
session.StatusTokenNotFound, 4096,
|
||||||
|
session.StatusTokenExpired, 4097,
|
||||||
|
)
|
||||||
|
}
|
BIN
status/grpc/types.pb.go
generated
BIN
status/grpc/types.pb.go
generated
Binary file not shown.
|
@ -80,3 +80,17 @@ func LocalizeCommonFail(c *Code) {
|
||||||
func GlobalizeCommonFail(c *Code) {
|
func GlobalizeCommonFail(c *Code) {
|
||||||
c.GlobalizeSection(sectionAfterSuccess(sectionCommon))
|
c.GlobalizeSection(sectionAfterSuccess(sectionCommon))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// LocalizeIfInSection checks if passed global status.Code belongs to the section and:
|
||||||
|
// then localizes the code and returns true,
|
||||||
|
// else leaves the code unchanged and returns false.
|
||||||
|
//
|
||||||
|
// Arg must not be nil.
|
||||||
|
func LocalizeIfInSection(c *Code, sec uint32) bool {
|
||||||
|
if IsInSection(*c, sec) {
|
||||||
|
c.LocalizeSection(sec)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
28
status/test/codes.go
Normal file
28
status/test/codes.go
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
package statustest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neofs-api-go/v2/status"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
// TestCodes checks mapping of status codes to the numbers.
|
||||||
|
// Args must be pairs (status.Code, int).
|
||||||
|
func TestCodes(t *testing.T,
|
||||||
|
localizer func(*status.Code) bool,
|
||||||
|
globalizer func(code *status.Code),
|
||||||
|
vals ...interface{},
|
||||||
|
) {
|
||||||
|
for i := 0; i < len(vals); i += 2 {
|
||||||
|
c := vals[i].(status.Code)
|
||||||
|
cp := c
|
||||||
|
|
||||||
|
globalizer(&cp)
|
||||||
|
require.True(t, cp.EqualNumber(uint32(vals[i+1].(int))), c)
|
||||||
|
|
||||||
|
require.True(t, localizer(&cp), c)
|
||||||
|
|
||||||
|
require.Equal(t, cp, c, c)
|
||||||
|
}
|
||||||
|
}
|
|
@ -42,6 +42,11 @@ func (x *Detail) SetValue(val []byte) {
|
||||||
// Code represents NeoFS API V2-compatible status code.
|
// Code represents NeoFS API V2-compatible status code.
|
||||||
type Code uint32
|
type Code uint32
|
||||||
|
|
||||||
|
// EqualNumber checks if the numerical Code equals num.
|
||||||
|
func (x Code) EqualNumber(num uint32) bool {
|
||||||
|
return uint32(x) == num
|
||||||
|
}
|
||||||
|
|
||||||
// Status represents structure of NeoFS API V2-compatible status return message.
|
// Status represents structure of NeoFS API V2-compatible status return message.
|
||||||
type Status struct {
|
type Status struct {
|
||||||
code Code
|
code Code
|
||||||
|
|
BIN
storagegroup/grpc/types.pb.go
generated
BIN
storagegroup/grpc/types.pb.go
generated
Binary file not shown.
BIN
subnet/grpc/types.pb.go
generated
BIN
subnet/grpc/types.pb.go
generated
Binary file not shown.
BIN
tombstone/grpc/types.pb.go
generated
BIN
tombstone/grpc/types.pb.go
generated
Binary file not shown.
BIN
util/proto/test/test.pb.go
generated
BIN
util/proto/test/test.pb.go
generated
Binary file not shown.
Loading…
Reference in a new issue