frostfs-node/services/public/object/ranges_test.go

779 lines
18 KiB
Go
Raw Normal View History

package object
import (
"bytes"
"context"
"testing"
"time"
"github.com/nspcc-dev/neofs-api-go/hash"
"github.com/nspcc-dev/neofs-api-go/object"
"github.com/nspcc-dev/neofs-api-go/service"
"github.com/nspcc-dev/neofs-node/internal"
"github.com/nspcc-dev/neofs-node/lib/objio"
"github.com/nspcc-dev/neofs-node/lib/transport"
"github.com/stretchr/testify/require"
"go.uber.org/zap"
)
type (
// Entity for mocking interfaces.
// Implementation of any interface intercepts arguments via f (if not nil).
// If err is not nil, it returns as it is. Otherwise, casted to needed type res returns w/o error.
testRangeEntity struct {
// Set of interfaces which entity must implement, but some methods from those does not call.
RangeChopper
object.Service_GetRangeServer
// Argument interceptor. Used for ascertain of correct parameter passage between components.
f func(...interface{})
// Mocked result of any interface.
res interface{}
// Mocked error of any interface.
err error
}
)
var (
_ objio.RelativeReceiver = (*testRangeEntity)(nil)
_ RangeChopper = (*testRangeEntity)(nil)
_ operationExecutor = (*testRangeEntity)(nil)
_ requestHandler = (*testRangeEntity)(nil)
_ rangeRevealer = (*testRangeEntity)(nil)
_ objectRangeReceiver = (*testRangeEntity)(nil)
_ object.Service_GetRangeServer = (*testRangeEntity)(nil)
_ responsePreparer = (*testRangeEntity)(nil)
)
func (s *testRangeEntity) prepareResponse(_ context.Context, req serviceRequest, resp serviceResponse) error {
if s.f != nil {
s.f(req, resp)
}
return s.err
}
func (s *testRangeEntity) Context() context.Context { return context.TODO() }
func (s *testRangeEntity) Send(r *GetRangeResponse) error {
if s.f != nil {
s.f(r)
}
return s.err
}
func (s *testRangeEntity) getRange(_ context.Context, t rangeTool) (interface{}, error) {
if s.f != nil {
s.f(t)
}
return s.res, s.err
}
func (s *testRangeEntity) reveal(_ context.Context, r *RangeDescriptor) ([]RangeDescriptor, error) {
if s.f != nil {
s.f(r)
}
if s.err != nil {
return nil, s.err
}
return s.res.([]RangeDescriptor), nil
}
func (s *testRangeEntity) Base(ctx context.Context, addr Address) (RangeDescriptor, error) {
if s.f != nil {
s.f(addr)
}
if s.err != nil {
return RangeDescriptor{}, s.err
}
return s.res.(RangeDescriptor), nil
}
func (s *testRangeEntity) Neighbor(ctx context.Context, addr Address, left bool) (RangeDescriptor, error) {
if s.f != nil {
s.f(addr, left)
}
if s.err != nil {
return RangeDescriptor{}, s.err
}
return s.res.(RangeDescriptor), nil
}
func (s *testRangeEntity) Chop(ctx context.Context, length, offset int64, fromStart bool) ([]RangeDescriptor, error) {
if s.f != nil {
s.f(length, offset, fromStart)
}
if s.err != nil {
return nil, s.err
}
return s.res.([]RangeDescriptor), nil
}
func (s *testRangeEntity) Closed() bool { return s.res.(bool) }
func (s *testRangeEntity) PutChopper(addr Address, chopper RangeChopper) error {
if s.f != nil {
s.f(addr, chopper)
}
return s.err
}
func (s *testRangeEntity) GetChopper(addr Address, rc objio.RCType) (RangeChopper, error) {
if s.f != nil {
s.f(addr, rc)
}
if s.err != nil {
return nil, s.err
}
return s.res.(RangeChopper), nil
}
func (s *testRangeEntity) executeOperation(_ context.Context, i transport.MetaInfo, h responseItemHandler) error {
if s.f != nil {
s.f(i, h)
}
return s.err
}
func (s *testRangeEntity) handleRequest(_ context.Context, p handleRequestParams) (interface{}, error) {
if s.f != nil {
s.f(p)
}
return s.res, s.err
}
func Test_objectService_GetRange(t *testing.T) {
req := &GetRangeRequest{Address: testObjectAddress(t)}
t.Run("request handler error", func(t *testing.T) {
rhErr := internal.Error("test error for request handler")
s := &objectService{
statusCalculator: newStatusCalculator(),
}
s.requestHandler = &testRangeEntity{
f: func(items ...interface{}) {
t.Run("correct request handler params", func(t *testing.T) {
p := items[0].(handleRequestParams)
require.Equal(t, s, p.executor)
require.Equal(t, req, p.request)
})
},
err: rhErr, // force requestHandler to return rhErr
}
// ascertain that error returns as expected
require.EqualError(t, s.GetRange(req, new(testRangeEntity)), rhErr.Error())
})
t.Run("correct result", func(t *testing.T) {
fragment := testData(t, 10)
resp := &GetRangeResponse{Fragment: fragment}
s := objectService{
requestHandler: &testRangeEntity{
res: bytes.NewReader(fragment), // force requestHandler to return fragment
},
rangeChunkPreparer: &testRangeEntity{
f: func(items ...interface{}) {
require.Equal(t, req, items[0])
require.Equal(t, makeRangeResponse(fragment), items[1])
},
res: resp,
},
statusCalculator: newStatusCalculator(),
}
srv := &testRangeEntity{
f: func(items ...interface{}) {
require.Equal(t, resp, items[0])
},
}
require.NoError(t, s.GetRange(req, srv))
})
}
func Test_objectService_GetRangeHash(t *testing.T) {
ctx := context.TODO()
req := &GetRangeHashRequest{Address: testObjectAddress(t)}
t.Run("request handler error", func(t *testing.T) {
rhErr := internal.Error("test error for request handler")
s := &objectService{
statusCalculator: newStatusCalculator(),
}
s.requestHandler = &testRangeEntity{
f: func(items ...interface{}) {
t.Run("correct request handler params", func(t *testing.T) {
p := items[0].(handleRequestParams)
require.Equal(t, s, p.executor)
require.Equal(t, req, p.request)
})
},
err: rhErr, // force requestHandler to return rhErr
}
// ascertain that error returns as expected
res, err := s.GetRangeHash(ctx, req)
require.EqualError(t, err, rhErr.Error())
require.Nil(t, res)
})
t.Run("correct result", func(t *testing.T) {
hCount := 5
hashes := make([]Hash, 0, hCount)
for i := 0; i < hCount; i++ {
hashes = append(hashes, hash.Sum(testData(t, 10)))
}
s := objectService{
requestHandler: &testRangeEntity{
res: hashes, // force requestHandler to return fragments
},
respPreparer: &testRangeEntity{
f: func(items ...interface{}) {
require.Equal(t, req, items[0])
require.Equal(t, makeRangeHashResponse(hashes), items[1])
},
res: &GetRangeHashResponse{Hashes: hashes},
},
statusCalculator: newStatusCalculator(),
}
res, err := s.GetRangeHash(ctx, req)
require.NoError(t, err)
require.Equal(t, hashes, res.Hashes)
})
}
func Test_coreRangeReceiver(t *testing.T) {
ctx := context.TODO()
log := zap.L()
t.Run("range reveal failure", func(t *testing.T) {
revErr := internal.Error("test error for range revealer")
rt := newRawRangeHashInfo()
rt.setTTL(service.NonForwardingTTL)
rt.setAddress(testObjectAddress(t))
rt.setRanges([]Range{
{
Offset: 1,
Length: 2,
},
})
revealer := &testRangeEntity{
f: func(items ...interface{}) {
require.Equal(t, &RangeDescriptor{
Size: int64(rt.rngList[0].Length),
Offset: int64(rt.rngList[0].Offset),
Addr: rt.addr,
}, items[0])
},
err: revErr,
}
s := &coreRangeReceiver{
rngRevealer: revealer,
log: log,
}
res, err := s.getRange(ctx, rt)
require.EqualError(t, err, errPayloadRangeNotFound.Error())
require.Nil(t, res)
revealer.err = nil
revealer.res = make([]RangeDescriptor, 0)
res, err = s.getRange(ctx, rt)
require.EqualError(t, err, errPayloadRangeNotFound.Error())
require.Nil(t, res)
})
t.Run("get sub range failure", func(t *testing.T) {
gErr := internal.Error("test error for get range")
rt := newRawRangeHashInfo()
rt.setTTL(service.NonForwardingTTL)
rt.setAddress(testObjectAddress(t))
rt.setRanges([]Range{
{
Offset: 1,
Length: 2,
},
})
revealer := &testRangeEntity{
res: []RangeDescriptor{{Size: 3, Offset: 4, Addr: testObjectAddress(t)}},
}
called := false
revealer.f = func(items ...interface{}) {
if called {
revealer.err = gErr
return
}
called = true
}
s := &coreRangeReceiver{
rngRevealer: revealer,
log: log,
}
res, err := s.getRange(ctx, rt)
require.EqualError(t, err, errPayloadRangeNotFound.Error())
require.Nil(t, res)
})
t.Run("non-forwarding behavior", func(t *testing.T) {
rt := newRawRangeHashInfo()
rt.setTTL(service.NonForwardingTTL - 1)
rt.setAddress(testObjectAddress(t))
rt.setRanges([]Range{
{
Offset: 1,
Length: 2,
},
})
rd := RangeDescriptor{
Size: int64(rt.rngList[0].Length),
Offset: int64(rt.rngList[0].Offset),
Addr: rt.addr,
}
d := hash.Sum(testData(t, 10))
s := &coreRangeReceiver{
straightRngRecv: &testRangeEntity{
f: func(items ...interface{}) {
require.Equal(t, rt.budOff(&rd), items[0])
},
res: d,
},
}
res, err := s.getRange(ctx, rt)
require.NoError(t, err)
require.Equal(t, d, res)
})
t.Run("correct result concat", func(t *testing.T) {
rt := newRawRangeHashInfo()
rt.setTTL(service.NonForwardingTTL)
rt.setRanges([]Range{
{},
})
revealer := new(testRangeEntity)
revCalled := false
revealer.f = func(items ...interface{}) {
if revCalled {
revealer.res = []RangeDescriptor{items[0].(RangeDescriptor)}
} else {
revealer.res = make([]RangeDescriptor, 2)
}
revCalled = true
}
h1, h2 := hash.Sum(testData(t, 10)), hash.Sum(testData(t, 10))
recvCalled := false
receiver := new(testRangeEntity)
receiver.f = func(...interface{}) {
if recvCalled {
receiver.res = h2
} else {
receiver.res = h1
}
recvCalled = true
}
s := &coreRangeReceiver{
rngRevealer: revealer,
straightRngRecv: receiver,
}
exp, err := hash.Concat([]Hash{h1, h2})
require.NoError(t, err)
res, err := s.getRange(ctx, rt)
require.NoError(t, err)
require.Equal(t, exp, res)
})
}
func Test_straightRangeReceiver_getRange(t *testing.T) {
ctx := context.TODO()
req := new(transportRequest)
t.Run("executor error", func(t *testing.T) {
exErr := internal.Error("test error for executor")
s := &straightRangeReceiver{
executor: &testRangeEntity{
f: func(items ...interface{}) {
t.Run("correct executor params", func(t *testing.T) {
require.Equal(t, req, items[0])
require.Equal(t, newSingleItemHandler(), items[1])
})
},
err: exErr, // force operationExecutor to return exErr
},
}
res, err := s.getRange(ctx, req)
require.EqualError(t, err, exErr.Error())
require.Nil(t, res)
})
t.Run("correct result", func(t *testing.T) {
v := testData(t, 10)
s := &straightRangeReceiver{
executor: &testRangeEntity{
f: func(items ...interface{}) {
items[1].(rangeItemAccumulator).handleItem(v)
},
err: nil, // force operationExecutor to return nil error
},
}
res, err := s.getRange(ctx, req)
require.NoError(t, err)
require.Equal(t, v, res)
})
}
func Test_coreRngRevealer_reveal(t *testing.T) {
ctx := context.TODO()
rd := RangeDescriptor{
Size: 5,
Offset: 6,
Addr: testObjectAddress(t),
}
t.Run("charybdis chopper presence", func(t *testing.T) {
cErr := internal.Error("test error for charybdis")
s := &coreRngRevealer{
chopTable: &testRangeEntity{
f: func(items ...interface{}) {
t.Run("correct chopper table params", func(t *testing.T) {
require.Equal(t, rd.Addr, items[0])
require.Equal(t, objio.RCCharybdis, items[1])
})
},
res: &testRangeEntity{
f: func(items ...interface{}) {
t.Run("correct chopper params", func(t *testing.T) {
require.Equal(t, rd.Size, items[0])
require.Equal(t, rd.Offset, items[1])
require.True(t, items[2].(bool))
})
},
res: true, // close chopper
err: cErr, // force RangeChopper to return cErr
},
},
}
res, err := s.reveal(ctx, &rd)
require.EqualError(t, err, cErr.Error())
require.Empty(t, res)
})
t.Run("scylla chopper presence", func(t *testing.T) {
scErr := internal.Error("test error for scylla")
scylla := &testRangeEntity{
err: scErr, // force RangeChopper to return scErr
}
ct := new(testRangeEntity)
ct.f = func(items ...interface{}) {
if items[1].(objio.RCType) == objio.RCCharybdis {
ct.err = internal.Error("")
} else {
ct.res = scylla
ct.err = nil
}
}
s := &coreRngRevealer{
chopTable: ct,
}
res, err := s.reveal(ctx, &rd)
require.EqualError(t, err, scErr.Error())
require.Empty(t, res)
})
t.Run("new scylla", func(t *testing.T) {
t.Run("error", func(t *testing.T) {
s := &coreRngRevealer{
relativeRecv: nil, // pass empty relation receiver to fail constructor
chopTable: &testRangeEntity{
err: internal.Error(""), // force ChopperTable to return non-nil error
},
}
res, err := s.reveal(ctx, &rd)
require.Error(t, err)
require.Nil(t, res)
})
t.Run("success", func(t *testing.T) {
rrErr := internal.Error("test error for relative receiver")
relRecv := &testRangeEntity{
err: rrErr, // force relative receiver to return rrErr
}
scylla, err := objio.NewScylla(&objio.ChopperParams{
RelativeReceiver: relRecv,
Addr: rd.Addr,
})
require.NoError(t, err)
callNum := 0
s := &coreRngRevealer{
relativeRecv: relRecv,
chopTable: &testRangeEntity{
f: func(items ...interface{}) {
t.Run("correct put chopper params", func(t *testing.T) {
if callNum >= 2 {
require.Equal(t, rd.Addr, items[0])
require.Equal(t, scylla, items[1])
}
})
},
err: internal.Error(""), // force ChopperTable to return non-nil error
},
}
expRes, expErr := scylla.Chop(ctx, rd.Size, rd.Offset, true)
require.Error(t, expErr)
res, err := s.reveal(ctx, &rd)
require.EqualError(t, err, expErr.Error())
require.Equal(t, expRes, res)
})
})
}
func Test_transportRequest_rangeTool(t *testing.T) {
t.Run("get ranges", func(t *testing.T) {
rngs := []Range{
{Offset: 1, Length: 2},
{Offset: 3, Length: 4},
}
reqs := []transportRequest{
{serviceRequest: &GetRangeHashRequest{Ranges: rngs}},
}
for i := range reqs {
require.Equal(t, reqs[i].GetRanges(), rngs)
}
})
t.Run("bud off", func(t *testing.T) {
var (
timeout = 6 * time.Second
ttl = uint32(16)
rd = RangeDescriptor{
Size: 1,
Offset: 2,
Addr: testObjectAddress(t),
}
)
t.Run("get range hash request", func(t *testing.T) {
salt := testData(t, 10)
r := &GetRangeHashRequest{Salt: salt}
r.SetToken(new(service.Token))
req := &transportRequest{
serviceRequest: r,
timeout: timeout,
}
req.SetTTL(ttl)
tool := req.budOff(&rd).(transport.RangeHashInfo)
require.Equal(t, timeout, tool.GetTimeout())
require.Equal(t, ttl, tool.GetTTL())
require.Equal(t, rd.Addr, tool.GetAddress())
require.Equal(t, []Range{{Offset: uint64(rd.Offset), Length: uint64(rd.Size)}}, tool.GetRanges())
require.Equal(t, salt, tool.GetSalt())
require.Equal(t, r.GetSessionToken(), tool.GetSessionToken())
})
})
t.Run("handler", func(t *testing.T) {
t.Run("get range request", func(t *testing.T) {
req := &transportRequest{serviceRequest: new(GetRangeHashRequest)}
handler := req.handler()
require.Equal(t, new(rangeHashAccum), handler)
})
})
}
func Test_rawRangeHashInfo(t *testing.T) {
t.Run("get ranges", func(t *testing.T) {
rngs := []Range{
{Offset: 1, Length: 2},
{Offset: 3, Length: 4},
}
r := newRawRangeHashInfo()
r.setRanges(rngs)
require.Equal(t, rngs, r.GetRanges())
})
t.Run("handler", func(t *testing.T) {
require.Equal(t,
&rangeHashAccum{concat: true},
newRawRangeHashInfo().handler(),
)
})
t.Run("bud off", func(t *testing.T) {
var (
ttl = uint32(12)
timeout = 7 * time.Hour
)
r := newRawRangeHashInfo()
r.setTTL(ttl)
r.setTimeout(timeout)
r.setSalt(testData(t, 20))
r.setSessionToken(new(service.Token))
rd := RangeDescriptor{
Size: 120,
Offset: 71,
Addr: testObjectAddress(t),
}
tool := r.budOff(&rd)
require.Equal(t, ttl, tool.GetTTL())
require.Equal(t, timeout, tool.GetTimeout())
require.Equal(t, rd.Addr, tool.GetAddress())
require.Equal(t, []Range{{Offset: uint64(rd.Offset), Length: uint64(rd.Size)}}, tool.GetRanges())
require.Equal(t, r.GetSessionToken(), tool.GetSessionToken())
require.Equal(t,
loopData(r.salt, int64(len(r.salt)), rd.Offset),
tool.(transport.RangeHashInfo).GetSalt(),
)
})
}
func Test_rawRangeInfo(t *testing.T) {
t.Run("get ranges", func(t *testing.T) {
rng := Range{Offset: 1, Length: 2}
r := newRawRangeInfo()
r.setRange(rng)
require.Equal(t, rng, r.GetRange())
})
}
func Test_loopSalt(t *testing.T) {
t.Run("empty data", func(t *testing.T) {
require.Empty(t, loopData(nil, 20, 10))
require.Empty(t, loopData(make([]byte, 0), 20, 10))
})
t.Run("data part", func(t *testing.T) {
var (
off, size int64 = 10, 20
d = testData(t, 40)
)
require.Equal(t, d[off:off+size], loopData(d, size, off))
})
t.Run("with recycle", func(t *testing.T) {
var (
d = testData(t, 40)
off = int64(len(d) / 2)
size = 2 * off
)
require.Equal(t,
append(d[off:], d[:size-off]...),
loopData(d, size, off),
)
})
}
func Test_rangeHashAccum(t *testing.T) {
t.Run("handle item", func(t *testing.T) {
s := &rangeHashAccum{
h: []Hash{hash.Sum(testData(t, 10))},
}
h := hash.Sum(testData(t, 10))
exp := append(s.h, h)
s.handleItem(h)
require.Equal(t, exp, s.h)
exp = append(s.h, s.h...)
s.handleItem(s.h)
require.Equal(t, exp, s.h)
})
t.Run("collect", func(t *testing.T) {
hashes := []Hash{hash.Sum(testData(t, 10)), hash.Sum(testData(t, 10))}
t.Run("w/ concat", func(t *testing.T) {
s := &rangeHashAccum{
concat: true,
h: hashes,
}
expRes, expErr := hash.Concat(hashes)
res, err := s.collect()
require.Equal(t, expRes, res)
require.Equal(t, expErr, err)
})
t.Run("w/o concat", func(t *testing.T) {
s := &rangeHashAccum{
concat: false,
h: hashes,
}
res, err := s.collect()
require.NoError(t, err)
require.Equal(t, hashes, res)
})
})
}