Add stable marshal of acl table

Signed-off-by: Alex Vanin <alexey@nspcc.ru>
This commit is contained in:
Alex Vanin 2020-08-17 16:57:51 +03:00 committed by Stanislav Bogatyrev
parent 6e42ba8d22
commit ddbeaa93b3
2 changed files with 124 additions and 33 deletions

View file

@ -7,26 +7,91 @@ import (
) )
const ( const (
FilterHeaderTypeField = 1 filterHeaderTypeField = 1
FilterMatchTypeField = 2 filterMatchTypeField = 2
FilterNameField = 3 filterNameField = 3
FilterValueField = 4 filterValueField = 4
TargetTypeField = 1 targetTypeField = 1
TargetKeysField = 2 targetKeysField = 2
RecordOperationField = 1 recordOperationField = 1
RecordActionField = 2 recordActionField = 2
RecordFiltersField = 3 recordFiltersField = 3
RecordTargetsField = 4 recordTargetsField = 4
tableContainerIDField = 1
tableRecordsField = 2
) )
func (t *Table) StableMarshal(buf []byte) ([]byte, error) { func (t *Table) StableMarshal(buf []byte) ([]byte, error) {
panic("not implemented") if t == nil {
return []byte{}, nil
} }
func (t *Table) StableSize() int { if buf == nil {
panic("not implemented") buf = make([]byte, t.StableSize())
}
var (
offset, n int
prefix uint64
err error
)
if t.cid != nil {
prefix, _ = proto.NestedStructurePrefix(tableContainerIDField)
offset += binary.PutUvarint(buf[offset:], prefix)
n = t.cid.StableSize()
offset += binary.PutUvarint(buf[offset:], uint64(n))
_, err = t.cid.StableMarshal(buf[offset:])
if err != nil {
return nil, err
}
offset += n
}
prefix, _ = proto.NestedStructurePrefix(tableRecordsField)
for i := range t.records {
offset += binary.PutUvarint(buf[offset:], prefix)
n = t.records[i].StableSize()
offset += binary.PutUvarint(buf[offset:], uint64(n))
_, err = t.records[i].StableMarshal(buf[offset:])
if err != nil {
return nil, err
}
offset += n
}
return buf, nil
}
func (t *Table) StableSize() (size int) {
if t == nil {
return 0
}
if t.cid != nil {
_, ln := proto.NestedStructurePrefix(tableContainerIDField)
n := t.cid.StableSize()
size += ln + proto.VarUIntSize(uint64(n)) + n
}
_, ln := proto.NestedStructurePrefix(tableRecordsField)
for i := range t.records {
n := t.records[i].StableSize()
size += ln + proto.VarUIntSize(uint64(n)) + n
}
return size
} }
func (r *Record) StableMarshal(buf []byte) ([]byte, error) { func (r *Record) StableMarshal(buf []byte) ([]byte, error) {
@ -44,21 +109,21 @@ func (r *Record) StableMarshal(buf []byte) ([]byte, error) {
err error err error
) )
n, err = proto.EnumMarshal(RecordOperationField, buf, int32(r.op)) n, err = proto.EnumMarshal(recordOperationField, buf, int32(r.op))
if err != nil { if err != nil {
return nil, err return nil, err
} }
offset += n offset += n
n, err = proto.EnumMarshal(RecordActionField, buf[offset:], int32(r.action)) n, err = proto.EnumMarshal(recordActionField, buf[offset:], int32(r.action))
if err != nil { if err != nil {
return nil, err return nil, err
} }
offset += n offset += n
prefix, _ = proto.NestedStructurePrefix(RecordFiltersField) prefix, _ = proto.NestedStructurePrefix(recordFiltersField)
for i := range r.filters { for i := range r.filters {
offset += binary.PutUvarint(buf[offset:], prefix) offset += binary.PutUvarint(buf[offset:], prefix)
@ -74,7 +139,7 @@ func (r *Record) StableMarshal(buf []byte) ([]byte, error) {
offset += n offset += n
} }
prefix, _ = proto.NestedStructurePrefix(RecordTargetsField) prefix, _ = proto.NestedStructurePrefix(recordTargetsField)
for i := range r.targets { for i := range r.targets {
offset += binary.PutUvarint(buf[offset:], prefix) offset += binary.PutUvarint(buf[offset:], prefix)
@ -98,17 +163,17 @@ func (r *Record) StableSize() (size int) {
return 0 return 0
} }
size += proto.EnumSize(RecordOperationField, int32(r.op)) size += proto.EnumSize(recordOperationField, int32(r.op))
size += proto.EnumSize(RecordActionField, int32(r.op)) size += proto.EnumSize(recordActionField, int32(r.op))
_, ln := proto.NestedStructurePrefix(RecordFiltersField) _, ln := proto.NestedStructurePrefix(recordFiltersField)
for i := range r.filters { for i := range r.filters {
n := r.filters[i].StableSize() n := r.filters[i].StableSize()
size += ln + proto.VarUIntSize(uint64(n)) + n size += ln + proto.VarUIntSize(uint64(n)) + n
} }
_, ln = proto.NestedStructurePrefix(RecordTargetsField) _, ln = proto.NestedStructurePrefix(recordTargetsField)
for i := range r.targets { for i := range r.targets {
n := r.targets[i].StableSize() n := r.targets[i].StableSize()
@ -132,28 +197,28 @@ func (f *HeaderFilter) StableMarshal(buf []byte) ([]byte, error) {
err error err error
) )
n, err = proto.EnumMarshal(FilterHeaderTypeField, buf, int32(f.hdrType)) n, err = proto.EnumMarshal(filterHeaderTypeField, buf, int32(f.hdrType))
if err != nil { if err != nil {
return nil, err return nil, err
} }
offset += n offset += n
n, err = proto.EnumMarshal(FilterMatchTypeField, buf[offset:], int32(f.matchType)) n, err = proto.EnumMarshal(filterMatchTypeField, buf[offset:], int32(f.matchType))
if err != nil { if err != nil {
return nil, err return nil, err
} }
offset += n offset += n
n, err = proto.StringMarshal(FilterNameField, buf[offset:], f.name) n, err = proto.StringMarshal(filterNameField, buf[offset:], f.name)
if err != nil { if err != nil {
return nil, err return nil, err
} }
offset += n offset += n
n, err = proto.StringMarshal(FilterValueField, buf[offset:], f.value) n, err = proto.StringMarshal(filterValueField, buf[offset:], f.value)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -166,10 +231,10 @@ func (f *HeaderFilter) StableSize() (size int) {
return 0 return 0
} }
size += proto.EnumSize(FilterHeaderTypeField, int32(f.hdrType)) size += proto.EnumSize(filterHeaderTypeField, int32(f.hdrType))
size += proto.EnumSize(FilterMatchTypeField, int32(f.matchType)) size += proto.EnumSize(filterMatchTypeField, int32(f.matchType))
size += proto.StringSize(FilterNameField, f.name) size += proto.StringSize(filterNameField, f.name)
size += proto.StringSize(FilterValueField, f.value) size += proto.StringSize(filterValueField, f.value)
return size return size
} }
@ -188,14 +253,14 @@ func (t *TargetInfo) StableMarshal(buf []byte) ([]byte, error) {
err error err error
) )
n, err = proto.EnumMarshal(TargetTypeField, buf, int32(t.target)) n, err = proto.EnumMarshal(targetTypeField, buf, int32(t.target))
if err != nil { if err != nil {
return nil, err return nil, err
} }
offset += n offset += n
n, err = proto.RepeatedBytesMarshal(TargetKeysField, buf[offset:], t.keys) n, err = proto.RepeatedBytesMarshal(targetKeysField, buf[offset:], t.keys)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -208,8 +273,8 @@ func (t *TargetInfo) StableSize() (size int) {
return 0 return 0
} }
size += proto.EnumSize(TargetTypeField, int32(t.target)) size += proto.EnumSize(targetTypeField, int32(t.target))
size += proto.RepeatedBytesSize(TargetKeysField, t.keys) size += proto.RepeatedBytesSize(targetKeysField, t.keys)
return size return size
} }

