forked from TrueCloudLab/frostfs-node
779 lines
18 KiB
Go
779 lines
18 KiB
Go
|
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)
|
||
|
})
|
||
|
})
|
||
|
}
|