210 lines
6.6 KiB
Go
210 lines
6.6 KiB
Go
package syncservice
|
|
|
|
import (
|
|
"context"
|
|
"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/spacesyncproto"
|
|
"github.com/anytypeio/go-anytype-infrastructure-experiments/common/commonspace/syncservice/mock_syncservice"
|
|
"github.com/anytypeio/go-anytype-infrastructure-experiments/pkg/acl/tree"
|
|
mock_tree "github.com/anytypeio/go-anytype-infrastructure-experiments/pkg/acl/tree/mock_objecttree"
|
|
"github.com/anytypeio/go-anytype-infrastructure-experiments/pkg/acl/treechangeproto"
|
|
"github.com/golang/mock/gomock"
|
|
"github.com/stretchr/testify/require"
|
|
"testing"
|
|
)
|
|
|
|
type treeContainer struct {
|
|
objTree tree.ObjectTree
|
|
}
|
|
|
|
func (t treeContainer) Tree() tree.ObjectTree {
|
|
return t.objTree
|
|
}
|
|
|
|
func TestSyncHandler_HandleHeadUpdate(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 := mock_tree.NewMockObjectTree(ctrl)
|
|
|
|
syncHandler := newSyncHandler(spaceId, cacheMock, syncClientMock)
|
|
t.Run("head update non empty all heads added", func(t *testing.T) {
|
|
treeId := "treeId"
|
|
senderId := "senderId"
|
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
|
headUpdate := &spacesyncproto.ObjectHeadUpdate{
|
|
Heads: []string{"h1"},
|
|
Changes: []*treechangeproto.RawTreeChangeWithId{chWithId},
|
|
SnapshotPath: []string{"h1"},
|
|
}
|
|
msg := spacesyncproto.WrapHeadUpdate(headUpdate, chWithId, treeId, "")
|
|
cacheMock.EXPECT().
|
|
GetTree(gomock.Any(), spaceId, treeId).
|
|
Return(cache.TreeResult{
|
|
Release: func() {},
|
|
TreeContainer: treeContainer{objectTreeMock},
|
|
}, nil)
|
|
objectTreeMock.EXPECT().
|
|
Lock()
|
|
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)
|
|
objectTreeMock.EXPECT().
|
|
Heads().
|
|
Return([]string{"h2", "h1"})
|
|
objectTreeMock.EXPECT().
|
|
HasChanges(gomock.Eq([]string{"h1"})).
|
|
Return(true)
|
|
objectTreeMock.EXPECT().
|
|
Unlock()
|
|
|
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
|
require.NoError(t, err)
|
|
})
|
|
|
|
t.Run("head update non empty heads not added", func(t *testing.T) {
|
|
treeId := "treeId"
|
|
senderId := "senderId"
|
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
|
headUpdate := &spacesyncproto.ObjectHeadUpdate{
|
|
Heads: []string{"h1"},
|
|
Changes: []*treechangeproto.RawTreeChangeWithId{chWithId},
|
|
SnapshotPath: []string{"h1"},
|
|
}
|
|
fullRequest := &spacesyncproto.ObjectSyncMessage{}
|
|
msg := spacesyncproto.WrapHeadUpdate(headUpdate, chWithId, treeId, "")
|
|
cacheMock.EXPECT().
|
|
GetTree(gomock.Any(), spaceId, treeId).
|
|
Return(cache.TreeResult{
|
|
Release: func() {},
|
|
TreeContainer: treeContainer{objectTreeMock},
|
|
}, nil)
|
|
objectTreeMock.EXPECT().
|
|
Lock()
|
|
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)
|
|
objectTreeMock.EXPECT().
|
|
Heads().
|
|
Return([]string{"h2"})
|
|
objectTreeMock.EXPECT().
|
|
HasChanges(gomock.Eq([]string{"h1"})).
|
|
Return(false)
|
|
syncClientMock.EXPECT().
|
|
CreateFullSyncRequest(gomock.Eq(objectTreeMock), gomock.Eq([]string{"h1"}), gomock.Eq([]string{"h1"}), gomock.Eq("")).
|
|
Return(fullRequest, nil)
|
|
objectTreeMock.EXPECT().
|
|
Unlock()
|
|
|
|
syncClientMock.EXPECT().SendAsync(gomock.Eq([]string{senderId}), gomock.Eq(fullRequest))
|
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
|
require.NoError(t, err)
|
|
})
|
|
|
|
t.Run("head update non empty equal heads", func(t *testing.T) {
|
|
treeId := "treeId"
|
|
senderId := "senderId"
|
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
|
headUpdate := &spacesyncproto.ObjectHeadUpdate{
|
|
Heads: []string{"h1"},
|
|
Changes: []*treechangeproto.RawTreeChangeWithId{chWithId},
|
|
SnapshotPath: []string{"h1"},
|
|
}
|
|
msg := spacesyncproto.WrapHeadUpdate(headUpdate, chWithId, treeId, "")
|
|
cacheMock.EXPECT().
|
|
GetTree(gomock.Any(), spaceId, treeId).
|
|
Return(cache.TreeResult{
|
|
Release: func() {},
|
|
TreeContainer: treeContainer{objectTreeMock},
|
|
}, nil)
|
|
objectTreeMock.EXPECT().
|
|
Lock()
|
|
objectTreeMock.EXPECT().
|
|
Heads().
|
|
Return([]string{"h1"})
|
|
objectTreeMock.EXPECT().
|
|
Unlock()
|
|
|
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
|
require.NoError(t, err)
|
|
})
|
|
|
|
t.Run("head update empty", func(t *testing.T) {
|
|
treeId := "treeId"
|
|
senderId := "senderId"
|
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
|
headUpdate := &spacesyncproto.ObjectHeadUpdate{
|
|
Heads: []string{"h1"},
|
|
Changes: nil,
|
|
SnapshotPath: []string{"h1"},
|
|
}
|
|
fullRequest := &spacesyncproto.ObjectSyncMessage{}
|
|
msg := spacesyncproto.WrapHeadUpdate(headUpdate, chWithId, treeId, "")
|
|
cacheMock.EXPECT().
|
|
GetTree(gomock.Any(), spaceId, treeId).
|
|
Return(cache.TreeResult{
|
|
Release: func() {},
|
|
TreeContainer: treeContainer{objectTreeMock},
|
|
}, nil)
|
|
objectTreeMock.EXPECT().
|
|
Lock()
|
|
objectTreeMock.EXPECT().
|
|
Heads().
|
|
Return([]string{"h2"})
|
|
syncClientMock.EXPECT().
|
|
CreateFullSyncRequest(gomock.Eq(objectTreeMock), gomock.Eq([]string{"h1"}), gomock.Eq([]string{"h1"}), gomock.Eq("")).
|
|
Return(fullRequest, nil)
|
|
objectTreeMock.EXPECT().
|
|
Unlock()
|
|
|
|
syncClientMock.EXPECT().SendAsync(gomock.Eq([]string{senderId}), gomock.Eq(fullRequest))
|
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
|
require.NoError(t, err)
|
|
})
|
|
|
|
t.Run("head update empty equal heads", func(t *testing.T) {
|
|
treeId := "treeId"
|
|
senderId := "senderId"
|
|
chWithId := &treechangeproto.RawTreeChangeWithId{}
|
|
headUpdate := &spacesyncproto.ObjectHeadUpdate{
|
|
Heads: []string{"h1"},
|
|
Changes: nil,
|
|
SnapshotPath: []string{"h1"},
|
|
}
|
|
msg := spacesyncproto.WrapHeadUpdate(headUpdate, chWithId, treeId, "")
|
|
cacheMock.EXPECT().
|
|
GetTree(gomock.Any(), spaceId, treeId).
|
|
Return(cache.TreeResult{
|
|
Release: func() {},
|
|
TreeContainer: treeContainer{objectTreeMock},
|
|
}, nil)
|
|
objectTreeMock.EXPECT().
|
|
Lock()
|
|
objectTreeMock.EXPECT().
|
|
Heads().
|
|
Return([]string{"h1"})
|
|
objectTreeMock.EXPECT().
|
|
Unlock()
|
|
|
|
err := syncHandler.HandleMessage(ctx, senderId, msg)
|
|
require.NoError(t, err)
|
|
})
|
|
}
|