View file

@ -6,6 +6,7 @@ import (
"github.com/nspcc-dev/neofs-api-go/v2/acl" "github.com/nspcc-dev/neofs-api-go/v2/acl"
grpc "github.com/nspcc-dev/neofs-api-go/v2/acl/grpc" grpc "github.com/nspcc-dev/neofs-api-go/v2/acl/grpc"
"github.com/nspcc-dev/neofs-api-go/v2/refs"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
) )
@ -120,3 +121,28 @@ func TestRecord_StableMarshal(t *testing.T) {
require.Equal(t, recordFrom, recordTo) require.Equal(t, recordFrom, recordTo)
}) })
} }
func TestTable_StableMarshal(t *testing.T) {
tableFrom := new(acl.Table)
transport := new(grpc.EACLTable)
t.Run("non empty", func(t *testing.T) {
cid := new(refs.ContainerID)
cid.SetValue([]byte("Container ID"))
r1 := generateRecord(false)
r2 := generateRecord(true)
tableFrom.SetContainerID(cid)
tableFrom.SetRecords([]*acl.Record{r1, r2})
wire, err := tableFrom.StableMarshal(nil)
require.NoError(t, err)
err = transport.Unmarshal(wire)
require.NoError(t, err)
tableTo := acl.TableFromGRPCMessage(transport)
require.Equal(t, tableFrom, tableTo)
})
}