Add sync handler tests
This commit is contained in:
parent
9c41f8d51f
commit
0da185508b
@ -123,7 +123,7 @@ func (s *syncHandler) handleFullSyncRequest(
|
|||||||
header = objTree.Header()
|
header = objTree.Header()
|
||||||
}
|
}
|
||||||
|
|
||||||
if !s.alreadyHasHeads(objTree, request.Heads) {
|
if len(request.Changes) != 0 && !s.alreadyHasHeads(objTree, request.Heads) {
|
||||||
_, err = objTree.AddRawChanges(ctx, request.Changes...)
|
_, err = objTree.AddRawChanges(ctx, request.Changes...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
|||||||
@ -2,6 +2,7 @@ package syncservice
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"fmt"
|
||||||
"github.com/anytypeio/go-anytype-infrastructure-experiments/common/commonspace/cache"
|
"github.com/anytypeio/go-anytype-infrastructure-experiments/common/commonspace/cache"
|
||||||
"github.com/anytypeio/go-anytype-infrastructure-experiments/common/commonspace/cache/mock_cache"
|
"github.com/anytypeio/go-anytype-infrastructure-experiments/common/commonspace/cache/mock_cache"
|
||||||
"github.com/anytypeio/go-anytype-infrastructure-experiments/common/commonspace/spacesyncproto"
|
"github.com/anytypeio/go-anytype-infrastructure-experiments/common/commonspace/spacesyncproto"
|
||||||
@ -11,6 +12,7 @@ import (
|
|||||||
"github.com/anytypeio/go-anytype-infrastructure-experiments/pkg/acl/treechangeproto"
|
"github.com/anytypeio/go-anytype-infrastructure-experiments/pkg/acl/treechangeproto"
|
||||||
"github.com/golang/mock/gomock"
|
"github.com/golang/mock/gomock"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -22,6 +24,25 @@ func (t treeContainer) Tree() tree.ObjectTree {
|
|||||||
return t.objTree
|
return t.objTree
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type testObjTreeMock struct {
|
||||||
|
*mock_tree.MockObjectTree
|
||||||
|
m sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func newTestObjMock(mockTree *mock_tree.MockObjectTree) *testObjTreeMock {
|
||||||
|
return &testObjTreeMock{
|
||||||
|
MockObjectTree: mockTree,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *testObjTreeMock) Lock() {
|
||||||
|
t.m.Lock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *testObjTreeMock) Unlock() {
|
||||||
|
t.m.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
||||||
ctrl := gomock.NewController(t)
|
ctrl := gomock.NewController(t)
|
||||||
defer ctrl.Finish()
|
defer ctrl.Finish()
|
||||||
@ -30,7 +51,7 @@ func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
|||||||
spaceId := "spaceId"
|
spaceId := "spaceId"
|
||||||
cacheMock := mock_cache.NewMockTreeCache(ctrl)
|
cacheMock := mock_cache.NewMockTreeCache(ctrl)
|
||||||
syncClientMock := mock_syncservice.NewMockSyncClient(ctrl)
|
syncClientMock := mock_syncservice.NewMockSyncClient(ctrl)
|
||||||
objectTreeMock := mock_tree.NewMockObjectTree(ctrl)
|
objectTreeMock := newTestObjMock(mock_tree.NewMockObjectTree(ctrl))
|
||||||
|
|
||||||
syncHandler := newSyncHandler(spaceId, cacheMock, syncClientMock)
|
syncHandler := newSyncHandler(spaceId, cacheMock, syncClientMock)
|
||||||
t.Run("head update non empty all heads added", func(t *testing.T) {
|
t.Run("head update non empty all heads added", func(t *testing.T) {
|
||||||
@ -49,8 +70,6 @@ func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
|||||||
Release: func() {},
|
Release: func() {},
|
||||||
TreeContainer: treeContainer{objectTreeMock},
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
}, nil)
|
}, nil)
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Lock()
|
|
||||||
objectTreeMock.EXPECT().
|
objectTreeMock.EXPECT().
|
||||||
Heads().
|
Heads().
|
||||||
Return([]string{"h2"})
|
Return([]string{"h2"})
|
||||||
@ -66,8 +85,6 @@ func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
|||||||
objectTreeMock.EXPECT().
|
objectTreeMock.EXPECT().
|
||||||
HasChanges(gomock.Eq([]string{"h1"})).
|
HasChanges(gomock.Eq([]string{"h1"})).
|
||||||
Return(true)
|
Return(true)
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Unlock()
|
|
||||||
|
|
||||||
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
@ -90,8 +107,6 @@ func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
|||||||
Release: func() {},
|
Release: func() {},
|
||||||
TreeContainer: treeContainer{objectTreeMock},
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
}, nil)
|
}, nil)
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Lock()
|
|
||||||
objectTreeMock.EXPECT().
|
objectTreeMock.EXPECT().
|
||||||
Heads().
|
Heads().
|
||||||
Return([]string{"h2"})
|
Return([]string{"h2"})
|
||||||
@ -110,8 +125,6 @@ func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
|||||||
syncClientMock.EXPECT().
|
syncClientMock.EXPECT().
|
||||||
CreateFullSyncRequest(gomock.Eq(objectTreeMock), gomock.Eq([]string{"h1"}), gomock.Eq([]string{"h1"}), gomock.Eq("")).
|
CreateFullSyncRequest(gomock.Eq(objectTreeMock), gomock.Eq([]string{"h1"}), gomock.Eq([]string{"h1"}), gomock.Eq("")).
|
||||||
Return(fullRequest, nil)
|
Return(fullRequest, nil)
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Unlock()
|
|
||||||
|
|
||||||
syncClientMock.EXPECT().SendAsync(gomock.Eq([]string{senderId}), gomock.Eq(fullRequest))
|
syncClientMock.EXPECT().SendAsync(gomock.Eq([]string{senderId}), gomock.Eq(fullRequest))
|
||||||
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
@ -134,13 +147,9 @@ func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
|||||||
Release: func() {},
|
Release: func() {},
|
||||||
TreeContainer: treeContainer{objectTreeMock},
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
}, nil)
|
}, nil)
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Lock()
|
|
||||||
objectTreeMock.EXPECT().
|
objectTreeMock.EXPECT().
|
||||||
Heads().
|
Heads().
|
||||||
Return([]string{"h1"})
|
Return([]string{"h1"})
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Unlock()
|
|
||||||
|
|
||||||
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
@ -163,16 +172,12 @@ func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
|||||||
Release: func() {},
|
Release: func() {},
|
||||||
TreeContainer: treeContainer{objectTreeMock},
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
}, nil)
|
}, nil)
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Lock()
|
|
||||||
objectTreeMock.EXPECT().
|
objectTreeMock.EXPECT().
|
||||||
Heads().
|
Heads().
|
||||||
Return([]string{"h2"})
|
Return([]string{"h2"})
|
||||||
syncClientMock.EXPECT().
|
syncClientMock.EXPECT().
|
||||||
CreateFullSyncRequest(gomock.Eq(objectTreeMock), gomock.Eq([]string{"h1"}), gomock.Eq([]string{"h1"}), gomock.Eq("")).
|
CreateFullSyncRequest(gomock.Eq(objectTreeMock), gomock.Eq([]string{"h1"}), gomock.Eq([]string{"h1"}), gomock.Eq("")).
|
||||||
Return(fullRequest, nil)
|
Return(fullRequest, nil)
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Unlock()
|
|
||||||
|
|
||||||
syncClientMock.EXPECT().SendAsync(gomock.Eq([]string{senderId}), gomock.Eq(fullRequest))
|
syncClientMock.EXPECT().SendAsync(gomock.Eq([]string{senderId}), gomock.Eq(fullRequest))
|
||||||
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
@ -195,13 +200,196 @@ func TestSyncHandler_HandleHeadUpdate(t *testing.T) {
|
|||||||
Release: func() {},
|
Release: func() {},
|
||||||
TreeContainer: treeContainer{objectTreeMock},
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
}, nil)
|
}, nil)
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Lock()
|
objectTreeMock.EXPECT().
|
||||||
|
Heads().
|
||||||
|
Return([]string{"h1"})
|
||||||
|
|
||||||
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
|
require.NoError(t, err)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSyncHandler_HandleFullSyncRequest(t *testing.T) {
|
||||||
|
ctrl := gomock.NewController(t)
|
||||||
|
defer ctrl.Finish()
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
spaceId := "spaceId"
|
||||||
|
cacheMock := mock_cache.NewMockTreeCache(ctrl)
|
||||||
|
syncClientMock := mock_syncservice.NewMockSyncClient(ctrl)
|
||||||
|
objectTreeMock := newTestObjMock(mock_tree.NewMockObjectTree(ctrl))
|
||||||
|
|
||||||
|
syncHandler := newSyncHandler(spaceId, cacheMock, syncClientMock)
|
||||||
|
t.Run("full sync request with change", func(t *testing.T) {
|
||||||
|
treeId := "treeId"
|
||||||
|
senderId := "senderId"
|
||||||
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
||||||
|
msg := spacesyncproto.WrapFullRequest(&spacesyncproto.ObjectFullSyncRequest{
|
||||||
|
Heads: []string{"h1"},
|
||||||
|
Changes: []*treechangeproto.RawTreeChangeWithId{chWithId},
|
||||||
|
SnapshotPath: []string{"h1"},
|
||||||
|
}, chWithId, treeId, "")
|
||||||
|
fullRequest := &spacesyncproto.ObjectSyncMessage{}
|
||||||
|
|
||||||
|
cacheMock.EXPECT().
|
||||||
|
GetTree(gomock.Any(), spaceId, treeId).
|
||||||
|
Return(cache.TreeResult{
|
||||||
|
Release: func() {},
|
||||||
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
|
}, nil)
|
||||||
|
objectTreeMock.EXPECT().
|
||||||
|
Heads().
|
||||||
|
Return([]string{"h2"})
|
||||||
|
objectTreeMock.EXPECT().
|
||||||
|
HasChanges(gomock.Eq([]string{"h1"})).
|
||||||
|
Return(false)
|
||||||
|
objectTreeMock.EXPECT().
|
||||||
|
AddRawChanges(gomock.Any(), gomock.Eq([]*treechangeproto.RawTreeChangeWithId{chWithId})).
|
||||||
|
Return(tree.AddResult{}, nil)
|
||||||
|
syncClientMock.EXPECT().
|
||||||
|
CreateFullSyncResponse(gomock.Eq(objectTreeMock), gomock.Eq([]string{"h1"}), gomock.Eq([]string{"h1"}), gomock.Eq("")).
|
||||||
|
Return(fullRequest, nil)
|
||||||
|
|
||||||
|
syncClientMock.EXPECT().SendAsync(gomock.Eq([]string{senderId}), gomock.Eq(fullRequest))
|
||||||
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
|
require.NoError(t, err)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("full sync request with change same heads", func(t *testing.T) {
|
||||||
|
treeId := "treeId"
|
||||||
|
senderId := "senderId"
|
||||||
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
||||||
|
msg := spacesyncproto.WrapFullRequest(&spacesyncproto.ObjectFullSyncRequest{
|
||||||
|
Heads: []string{"h2"},
|
||||||
|
Changes: []*treechangeproto.RawTreeChangeWithId{chWithId},
|
||||||
|
SnapshotPath: []string{"h2"},
|
||||||
|
}, chWithId, treeId, "")
|
||||||
|
fullRequest := &spacesyncproto.ObjectSyncMessage{}
|
||||||
|
|
||||||
|
cacheMock.EXPECT().
|
||||||
|
GetTree(gomock.Any(), spaceId, treeId).
|
||||||
|
Return(cache.TreeResult{
|
||||||
|
Release: func() {},
|
||||||
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
|
}, nil)
|
||||||
|
objectTreeMock.EXPECT().
|
||||||
|
Heads().
|
||||||
|
Return([]string{"h2"})
|
||||||
|
syncClientMock.EXPECT().
|
||||||
|
CreateFullSyncResponse(gomock.Eq(objectTreeMock), gomock.Eq([]string{"h2"}), gomock.Eq([]string{"h2"}), gomock.Eq("")).
|
||||||
|
Return(fullRequest, nil)
|
||||||
|
|
||||||
|
syncClientMock.EXPECT().SendAsync(gomock.Eq([]string{senderId}), gomock.Eq(fullRequest))
|
||||||
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
|
require.NoError(t, err)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("full sync request without change", func(t *testing.T) {
|
||||||
|
treeId := "treeId"
|
||||||
|
senderId := "senderId"
|
||||||
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
||||||
|
msg := spacesyncproto.WrapFullRequest(&spacesyncproto.ObjectFullSyncRequest{
|
||||||
|
Heads: []string{"h1"},
|
||||||
|
SnapshotPath: []string{"h1"},
|
||||||
|
}, chWithId, treeId, "")
|
||||||
|
fullRequest := &spacesyncproto.ObjectSyncMessage{}
|
||||||
|
|
||||||
|
cacheMock.EXPECT().
|
||||||
|
GetTree(gomock.Any(), spaceId, treeId).
|
||||||
|
Return(cache.TreeResult{
|
||||||
|
Release: func() {},
|
||||||
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
|
}, nil)
|
||||||
|
syncClientMock.EXPECT().
|
||||||
|
CreateFullSyncResponse(gomock.Eq(objectTreeMock), gomock.Eq([]string{"h1"}), gomock.Eq([]string{"h1"}), gomock.Eq("")).
|
||||||
|
Return(fullRequest, nil)
|
||||||
|
|
||||||
|
syncClientMock.EXPECT().SendAsync(gomock.Eq([]string{senderId}), gomock.Eq(fullRequest))
|
||||||
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
|
require.NoError(t, err)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("full sync request with get tree error", func(t *testing.T) {
|
||||||
|
treeId := "treeId"
|
||||||
|
senderId := "senderId"
|
||||||
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
||||||
|
msg := spacesyncproto.WrapFullRequest(&spacesyncproto.ObjectFullSyncRequest{
|
||||||
|
Heads: []string{"h1"},
|
||||||
|
SnapshotPath: []string{"h1"},
|
||||||
|
}, chWithId, treeId, "")
|
||||||
|
|
||||||
|
cacheMock.EXPECT().
|
||||||
|
GetTree(gomock.Any(), spaceId, treeId).
|
||||||
|
Return(cache.TreeResult{}, fmt.Errorf("some"))
|
||||||
|
|
||||||
|
syncClientMock.EXPECT().
|
||||||
|
SendAsync(gomock.Eq([]string{senderId}), gomock.Any())
|
||||||
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
|
require.Error(t, err)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSyncHandler_HandleFullSyncResponse(t *testing.T) {
|
||||||
|
ctrl := gomock.NewController(t)
|
||||||
|
defer ctrl.Finish()
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
spaceId := "spaceId"
|
||||||
|
cacheMock := mock_cache.NewMockTreeCache(ctrl)
|
||||||
|
syncClientMock := mock_syncservice.NewMockSyncClient(ctrl)
|
||||||
|
objectTreeMock := newTestObjMock(mock_tree.NewMockObjectTree(ctrl))
|
||||||
|
|
||||||
|
syncHandler := newSyncHandler(spaceId, cacheMock, syncClientMock)
|
||||||
|
t.Run("full sync response with change", func(t *testing.T) {
|
||||||
|
treeId := "treeId"
|
||||||
|
senderId := "senderId"
|
||||||
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
||||||
|
msg := spacesyncproto.WrapFullResponse(&spacesyncproto.ObjectFullSyncResponse{
|
||||||
|
Heads: []string{"h1"},
|
||||||
|
Changes: []*treechangeproto.RawTreeChangeWithId{chWithId},
|
||||||
|
SnapshotPath: []string{"h1"},
|
||||||
|
}, chWithId, treeId, "")
|
||||||
|
|
||||||
|
cacheMock.EXPECT().
|
||||||
|
GetTree(gomock.Any(), spaceId, treeId).
|
||||||
|
Return(cache.TreeResult{
|
||||||
|
Release: func() {},
|
||||||
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
|
}, nil)
|
||||||
|
objectTreeMock.EXPECT().
|
||||||
|
Heads().
|
||||||
|
Return([]string{"h2"})
|
||||||
|
objectTreeMock.EXPECT().
|
||||||
|
HasChanges(gomock.Eq([]string{"h1"})).
|
||||||
|
Return(false)
|
||||||
|
objectTreeMock.EXPECT().
|
||||||
|
AddRawChanges(gomock.Any(), gomock.Eq([]*treechangeproto.RawTreeChangeWithId{chWithId})).
|
||||||
|
Return(tree.AddResult{}, nil)
|
||||||
|
|
||||||
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
|
require.NoError(t, err)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("full sync response same heads", func(t *testing.T) {
|
||||||
|
treeId := "treeId"
|
||||||
|
senderId := "senderId"
|
||||||
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
||||||
|
msg := spacesyncproto.WrapFullResponse(&spacesyncproto.ObjectFullSyncResponse{
|
||||||
|
Heads: []string{"h1"},
|
||||||
|
Changes: []*treechangeproto.RawTreeChangeWithId{chWithId},
|
||||||
|
SnapshotPath: []string{"h1"},
|
||||||
|
}, chWithId, treeId, "")
|
||||||
|
|
||||||
|
cacheMock.EXPECT().
|
||||||
|
GetTree(gomock.Any(), spaceId, treeId).
|
||||||
|
Return(cache.TreeResult{
|
||||||
|
Release: func() {},
|
||||||
|
TreeContainer: treeContainer{objectTreeMock},
|
||||||
|
}, nil)
|
||||||
objectTreeMock.EXPECT().
|
objectTreeMock.EXPECT().
|
||||||
Heads().
|
Heads().
|
||||||
Return([]string{"h1"})
|
Return([]string{"h1"})
|
||||||
objectTreeMock.EXPECT().
|
|
||||||
Unlock()
|
|
||||||
|
|
||||||
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user