From ea77dc2bb942ca455df43ce149a0adbfa5bfac82 Mon Sep 17 00:00:00 2001 From: Fred Carle Date: Mon, 1 Jul 2024 19:15:51 -0400 Subject: [PATCH] refactor: Decouple client.DB from net (#2768) ## Relevant issue(s) Resolves #2767 ## Description This PR make the `net` package no longer dependent on `client.DB`. Instead, on creation, the peer takes in a rootstore, a blockstore and an event bus. The p2p collections and replicators have been moved to `db`. Most of the code is reused but it now makes use of the event bus to update the peer's related state (pubsub topics and peer connections). --- client/db.go | 11 +- client/errors.go | 1 + client/mocks/db.go | 391 ++++++- client/p2p.go | 2 - datastore/blockstore.go | 2 +- datastore/mocks/root_store.go | 178 ++-- datastore/mocks/txn.go | 74 +- datastore/mocks/utils.go | 6 +- datastore/multi.go | 6 +- datastore/store.go | 14 +- event/event.go | 32 + go.mod | 2 +- http/client.go | 4 +- http/client_tx.go | 2 +- internal/db/db.go | 22 +- internal/db/errors.go | 28 + internal/db/fetcher/versioned.go | 12 +- internal/db/index_test.go | 26 +- internal/db/merge.go | 50 +- internal/db/merge_test.go | 6 +- internal/db/messages.go | 84 ++ internal/db/p2p_replicator.go | 346 +++++++ internal/db/p2p_replicator_test.go | 320 ++++++ .../db/p2p_schema_root.go | 140 +-- internal/db/p2p_schema_root_test.go | 307 ++++++ internal/merkle/clock/clock.go | 18 +- internal/merkle/clock/clock_test.go | 6 +- internal/merkle/crdt/composite.go | 2 +- internal/merkle/crdt/counter.go | 2 +- internal/merkle/crdt/lwwreg.go | 2 +- internal/merkle/crdt/merklecrdt.go | 2 +- internal/merkle/crdt/merklecrdt_test.go | 2 +- internal/planner/commit.go | 2 +- net/client_test.go | 50 +- net/dialer_test.go | 42 +- net/errors.go | 33 +- net/node.go | 282 ----- net/node_test.go | 204 ---- net/peer.go | 429 ++++---- net/peer_replicator.go | 230 ----- net/peer_test.go | 965 +++++------------- net/server.go | 110 +- net/server_test.go | 163 +-- node/node.go | 28 +- node/store.go | 2 +- tests/bench/bench_util.go | 2 +- tests/bench/query/planner/utils.go | 2 +- tests/bench/storage/utils.go | 4 +- tests/clients/cli/wrapper.go | 32 +- tests/clients/cli/wrapper_tx.go | 4 +- tests/clients/clients.go | 2 +- tests/clients/http/wrapper.go | 32 +- tests/clients/http/wrapper_tx.go | 4 +- tests/integration/client.go | 33 +- tests/integration/db.go | 6 +- tests/integration/p2p.go | 36 +- tests/integration/utils2.go | 75 +- 57 files changed, 2643 insertions(+), 2229 deletions(-) create mode 100644 internal/db/messages.go create mode 100644 internal/db/p2p_replicator.go create mode 100644 internal/db/p2p_replicator_test.go rename net/peer_collection.go => internal/db/p2p_schema_root.go (59%) create mode 100644 internal/db/p2p_schema_root_test.go delete mode 100644 net/node.go delete mode 100644 net/node_test.go delete mode 100644 net/peer_replicator.go diff --git a/client/db.go b/client/db.go index e52dfed60a..ad2229cdb0 100644 --- a/client/db.go +++ b/client/db.go @@ -42,13 +42,13 @@ type DB interface { // can safely operate on it concurrently. NewConcurrentTxn(context.Context, bool) (datastore.Txn, error) - // Root returns the underlying root store, within which all data managed by DefraDB is held. - Root() datastore.RootStore + // Rootstore returns the underlying root store, within which all data managed by DefraDB is held. + Rootstore() datastore.Rootstore // Blockstore returns the blockstore, within which all blocks (commits) managed by DefraDB are held. // // It sits within the rootstore returned by [Root]. - Blockstore() datastore.DAGStore + Blockstore() datastore.Blockstore // Peerstore returns the peerstore where known host information is stored. // @@ -106,6 +106,11 @@ type Store interface { // Backup holds the backup related methods that must be implemented by the database. Backup + // P2P contains functions related to the P2P system. + // + // These functions are only useful if there is a configured network peer. + P2P + // AddSchema takes the provided GQL schema in SDL format, and applies it to the [Store], // creating the necessary collections, request types, etc. // diff --git a/client/errors.go b/client/errors.go index 460392a030..dac8ebcc87 100644 --- a/client/errors.go +++ b/client/errors.go @@ -56,6 +56,7 @@ var ( ErrCanNotNormalizeValue = errors.New(errCanNotNormalizeValue) ErrCanNotTurnNormalValueIntoArray = errors.New(errCanNotTurnNormalValueIntoArray) ErrCanNotMakeNormalNilFromFieldKind = errors.New(errCanNotMakeNormalNilFromFieldKind) + ErrCollectionNotFound = errors.New(errCollectionNotFound) ) // NewErrFieldNotExist returns an error indicating that the given field does not exist. diff --git a/client/mocks/db.go b/client/mocks/db.go index 396bc5397c..089e41c159 100644 --- a/client/mocks/db.go +++ b/client/mocks/db.go @@ -18,6 +18,8 @@ import ( mock "github.com/stretchr/testify/mock" model "github.com/lens-vm/lens/host-go/config/model" + + peer "github.com/libp2p/go-libp2p/core/peer" ) // DB is an autogenerated mock type for the DB type @@ -33,6 +35,53 @@ func (_m *DB) EXPECT() *DB_Expecter { return &DB_Expecter{mock: &_m.Mock} } +// AddP2PCollections provides a mock function with given fields: ctx, collectionIDs +func (_m *DB) AddP2PCollections(ctx context.Context, collectionIDs []string) error { + ret := _m.Called(ctx, collectionIDs) + + if len(ret) == 0 { + panic("no return value specified for AddP2PCollections") + } + + var r0 error + if rf, ok := ret.Get(0).(func(context.Context, []string) error); ok { + r0 = rf(ctx, collectionIDs) + } else { + r0 = ret.Error(0) + } + + return r0 +} + +// DB_AddP2PCollections_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'AddP2PCollections' +type DB_AddP2PCollections_Call struct { + *mock.Call +} + +// AddP2PCollections is a helper method to define mock.On call +// - ctx context.Context +// - collectionIDs []string +func (_e *DB_Expecter) AddP2PCollections(ctx interface{}, collectionIDs interface{}) *DB_AddP2PCollections_Call { + return &DB_AddP2PCollections_Call{Call: _e.mock.On("AddP2PCollections", ctx, collectionIDs)} +} + +func (_c *DB_AddP2PCollections_Call) Run(run func(ctx context.Context, collectionIDs []string)) *DB_AddP2PCollections_Call { + _c.Call.Run(func(args mock.Arguments) { + run(args[0].(context.Context), args[1].([]string)) + }) + return _c +} + +func (_c *DB_AddP2PCollections_Call) Return(_a0 error) *DB_AddP2PCollections_Call { + _c.Call.Return(_a0) + return _c +} + +func (_c *DB_AddP2PCollections_Call) RunAndReturn(run func(context.Context, []string) error) *DB_AddP2PCollections_Call { + _c.Call.Return(run) + return _c +} + // AddPolicy provides a mock function with given fields: ctx, policy func (_m *DB) AddPolicy(ctx context.Context, policy string) (client.AddPolicyResult, error) { ret := _m.Called(ctx, policy) @@ -305,19 +354,19 @@ func (_c *DB_BasicImport_Call) RunAndReturn(run func(context.Context, string) er } // Blockstore provides a mock function with given fields: -func (_m *DB) Blockstore() datastore.DAGStore { +func (_m *DB) Blockstore() datastore.Blockstore { ret := _m.Called() if len(ret) == 0 { panic("no return value specified for Blockstore") } - var r0 datastore.DAGStore - if rf, ok := ret.Get(0).(func() datastore.DAGStore); ok { + var r0 datastore.Blockstore + if rf, ok := ret.Get(0).(func() datastore.Blockstore); ok { r0 = rf() } else { if ret.Get(0) != nil { - r0 = ret.Get(0).(datastore.DAGStore) + r0 = ret.Get(0).(datastore.Blockstore) } } @@ -341,12 +390,12 @@ func (_c *DB_Blockstore_Call) Run(run func()) *DB_Blockstore_Call { return _c } -func (_c *DB_Blockstore_Call) Return(_a0 datastore.DAGStore) *DB_Blockstore_Call { +func (_c *DB_Blockstore_Call) Return(_a0 datastore.Blockstore) *DB_Blockstore_Call { _c.Call.Return(_a0) return _c } -func (_c *DB_Blockstore_Call) RunAndReturn(run func() datastore.DAGStore) *DB_Blockstore_Call { +func (_c *DB_Blockstore_Call) RunAndReturn(run func() datastore.Blockstore) *DB_Blockstore_Call { _c.Call.Return(run) return _c } @@ -383,6 +432,53 @@ func (_c *DB_Close_Call) RunAndReturn(run func()) *DB_Close_Call { return _c } +// DeleteReplicator provides a mock function with given fields: ctx, rep +func (_m *DB) DeleteReplicator(ctx context.Context, rep client.Replicator) error { + ret := _m.Called(ctx, rep) + + if len(ret) == 0 { + panic("no return value specified for DeleteReplicator") + } + + var r0 error + if rf, ok := ret.Get(0).(func(context.Context, client.Replicator) error); ok { + r0 = rf(ctx, rep) + } else { + r0 = ret.Error(0) + } + + return r0 +} + +// DB_DeleteReplicator_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'DeleteReplicator' +type DB_DeleteReplicator_Call struct { + *mock.Call +} + +// DeleteReplicator is a helper method to define mock.On call +// - ctx context.Context +// - rep client.Replicator +func (_e *DB_Expecter) DeleteReplicator(ctx interface{}, rep interface{}) *DB_DeleteReplicator_Call { + return &DB_DeleteReplicator_Call{Call: _e.mock.On("DeleteReplicator", ctx, rep)} +} + +func (_c *DB_DeleteReplicator_Call) Run(run func(ctx context.Context, rep client.Replicator)) *DB_DeleteReplicator_Call { + _c.Call.Run(func(args mock.Arguments) { + run(args[0].(context.Context), args[1].(client.Replicator)) + }) + return _c +} + +func (_c *DB_DeleteReplicator_Call) Return(_a0 error) *DB_DeleteReplicator_Call { + _c.Call.Return(_a0) + return _c +} + +func (_c *DB_DeleteReplicator_Call) RunAndReturn(run func(context.Context, client.Replicator) error) *DB_DeleteReplicator_Call { + _c.Call.Return(run) + return _c +} + // Events provides a mock function with given fields: func (_m *DB) Events() *event.Bus { ret := _m.Called() @@ -537,6 +633,122 @@ func (_c *DB_GetAllIndexes_Call) RunAndReturn(run func(context.Context) (map[str return _c } +// GetAllP2PCollections provides a mock function with given fields: ctx +func (_m *DB) GetAllP2PCollections(ctx context.Context) ([]string, error) { + ret := _m.Called(ctx) + + if len(ret) == 0 { + panic("no return value specified for GetAllP2PCollections") + } + + var r0 []string + var r1 error + if rf, ok := ret.Get(0).(func(context.Context) ([]string, error)); ok { + return rf(ctx) + } + if rf, ok := ret.Get(0).(func(context.Context) []string); ok { + r0 = rf(ctx) + } else { + if ret.Get(0) != nil { + r0 = ret.Get(0).([]string) + } + } + + if rf, ok := ret.Get(1).(func(context.Context) error); ok { + r1 = rf(ctx) + } else { + r1 = ret.Error(1) + } + + return r0, r1 +} + +// DB_GetAllP2PCollections_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'GetAllP2PCollections' +type DB_GetAllP2PCollections_Call struct { + *mock.Call +} + +// GetAllP2PCollections is a helper method to define mock.On call +// - ctx context.Context +func (_e *DB_Expecter) GetAllP2PCollections(ctx interface{}) *DB_GetAllP2PCollections_Call { + return &DB_GetAllP2PCollections_Call{Call: _e.mock.On("GetAllP2PCollections", ctx)} +} + +func (_c *DB_GetAllP2PCollections_Call) Run(run func(ctx context.Context)) *DB_GetAllP2PCollections_Call { + _c.Call.Run(func(args mock.Arguments) { + run(args[0].(context.Context)) + }) + return _c +} + +func (_c *DB_GetAllP2PCollections_Call) Return(_a0 []string, _a1 error) *DB_GetAllP2PCollections_Call { + _c.Call.Return(_a0, _a1) + return _c +} + +func (_c *DB_GetAllP2PCollections_Call) RunAndReturn(run func(context.Context) ([]string, error)) *DB_GetAllP2PCollections_Call { + _c.Call.Return(run) + return _c +} + +// GetAllReplicators provides a mock function with given fields: ctx +func (_m *DB) GetAllReplicators(ctx context.Context) ([]client.Replicator, error) { + ret := _m.Called(ctx) + + if len(ret) == 0 { + panic("no return value specified for GetAllReplicators") + } + + var r0 []client.Replicator + var r1 error + if rf, ok := ret.Get(0).(func(context.Context) ([]client.Replicator, error)); ok { + return rf(ctx) + } + if rf, ok := ret.Get(0).(func(context.Context) []client.Replicator); ok { + r0 = rf(ctx) + } else { + if ret.Get(0) != nil { + r0 = ret.Get(0).([]client.Replicator) + } + } + + if rf, ok := ret.Get(1).(func(context.Context) error); ok { + r1 = rf(ctx) + } else { + r1 = ret.Error(1) + } + + return r0, r1 +} + +// DB_GetAllReplicators_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'GetAllReplicators' +type DB_GetAllReplicators_Call struct { + *mock.Call +} + +// GetAllReplicators is a helper method to define mock.On call +// - ctx context.Context +func (_e *DB_Expecter) GetAllReplicators(ctx interface{}) *DB_GetAllReplicators_Call { + return &DB_GetAllReplicators_Call{Call: _e.mock.On("GetAllReplicators", ctx)} +} + +func (_c *DB_GetAllReplicators_Call) Run(run func(ctx context.Context)) *DB_GetAllReplicators_Call { + _c.Call.Run(func(args mock.Arguments) { + run(args[0].(context.Context)) + }) + return _c +} + +func (_c *DB_GetAllReplicators_Call) Return(_a0 []client.Replicator, _a1 error) *DB_GetAllReplicators_Call { + _c.Call.Return(_a0, _a1) + return _c +} + +func (_c *DB_GetAllReplicators_Call) RunAndReturn(run func(context.Context) ([]client.Replicator, error)) *DB_GetAllReplicators_Call { + _c.Call.Return(run) + return _c +} + // GetCollectionByName provides a mock function with given fields: _a0, _a1 func (_m *DB) GetCollectionByName(_a0 context.Context, _a1 string) (client.Collection, error) { ret := _m.Called(_a0, _a1) @@ -1124,6 +1336,51 @@ func (_c *DB_PatchSchema_Call) RunAndReturn(run func(context.Context, string, im return _c } +// PeerInfo provides a mock function with given fields: +func (_m *DB) PeerInfo() peer.AddrInfo { + ret := _m.Called() + + if len(ret) == 0 { + panic("no return value specified for PeerInfo") + } + + var r0 peer.AddrInfo + if rf, ok := ret.Get(0).(func() peer.AddrInfo); ok { + r0 = rf() + } else { + r0 = ret.Get(0).(peer.AddrInfo) + } + + return r0 +} + +// DB_PeerInfo_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'PeerInfo' +type DB_PeerInfo_Call struct { + *mock.Call +} + +// PeerInfo is a helper method to define mock.On call +func (_e *DB_Expecter) PeerInfo() *DB_PeerInfo_Call { + return &DB_PeerInfo_Call{Call: _e.mock.On("PeerInfo")} +} + +func (_c *DB_PeerInfo_Call) Run(run func()) *DB_PeerInfo_Call { + _c.Call.Run(func(args mock.Arguments) { + run() + }) + return _c +} + +func (_c *DB_PeerInfo_Call) Return(_a0 peer.AddrInfo) *DB_PeerInfo_Call { + _c.Call.Return(_a0) + return _c +} + +func (_c *DB_PeerInfo_Call) RunAndReturn(run func() peer.AddrInfo) *DB_PeerInfo_Call { + _c.Call.Return(run) + return _c +} + // Peerstore provides a mock function with given fields: func (_m *DB) Peerstore() datastore.DSBatching { ret := _m.Called() @@ -1217,49 +1474,96 @@ func (_c *DB_PrintDump_Call) RunAndReturn(run func(context.Context) error) *DB_P return _c } -// Root provides a mock function with given fields: -func (_m *DB) Root() datastore.RootStore { +// RemoveP2PCollections provides a mock function with given fields: ctx, collectionIDs +func (_m *DB) RemoveP2PCollections(ctx context.Context, collectionIDs []string) error { + ret := _m.Called(ctx, collectionIDs) + + if len(ret) == 0 { + panic("no return value specified for RemoveP2PCollections") + } + + var r0 error + if rf, ok := ret.Get(0).(func(context.Context, []string) error); ok { + r0 = rf(ctx, collectionIDs) + } else { + r0 = ret.Error(0) + } + + return r0 +} + +// DB_RemoveP2PCollections_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'RemoveP2PCollections' +type DB_RemoveP2PCollections_Call struct { + *mock.Call +} + +// RemoveP2PCollections is a helper method to define mock.On call +// - ctx context.Context +// - collectionIDs []string +func (_e *DB_Expecter) RemoveP2PCollections(ctx interface{}, collectionIDs interface{}) *DB_RemoveP2PCollections_Call { + return &DB_RemoveP2PCollections_Call{Call: _e.mock.On("RemoveP2PCollections", ctx, collectionIDs)} +} + +func (_c *DB_RemoveP2PCollections_Call) Run(run func(ctx context.Context, collectionIDs []string)) *DB_RemoveP2PCollections_Call { + _c.Call.Run(func(args mock.Arguments) { + run(args[0].(context.Context), args[1].([]string)) + }) + return _c +} + +func (_c *DB_RemoveP2PCollections_Call) Return(_a0 error) *DB_RemoveP2PCollections_Call { + _c.Call.Return(_a0) + return _c +} + +func (_c *DB_RemoveP2PCollections_Call) RunAndReturn(run func(context.Context, []string) error) *DB_RemoveP2PCollections_Call { + _c.Call.Return(run) + return _c +} + +// Rootstore provides a mock function with given fields: +func (_m *DB) Rootstore() datastore.Rootstore { ret := _m.Called() if len(ret) == 0 { - panic("no return value specified for Root") + panic("no return value specified for Rootstore") } - var r0 datastore.RootStore - if rf, ok := ret.Get(0).(func() datastore.RootStore); ok { + var r0 datastore.Rootstore + if rf, ok := ret.Get(0).(func() datastore.Rootstore); ok { r0 = rf() } else { if ret.Get(0) != nil { - r0 = ret.Get(0).(datastore.RootStore) + r0 = ret.Get(0).(datastore.Rootstore) } } return r0 } -// DB_Root_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Root' -type DB_Root_Call struct { +// DB_Rootstore_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Rootstore' +type DB_Rootstore_Call struct { *mock.Call } -// Root is a helper method to define mock.On call -func (_e *DB_Expecter) Root() *DB_Root_Call { - return &DB_Root_Call{Call: _e.mock.On("Root")} +// Rootstore is a helper method to define mock.On call +func (_e *DB_Expecter) Rootstore() *DB_Rootstore_Call { + return &DB_Rootstore_Call{Call: _e.mock.On("Rootstore")} } -func (_c *DB_Root_Call) Run(run func()) *DB_Root_Call { +func (_c *DB_Rootstore_Call) Run(run func()) *DB_Rootstore_Call { _c.Call.Run(func(args mock.Arguments) { run() }) return _c } -func (_c *DB_Root_Call) Return(_a0 datastore.RootStore) *DB_Root_Call { +func (_c *DB_Rootstore_Call) Return(_a0 datastore.Rootstore) *DB_Rootstore_Call { _c.Call.Return(_a0) return _c } -func (_c *DB_Root_Call) RunAndReturn(run func() datastore.RootStore) *DB_Root_Call { +func (_c *DB_Rootstore_Call) RunAndReturn(run func() datastore.Rootstore) *DB_Rootstore_Call { _c.Call.Return(run) return _c } @@ -1358,6 +1662,53 @@ func (_c *DB_SetMigration_Call) RunAndReturn(run func(context.Context, client.Le return _c } +// SetReplicator provides a mock function with given fields: ctx, rep +func (_m *DB) SetReplicator(ctx context.Context, rep client.Replicator) error { + ret := _m.Called(ctx, rep) + + if len(ret) == 0 { + panic("no return value specified for SetReplicator") + } + + var r0 error + if rf, ok := ret.Get(0).(func(context.Context, client.Replicator) error); ok { + r0 = rf(ctx, rep) + } else { + r0 = ret.Error(0) + } + + return r0 +} + +// DB_SetReplicator_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'SetReplicator' +type DB_SetReplicator_Call struct { + *mock.Call +} + +// SetReplicator is a helper method to define mock.On call +// - ctx context.Context +// - rep client.Replicator +func (_e *DB_Expecter) SetReplicator(ctx interface{}, rep interface{}) *DB_SetReplicator_Call { + return &DB_SetReplicator_Call{Call: _e.mock.On("SetReplicator", ctx, rep)} +} + +func (_c *DB_SetReplicator_Call) Run(run func(ctx context.Context, rep client.Replicator)) *DB_SetReplicator_Call { + _c.Call.Run(func(args mock.Arguments) { + run(args[0].(context.Context), args[1].(client.Replicator)) + }) + return _c +} + +func (_c *DB_SetReplicator_Call) Return(_a0 error) *DB_SetReplicator_Call { + _c.Call.Return(_a0) + return _c +} + +func (_c *DB_SetReplicator_Call) RunAndReturn(run func(context.Context, client.Replicator) error) *DB_SetReplicator_Call { + _c.Call.Return(run) + return _c +} + // NewDB creates a new instance of DB. It also registers a testing interface on the mock and a cleanup function to assert the mocks expectations. // The first argument is typically a *testing.T value. func NewDB(t interface { diff --git a/client/p2p.go b/client/p2p.go index 12be6ebf8d..d3d3c699b3 100644 --- a/client/p2p.go +++ b/client/p2p.go @@ -18,8 +18,6 @@ import ( // P2P is a peer connected database implementation. type P2P interface { - DB - // PeerInfo returns the p2p host id and listening addresses. PeerInfo() peer.AddrInfo diff --git a/datastore/blockstore.go b/datastore/blockstore.go index f9f92198cd..408e209eff 100644 --- a/datastore/blockstore.go +++ b/datastore/blockstore.go @@ -62,7 +62,7 @@ type bstore struct { } var _ blockstore.Blockstore = (*bstore)(nil) -var _ DAGStore = (*bstore)(nil) +var _ Blockstore = (*bstore)(nil) // AsIPLDStorage returns an IPLDStorage instance. // diff --git a/datastore/mocks/root_store.go b/datastore/mocks/root_store.go index 94d2694721..1cd09f9e7c 100644 --- a/datastore/mocks/root_store.go +++ b/datastore/mocks/root_store.go @@ -12,21 +12,21 @@ import ( query "github.com/ipfs/go-datastore/query" ) -// RootStore is an autogenerated mock type for the RootStore type -type RootStore struct { +// Rootstore is an autogenerated mock type for the Rootstore type +type Rootstore struct { mock.Mock } -type RootStore_Expecter struct { +type Rootstore_Expecter struct { mock *mock.Mock } -func (_m *RootStore) EXPECT() *RootStore_Expecter { - return &RootStore_Expecter{mock: &_m.Mock} +func (_m *Rootstore) EXPECT() *Rootstore_Expecter { + return &Rootstore_Expecter{mock: &_m.Mock} } // Batch provides a mock function with given fields: ctx -func (_m *RootStore) Batch(ctx context.Context) (datastore.Batch, error) { +func (_m *Rootstore) Batch(ctx context.Context) (datastore.Batch, error) { ret := _m.Called(ctx) if len(ret) == 0 { @@ -55,36 +55,36 @@ func (_m *RootStore) Batch(ctx context.Context) (datastore.Batch, error) { return r0, r1 } -// RootStore_Batch_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Batch' -type RootStore_Batch_Call struct { +// Rootstore_Batch_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Batch' +type Rootstore_Batch_Call struct { *mock.Call } // Batch is a helper method to define mock.On call // - ctx context.Context -func (_e *RootStore_Expecter) Batch(ctx interface{}) *RootStore_Batch_Call { - return &RootStore_Batch_Call{Call: _e.mock.On("Batch", ctx)} +func (_e *Rootstore_Expecter) Batch(ctx interface{}) *Rootstore_Batch_Call { + return &Rootstore_Batch_Call{Call: _e.mock.On("Batch", ctx)} } -func (_c *RootStore_Batch_Call) Run(run func(ctx context.Context)) *RootStore_Batch_Call { +func (_c *Rootstore_Batch_Call) Run(run func(ctx context.Context)) *Rootstore_Batch_Call { _c.Call.Run(func(args mock.Arguments) { run(args[0].(context.Context)) }) return _c } -func (_c *RootStore_Batch_Call) Return(_a0 datastore.Batch, _a1 error) *RootStore_Batch_Call { +func (_c *Rootstore_Batch_Call) Return(_a0 datastore.Batch, _a1 error) *Rootstore_Batch_Call { _c.Call.Return(_a0, _a1) return _c } -func (_c *RootStore_Batch_Call) RunAndReturn(run func(context.Context) (datastore.Batch, error)) *RootStore_Batch_Call { +func (_c *Rootstore_Batch_Call) RunAndReturn(run func(context.Context) (datastore.Batch, error)) *Rootstore_Batch_Call { _c.Call.Return(run) return _c } // Close provides a mock function with given fields: -func (_m *RootStore) Close() error { +func (_m *Rootstore) Close() error { ret := _m.Called() if len(ret) == 0 { @@ -101,35 +101,35 @@ func (_m *RootStore) Close() error { return r0 } -// RootStore_Close_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Close' -type RootStore_Close_Call struct { +// Rootstore_Close_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Close' +type Rootstore_Close_Call struct { *mock.Call } // Close is a helper method to define mock.On call -func (_e *RootStore_Expecter) Close() *RootStore_Close_Call { - return &RootStore_Close_Call{Call: _e.mock.On("Close")} +func (_e *Rootstore_Expecter) Close() *Rootstore_Close_Call { + return &Rootstore_Close_Call{Call: _e.mock.On("Close")} } -func (_c *RootStore_Close_Call) Run(run func()) *RootStore_Close_Call { +func (_c *Rootstore_Close_Call) Run(run func()) *Rootstore_Close_Call { _c.Call.Run(func(args mock.Arguments) { run() }) return _c } -func (_c *RootStore_Close_Call) Return(_a0 error) *RootStore_Close_Call { +func (_c *Rootstore_Close_Call) Return(_a0 error) *Rootstore_Close_Call { _c.Call.Return(_a0) return _c } -func (_c *RootStore_Close_Call) RunAndReturn(run func() error) *RootStore_Close_Call { +func (_c *Rootstore_Close_Call) RunAndReturn(run func() error) *Rootstore_Close_Call { _c.Call.Return(run) return _c } // Delete provides a mock function with given fields: ctx, key -func (_m *RootStore) Delete(ctx context.Context, key datastore.Key) error { +func (_m *Rootstore) Delete(ctx context.Context, key datastore.Key) error { ret := _m.Called(ctx, key) if len(ret) == 0 { @@ -146,37 +146,37 @@ func (_m *RootStore) Delete(ctx context.Context, key datastore.Key) error { return r0 } -// RootStore_Delete_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Delete' -type RootStore_Delete_Call struct { +// Rootstore_Delete_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Delete' +type Rootstore_Delete_Call struct { *mock.Call } // Delete is a helper method to define mock.On call // - ctx context.Context // - key datastore.Key -func (_e *RootStore_Expecter) Delete(ctx interface{}, key interface{}) *RootStore_Delete_Call { - return &RootStore_Delete_Call{Call: _e.mock.On("Delete", ctx, key)} +func (_e *Rootstore_Expecter) Delete(ctx interface{}, key interface{}) *Rootstore_Delete_Call { + return &Rootstore_Delete_Call{Call: _e.mock.On("Delete", ctx, key)} } -func (_c *RootStore_Delete_Call) Run(run func(ctx context.Context, key datastore.Key)) *RootStore_Delete_Call { +func (_c *Rootstore_Delete_Call) Run(run func(ctx context.Context, key datastore.Key)) *Rootstore_Delete_Call { _c.Call.Run(func(args mock.Arguments) { run(args[0].(context.Context), args[1].(datastore.Key)) }) return _c } -func (_c *RootStore_Delete_Call) Return(_a0 error) *RootStore_Delete_Call { +func (_c *Rootstore_Delete_Call) Return(_a0 error) *Rootstore_Delete_Call { _c.Call.Return(_a0) return _c } -func (_c *RootStore_Delete_Call) RunAndReturn(run func(context.Context, datastore.Key) error) *RootStore_Delete_Call { +func (_c *Rootstore_Delete_Call) RunAndReturn(run func(context.Context, datastore.Key) error) *Rootstore_Delete_Call { _c.Call.Return(run) return _c } // Get provides a mock function with given fields: ctx, key -func (_m *RootStore) Get(ctx context.Context, key datastore.Key) ([]byte, error) { +func (_m *Rootstore) Get(ctx context.Context, key datastore.Key) ([]byte, error) { ret := _m.Called(ctx, key) if len(ret) == 0 { @@ -205,37 +205,37 @@ func (_m *RootStore) Get(ctx context.Context, key datastore.Key) ([]byte, error) return r0, r1 } -// RootStore_Get_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Get' -type RootStore_Get_Call struct { +// Rootstore_Get_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Get' +type Rootstore_Get_Call struct { *mock.Call } // Get is a helper method to define mock.On call // - ctx context.Context // - key datastore.Key -func (_e *RootStore_Expecter) Get(ctx interface{}, key interface{}) *RootStore_Get_Call { - return &RootStore_Get_Call{Call: _e.mock.On("Get", ctx, key)} +func (_e *Rootstore_Expecter) Get(ctx interface{}, key interface{}) *Rootstore_Get_Call { + return &Rootstore_Get_Call{Call: _e.mock.On("Get", ctx, key)} } -func (_c *RootStore_Get_Call) Run(run func(ctx context.Context, key datastore.Key)) *RootStore_Get_Call { +func (_c *Rootstore_Get_Call) Run(run func(ctx context.Context, key datastore.Key)) *Rootstore_Get_Call { _c.Call.Run(func(args mock.Arguments) { run(args[0].(context.Context), args[1].(datastore.Key)) }) return _c } -func (_c *RootStore_Get_Call) Return(value []byte, err error) *RootStore_Get_Call { +func (_c *Rootstore_Get_Call) Return(value []byte, err error) *Rootstore_Get_Call { _c.Call.Return(value, err) return _c } -func (_c *RootStore_Get_Call) RunAndReturn(run func(context.Context, datastore.Key) ([]byte, error)) *RootStore_Get_Call { +func (_c *Rootstore_Get_Call) RunAndReturn(run func(context.Context, datastore.Key) ([]byte, error)) *Rootstore_Get_Call { _c.Call.Return(run) return _c } // GetSize provides a mock function with given fields: ctx, key -func (_m *RootStore) GetSize(ctx context.Context, key datastore.Key) (int, error) { +func (_m *Rootstore) GetSize(ctx context.Context, key datastore.Key) (int, error) { ret := _m.Called(ctx, key) if len(ret) == 0 { @@ -262,37 +262,37 @@ func (_m *RootStore) GetSize(ctx context.Context, key datastore.Key) (int, error return r0, r1 } -// RootStore_GetSize_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'GetSize' -type RootStore_GetSize_Call struct { +// Rootstore_GetSize_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'GetSize' +type Rootstore_GetSize_Call struct { *mock.Call } // GetSize is a helper method to define mock.On call // - ctx context.Context // - key datastore.Key -func (_e *RootStore_Expecter) GetSize(ctx interface{}, key interface{}) *RootStore_GetSize_Call { - return &RootStore_GetSize_Call{Call: _e.mock.On("GetSize", ctx, key)} +func (_e *Rootstore_Expecter) GetSize(ctx interface{}, key interface{}) *Rootstore_GetSize_Call { + return &Rootstore_GetSize_Call{Call: _e.mock.On("GetSize", ctx, key)} } -func (_c *RootStore_GetSize_Call) Run(run func(ctx context.Context, key datastore.Key)) *RootStore_GetSize_Call { +func (_c *Rootstore_GetSize_Call) Run(run func(ctx context.Context, key datastore.Key)) *Rootstore_GetSize_Call { _c.Call.Run(func(args mock.Arguments) { run(args[0].(context.Context), args[1].(datastore.Key)) }) return _c } -func (_c *RootStore_GetSize_Call) Return(size int, err error) *RootStore_GetSize_Call { +func (_c *Rootstore_GetSize_Call) Return(size int, err error) *Rootstore_GetSize_Call { _c.Call.Return(size, err) return _c } -func (_c *RootStore_GetSize_Call) RunAndReturn(run func(context.Context, datastore.Key) (int, error)) *RootStore_GetSize_Call { +func (_c *Rootstore_GetSize_Call) RunAndReturn(run func(context.Context, datastore.Key) (int, error)) *Rootstore_GetSize_Call { _c.Call.Return(run) return _c } // Has provides a mock function with given fields: ctx, key -func (_m *RootStore) Has(ctx context.Context, key datastore.Key) (bool, error) { +func (_m *Rootstore) Has(ctx context.Context, key datastore.Key) (bool, error) { ret := _m.Called(ctx, key) if len(ret) == 0 { @@ -319,37 +319,37 @@ func (_m *RootStore) Has(ctx context.Context, key datastore.Key) (bool, error) { return r0, r1 } -// RootStore_Has_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Has' -type RootStore_Has_Call struct { +// Rootstore_Has_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Has' +type Rootstore_Has_Call struct { *mock.Call } // Has is a helper method to define mock.On call // - ctx context.Context // - key datastore.Key -func (_e *RootStore_Expecter) Has(ctx interface{}, key interface{}) *RootStore_Has_Call { - return &RootStore_Has_Call{Call: _e.mock.On("Has", ctx, key)} +func (_e *Rootstore_Expecter) Has(ctx interface{}, key interface{}) *Rootstore_Has_Call { + return &Rootstore_Has_Call{Call: _e.mock.On("Has", ctx, key)} } -func (_c *RootStore_Has_Call) Run(run func(ctx context.Context, key datastore.Key)) *RootStore_Has_Call { +func (_c *Rootstore_Has_Call) Run(run func(ctx context.Context, key datastore.Key)) *Rootstore_Has_Call { _c.Call.Run(func(args mock.Arguments) { run(args[0].(context.Context), args[1].(datastore.Key)) }) return _c } -func (_c *RootStore_Has_Call) Return(exists bool, err error) *RootStore_Has_Call { +func (_c *Rootstore_Has_Call) Return(exists bool, err error) *Rootstore_Has_Call { _c.Call.Return(exists, err) return _c } -func (_c *RootStore_Has_Call) RunAndReturn(run func(context.Context, datastore.Key) (bool, error)) *RootStore_Has_Call { +func (_c *Rootstore_Has_Call) RunAndReturn(run func(context.Context, datastore.Key) (bool, error)) *Rootstore_Has_Call { _c.Call.Return(run) return _c } // NewTransaction provides a mock function with given fields: ctx, readOnly -func (_m *RootStore) NewTransaction(ctx context.Context, readOnly bool) (datastore.Txn, error) { +func (_m *Rootstore) NewTransaction(ctx context.Context, readOnly bool) (datastore.Txn, error) { ret := _m.Called(ctx, readOnly) if len(ret) == 0 { @@ -378,37 +378,37 @@ func (_m *RootStore) NewTransaction(ctx context.Context, readOnly bool) (datasto return r0, r1 } -// RootStore_NewTransaction_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'NewTransaction' -type RootStore_NewTransaction_Call struct { +// Rootstore_NewTransaction_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'NewTransaction' +type Rootstore_NewTransaction_Call struct { *mock.Call } // NewTransaction is a helper method to define mock.On call // - ctx context.Context // - readOnly bool -func (_e *RootStore_Expecter) NewTransaction(ctx interface{}, readOnly interface{}) *RootStore_NewTransaction_Call { - return &RootStore_NewTransaction_Call{Call: _e.mock.On("NewTransaction", ctx, readOnly)} +func (_e *Rootstore_Expecter) NewTransaction(ctx interface{}, readOnly interface{}) *Rootstore_NewTransaction_Call { + return &Rootstore_NewTransaction_Call{Call: _e.mock.On("NewTransaction", ctx, readOnly)} } -func (_c *RootStore_NewTransaction_Call) Run(run func(ctx context.Context, readOnly bool)) *RootStore_NewTransaction_Call { +func (_c *Rootstore_NewTransaction_Call) Run(run func(ctx context.Context, readOnly bool)) *Rootstore_NewTransaction_Call { _c.Call.Run(func(args mock.Arguments) { run(args[0].(context.Context), args[1].(bool)) }) return _c } -func (_c *RootStore_NewTransaction_Call) Return(_a0 datastore.Txn, _a1 error) *RootStore_NewTransaction_Call { +func (_c *Rootstore_NewTransaction_Call) Return(_a0 datastore.Txn, _a1 error) *Rootstore_NewTransaction_Call { _c.Call.Return(_a0, _a1) return _c } -func (_c *RootStore_NewTransaction_Call) RunAndReturn(run func(context.Context, bool) (datastore.Txn, error)) *RootStore_NewTransaction_Call { +func (_c *Rootstore_NewTransaction_Call) RunAndReturn(run func(context.Context, bool) (datastore.Txn, error)) *Rootstore_NewTransaction_Call { _c.Call.Return(run) return _c } // Put provides a mock function with given fields: ctx, key, value -func (_m *RootStore) Put(ctx context.Context, key datastore.Key, value []byte) error { +func (_m *Rootstore) Put(ctx context.Context, key datastore.Key, value []byte) error { ret := _m.Called(ctx, key, value) if len(ret) == 0 { @@ -425,8 +425,8 @@ func (_m *RootStore) Put(ctx context.Context, key datastore.Key, value []byte) e return r0 } -// RootStore_Put_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Put' -type RootStore_Put_Call struct { +// Rootstore_Put_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Put' +type Rootstore_Put_Call struct { *mock.Call } @@ -434,29 +434,29 @@ type RootStore_Put_Call struct { // - ctx context.Context // - key datastore.Key // - value []byte -func (_e *RootStore_Expecter) Put(ctx interface{}, key interface{}, value interface{}) *RootStore_Put_Call { - return &RootStore_Put_Call{Call: _e.mock.On("Put", ctx, key, value)} +func (_e *Rootstore_Expecter) Put(ctx interface{}, key interface{}, value interface{}) *Rootstore_Put_Call { + return &Rootstore_Put_Call{Call: _e.mock.On("Put", ctx, key, value)} } -func (_c *RootStore_Put_Call) Run(run func(ctx context.Context, key datastore.Key, value []byte)) *RootStore_Put_Call { +func (_c *Rootstore_Put_Call) Run(run func(ctx context.Context, key datastore.Key, value []byte)) *Rootstore_Put_Call { _c.Call.Run(func(args mock.Arguments) { run(args[0].(context.Context), args[1].(datastore.Key), args[2].([]byte)) }) return _c } -func (_c *RootStore_Put_Call) Return(_a0 error) *RootStore_Put_Call { +func (_c *Rootstore_Put_Call) Return(_a0 error) *Rootstore_Put_Call { _c.Call.Return(_a0) return _c } -func (_c *RootStore_Put_Call) RunAndReturn(run func(context.Context, datastore.Key, []byte) error) *RootStore_Put_Call { +func (_c *Rootstore_Put_Call) RunAndReturn(run func(context.Context, datastore.Key, []byte) error) *Rootstore_Put_Call { _c.Call.Return(run) return _c } // Query provides a mock function with given fields: ctx, q -func (_m *RootStore) Query(ctx context.Context, q query.Query) (query.Results, error) { +func (_m *Rootstore) Query(ctx context.Context, q query.Query) (query.Results, error) { ret := _m.Called(ctx, q) if len(ret) == 0 { @@ -485,37 +485,37 @@ func (_m *RootStore) Query(ctx context.Context, q query.Query) (query.Results, e return r0, r1 } -// RootStore_Query_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Query' -type RootStore_Query_Call struct { +// Rootstore_Query_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Query' +type Rootstore_Query_Call struct { *mock.Call } // Query is a helper method to define mock.On call // - ctx context.Context // - q query.Query -func (_e *RootStore_Expecter) Query(ctx interface{}, q interface{}) *RootStore_Query_Call { - return &RootStore_Query_Call{Call: _e.mock.On("Query", ctx, q)} +func (_e *Rootstore_Expecter) Query(ctx interface{}, q interface{}) *Rootstore_Query_Call { + return &Rootstore_Query_Call{Call: _e.mock.On("Query", ctx, q)} } -func (_c *RootStore_Query_Call) Run(run func(ctx context.Context, q query.Query)) *RootStore_Query_Call { +func (_c *Rootstore_Query_Call) Run(run func(ctx context.Context, q query.Query)) *Rootstore_Query_Call { _c.Call.Run(func(args mock.Arguments) { run(args[0].(context.Context), args[1].(query.Query)) }) return _c } -func (_c *RootStore_Query_Call) Return(_a0 query.Results, _a1 error) *RootStore_Query_Call { +func (_c *Rootstore_Query_Call) Return(_a0 query.Results, _a1 error) *Rootstore_Query_Call { _c.Call.Return(_a0, _a1) return _c } -func (_c *RootStore_Query_Call) RunAndReturn(run func(context.Context, query.Query) (query.Results, error)) *RootStore_Query_Call { +func (_c *Rootstore_Query_Call) RunAndReturn(run func(context.Context, query.Query) (query.Results, error)) *Rootstore_Query_Call { _c.Call.Return(run) return _c } // Sync provides a mock function with given fields: ctx, prefix -func (_m *RootStore) Sync(ctx context.Context, prefix datastore.Key) error { +func (_m *Rootstore) Sync(ctx context.Context, prefix datastore.Key) error { ret := _m.Called(ctx, prefix) if len(ret) == 0 { @@ -532,42 +532,42 @@ func (_m *RootStore) Sync(ctx context.Context, prefix datastore.Key) error { return r0 } -// RootStore_Sync_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Sync' -type RootStore_Sync_Call struct { +// Rootstore_Sync_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Sync' +type Rootstore_Sync_Call struct { *mock.Call } // Sync is a helper method to define mock.On call // - ctx context.Context // - prefix datastore.Key -func (_e *RootStore_Expecter) Sync(ctx interface{}, prefix interface{}) *RootStore_Sync_Call { - return &RootStore_Sync_Call{Call: _e.mock.On("Sync", ctx, prefix)} +func (_e *Rootstore_Expecter) Sync(ctx interface{}, prefix interface{}) *Rootstore_Sync_Call { + return &Rootstore_Sync_Call{Call: _e.mock.On("Sync", ctx, prefix)} } -func (_c *RootStore_Sync_Call) Run(run func(ctx context.Context, prefix datastore.Key)) *RootStore_Sync_Call { +func (_c *Rootstore_Sync_Call) Run(run func(ctx context.Context, prefix datastore.Key)) *Rootstore_Sync_Call { _c.Call.Run(func(args mock.Arguments) { run(args[0].(context.Context), args[1].(datastore.Key)) }) return _c } -func (_c *RootStore_Sync_Call) Return(_a0 error) *RootStore_Sync_Call { +func (_c *Rootstore_Sync_Call) Return(_a0 error) *Rootstore_Sync_Call { _c.Call.Return(_a0) return _c } -func (_c *RootStore_Sync_Call) RunAndReturn(run func(context.Context, datastore.Key) error) *RootStore_Sync_Call { +func (_c *Rootstore_Sync_Call) RunAndReturn(run func(context.Context, datastore.Key) error) *Rootstore_Sync_Call { _c.Call.Return(run) return _c } -// NewRootStore creates a new instance of RootStore. It also registers a testing interface on the mock and a cleanup function to assert the mocks expectations. +// NewRootstore creates a new instance of Rootstore. It also registers a testing interface on the mock and a cleanup function to assert the mocks expectations. // The first argument is typically a *testing.T value. -func NewRootStore(t interface { +func NewRootstore(t interface { mock.TestingT Cleanup(func()) -}) *RootStore { - mock := &RootStore{} +}) *Rootstore { + mock := &Rootstore{} mock.Mock.Test(t) t.Cleanup(func() { mock.AssertExpectations(t) }) diff --git a/datastore/mocks/txn.go b/datastore/mocks/txn.go index 7c9872dfb2..f29c045dcd 100644 --- a/datastore/mocks/txn.go +++ b/datastore/mocks/txn.go @@ -22,95 +22,95 @@ func (_m *Txn) EXPECT() *Txn_Expecter { return &Txn_Expecter{mock: &_m.Mock} } -// Commit provides a mock function with given fields: ctx -func (_m *Txn) Commit(ctx context.Context) error { - ret := _m.Called(ctx) +// Blockstore provides a mock function with given fields: +func (_m *Txn) Blockstore() datastore.Blockstore { + ret := _m.Called() if len(ret) == 0 { - panic("no return value specified for Commit") + panic("no return value specified for Blockstore") } - var r0 error - if rf, ok := ret.Get(0).(func(context.Context) error); ok { - r0 = rf(ctx) + var r0 datastore.Blockstore + if rf, ok := ret.Get(0).(func() datastore.Blockstore); ok { + r0 = rf() } else { - r0 = ret.Error(0) + if ret.Get(0) != nil { + r0 = ret.Get(0).(datastore.Blockstore) + } } return r0 } -// Txn_Commit_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Commit' -type Txn_Commit_Call struct { +// Txn_Blockstore_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Blockstore' +type Txn_Blockstore_Call struct { *mock.Call } -// Commit is a helper method to define mock.On call -// - ctx context.Context -func (_e *Txn_Expecter) Commit(ctx interface{}) *Txn_Commit_Call { - return &Txn_Commit_Call{Call: _e.mock.On("Commit", ctx)} +// Blockstore is a helper method to define mock.On call +func (_e *Txn_Expecter) Blockstore() *Txn_Blockstore_Call { + return &Txn_Blockstore_Call{Call: _e.mock.On("Blockstore")} } -func (_c *Txn_Commit_Call) Run(run func(ctx context.Context)) *Txn_Commit_Call { +func (_c *Txn_Blockstore_Call) Run(run func()) *Txn_Blockstore_Call { _c.Call.Run(func(args mock.Arguments) { - run(args[0].(context.Context)) + run() }) return _c } -func (_c *Txn_Commit_Call) Return(_a0 error) *Txn_Commit_Call { +func (_c *Txn_Blockstore_Call) Return(_a0 datastore.Blockstore) *Txn_Blockstore_Call { _c.Call.Return(_a0) return _c } -func (_c *Txn_Commit_Call) RunAndReturn(run func(context.Context) error) *Txn_Commit_Call { +func (_c *Txn_Blockstore_Call) RunAndReturn(run func() datastore.Blockstore) *Txn_Blockstore_Call { _c.Call.Return(run) return _c } -// DAGstore provides a mock function with given fields: -func (_m *Txn) DAGstore() datastore.DAGStore { - ret := _m.Called() +// Commit provides a mock function with given fields: ctx +func (_m *Txn) Commit(ctx context.Context) error { + ret := _m.Called(ctx) if len(ret) == 0 { - panic("no return value specified for DAGstore") + panic("no return value specified for Commit") } - var r0 datastore.DAGStore - if rf, ok := ret.Get(0).(func() datastore.DAGStore); ok { - r0 = rf() + var r0 error + if rf, ok := ret.Get(0).(func(context.Context) error); ok { + r0 = rf(ctx) } else { - if ret.Get(0) != nil { - r0 = ret.Get(0).(datastore.DAGStore) - } + r0 = ret.Error(0) } return r0 } -// Txn_DAGstore_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'DAGstore' -type Txn_DAGstore_Call struct { +// Txn_Commit_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Commit' +type Txn_Commit_Call struct { *mock.Call } -// DAGstore is a helper method to define mock.On call -func (_e *Txn_Expecter) DAGstore() *Txn_DAGstore_Call { - return &Txn_DAGstore_Call{Call: _e.mock.On("DAGstore")} +// Commit is a helper method to define mock.On call +// - ctx context.Context +func (_e *Txn_Expecter) Commit(ctx interface{}) *Txn_Commit_Call { + return &Txn_Commit_Call{Call: _e.mock.On("Commit", ctx)} } -func (_c *Txn_DAGstore_Call) Run(run func()) *Txn_DAGstore_Call { +func (_c *Txn_Commit_Call) Run(run func(ctx context.Context)) *Txn_Commit_Call { _c.Call.Run(func(args mock.Arguments) { - run() + run(args[0].(context.Context)) }) return _c } -func (_c *Txn_DAGstore_Call) Return(_a0 datastore.DAGStore) *Txn_DAGstore_Call { +func (_c *Txn_Commit_Call) Return(_a0 error) *Txn_Commit_Call { _c.Call.Return(_a0) return _c } -func (_c *Txn_DAGstore_Call) RunAndReturn(run func() datastore.DAGStore) *Txn_DAGstore_Call { +func (_c *Txn_Commit_Call) RunAndReturn(run func(context.Context) error) *Txn_Commit_Call { _c.Call.Return(run) return _c } diff --git a/datastore/mocks/utils.go b/datastore/mocks/utils.go index 50131a8538..af3c49fd0c 100644 --- a/datastore/mocks/utils.go +++ b/datastore/mocks/utils.go @@ -36,7 +36,7 @@ func prepareDataStore(t *testing.T) *DSReaderWriter { return dataStore } -func prepareRootStore(t *testing.T) *DSReaderWriter { +func prepareRootstore(t *testing.T) *DSReaderWriter { return NewDSReaderWriter(t) } @@ -73,7 +73,7 @@ func NewTxnWithMultistore(t *testing.T) *MultiStoreTxn { result := &MultiStoreTxn{ Txn: txn, t: t, - MockRootstore: prepareRootStore(t), + MockRootstore: prepareRootstore(t), MockDatastore: prepareDataStore(t), MockHeadstore: prepareHeadStore(t), MockDAGstore: prepareDAGStore(t), @@ -83,7 +83,7 @@ func NewTxnWithMultistore(t *testing.T) *MultiStoreTxn { txn.EXPECT().Rootstore().Return(result.MockRootstore).Maybe() txn.EXPECT().Datastore().Return(result.MockDatastore).Maybe() txn.EXPECT().Headstore().Return(result.MockHeadstore).Maybe() - txn.EXPECT().DAGstore().Return(result.MockDAGstore).Maybe() + txn.EXPECT().Blockstore().Return(result.MockDAGstore).Maybe() txn.EXPECT().Systemstore().Return(result.MockSystemstore).Maybe() return result diff --git a/datastore/multi.go b/datastore/multi.go index bbd333ba19..a70a24a60d 100644 --- a/datastore/multi.go +++ b/datastore/multi.go @@ -32,7 +32,7 @@ type multistore struct { peer DSBatching system DSReaderWriter // block DSReaderWriter - dag DAGStore + dag Blockstore } var _ MultiStore = (*multistore)(nil) @@ -67,8 +67,8 @@ func (ms multistore) Peerstore() DSBatching { return ms.peer } -// DAGstore implements MultiStore. -func (ms multistore) DAGstore() DAGStore { +// Blockstore implements MultiStore. +func (ms multistore) Blockstore() Blockstore { return ms.dag } diff --git a/datastore/store.go b/datastore/store.go index 7954eb5014..66501270d1 100644 --- a/datastore/store.go +++ b/datastore/store.go @@ -24,8 +24,8 @@ var ( log = corelog.NewLogger("store") ) -// RootStore wraps Batching and TxnDatastore requiring datastore to support both batching and transactions. -type RootStore interface { +// Rootstore wraps Batching and TxnDatastore requiring datastore to support both batching and transactions. +type Rootstore interface { ds.Batching ds.TxnDatastore } @@ -47,10 +47,10 @@ type MultiStore interface { // under the /peers namespace Peerstore() DSBatching - // DAGstore is a wrapped root DSReaderWriter - // as a Blockstore, embedded into a DAGStore + // Blockstore is a wrapped root DSReaderWriter + // as a Blockstore, embedded into a Blockstore // under the /blocks namespace - DAGstore() DAGStore + Blockstore() Blockstore // Headstore is a wrapped root DSReaderWriter // under the /system namespace @@ -70,8 +70,8 @@ type DSReaderWriter interface { iterable.Iterable } -// DAGStore proxies the ipld.DAGService under the /core namespace for future-proofing -type DAGStore interface { +// Blockstore proxies the ipld.DAGService under the /core namespace for future-proofing +type Blockstore interface { blockstore.Blockstore AsIPLDStorage() IPLDStorage } diff --git a/event/event.go b/event/event.go index e9afdf1a57..fa557cc03c 100644 --- a/event/event.go +++ b/event/event.go @@ -32,6 +32,16 @@ const ( PubSubName = Name("pubsub") // PeerName is the name of the network connect event. PeerName = Name("peer") + // P2PTopicName is the name of the network p2p topic update event. + P2PTopicName = Name("p2p-topic") + // PeerInfoName is the name of the network peer info event. + PeerInfoName = Name("peer-info") + // ReplicatorName is the name of the replicator event. + ReplicatorName = Name("replicator") + // P2PTopicCompletedName is the name of the network p2p topic update completed event. + P2PTopicCompletedName = Name("p2p-topic-completed") + // ReplicatorCompletedName is the name of the replicator completed event. + ReplicatorCompletedName = Name("replicator-completed") ) // Peer is an event that is published when @@ -110,3 +120,25 @@ type Subscription struct { func (s *Subscription) Message() <-chan Message { return s.value } + +// P2PTopic is an event that is published when a peer has updated the topics it is subscribed to. +type P2PTopic struct { + ToAdd []string + ToRemove []string +} + +// PeerInfo is an event that is published when the node has updated its peer info. +type PeerInfo struct { + Info peer.AddrInfo +} + +// Replicator is an event that is published when a replicator is added or updated. +type Replicator struct { + // The peer info for the replicator instance. + Info peer.AddrInfo + // The map of schema roots that the replicator will receive updates for. + Schemas map[string]struct{} + // Docs will receive Updates if new collections have been added to the replicator + // and those collections have documents to be replicated. + Docs <-chan Update +} diff --git a/go.mod b/go.mod index 6efc0e1590..350e183739 100644 --- a/go.mod +++ b/go.mod @@ -33,6 +33,7 @@ require ( github.com/libp2p/go-libp2p-kad-dht v0.25.2 github.com/libp2p/go-libp2p-pubsub v0.11.0 github.com/libp2p/go-libp2p-record v0.2.0 + github.com/mr-tron/base58 v1.2.0 github.com/multiformats/go-multiaddr v0.12.4 github.com/multiformats/go-multibase v0.2.0 github.com/multiformats/go-multicodec v0.9.0 @@ -191,7 +192,6 @@ require ( github.com/minio/sha256-simd v1.0.1 // indirect github.com/mitchellh/mapstructure v1.5.0 // indirect github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826 // indirect - github.com/mr-tron/base58 v1.2.0 // indirect github.com/multiformats/go-base32 v0.1.0 // indirect github.com/multiformats/go-base36 v0.2.0 // indirect github.com/multiformats/go-multiaddr-dns v0.3.1 // indirect diff --git a/http/client.go b/http/client.go index 2843ee4f2d..2082604599 100644 --- a/http/client.go +++ b/http/client.go @@ -435,11 +435,11 @@ func (c *Client) Close() { // do nothing } -func (c *Client) Root() datastore.RootStore { +func (c *Client) Rootstore() datastore.Rootstore { panic("client side database") } -func (c *Client) Blockstore() datastore.DAGStore { +func (c *Client) Blockstore() datastore.Blockstore { panic("client side database") } diff --git a/http/client_tx.go b/http/client_tx.go index 19e5814b51..a804b934f1 100644 --- a/http/client_tx.go +++ b/http/client_tx.go @@ -99,7 +99,7 @@ func (c *Transaction) Peerstore() datastore.DSBatching { panic("client side transaction") } -func (c *Transaction) DAGstore() datastore.DAGStore { +func (c *Transaction) Blockstore() datastore.Blockstore { panic("client side transaction") } diff --git a/internal/db/db.go b/internal/db/db.go index a6fb37f643..197ab493d5 100644 --- a/internal/db/db.go +++ b/internal/db/db.go @@ -55,7 +55,7 @@ const ( type db struct { glock sync.RWMutex - rootstore datastore.RootStore + rootstore datastore.Rootstore multistore datastore.MultiStore events *event.Bus @@ -75,12 +75,16 @@ type db struct { // Contains ACP if it exists acp immutable.Option[acp.ACP] + + // The peer ID and network address information for the current node + // if network is enabled. The `atomic.Value` should hold a `peer.AddrInfo` struct. + peerInfo atomic.Value } // NewDB creates a new instance of the DB using the given options. func NewDB( ctx context.Context, - rootstore datastore.RootStore, + rootstore datastore.Rootstore, acp immutable.Option[acp.ACP], lens client.LensRegistry, options ...Option, @@ -90,7 +94,7 @@ func NewDB( func newDB( ctx context.Context, - rootstore datastore.RootStore, + rootstore datastore.Rootstore, acp immutable.Option[acp.ACP], lens client.LensRegistry, options ...Option, @@ -126,11 +130,11 @@ func newDB( return nil, err } - sub, err := db.events.Subscribe(event.MergeName) + sub, err := db.events.Subscribe(event.MergeName, event.PeerInfoName) if err != nil { return nil, err } - go db.handleMerges(ctx, sub) + go db.handleMessages(ctx, sub) return db, nil } @@ -147,14 +151,14 @@ func (db *db) NewConcurrentTxn(ctx context.Context, readonly bool) (datastore.Tx return datastore.NewConcurrentTxnFrom(ctx, db.rootstore, txnId, readonly) } -// Root returns the root datastore. -func (db *db) Root() datastore.RootStore { +// Rootstore returns the root datastore. +func (db *db) Rootstore() datastore.Rootstore { return db.rootstore } // Blockstore returns the internal DAG store which contains IPLD blocks. -func (db *db) Blockstore() datastore.DAGStore { - return db.multistore.DAGstore() +func (db *db) Blockstore() datastore.Blockstore { + return db.multistore.Blockstore() } // Peerstore returns the internal DAG store which contains IPLD blocks. diff --git a/internal/db/errors.go b/internal/db/errors.go index 7a81824efe..603cee8130 100644 --- a/internal/db/errors.go +++ b/internal/db/errors.go @@ -11,6 +11,8 @@ package db import ( + "github.com/libp2p/go-libp2p/core/peer" + "github.com/sourcenetwork/defradb/client" "github.com/sourcenetwork/defradb/errors" ) @@ -94,6 +96,10 @@ const ( errSecondaryFieldOnSchema string = "secondary relation fields cannot be defined on the schema" errRelationMissingField string = "relation missing field" errNoTransactionInContext string = "no transaction in context" + errReplicatorExists string = "replicator already exists for %s with peerID %s" + errReplicatorDocID string = "failed to get docID for replicator" + errReplicatorCollections string = "failed to get collections for replicator" + errReplicatorNotFound string = "replicator not found" ) var ( @@ -127,7 +133,13 @@ var ( ErrSecondaryFieldOnSchema = errors.New(errSecondaryFieldOnSchema) ErrRelationMissingField = errors.New(errRelationMissingField) ErrMultipleRelationPrimaries = errors.New("relation can only have a single field set as primary") + ErrP2PColHasPolicy = errors.New("p2p collection specified has a policy on it") ErrNoTransactionInContext = errors.New(errNoTransactionInContext) + ErrReplicatorColHasPolicy = errors.New("replicator collection specified has a policy on it") + ErrReplicatorSomeColsHavePolicy = errors.New("replicator can not use all collections, as some have policy") + ErrSelfTargetForReplicator = errors.New("can't target ourselves as a replicator") + ErrReplicatorCollections = errors.New(errReplicatorCollections) + ErrReplicatorNotFound = errors.New(errReplicatorNotFound) ) // NewErrFailedToGetHeads returns a new error indicating that the heads of a document @@ -617,3 +629,19 @@ func NewErrRelationMissingField(objectName, relationName string) error { errors.NewKV("RelationName", relationName), ) } + +func NewErrReplicatorExists(collection string, peerID peer.ID) error { + return errors.New( + errReplicatorExists, + errors.NewKV("Collection", collection), + errors.NewKV("PeerID", peerID.String()), + ) +} + +func NewErrReplicatorDocID(inner error, kv ...errors.KV) error { + return errors.Wrap(errReplicatorDocID, inner, kv...) +} + +func NewErrReplicatorCollections(inner error, kv ...errors.KV) error { + return errors.Wrap(errReplicatorCollections, inner, kv...) +} diff --git a/internal/db/fetcher/versioned.go b/internal/db/fetcher/versioned.go index 6ce8f94ebc..892b84e329 100644 --- a/internal/db/fetcher/versioned.go +++ b/internal/db/fetcher/versioned.go @@ -86,7 +86,7 @@ type VersionedFetcher struct { ctx context.Context // Transient version store - root datastore.RootStore + root datastore.Rootstore store datastore.Txn dsKey core.DataStoreKey @@ -281,7 +281,7 @@ func (vf *VersionedFetcher) seekNext(c cid.Cid, topParent bool) error { // @body: We could possibly append the DocID to the CID either as a // child key, or an instance on the CID key. - hasLocalBlock, err := vf.store.DAGstore().Has(vf.ctx, c) + hasLocalBlock, err := vf.store.Blockstore().Has(vf.ctx, c) if err != nil { return NewErrVFetcherFailedToFindBlock(err) } @@ -290,13 +290,13 @@ func (vf *VersionedFetcher) seekNext(c cid.Cid, topParent bool) error { return nil } - blk, err := vf.txn.DAGstore().Get(vf.ctx, c) + blk, err := vf.txn.Blockstore().Get(vf.ctx, c) if err != nil { return NewErrVFetcherFailedToGetBlock(err) } // store the block in the local (transient store) - if err := vf.store.DAGstore().Put(vf.ctx, blk); err != nil { + if err := vf.store.Blockstore().Put(vf.ctx, blk); err != nil { return NewErrVFetcherFailedToWriteBlock(err) } @@ -336,7 +336,7 @@ func (vf *VersionedFetcher) seekNext(c cid.Cid, topParent bool) error { } // merge in the state of the IPLD Block identified by CID c into the VersionedFetcher state. -// Requires the CID to already exist in the DAGStore. +// Requires the CID to already exist in the Blockstore. // This function only works for merging Composite MerkleCRDT objects. // // First it checks for the existence of the block, @@ -421,7 +421,7 @@ func (vf *VersionedFetcher) processBlock( func (vf *VersionedFetcher) getDAGBlock(c cid.Cid) (*coreblock.Block, error) { // get Block - blk, err := vf.store.DAGstore().Get(vf.ctx, c) + blk, err := vf.store.Blockstore().Get(vf.ctx, c) if err != nil { return nil, NewErrFailedToGetDagNode(err) } diff --git a/internal/db/index_test.go b/internal/db/index_test.go index b7f4bbfb96..9226f92efd 100644 --- a/internal/db/index_test.go +++ b/internal/db/index_test.go @@ -473,11 +473,11 @@ func TestCreateIndex_IfFailsToCreateTxn_ReturnError(t *testing.T) { testErr := errors.New("test error") - mockedRootStore := mocks.NewRootStore(t) - mockedRootStore.On("Close").Return(nil) + mockedRootstore := mocks.NewRootstore(t) + mockedRootstore.On("Close").Return(nil) - mockedRootStore.EXPECT().NewTransaction(mock.Anything, mock.Anything).Return(nil, testErr) - f.db.rootstore = mockedRootStore + mockedRootstore.EXPECT().NewTransaction(mock.Anything, mock.Anything).Return(nil, testErr) + f.db.rootstore = mockedRootstore _, err := f.users.CreateIndex(f.ctx, getUsersIndexDescOnName()) require.ErrorIs(t, err, testErr) @@ -859,15 +859,15 @@ func TestCollectionGetIndexes_IfFailsToCreateTxn_ShouldNotCache(t *testing.T) { testErr := errors.New("test error") - workingRootStore := f.db.rootstore - mockedRootStore := mocks.NewRootStore(t) - f.db.rootstore = mockedRootStore - mockedRootStore.EXPECT().NewTransaction(mock.Anything, mock.Anything).Return(nil, testErr) + workingRootstore := f.db.rootstore + mockedRootstore := mocks.NewRootstore(t) + f.db.rootstore = mockedRootstore + mockedRootstore.EXPECT().NewTransaction(mock.Anything, mock.Anything).Return(nil, testErr) _, err := f.users.GetIndexes(f.ctx) require.ErrorIs(t, err, testErr) - f.db.rootstore = workingRootStore + f.db.rootstore = workingRootstore indexes, err := f.users.GetIndexes(f.ctx) require.NoError(t, err) @@ -1075,11 +1075,11 @@ func TestDropIndex_IfFailsToCreateTxn_ReturnError(t *testing.T) { testErr := errors.New("test error") - mockedRootStore := mocks.NewRootStore(t) - mockedRootStore.On("Close").Return(nil) + mockedRootstore := mocks.NewRootstore(t) + mockedRootstore.On("Close").Return(nil) - mockedRootStore.EXPECT().NewTransaction(mock.Anything, mock.Anything).Return(nil, testErr) - f.db.rootstore = mockedRootStore + mockedRootstore.EXPECT().NewTransaction(mock.Anything, mock.Anything).Return(nil, testErr) + f.db.rootstore = mockedRootstore err := f.users.DropIndex(f.ctx, testUsersColIndexName) require.ErrorIs(t, err, testErr) diff --git a/internal/db/merge.go b/internal/db/merge.go index e7d4c8252c..bbfedd98d8 100644 --- a/internal/db/merge.go +++ b/internal/db/merge.go @@ -19,12 +19,10 @@ import ( "github.com/ipld/go-ipld-prime/linking" cidlink "github.com/ipld/go-ipld-prime/linking/cid" - "github.com/sourcenetwork/corelog" "github.com/sourcenetwork/immutable" "github.com/sourcenetwork/defradb/client" "github.com/sourcenetwork/defradb/datastore" - "github.com/sourcenetwork/defradb/datastore/badger/v4" "github.com/sourcenetwork/defradb/errors" "github.com/sourcenetwork/defradb/event" "github.com/sourcenetwork/defradb/internal/core" @@ -34,50 +32,6 @@ import ( merklecrdt "github.com/sourcenetwork/defradb/internal/merkle/crdt" ) -func (db *db) handleMerges(ctx context.Context, sub *event.Subscription) { - queue := newMergeQueue() - for { - select { - case <-ctx.Done(): - return - case msg, ok := <-sub.Message(): - if !ok { - return - } - merge, ok := msg.Data.(event.Merge) - if !ok { - continue - } - go func() { - // ensure only one merge per docID - queue.add(merge.DocID) - defer queue.done(merge.DocID) - - // retry the merge process if a conflict occurs - // - // conficts occur when a user updates a document - // while a merge is in progress. - var err error - for i := 0; i < db.MaxTxnRetries(); i++ { - err = db.executeMerge(ctx, merge) - if errors.Is(err, badger.ErrTxnConflict) { - continue // retry merge - } - break // merge success or error - } - - if err != nil { - log.ErrorContextE( - ctx, - "Failed to execute merge", - err, - corelog.Any("Event", merge)) - } - }() - } - } -} - func (db *db) executeMerge(ctx context.Context, dagMerge event.Merge) error { ctx, txn, err := ensureContextTxn(ctx, db, false) if err != nil { @@ -91,7 +45,7 @@ func (db *db) executeMerge(ctx context.Context, dagMerge event.Merge) error { } ls := cidlink.DefaultLinkSystem() - ls.SetReadStorage(txn.DAGstore().AsIPLDStorage()) + ls.SetReadStorage(txn.Blockstore().AsIPLDStorage()) docID, err := client.NewDocIDFromString(dagMerge.DocID) if err != nil { @@ -409,7 +363,7 @@ func getHeadsAsMergeTarget(ctx context.Context, txn datastore.Txn, dsKey core.Da mt := newMergeTarget() for _, cid := range cids { - b, err := txn.DAGstore().Get(ctx, cid) + b, err := txn.Blockstore().Get(ctx, cid) if err != nil { return mergeTarget{}, err } diff --git a/internal/db/merge_test.go b/internal/db/merge_test.go index f620003bbe..a78fd59983 100644 --- a/internal/db/merge_test.go +++ b/internal/db/merge_test.go @@ -48,7 +48,7 @@ func TestMerge_SingleBranch_NoError(t *testing.T) { require.NoError(t, err) lsys := cidlink.DefaultLinkSystem() - lsys.SetWriteStorage(db.multistore.DAGstore().AsIPLDStorage()) + lsys.SetWriteStorage(db.multistore.Blockstore().AsIPLDStorage()) initialDocState := map[string]any{ "name": "John", @@ -93,7 +93,7 @@ func TestMerge_DualBranch_NoError(t *testing.T) { require.NoError(t, err) lsys := cidlink.DefaultLinkSystem() - lsys.SetWriteStorage(db.multistore.DAGstore().AsIPLDStorage()) + lsys.SetWriteStorage(db.multistore.Blockstore().AsIPLDStorage()) initialDocState := map[string]any{ "name": "John", @@ -151,7 +151,7 @@ func TestMerge_DualBranchWithOneIncomplete_CouldNotFindCID(t *testing.T) { require.NoError(t, err) lsys := cidlink.DefaultLinkSystem() - lsys.SetWriteStorage(db.multistore.DAGstore().AsIPLDStorage()) + lsys.SetWriteStorage(db.multistore.Blockstore().AsIPLDStorage()) initialDocState := map[string]any{ "name": "John", diff --git a/internal/db/messages.go b/internal/db/messages.go new file mode 100644 index 0000000000..1967c0e238 --- /dev/null +++ b/internal/db/messages.go @@ -0,0 +1,84 @@ +// Copyright 2024 Democratized Data Foundation +// +// Use of this software is governed by the Business Source License +// included in the file licenses/BSL.txt. +// +// As of the Change Date specified in that file, in accordance with +// the Business Source License, use of this software will be governed +// by the Apache License, Version 2.0, included in the file +// licenses/APL.txt. + +package db + +import ( + "context" + "sync" + + "github.com/sourcenetwork/corelog" + + "github.com/sourcenetwork/defradb/datastore/badger/v4" + "github.com/sourcenetwork/defradb/errors" + "github.com/sourcenetwork/defradb/event" +) + +func (db *db) handleMessages(ctx context.Context, sub *event.Subscription) { + queue := newMergeQueue() + // This is used to ensure we only trigger loadAndPublishP2PCollections and loadAndPublishReplicators + // once per db instanciation. + loadOnce := sync.Once{} + for { + select { + case <-ctx.Done(): + return + case msg, ok := <-sub.Message(): + if !ok { + return + } + switch evt := msg.Data.(type) { + case event.Merge: + go func() { + // ensure only one merge per docID + queue.add(evt.DocID) + defer queue.done(evt.DocID) + + // retry the merge process if a conflict occurs + // + // conficts occur when a user updates a document + // while a merge is in progress. + var err error + for i := 0; i < db.MaxTxnRetries(); i++ { + err = db.executeMerge(ctx, evt) + if errors.Is(err, badger.ErrTxnConflict) { + continue // retry merge + } + break // merge success or error + } + + if err != nil { + log.ErrorContextE( + ctx, + "Failed to execute merge", + err, + corelog.Any("Event", evt)) + } + }() + case event.PeerInfo: + db.peerInfo.Store(evt.Info) + // Load and publish P2P collections and replicators once per db instance start. + // A Go routine is used to ensure the message handler is not blocked by these potentially + // long running operations. + go loadOnce.Do(func() { + err := db.loadAndPublishP2PCollections(ctx) + if err != nil { + log.ErrorContextE(ctx, "Failed to load P2P collections", err) + } + + err = db.loadAndPublishReplicators(ctx) + if err != nil { + log.ErrorContextE(ctx, "Failed to load replicators", err) + } + }) + } + } + } +} diff --git a/internal/db/p2p_replicator.go b/internal/db/p2p_replicator.go new file mode 100644 index 0000000000..9f2fd84215 --- /dev/null +++ b/internal/db/p2p_replicator.go @@ -0,0 +1,346 @@ +// Copyright 2024 Democratized Data Foundation +// +// Use of this software is governed by the Business Source License +// included in the file licenses/BSL.txt. +// +// As of the Change Date specified in that file, in accordance with +// the Business Source License, use of this software will be governed +// by the Apache License, Version 2.0, included in the file +// licenses/APL.txt. + +package db + +import ( + "context" + "encoding/json" + + dsq "github.com/ipfs/go-datastore/query" + "github.com/libp2p/go-libp2p/core/peer" + + "github.com/sourcenetwork/corelog" + + "github.com/sourcenetwork/defradb/client" + "github.com/sourcenetwork/defradb/errors" + "github.com/sourcenetwork/defradb/event" + "github.com/sourcenetwork/defradb/internal/core" + "github.com/sourcenetwork/defradb/internal/merkle/clock" +) + +func (db *db) SetReplicator(ctx context.Context, rep client.Replicator) error { + txn, err := db.NewTxn(ctx, false) + if err != nil { + return err + } + defer txn.Discard(ctx) + + if err := rep.Info.ID.Validate(); err != nil { + return err + } + + peerInfo := peer.AddrInfo{} + if info := db.peerInfo.Load(); info != nil { + peerInfo = info.(peer.AddrInfo) + } + if rep.Info.ID == peerInfo.ID { + return ErrSelfTargetForReplicator + } + + // TODO-ACP: Support ACP <> P2P - https://github.com/sourcenetwork/defradb/issues/2366 + // ctx = db.SetContextIdentity(ctx, identity) + ctx = SetContextTxn(ctx, txn) + + storedRep := client.Replicator{} + storedSchemas := make(map[string]struct{}) + repKey := core.NewReplicatorKey(rep.Info.ID.String()) + hasOldRep, err := txn.Systemstore().Has(ctx, repKey.ToDS()) + if err != nil { + return err + } + if hasOldRep { + repBytes, err := txn.Systemstore().Get(ctx, repKey.ToDS()) + if err != nil { + return err + } + err = json.Unmarshal(repBytes, &storedRep) + if err != nil { + return err + } + for _, schema := range storedRep.Schemas { + storedSchemas[schema] = struct{}{} + } + } else { + storedRep.Info = rep.Info + } + + var collections []client.Collection + switch { + case len(rep.Schemas) > 0: + // if specific collections are chosen get them by name + for _, name := range rep.Schemas { + col, err := db.GetCollectionByName(ctx, name) + if err != nil { + return NewErrReplicatorCollections(err) + } + + if col.Description().Policy.HasValue() { + return ErrReplicatorColHasPolicy + } + + collections = append(collections, col) + } + + default: + // default to all collections (unless a collection contains a policy). + // TODO-ACP: default to all collections after resolving https://github.com/sourcenetwork/defradb/issues/2366 + allCollections, err := db.GetCollections(ctx, client.CollectionFetchOptions{}) + if err != nil { + return NewErrReplicatorCollections(err) + } + + for _, col := range allCollections { + // Can not default to all collections if any collection has a policy. + // TODO-ACP: remove this check/loop after https://github.com/sourcenetwork/defradb/issues/2366 + if col.Description().Policy.HasValue() { + return ErrReplicatorSomeColsHavePolicy + } + } + collections = allCollections + } + + addedCols := []client.Collection{} + for _, col := range collections { + if _, ok := storedSchemas[col.SchemaRoot()]; !ok { + storedSchemas[col.SchemaRoot()] = struct{}{} + addedCols = append(addedCols, col) + storedRep.Schemas = append(storedRep.Schemas, col.SchemaRoot()) + } + } + + // persist replicator to the datastore + newRepBytes, err := json.Marshal(storedRep) + if err != nil { + return err + } + + err = txn.Systemstore().Put(ctx, repKey.ToDS(), newRepBytes) + if err != nil { + return err + } + + txn.OnSuccess(func() { + db.events.Publish(event.NewMessage(event.ReplicatorName, event.Replicator{ + Info: rep.Info, + Schemas: storedSchemas, + Docs: db.getDocsHeads(context.Background(), addedCols), + })) + }) + + return txn.Commit(ctx) +} + +func (db *db) getDocsHeads( + ctx context.Context, + cols []client.Collection, +) <-chan event.Update { + updateChan := make(chan event.Update) + go func() { + defer close(updateChan) + txn, err := db.NewTxn(ctx, true) + if err != nil { + log.ErrorContextE(ctx, "Failed to get transaction", err) + return + } + defer txn.Discard(ctx) + ctx = SetContextTxn(ctx, txn) + for _, col := range cols { + keysCh, err := col.GetAllDocIDs(ctx) + if err != nil { + log.ErrorContextE( + ctx, + "Failed to get all docIDs", + NewErrReplicatorDocID(err, errors.NewKV("Collection", col.Name().Value())), + ) + continue + } + for docIDResult := range keysCh { + if docIDResult.Err != nil { + log.ErrorContextE(ctx, "Key channel error", docIDResult.Err) + continue + } + docID := core.DataStoreKeyFromDocID(docIDResult.ID) + headset := clock.NewHeadSet( + txn.Headstore(), + docID.WithFieldId(core.COMPOSITE_NAMESPACE).ToHeadStoreKey(), + ) + cids, _, err := headset.List(ctx) + if err != nil { + log.ErrorContextE( + ctx, + "Failed to get heads", + err, + corelog.String("DocID", docIDResult.ID.String()), + corelog.Any("Collection", col.Name())) + continue + } + // loop over heads, get block, make the required logs, and send + for _, c := range cids { + blk, err := txn.Blockstore().Get(ctx, c) + if err != nil { + log.ErrorContextE(ctx, "Failed to get block", err, + corelog.Any("CID", c), + corelog.Any("Collection", col.Name())) + continue + } + + updateChan <- event.Update{ + DocID: docIDResult.ID.String(), + Cid: c, + SchemaRoot: col.SchemaRoot(), + Block: blk.RawData(), + } + } + } + } + }() + + return updateChan +} + +func (db *db) DeleteReplicator(ctx context.Context, rep client.Replicator) error { + txn, err := db.NewTxn(ctx, false) + if err != nil { + return err + } + defer txn.Discard(ctx) + + if err := rep.Info.ID.Validate(); err != nil { + return err + } + + // set transaction for all operations + ctx = SetContextTxn(ctx, txn) + + storedRep := client.Replicator{} + storedSchemas := make(map[string]struct{}) + repKey := core.NewReplicatorKey(rep.Info.ID.String()) + hasOldRep, err := txn.Systemstore().Has(ctx, repKey.ToDS()) + if err != nil { + return err + } + if !hasOldRep { + return ErrReplicatorNotFound + } + repBytes, err := txn.Systemstore().Get(ctx, repKey.ToDS()) + if err != nil { + return err + } + err = json.Unmarshal(repBytes, &storedRep) + if err != nil { + return err + } + for _, schema := range storedRep.Schemas { + storedSchemas[schema] = struct{}{} + } + + var collections []client.Collection + if len(rep.Schemas) > 0 { + // if specific collections are chosen get them by name + for _, name := range rep.Schemas { + col, err := db.GetCollectionByName(ctx, name) + if err != nil { + return NewErrReplicatorCollections(err) + } + collections = append(collections, col) + } + // make sure the replicator exists in the datastore + key := core.NewReplicatorKey(rep.Info.ID.String()) + _, err = txn.Systemstore().Get(ctx, key.ToDS()) + if err != nil { + return err + } + } else { + storedSchemas = make(map[string]struct{}) + } + + for _, col := range collections { + delete(storedSchemas, col.SchemaRoot()) + } + // Update the list of schemas for this replicator prior to persisting. + storedRep.Schemas = []string{} + for schema := range storedSchemas { + storedRep.Schemas = append(storedRep.Schemas, schema) + } + + // Persist the replicator to the store, deleting it if no schemas remain + key := core.NewReplicatorKey(rep.Info.ID.String()) + if len(rep.Schemas) == 0 { + err := txn.Systemstore().Delete(ctx, key.ToDS()) + if err != nil { + return err + } + } else { + repBytes, err := json.Marshal(rep) + if err != nil { + return err + } + err = txn.Systemstore().Put(ctx, key.ToDS(), repBytes) + if err != nil { + return err + } + } + + txn.OnSuccess(func() { + db.events.Publish(event.NewMessage(event.ReplicatorName, event.Replicator{ + Info: rep.Info, + Schemas: storedSchemas, + })) + }) + + return txn.Commit(ctx) +} + +func (db *db) GetAllReplicators(ctx context.Context) ([]client.Replicator, error) { + txn, err := db.NewTxn(ctx, true) + if err != nil { + return nil, err + } + defer txn.Discard(ctx) + + // create collection system prefix query + query := dsq.Query{ + Prefix: core.NewReplicatorKey("").ToString(), + } + results, err := txn.Systemstore().Query(ctx, query) + if err != nil { + return nil, err + } + + var reps []client.Replicator + for result := range results.Next() { + var rep client.Replicator + if err = json.Unmarshal(result.Value, &rep); err != nil { + return nil, err + } + reps = append(reps, rep) + } + return reps, nil +} + +func (db *db) loadAndPublishReplicators(ctx context.Context) error { + replicators, err := db.GetAllReplicators(ctx) + if err != nil { + return err + } + + for _, rep := range replicators { + schemaMap := make(map[string]struct{}) + for _, schema := range rep.Schemas { + schemaMap[schema] = struct{}{} + } + db.events.Publish(event.NewMessage(event.ReplicatorName, event.Replicator{ + Info: rep.Info, + Schemas: schemaMap, + })) + } + return nil +} diff --git a/internal/db/p2p_replicator_test.go b/internal/db/p2p_replicator_test.go new file mode 100644 index 0000000000..b287101a54 --- /dev/null +++ b/internal/db/p2p_replicator_test.go @@ -0,0 +1,320 @@ +// Copyright 2024 Democratized Data Foundation +// +// Use of this software is governed by the Business Source License +// included in the file licenses/BSL.txt. +// +// As of the Change Date specified in that file, in accordance with +// the Business Source License, use of this software will be governed +// by the Apache License, Version 2.0, included in the file +// licenses/APL.txt. + +package db + +import ( + "context" + "testing" + "time" + + "github.com/libp2p/go-libp2p/core/peer" + b58 "github.com/mr-tron/base58/base58" + "github.com/stretchr/testify/require" + + "github.com/sourcenetwork/defradb/client" + "github.com/sourcenetwork/defradb/event" +) + +func waitForPeerInfo(db *db, sub *event.Subscription) { + for msg := range sub.Message() { + if msg.Name == event.PeerInfoName { + hasPeerInfo := false + if db.peerInfo.Load() != nil { + hasPeerInfo = true + } + if !hasPeerInfo { + time.Sleep(1 * time.Millisecond) + } + break + } + } +} + +func TestSetReplicator_WithEmptyPeerInfo_ShouldError(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + err = db.SetReplicator(ctx, client.Replicator{}) + require.ErrorContains(t, err, "empty peer ID") +} + +func TestSetReplicator_WithSelfTarget_ShouldError(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.PeerInfoName) + require.NoError(t, err) + db.events.Publish(event.NewMessage(event.PeerInfoName, event.PeerInfo{Info: peer.AddrInfo{ID: "self"}})) + waitForPeerInfo(db, sub) + err = db.SetReplicator(ctx, client.Replicator{Info: peer.AddrInfo{ID: "self"}}) + require.ErrorIs(t, err, ErrSelfTargetForReplicator) +} + +func TestSetReplicator_WithInvalidCollection_ShouldError(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.PeerInfoName) + require.NoError(t, err) + db.events.Publish(event.NewMessage(event.PeerInfoName, event.PeerInfo{Info: peer.AddrInfo{ID: "self"}})) + waitForPeerInfo(db, sub) + err = db.SetReplicator(ctx, client.Replicator{ + Info: peer.AddrInfo{ID: "other"}, + Schemas: []string{"invalidCollection"}, + }) + require.ErrorIs(t, err, ErrReplicatorCollections) +} + +func TestSetReplicator_WithValidCollection_ShouldSucceed(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.ReplicatorName) + require.NoError(t, err) + cols, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema, err := db.GetSchemaByVersionID(ctx, cols[0].SchemaVersionID) + require.NoError(t, err) + err = db.SetReplicator(ctx, client.Replicator{ + Info: peer.AddrInfo{ID: "other"}, + Schemas: []string{"User"}, + }) + require.NoError(t, err) + msg := <-sub.Message() + replicator := msg.Data.(event.Replicator) + require.Equal(t, peer.ID("other"), replicator.Info.ID) + require.Equal(t, map[string]struct{}{schema.Root: {}}, replicator.Schemas) +} + +func TestSetReplicator_WithValidCollectionsOnSeparateSet_ShouldSucceed(t *testing.T) { + b, err := b58.Decode("12D3KooWB8Na2fKhdGtej5GjoVhmBBYFvqXiqFCSkR7fJFWHUbNr") + require.NoError(t, err) + peerID, err := peer.IDFromBytes(b) + require.NoError(t, err) + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.ReplicatorName) + require.NoError(t, err) + cols1, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema1, err := db.GetSchemaByVersionID(ctx, cols1[0].SchemaVersionID) + require.NoError(t, err) + err = db.SetReplicator(ctx, client.Replicator{ + Info: peer.AddrInfo{ID: peerID}, + Schemas: []string{"User"}, + }) + require.NoError(t, err) + msg := <-sub.Message() + replicator := msg.Data.(event.Replicator) + require.Equal(t, peerID, replicator.Info.ID) + require.Equal(t, map[string]struct{}{schema1.Root: {}}, replicator.Schemas) + + cols2, err := db.AddSchema(ctx, `type Book { name: String }`) + require.NoError(t, err) + schema2, err := db.GetSchemaByVersionID(ctx, cols2[0].SchemaVersionID) + require.NoError(t, err) + err = db.SetReplicator(ctx, client.Replicator{ + Info: peer.AddrInfo{ID: peerID}, + Schemas: []string{"Book"}, + }) + require.NoError(t, err) + msg = <-sub.Message() + replicator = msg.Data.(event.Replicator) + require.Equal(t, peerID, replicator.Info.ID) + require.Equal(t, map[string]struct{}{schema1.Root: {}, schema2.Root: {}}, replicator.Schemas) +} + +func TestSetReplicator_WithValidCollectionWithDoc_ShouldSucceed(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.ReplicatorName) + require.NoError(t, err) + cols, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + col, err := db.GetCollectionByName(ctx, cols[0].Name.Value()) + require.NoError(t, err) + doc, err := client.NewDocFromMap(map[string]any{"name": "Alice"}, col.Definition()) + require.NoError(t, err) + err = col.Create(ctx, doc) + require.NoError(t, err) + + err = db.SetReplicator(ctx, client.Replicator{ + Info: peer.AddrInfo{ID: "other"}, + Schemas: []string{"User"}, + }) + require.NoError(t, err) + msg := <-sub.Message() + replicator := msg.Data.(event.Replicator) + require.Equal(t, peer.ID("other"), replicator.Info.ID) + require.Equal(t, map[string]struct{}{col.SchemaRoot(): {}}, replicator.Schemas) + for docEvt := range replicator.Docs { + require.Equal(t, doc.ID().String(), docEvt.DocID) + } +} + +func TestDeleteReplicator_WithEmptyPeerInfo_ShouldError(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + err = db.DeleteReplicator(ctx, client.Replicator{}) + require.ErrorContains(t, err, "empty peer ID") +} + +func TestDeleteReplicator_WithNonExistantReplicator_ShouldError(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + err = db.DeleteReplicator(ctx, client.Replicator{Info: peer.AddrInfo{ID: "other"}}) + require.ErrorIs(t, err, ErrReplicatorNotFound) +} + +func TestDeleteReplicator_WithValidCollection_ShouldSucceed(t *testing.T) { + b, err := b58.Decode("12D3KooWB8Na2fKhdGtej5GjoVhmBBYFvqXiqFCSkR7fJFWHUbNr") + require.NoError(t, err) + peerID, err := peer.IDFromBytes(b) + require.NoError(t, err) + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.ReplicatorName) + require.NoError(t, err) + cols, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema, err := db.GetSchemaByVersionID(ctx, cols[0].SchemaVersionID) + require.NoError(t, err) + err = db.SetReplicator(ctx, client.Replicator{ + Info: peer.AddrInfo{ID: peerID}, + Schemas: []string{"User"}, + }) + require.NoError(t, err) + msg := <-sub.Message() + replicator := msg.Data.(event.Replicator) + require.Equal(t, peerID, replicator.Info.ID) + require.Equal(t, map[string]struct{}{schema.Root: {}}, replicator.Schemas) + err = db.DeleteReplicator(ctx, client.Replicator{Info: peer.AddrInfo{ID: peerID}}) + require.NoError(t, err) + msg = <-sub.Message() + replicator = msg.Data.(event.Replicator) + require.Equal(t, peerID, replicator.Info.ID) + require.Equal(t, map[string]struct{}{}, replicator.Schemas) +} + +func TestDeleteReplicator_PartialWithValidCollections_ShouldSucceed(t *testing.T) { + b, err := b58.Decode("12D3KooWB8Na2fKhdGtej5GjoVhmBBYFvqXiqFCSkR7fJFWHUbNr") + require.NoError(t, err) + peerID, err := peer.IDFromBytes(b) + require.NoError(t, err) + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.ReplicatorName) + require.NoError(t, err) + cols1, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema1, err := db.GetSchemaByVersionID(ctx, cols1[0].SchemaVersionID) + require.NoError(t, err) + cols2, err := db.AddSchema(ctx, `type Book { name: String }`) + require.NoError(t, err) + schema2, err := db.GetSchemaByVersionID(ctx, cols2[0].SchemaVersionID) + require.NoError(t, err) + err = db.SetReplicator(ctx, client.Replicator{ + Info: peer.AddrInfo{ID: peerID}, + Schemas: []string{"User", "Book"}, + }) + require.NoError(t, err) + msg := <-sub.Message() + replicator := msg.Data.(event.Replicator) + require.Equal(t, peerID, replicator.Info.ID) + require.Equal(t, map[string]struct{}{schema1.Root: {}, schema2.Root: {}}, replicator.Schemas) + + err = db.DeleteReplicator(ctx, client.Replicator{Info: peer.AddrInfo{ID: peerID}, Schemas: []string{"User"}}) + require.NoError(t, err) + msg = <-sub.Message() + replicator = msg.Data.(event.Replicator) + require.Equal(t, peerID, replicator.Info.ID) + require.Equal(t, map[string]struct{}{schema2.Root: {}}, replicator.Schemas) +} + +func TestGetAllReplicators_WithValidCollection_ShouldSucceed(t *testing.T) { + b, err := b58.Decode("12D3KooWB8Na2fKhdGtej5GjoVhmBBYFvqXiqFCSkR7fJFWHUbNr") + require.NoError(t, err) + peerID, err := peer.IDFromBytes(b) + require.NoError(t, err) + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.ReplicatorName) + require.NoError(t, err) + cols, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema, err := db.GetSchemaByVersionID(ctx, cols[0].SchemaVersionID) + require.NoError(t, err) + err = db.SetReplicator(ctx, client.Replicator{ + Info: peer.AddrInfo{ID: peerID}, + Schemas: []string{"User"}, + }) + require.NoError(t, err) + msg := <-sub.Message() + replicator := msg.Data.(event.Replicator) + require.Equal(t, peerID, replicator.Info.ID) + require.Equal(t, map[string]struct{}{schema.Root: {}}, replicator.Schemas) + + reps, err := db.GetAllReplicators(ctx) + require.NoError(t, err) + require.Equal(t, peerID, reps[0].Info.ID) + require.Equal(t, []string{schema.Root}, reps[0].Schemas) +} + +func TestLoadReplicators_WithValidCollection_ShouldSucceed(t *testing.T) { + b, err := b58.Decode("12D3KooWB8Na2fKhdGtej5GjoVhmBBYFvqXiqFCSkR7fJFWHUbNr") + require.NoError(t, err) + peerID, err := peer.IDFromBytes(b) + require.NoError(t, err) + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.ReplicatorName) + require.NoError(t, err) + cols, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema, err := db.GetSchemaByVersionID(ctx, cols[0].SchemaVersionID) + require.NoError(t, err) + err = db.SetReplicator(ctx, client.Replicator{ + Info: peer.AddrInfo{ID: peerID}, + Schemas: []string{"User"}, + }) + require.NoError(t, err) + msg := <-sub.Message() + replicator := msg.Data.(event.Replicator) + require.Equal(t, peerID, replicator.Info.ID) + require.Equal(t, map[string]struct{}{schema.Root: {}}, replicator.Schemas) + + err = db.loadAndPublishReplicators(ctx) + require.NoError(t, err) + msg = <-sub.Message() + replicator = msg.Data.(event.Replicator) + require.Equal(t, peerID, replicator.Info.ID) + require.Equal(t, map[string]struct{}{schema.Root: {}}, replicator.Schemas) +} diff --git a/net/peer_collection.go b/internal/db/p2p_schema_root.go similarity index 59% rename from net/peer_collection.go rename to internal/db/p2p_schema_root.go index 1676a7be43..b16af0ff4e 100644 --- a/net/peer_collection.go +++ b/internal/db/p2p_schema_root.go @@ -1,4 +1,4 @@ -// Copyright 2023 Democratized Data Foundation +// Copyright 2024 Democratized Data Foundation // // Use of this software is governed by the Business Source License // included in the file licenses/BSL.txt. @@ -8,23 +8,24 @@ // by the Apache License, Version 2.0, included in the file // licenses/APL.txt. -package net +package db import ( "context" dsq "github.com/ipfs/go-datastore/query" + "github.com/libp2p/go-libp2p/core/peer" "github.com/sourcenetwork/immutable" "github.com/sourcenetwork/defradb/client" + "github.com/sourcenetwork/defradb/event" "github.com/sourcenetwork/defradb/internal/core" - "github.com/sourcenetwork/defradb/internal/db" ) const marker = byte(0xff) -func (p *Peer) AddP2PCollections(ctx context.Context, collectionIDs []string) error { - txn, err := p.db.NewTxn(ctx, false) +func (db *db) AddP2PCollections(ctx context.Context, collectionIDs []string) error { + txn, err := db.NewTxn(ctx, false) if err != nil { return err } @@ -32,12 +33,12 @@ func (p *Peer) AddP2PCollections(ctx context.Context, collectionIDs []string) er // TODO-ACP: Support ACP <> P2P - https://github.com/sourcenetwork/defradb/issues/2366 // ctx = db.SetContextIdentity(ctx, identity) - ctx = db.SetContextTxn(ctx, txn) + ctx = SetContextTxn(ctx, txn) // first let's make sure the collections actually exists storeCollections := []client.Collection{} for _, col := range collectionIDs { - storeCol, err := p.db.GetCollections( + storeCol, err := db.GetCollections( ctx, client.CollectionFetchOptions{ SchemaRoot: immutable.Some(col), @@ -60,6 +61,8 @@ func (p *Peer) AddP2PCollections(ctx context.Context, collectionIDs []string) er } } + evt := event.P2PTopic{} + // Ensure we can add all the collections to the store on the transaction // before adding to topics. for _, col := range storeCollections { @@ -68,45 +71,28 @@ func (p *Peer) AddP2PCollections(ctx context.Context, collectionIDs []string) er if err != nil { return err } + evt.ToAdd = append(evt.ToAdd, col.SchemaRoot()) } - // Add pubsub topics and remove them if we get an error. - addedTopics := []string{} - for _, col := range collectionIDs { - err = p.server.addPubSubTopic(col, true) - if err != nil { - return p.rollbackAddPubSubTopics(addedTopics, err) - } - addedTopics = append(addedTopics, col) - } - - // After adding the collection topics, we remove the collections' documents - // from the pubsub topics to avoid receiving duplicate events. - removedTopics := []string{} for _, col := range storeCollections { keyChan, err := col.GetAllDocIDs(ctx) if err != nil { return err } for key := range keyChan { - err := p.server.removePubSubTopic(key.ID.String()) - if err != nil { - return p.rollbackRemovePubSubTopics(removedTopics, err) - } - removedTopics = append(removedTopics, key.ID.String()) + evt.ToRemove = append(evt.ToRemove, key.ID.String()) } } - if err = txn.Commit(ctx); err != nil { - err = p.rollbackRemovePubSubTopics(removedTopics, err) - return p.rollbackAddPubSubTopics(addedTopics, err) - } + txn.OnSuccess(func() { + db.events.Publish(event.NewMessage(event.P2PTopicName, evt)) + }) - return nil + return txn.Commit(ctx) } -func (p *Peer) RemoveP2PCollections(ctx context.Context, collectionIDs []string) error { - txn, err := p.db.NewTxn(ctx, false) +func (db *db) RemoveP2PCollections(ctx context.Context, collectionIDs []string) error { + txn, err := db.NewTxn(ctx, false) if err != nil { return err } @@ -114,12 +100,12 @@ func (p *Peer) RemoveP2PCollections(ctx context.Context, collectionIDs []string) // TODO-ACP: Support ACP <> P2P - https://github.com/sourcenetwork/defradb/issues/2366 // ctx = db.SetContextIdentity(ctx, identity) - ctx = db.SetContextTxn(ctx, txn) + ctx = SetContextTxn(ctx, txn) // first let's make sure the collections actually exists storeCollections := []client.Collection{} for _, col := range collectionIDs { - storeCol, err := p.db.GetCollections( + storeCol, err := db.GetCollections( ctx, client.CollectionFetchOptions{ SchemaRoot: immutable.Some(col), @@ -134,6 +120,8 @@ func (p *Peer) RemoveP2PCollections(ctx context.Context, collectionIDs []string) storeCollections = append(storeCollections, storeCol...) } + evt := event.P2PTopic{} + // Ensure we can remove all the collections to the store on the transaction // before adding to topics. for _, col := range storeCollections { @@ -142,49 +130,32 @@ func (p *Peer) RemoveP2PCollections(ctx context.Context, collectionIDs []string) if err != nil { return err } + evt.ToRemove = append(evt.ToRemove, col.SchemaRoot()) } - // Remove pubsub topics and add them back if we get an error. - removedTopics := []string{} - for _, col := range collectionIDs { - err = p.server.removePubSubTopic(col) - if err != nil { - return p.rollbackRemovePubSubTopics(removedTopics, err) - } - removedTopics = append(removedTopics, col) - } - - // After removing the collection topics, we add back the collections' documents - // to the pubsub topics. - addedTopics := []string{} for _, col := range storeCollections { keyChan, err := col.GetAllDocIDs(ctx) if err != nil { return err } for key := range keyChan { - err := p.server.addPubSubTopic(key.ID.String(), true) - if err != nil { - return p.rollbackAddPubSubTopics(addedTopics, err) - } - addedTopics = append(addedTopics, key.ID.String()) + evt.ToAdd = append(evt.ToAdd, key.ID.String()) } } - if err = txn.Commit(ctx); err != nil { - err = p.rollbackAddPubSubTopics(addedTopics, err) - return p.rollbackRemovePubSubTopics(removedTopics, err) - } + txn.OnSuccess(func() { + db.events.Publish(event.NewMessage(event.P2PTopicName, evt)) + }) - return nil + return txn.Commit(ctx) } -func (p *Peer) GetAllP2PCollections(ctx context.Context) ([]string, error) { - txn, err := p.db.NewTxn(p.ctx, true) +func (db *db) GetAllP2PCollections(ctx context.Context) ([]string, error) { + txn, err := db.NewTxn(ctx, true) if err != nil { return nil, err } - defer txn.Discard(p.ctx) + defer txn.Discard(ctx) query := dsq.Query{ Prefix: core.NewP2PCollectionKey("").ToString(), @@ -205,3 +176,52 @@ func (p *Peer) GetAllP2PCollections(ctx context.Context) ([]string, error) { return collectionIDs, nil } + +func (db *db) PeerInfo() peer.AddrInfo { + peerInfo := db.peerInfo.Load() + if peerInfo != nil { + return peerInfo.(peer.AddrInfo) + } + return peer.AddrInfo{} +} + +func (db *db) loadAndPublishP2PCollections(ctx context.Context) error { + schemaRoots, err := db.GetAllP2PCollections(ctx) + if err != nil { + return err + } + db.events.Publish(event.NewMessage(event.P2PTopicName, event.P2PTopic{ + ToAdd: schemaRoots, + })) + + // Get all DocIDs across all collections in the DB + cols, err := db.GetCollections(ctx, client.CollectionFetchOptions{}) + if err != nil { + return err + } + + // Index the schema roots for faster lookup. + colMap := make(map[string]struct{}) + for _, schemaRoot := range schemaRoots { + colMap[schemaRoot] = struct{}{} + } + + for _, col := range cols { + // If we subscribed to the collection, we skip subscribing to the collection's docIDs. + if _, ok := colMap[col.SchemaRoot()]; ok { + continue + } + // TODO-ACP: Support ACP <> P2P - https://github.com/sourcenetwork/defradb/issues/2366 + docIDChan, err := col.GetAllDocIDs(ctx) + if err != nil { + return err + } + + for docID := range docIDChan { + db.events.Publish(event.NewMessage(event.P2PTopicName, event.P2PTopic{ + ToAdd: []string{docID.ID.String()}, + })) + } + } + return nil +} diff --git a/internal/db/p2p_schema_root_test.go b/internal/db/p2p_schema_root_test.go new file mode 100644 index 0000000000..8039f815b0 --- /dev/null +++ b/internal/db/p2p_schema_root_test.go @@ -0,0 +1,307 @@ +// Copyright 2024 Democratized Data Foundation +// +// Use of this software is governed by the Business Source License +// included in the file licenses/BSL.txt. +// +// As of the Change Date specified in that file, in accordance with +// the Business Source License, use of this software will be governed +// by the Apache License, Version 2.0, included in the file +// licenses/APL.txt. + +package db + +import ( + "context" + "encoding/hex" + "fmt" + "testing" + + "github.com/decred/dcrd/dcrec/secp256k1/v4" + "github.com/stretchr/testify/require" + + "github.com/sourcenetwork/immutable" + + "github.com/sourcenetwork/defradb/acp" + acpIdentity "github.com/sourcenetwork/defradb/acp/identity" + "github.com/sourcenetwork/defradb/client" + "github.com/sourcenetwork/defradb/datastore/memory" + "github.com/sourcenetwork/defradb/event" +) + +func TestAddP2PCollection_WithInvalidCollection_ShouldError(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + err = db.AddP2PCollections(ctx, []string{"invalidCollection"}) + require.ErrorIs(t, err, client.ErrCollectionNotFound) +} + +func TestAddP2PCollection_WithValidCollection_ShouldSucceed(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.P2PTopicName) + require.NoError(t, err) + cols, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema, err := db.GetSchemaByVersionID(ctx, cols[0].SchemaVersionID) + require.NoError(t, err) + err = db.AddP2PCollections(ctx, []string{schema.Root}) + require.NoError(t, err) + // Check that the event was published + for msg := range sub.Message() { + p2pTopic := msg.Data.(event.P2PTopic) + require.Equal(t, []string{schema.Root}, p2pTopic.ToAdd) + break + } +} + +func TestAddP2PCollection_WithValidCollectionAndDoc_ShouldSucceed(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.P2PTopicName) + require.NoError(t, err) + cols, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + col, err := db.GetCollectionByName(ctx, cols[0].Name.Value()) + require.NoError(t, err) + doc, err := client.NewDocFromMap(map[string]any{"name": "Alice"}, col.Definition()) + require.NoError(t, err) + err = col.Create(ctx, doc) + require.NoError(t, err) + + err = db.AddP2PCollections(ctx, []string{col.SchemaRoot()}) + require.NoError(t, err) + // Check that the event was published + for msg := range sub.Message() { + p2pTopic := msg.Data.(event.P2PTopic) + require.Equal(t, []string{col.SchemaRoot()}, p2pTopic.ToAdd) + require.Equal(t, []string{doc.ID().String()}, p2pTopic.ToRemove) + break + } +} + +func TestAddP2PCollection_WithMultipleValidCollections_ShouldSucceed(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.P2PTopicName) + require.NoError(t, err) + cols1, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema1, err := db.GetSchemaByVersionID(ctx, cols1[0].SchemaVersionID) + require.NoError(t, err) + cols2, err := db.AddSchema(ctx, `type Books { name: String }`) + require.NoError(t, err) + schema2, err := db.GetSchemaByVersionID(ctx, cols2[0].SchemaVersionID) + require.NoError(t, err) + err = db.AddP2PCollections(ctx, []string{schema1.Root, schema2.Root}) + require.NoError(t, err) + // Check that the event was published + for msg := range sub.Message() { + p2pTopic := msg.Data.(event.P2PTopic) + require.Equal(t, []string{schema1.Root, schema2.Root}, p2pTopic.ToAdd) + break + } +} + +func TestRemoveP2PCollection_WithInvalidCollection_ShouldError(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + err = db.RemoveP2PCollections(ctx, []string{"invalidCollection"}) + require.ErrorIs(t, err, client.ErrCollectionNotFound) +} + +func TestRemoveP2PCollection_WithValidCollection_ShouldSucceed(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.P2PTopicName) + require.NoError(t, err) + cols, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema, err := db.GetSchemaByVersionID(ctx, cols[0].SchemaVersionID) + require.NoError(t, err) + err = db.AddP2PCollections(ctx, []string{schema.Root}) + require.NoError(t, err) + // Check that the event was published + for msg := range sub.Message() { + p2pTopic := msg.Data.(event.P2PTopic) + require.Equal(t, []string{schema.Root}, p2pTopic.ToAdd) + break + } + err = db.RemoveP2PCollections(ctx, []string{schema.Root}) + require.NoError(t, err) + // Check that the event was published + for msg := range sub.Message() { + p2pTopic := msg.Data.(event.P2PTopic) + require.Equal(t, []string{schema.Root}, p2pTopic.ToRemove) + break + } +} + +func TestRemoveP2PCollection_WithValidCollectionAndDoc_ShouldSucceed(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.P2PTopicName) + require.NoError(t, err) + cols, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + col, err := db.GetCollectionByName(ctx, cols[0].Name.Value()) + require.NoError(t, err) + doc, err := client.NewDocFromMap(map[string]any{"name": "Alice"}, col.Definition()) + require.NoError(t, err) + err = col.Create(ctx, doc) + require.NoError(t, err) + + err = db.AddP2PCollections(ctx, []string{col.SchemaRoot()}) + require.NoError(t, err) + // Check that the event was published + for msg := range sub.Message() { + p2pTopic := msg.Data.(event.P2PTopic) + require.Equal(t, []string{col.SchemaRoot()}, p2pTopic.ToAdd) + require.Equal(t, []string{doc.ID().String()}, p2pTopic.ToRemove) + break + } + err = db.RemoveP2PCollections(ctx, []string{col.SchemaRoot()}) + require.NoError(t, err) + // Check that the event was published + for msg := range sub.Message() { + p2pTopic := msg.Data.(event.P2PTopic) + require.Equal(t, []string{col.SchemaRoot()}, p2pTopic.ToRemove) + require.Equal(t, []string{doc.ID().String()}, p2pTopic.ToAdd) + break + } +} + +func TestLoadP2PCollection_WithValidCollectionsAndDocs_ShouldSucceed(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + sub, err := db.events.Subscribe(event.P2PTopicName) + require.NoError(t, err) + cols1, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + col1, err := db.GetCollectionByName(ctx, cols1[0].Name.Value()) + require.NoError(t, err) + doc1, err := client.NewDocFromMap(map[string]any{"name": "Alice"}, col1.Definition()) + require.NoError(t, err) + err = col1.Create(ctx, doc1) + require.NoError(t, err) + + cols2, err := db.AddSchema(ctx, `type Book { name: String }`) + require.NoError(t, err) + col2, err := db.GetCollectionByName(ctx, cols2[0].Name.Value()) + require.NoError(t, err) + doc2, err := client.NewDocFromMap(map[string]any{"name": "Some book"}, col2.Definition()) + require.NoError(t, err) + err = col2.Create(ctx, doc2) + require.NoError(t, err) + + err = db.AddP2PCollections(ctx, []string{col1.SchemaRoot()}) + require.NoError(t, err) + // Check that the event was published + for msg := range sub.Message() { + p2pTopic := msg.Data.(event.P2PTopic) + require.Equal(t, []string{col1.SchemaRoot()}, p2pTopic.ToAdd) + require.Equal(t, []string{doc1.ID().String()}, p2pTopic.ToRemove) + break + } + err = db.loadAndPublishP2PCollections(ctx) + require.NoError(t, err) + // Check that the event was published + msg := <-sub.Message() + p2pTopic := msg.Data.(event.P2PTopic) + require.Equal(t, []string{col1.SchemaRoot()}, p2pTopic.ToAdd) + msg = <-sub.Message() + p2pTopic = msg.Data.(event.P2PTopic) + require.Equal(t, []string{doc2.ID().String()}, p2pTopic.ToAdd) +} + +func TestGetAllP2PCollections_WithMultipleValidCollections_ShouldSucceed(t *testing.T) { + ctx := context.Background() + db, err := newDefraMemoryDB(ctx) + require.NoError(t, err) + defer db.Close() + cols1, err := db.AddSchema(ctx, `type User { name: String }`) + require.NoError(t, err) + schema1, err := db.GetSchemaByVersionID(ctx, cols1[0].SchemaVersionID) + require.NoError(t, err) + cols2, err := db.AddSchema(ctx, `type Books { name: String }`) + require.NoError(t, err) + schema2, err := db.GetSchemaByVersionID(ctx, cols2[0].SchemaVersionID) + require.NoError(t, err) + err = db.AddP2PCollections(ctx, []string{schema1.Root, schema2.Root}) + require.NoError(t, err) + cols, err := db.GetAllP2PCollections(ctx) + require.NoError(t, err) + require.Equal(t, []string{schema2.Root, schema1.Root}, cols) +} + +// This test documents that we don't allow adding p2p collections that have a policy +// until the following is implemented: +// TODO-ACP: ACP <> P2P https://github.com/sourcenetwork/defradb/issues/2366 +func TestAddP2PCollectionsWithPermissionedCollection_Error(t *testing.T) { + ctx := context.Background() + rootstore := memory.NewDatastore(ctx) + db, err := newDB(ctx, rootstore, immutable.Some[acp.ACP](acp.NewLocalACP()), nil) + require.NoError(t, err) + + policy := ` + name: test + description: a policy + actor: + name: actor + resources: + user: + permissions: + read: + expr: owner + write: + expr: owner + relations: + owner: + types: + - actor + ` + + privKeyBytes, err := hex.DecodeString("028d53f37a19afb9a0dbc5b4be30c65731479ee8cfa0c9bc8f8bf198cc3c075f") + require.NoError(t, err) + privKey := secp256k1.PrivKeyFromBytes(privKeyBytes) + identity, err := acpIdentity.FromPrivateKey(privKey) + require.NoError(t, err) + + ctx = SetContextIdentity(ctx, identity) + policyResult, err := db.AddPolicy(ctx, policy) + policyID := policyResult.PolicyID + require.NoError(t, err) + require.Equal(t, "7b5ed30570e8d9206027ef6d5469879a6c1ea4595625c6ca33a19063a6ed6214", policyID) + + schema := fmt.Sprintf(` + type User @policy(id: "%s", resource: "user") { + name: String + age: Int + } + `, policyID, + ) + _, err = db.AddSchema(ctx, schema) + require.NoError(t, err) + + col, err := db.GetCollectionByName(ctx, "User") + require.NoError(t, err) + + err = db.AddP2PCollections(ctx, []string{col.SchemaRoot()}) + require.Error(t, err) + require.ErrorIs(t, err, ErrP2PColHasPolicy) +} diff --git a/internal/merkle/clock/clock.go b/internal/merkle/clock/clock.go index 06cccb6467..d16d1d6a5b 100644 --- a/internal/merkle/clock/clock.go +++ b/internal/merkle/clock/clock.go @@ -32,8 +32,8 @@ var ( // MerkleClock is a MerkleCRDT clock that can be used to read/write events (deltas) to the clock. type MerkleClock struct { - headstore datastore.DSReaderWriter - dagstore datastore.DAGStore + headstore datastore.DSReaderWriter + blockstore datastore.Blockstore // dagSyncer headset *heads crdt core.ReplicatedData @@ -42,15 +42,15 @@ type MerkleClock struct { // NewMerkleClock returns a new MerkleClock. func NewMerkleClock( headstore datastore.DSReaderWriter, - dagstore datastore.DAGStore, + blockstore datastore.Blockstore, namespace core.HeadStoreKey, crdt core.ReplicatedData, ) *MerkleClock { return &MerkleClock{ - headstore: headstore, - dagstore: dagstore, - headset: NewHeadSet(headstore, namespace), - crdt: crdt, + headstore: headstore, + blockstore: blockstore, + headset: NewHeadSet(headstore, namespace), + crdt: crdt, } } @@ -60,7 +60,7 @@ func (mc *MerkleClock) putBlock( ) (cidlink.Link, error) { nd := block.GenerateNode() lsys := cidlink.DefaultLinkSystem() - lsys.SetWriteStorage(mc.dagstore.AsIPLDStorage()) + lsys.SetWriteStorage(mc.blockstore.AsIPLDStorage()) link, err := lsys.Store(linking.LinkContext{Ctx: ctx}, coreblock.GetLinkPrototype(), nd) if err != nil { return cidlink.Link{}, NewErrWritingBlock(err) @@ -155,7 +155,7 @@ func (mc *MerkleClock) ProcessBlock( continue } - known, err := mc.dagstore.Has(ctx, linkCid) + known, err := mc.blockstore.Has(ctx, linkCid) if err != nil { return NewErrCouldNotFindBlock(linkCid, err) } diff --git a/internal/merkle/clock/clock_test.go b/internal/merkle/clock/clock_test.go index c9a51a7a1e..7effc02fef 100644 --- a/internal/merkle/clock/clock_test.go +++ b/internal/merkle/clock/clock_test.go @@ -36,7 +36,7 @@ func newTestMerkleClock() *MerkleClock { reg := crdt.NewLWWRegister(multistore.Rootstore(), core.CollectionSchemaVersionKey{}, core.DataStoreKey{}, "") return NewMerkleClock( multistore.Headstore(), - multistore.DAGstore(), + multistore.Blockstore(), core.HeadStoreKey{DocID: request.DocIDArgName, FieldId: "1"}, reg, ) @@ -46,7 +46,7 @@ func TestNewMerkleClock(t *testing.T) { s := newDS() multistore := datastore.MultiStoreFrom(s) reg := crdt.NewLWWRegister(multistore.Rootstore(), core.CollectionSchemaVersionKey{}, core.DataStoreKey{}, "") - clk := NewMerkleClock(multistore.Headstore(), multistore.DAGstore(), core.HeadStoreKey{}, reg) + clk := NewMerkleClock(multistore.Headstore(), multistore.Blockstore(), core.HeadStoreKey{}, reg) if clk.headstore != multistore.Headstore() { t.Error("MerkleClock store not correctly set") @@ -146,7 +146,7 @@ func TestMerkleClockAddDeltaWithHeads(t *testing.T) { } numBlocks := 0 - cids, err := clk.dagstore.AllKeysChan(ctx) + cids, err := clk.blockstore.AllKeysChan(ctx) if err != nil { t.Error("Failed to get blockstore content for merkle clock:", err) return diff --git a/internal/merkle/crdt/composite.go b/internal/merkle/crdt/composite.go index 33d6ab2d3c..26ab4134e5 100644 --- a/internal/merkle/crdt/composite.go +++ b/internal/merkle/crdt/composite.go @@ -44,7 +44,7 @@ func NewMerkleCompositeDAG( fieldName, ) - clock := clock.NewMerkleClock(store.Headstore(), store.DAGstore(), key.ToHeadStoreKey(), compositeDag) + clock := clock.NewMerkleClock(store.Headstore(), store.Blockstore(), key.ToHeadStoreKey(), compositeDag) base := &baseMerkleCRDT{clock: clock, crdt: compositeDag} return &MerkleCompositeDAG{ diff --git a/internal/merkle/crdt/counter.go b/internal/merkle/crdt/counter.go index c5d3a7e8dd..2553dcfd2f 100644 --- a/internal/merkle/crdt/counter.go +++ b/internal/merkle/crdt/counter.go @@ -39,7 +39,7 @@ func NewMerkleCounter( kind client.ScalarKind, ) *MerkleCounter { register := crdt.NewCounter(store.Datastore(), schemaVersionKey, key, fieldName, allowDecrement, kind) - clk := clock.NewMerkleClock(store.Headstore(), store.DAGstore(), key.ToHeadStoreKey(), register) + clk := clock.NewMerkleClock(store.Headstore(), store.Blockstore(), key.ToHeadStoreKey(), register) base := &baseMerkleCRDT{clock: clk, crdt: register} return &MerkleCounter{ baseMerkleCRDT: base, diff --git a/internal/merkle/crdt/lwwreg.go b/internal/merkle/crdt/lwwreg.go index 6755eac639..b8132ccad5 100644 --- a/internal/merkle/crdt/lwwreg.go +++ b/internal/merkle/crdt/lwwreg.go @@ -37,7 +37,7 @@ func NewMerkleLWWRegister( fieldName string, ) *MerkleLWWRegister { register := corecrdt.NewLWWRegister(store.Datastore(), schemaVersionKey, key, fieldName) - clk := clock.NewMerkleClock(store.Headstore(), store.DAGstore(), key.ToHeadStoreKey(), register) + clk := clock.NewMerkleClock(store.Headstore(), store.Blockstore(), key.ToHeadStoreKey(), register) base := &baseMerkleCRDT{clock: clk, crdt: register} return &MerkleLWWRegister{ baseMerkleCRDT: base, diff --git a/internal/merkle/crdt/merklecrdt.go b/internal/merkle/crdt/merklecrdt.go index abc0ffeb51..fc3019b05c 100644 --- a/internal/merkle/crdt/merklecrdt.go +++ b/internal/merkle/crdt/merklecrdt.go @@ -26,7 +26,7 @@ import ( type Stores interface { Datastore() datastore.DSReaderWriter - DAGstore() datastore.DAGStore + Blockstore() datastore.Blockstore Headstore() datastore.DSReaderWriter } diff --git a/internal/merkle/crdt/merklecrdt_test.go b/internal/merkle/crdt/merklecrdt_test.go index bd42223509..29482b28bf 100644 --- a/internal/merkle/crdt/merklecrdt_test.go +++ b/internal/merkle/crdt/merklecrdt_test.go @@ -32,7 +32,7 @@ func newTestBaseMerkleCRDT() (*baseMerkleCRDT, datastore.DSReaderWriter) { multistore := datastore.MultiStoreFrom(s) reg := crdt.NewLWWRegister(multistore.Datastore(), core.CollectionSchemaVersionKey{}, core.DataStoreKey{}, "") - clk := clock.NewMerkleClock(multistore.Headstore(), multistore.DAGstore(), core.HeadStoreKey{}, reg) + clk := clock.NewMerkleClock(multistore.Headstore(), multistore.Blockstore(), core.HeadStoreKey{}, reg) return &baseMerkleCRDT{clock: clk, crdt: reg}, multistore.Rootstore() } diff --git a/internal/planner/commit.go b/internal/planner/commit.go index 2f9de44bac..3a5bec39f9 100644 --- a/internal/planner/commit.go +++ b/internal/planner/commit.go @@ -184,7 +184,7 @@ func (n *dagScanNode) Next() (bool, error) { n.execInfo.iterations++ var currentCid *cid.Cid - store := n.planner.txn.DAGstore() + store := n.planner.txn.Blockstore() if len(n.queuedCids) > 0 { currentCid = n.queuedCids[0] diff --git a/net/client_test.go b/net/client_test.go index 6a43805ae8..43c4ec1e01 100644 --- a/net/client_test.go +++ b/net/client_test.go @@ -45,8 +45,9 @@ var def = client.CollectionDefinition{ func TestPushlogWithDialFailure(t *testing.T) { ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() doc, err := client.NewDocFromJSON([]byte(`{"test": "test"}`), def) require.NoError(t, err) @@ -56,13 +57,13 @@ func TestPushlogWithDialFailure(t *testing.T) { cid, err := createCID(doc) require.NoError(t, err) - n.server.opts = append( - n.server.opts, + p.server.opts = append( + p.server.opts, grpc.WithTransportCredentials(nil), grpc.WithCredentialsBundle(nil), ) - err = n.server.pushLog(ctx, event.Update{ + err = p.server.pushLog(ctx, event.Update{ DocID: id.String(), Cid: cid, SchemaRoot: "test", @@ -73,8 +74,9 @@ func TestPushlogWithDialFailure(t *testing.T) { func TestPushlogWithInvalidPeerID(t *testing.T) { ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() doc, err := client.NewDocFromJSON([]byte(`{"test": "test"}`), def) require.NoError(t, err) @@ -84,7 +86,7 @@ func TestPushlogWithInvalidPeerID(t *testing.T) { cid, err := createCID(doc) require.NoError(t, err) - err = n.server.pushLog(ctx, event.Update{ + err = p.server.pushLog(ctx, event.Update{ DocID: id.String(), Cid: cid, SchemaRoot: "test", @@ -95,27 +97,29 @@ func TestPushlogWithInvalidPeerID(t *testing.T) { func TestPushlogW_WithValidPeerID_NoError(t *testing.T) { ctx := context.Background() - _, n1 := newTestNode(ctx, t) - defer n1.Close() - n1.Start() - _, n2 := newTestNode(ctx, t) - defer n2.Close() - n2.Start() + db1, p1 := newTestPeer(ctx, t) + defer db1.Close() + defer p1.Close() + p1.Start() + db2, p2 := newTestPeer(ctx, t) + defer p2.Close() + defer db2.Close() + p2.Start() - err := n1.host.Connect(ctx, n2.PeerInfo()) + err := p1.host.Connect(ctx, p2.PeerInfo()) require.NoError(t, err) - _, err = n1.db.AddSchema(ctx, `type User { + _, err = db1.AddSchema(ctx, `type User { name: String }`) require.NoError(t, err) - _, err = n2.db.AddSchema(ctx, `type User { + _, err = db2.AddSchema(ctx, `type User { name: String }`) require.NoError(t, err) - col, err := n1.db.GetCollectionByName(ctx, "User") + col, err := db1.GetCollectionByName(ctx, "User") require.NoError(t, err) doc, err := client.NewDocFromJSON([]byte(`{"name": "test"}`), col.Definition()) @@ -124,22 +128,22 @@ func TestPushlogW_WithValidPeerID_NoError(t *testing.T) { err = col.Save(ctx, doc) require.NoError(t, err) - col, err = n2.db.GetCollectionByName(ctx, "User") + col, err = db2.GetCollectionByName(ctx, "User") require.NoError(t, err) err = col.Save(ctx, doc) require.NoError(t, err) - headCID, err := getHead(ctx, n1.db, doc.ID()) + headCID, err := getHead(ctx, db1, doc.ID()) require.NoError(t, err) - b, err := n1.db.Blockstore().AsIPLDStorage().Get(ctx, headCID.KeyString()) + b, err := db1.Blockstore().AsIPLDStorage().Get(ctx, headCID.KeyString()) require.NoError(t, err) - err = n1.server.pushLog(ctx, event.Update{ + err = p1.server.pushLog(ctx, event.Update{ DocID: doc.ID().String(), Cid: headCID, SchemaRoot: col.SchemaRoot(), Block: b, - }, n2.PeerInfo().ID) + }, p2.PeerInfo().ID) require.NoError(t, err) } diff --git a/net/dialer_test.go b/net/dialer_test.go index 7f37611ec3..479d4d7e63 100644 --- a/net/dialer_test.go +++ b/net/dialer_test.go @@ -23,17 +23,23 @@ import ( func TestDial_WithConnectedPeer_NoError(t *testing.T) { db1 := FixtureNewMemoryDBWithBroadcaster(t) db2 := FixtureNewMemoryDBWithBroadcaster(t) + defer db1.Close() + defer db2.Close() ctx := context.Background() - n1, err := NewNode( + n1, err := NewPeer( ctx, - db1, + db1.Rootstore(), + db1.Blockstore(), + db1.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) assert.NoError(t, err) defer n1.Close() - n2, err := NewNode( + n2, err := NewPeer( ctx, - db2, + db2.Rootstore(), + db2.Blockstore(), + db2.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) assert.NoError(t, err) @@ -50,17 +56,23 @@ func TestDial_WithConnectedPeer_NoError(t *testing.T) { func TestDial_WithConnectedPeerAndSecondConnection_NoError(t *testing.T) { db1 := FixtureNewMemoryDBWithBroadcaster(t) db2 := FixtureNewMemoryDBWithBroadcaster(t) + defer db1.Close() + defer db2.Close() ctx := context.Background() - n1, err := NewNode( + n1, err := NewPeer( ctx, - db1, + db1.Rootstore(), + db1.Blockstore(), + db1.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) assert.NoError(t, err) defer n1.Close() - n2, err := NewNode( + n2, err := NewPeer( ctx, - db2, + db2.Rootstore(), + db2.Blockstore(), + db2.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) assert.NoError(t, err) @@ -80,17 +92,23 @@ func TestDial_WithConnectedPeerAndSecondConnection_NoError(t *testing.T) { func TestDial_WithConnectedPeerAndSecondConnectionWithConnectionShutdown_ClosingConnectionError(t *testing.T) { db1 := FixtureNewMemoryDBWithBroadcaster(t) db2 := FixtureNewMemoryDBWithBroadcaster(t) + defer db1.Close() + defer db2.Close() ctx := context.Background() - n1, err := NewNode( + n1, err := NewPeer( ctx, - db1, + db1.Rootstore(), + db1.Blockstore(), + db1.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) assert.NoError(t, err) defer n1.Close() - n2, err := NewNode( + n2, err := NewPeer( ctx, - db2, + db2.Rootstore(), + db2.Blockstore(), + db2.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) assert.NoError(t, err) diff --git a/net/errors.go b/net/errors.go index eb53a8e2a5..615f1088ef 100644 --- a/net/errors.go +++ b/net/errors.go @@ -13,8 +13,6 @@ package net import ( "fmt" - "github.com/libp2p/go-libp2p/core/peer" - "github.com/sourcenetwork/defradb/errors" ) @@ -23,23 +21,16 @@ const ( errFailedToGetDocID = "failed to get DocID from broadcast message" errPublishingToDocIDTopic = "can't publish log %s for docID %s" errPublishingToSchemaTopic = "can't publish log %s for schema %s" - errReplicatorExists = "replicator already exists for %s with peerID %s" - errReplicatorDocID = "failed to get docID for replicator %s with peerID %s" - errReplicatorCollections = "failed to get collections for replicator" errCheckingForExistingBlock = "failed to check for existing block" ) var ( - ErrP2PColHasPolicy = errors.New("p2p collection specified has a policy on it") - ErrReplicatorColHasPolicy = errors.New("replicator collection specified has a policy on it") - ErrReplicatorSomeColsHavePolicy = errors.New("replicator can not use all collections, as some have policy") - ErrPeerConnectionWaitTimout = errors.New("waiting for peer connection timed out") - ErrPubSubWaitTimeout = errors.New("waiting for pubsub timed out") - ErrPushLogWaitTimeout = errors.New("waiting for pushlog timed out") - ErrNilDB = errors.New("database object can't be nil") - ErrNilUpdateChannel = errors.New("tried to subscribe to update channel, but update channel is nil") - ErrSelfTargetForReplicator = errors.New("can't target ourselves as a replicator") - ErrCheckingForExistingBlock = errors.New(errCheckingForExistingBlock) + ErrPeerConnectionWaitTimout = errors.New("waiting for peer connection timed out") + ErrPubSubWaitTimeout = errors.New("waiting for pubsub timed out") + ErrPushLogWaitTimeout = errors.New("waiting for pushlog timed out") + ErrNilDB = errors.New("database object can't be nil") + ErrNilUpdateChannel = errors.New("tried to subscribe to update channel, but update channel is nil") + ErrCheckingForExistingBlock = errors.New(errCheckingForExistingBlock) ) func NewErrPushLog(inner error, kv ...errors.KV) error { @@ -58,18 +49,6 @@ func NewErrPublishingToSchemaTopic(inner error, cid, docID string, kv ...errors. return errors.Wrap(fmt.Sprintf(errPublishingToSchemaTopic, cid, docID), inner, kv...) } -func NewErrReplicatorExists(collection string, peerID peer.ID, kv ...errors.KV) error { - return errors.New(fmt.Sprintf(errReplicatorExists, collection, peerID), kv...) -} - -func NewErrReplicatorDocID(inner error, collection string, peerID peer.ID, kv ...errors.KV) error { - return errors.Wrap(fmt.Sprintf(errReplicatorDocID, collection, peerID), inner, kv...) -} - -func NewErrReplicatorCollections(inner error, kv ...errors.KV) error { - return errors.Wrap(errReplicatorCollections, inner, kv...) -} - func NewErrCheckingForExistingBlock(inner error, cid string) error { return errors.Wrap(errCheckingForExistingBlock, inner, errors.NewKV("cid", cid)) } diff --git a/net/node.go b/net/node.go deleted file mode 100644 index 3338ac0f04..0000000000 --- a/net/node.go +++ /dev/null @@ -1,282 +0,0 @@ -// Copyright 2023 Democratized Data Foundation -// -// Use of this software is governed by the Business Source License -// included in the file licenses/BSL.txt. -// -// As of the Change Date specified in that file, in accordance with -// the Business Source License, use of this software will be governed -// by the Apache License, Version 2.0, included in the file -// licenses/APL.txt. - -/* -Package node is responsible for interfacing a given DefraDB instance with a networked peer instance -and GRPC server. - -Basically it combines db/DB, net/Peer, and net/Server into a single Node object. -*/ -package net - -import ( - "context" - "fmt" - "sync" - "sync/atomic" - "time" - - "github.com/ipfs/boxo/ipns" - ds "github.com/ipfs/go-datastore" - libp2p "github.com/libp2p/go-libp2p" - dht "github.com/libp2p/go-libp2p-kad-dht" - dualdht "github.com/libp2p/go-libp2p-kad-dht/dual" - pubsub "github.com/libp2p/go-libp2p-pubsub" - record "github.com/libp2p/go-libp2p-record" - libp2pCrypto "github.com/libp2p/go-libp2p/core/crypto" - libp2pEvent "github.com/libp2p/go-libp2p/core/event" - "github.com/libp2p/go-libp2p/core/host" - "github.com/libp2p/go-libp2p/core/peer" - "github.com/libp2p/go-libp2p/core/routing" - "github.com/multiformats/go-multiaddr" - - "github.com/sourcenetwork/corelog" - "github.com/sourcenetwork/go-libp2p-pubsub-rpc/finalizer" - - // @TODO: https://github.com/sourcenetwork/defradb/issues/1902 - //nolint:staticcheck - "github.com/libp2p/go-libp2p/p2p/host/peerstore/pstoreds" - "github.com/libp2p/go-libp2p/p2p/net/connmgr" - - "github.com/sourcenetwork/defradb/client" - "github.com/sourcenetwork/defradb/crypto" - "github.com/sourcenetwork/defradb/event" -) - -var _ client.P2P = (*Node)(nil) - -// Node is a networked peer instance of DefraDB. -type Node struct { - // embed the DB interface into the node - client.DB - - *Peer - - ctx context.Context - cancel context.CancelFunc - dhtClose func() error -} - -// NewNode creates a new network node instance of DefraDB, wired into libp2p. -func NewNode( - ctx context.Context, - db client.DB, - opts ...NodeOpt, -) (node *Node, err error) { - options := DefaultOptions() - for _, opt := range opts { - opt(options) - } - - connManager, err := connmgr.NewConnManager(100, 400, connmgr.WithGracePeriod(time.Second*20)) - if err != nil { - return nil, err - } - - var listenAddresses []multiaddr.Multiaddr - for _, addr := range options.ListenAddresses { - listenAddress, err := multiaddr.NewMultiaddr(addr) - if err != nil { - return nil, err - } - listenAddresses = append(listenAddresses, listenAddress) - } - - fin := finalizer.NewFinalizer() - - ctx, cancel := context.WithCancel(ctx) - defer func() { - if node == nil { - cancel() - } - }() - - peerstore, err := pstoreds.NewPeerstore(ctx, db.Peerstore(), pstoreds.DefaultOpts()) - if err != nil { - return nil, fin.Cleanup(err) - } - fin.Add(peerstore) - - if options.PrivateKey == nil { - // generate an ephemeral private key - key, err := crypto.GenerateEd25519() - if err != nil { - return nil, fin.Cleanup(err) - } - options.PrivateKey = key - } - - // unmarshal the private key bytes - privateKey, err := libp2pCrypto.UnmarshalEd25519PrivateKey(options.PrivateKey) - if err != nil { - return nil, fin.Cleanup(err) - } - - var ddht *dualdht.DHT - - libp2pOpts := []libp2p.Option{ - libp2p.ConnectionManager(connManager), - libp2p.DefaultTransports, - libp2p.Identity(privateKey), - libp2p.ListenAddrs(listenAddresses...), - libp2p.Peerstore(peerstore), - libp2p.Routing(func(h host.Host) (routing.PeerRouting, error) { - // Delete this line and uncomment the next 6 lines once we remove batchable datastore support. - // var store ds.Batching - // // If `rootstore` doesn't implement `Batching`, `nil` will be passed - // // to newDHT which will cause the DHT to be stored in memory. - // if dsb, isBatching := rootstore.(ds.Batching); isBatching { - // store = dsb - // } - store := db.Root() // Delete this line once we remove batchable datastore support. - ddht, err = newDHT(ctx, h, store) - return ddht, err - }), - } - if !options.EnableRelay { - libp2pOpts = append(libp2pOpts, libp2p.DisableRelay()) - } - - h, err := libp2p.New(libp2pOpts...) - if err != nil { - return nil, fin.Cleanup(err) - } - log.InfoContext( - ctx, - "Created LibP2P host", - corelog.Any("PeerId", h.ID()), - corelog.Any("Address", options.ListenAddresses), - ) - - var ps *pubsub.PubSub - if options.EnablePubSub { - ps, err = pubsub.NewGossipSub( - ctx, - h, - pubsub.WithPeerExchange(true), - pubsub.WithFloodPublish(true), - ) - if err != nil { - return nil, fin.Cleanup(err) - } - } - - peer, err := NewPeer( - ctx, - db, - h, - ddht, - ps, - options.GRPCServerOptions, - options.GRPCDialOptions, - ) - if err != nil { - return nil, fin.Cleanup(err) - } - - sub, err := h.EventBus().Subscribe(&libp2pEvent.EvtPeerConnectednessChanged{}) - if err != nil { - return nil, fin.Cleanup(err) - } - // publish subscribed events to the event bus - go func() { - for val := range sub.Out() { - db.Events().Publish(event.NewMessage(event.PeerName, val)) - } - }() - - node = &Node{ - Peer: peer, - DB: db, - ctx: ctx, - cancel: cancel, - dhtClose: ddht.Close, - } - - return -} - -// Bootstrap connects to the given peers. -func (n *Node) Bootstrap(addrs []peer.AddrInfo) { - var connected uint64 - - var wg sync.WaitGroup - for _, pinfo := range addrs { - wg.Add(1) - go func(pinfo peer.AddrInfo) { - defer wg.Done() - err := n.host.Connect(n.ctx, pinfo) - if err != nil { - log.ErrorContextE(n.ctx, "Cannot connect to peer", err) - return - } - log.InfoContext(n.ctx, "Connected", corelog.Any("PeerID", pinfo.ID)) - atomic.AddUint64(&connected, 1) - }(pinfo) - } - - wg.Wait() - - if nPeers := len(addrs); int(connected) < nPeers/2 { - log.InfoContext(n.ctx, fmt.Sprintf("Only connected to %d bootstrap peers out of %d", connected, nPeers)) - } - - err := n.dht.Bootstrap(n.ctx) - if err != nil { - log.ErrorContextE(n.ctx, "Problem bootstraping using DHT", err) - return - } -} - -func (n *Node) PeerID() peer.ID { - return n.host.ID() -} - -func (n *Node) ListenAddrs() []multiaddr.Multiaddr { - return n.host.Network().ListenAddresses() -} - -func (n *Node) PeerInfo() peer.AddrInfo { - return peer.AddrInfo{ - ID: n.host.ID(), - Addrs: n.host.Network().ListenAddresses(), - } -} - -func newDHT(ctx context.Context, h host.Host, dsb ds.Batching) (*dualdht.DHT, error) { - dhtOpts := []dualdht.Option{ - dualdht.DHTOption(dht.NamespacedValidator("pk", record.PublicKeyValidator{})), - dualdht.DHTOption(dht.NamespacedValidator("ipns", ipns.Validator{KeyBook: h.Peerstore()})), - dualdht.DHTOption(dht.Concurrency(10)), - dualdht.DHTOption(dht.Mode(dht.ModeAuto)), - } - if dsb != nil { - dhtOpts = append(dhtOpts, dualdht.DHTOption(dht.Datastore(dsb))) - } - - return dualdht.New(ctx, h, dhtOpts...) -} - -// Close closes the node and all its services. -func (n Node) Close() { - if n.cancel != nil { - n.cancel() - } - if n.Peer != nil { - n.Peer.Close() - } - if n.dhtClose != nil { - err := n.dhtClose() - if err != nil { - log.ErrorContextE(n.ctx, "Failed to close DHT", err) - } - } - n.DB.Close() -} diff --git a/net/node_test.go b/net/node_test.go deleted file mode 100644 index f04e7c6bac..0000000000 --- a/net/node_test.go +++ /dev/null @@ -1,204 +0,0 @@ -// Copyright 2023 Democratized Data Foundation -// -// Use of this software is governed by the Business Source License -// included in the file licenses/BSL.txt. -// -// As of the Change Date specified in that file, in accordance with -// the Business Source License, use of this software will be governed -// by the Apache License, Version 2.0, included in the file -// licenses/APL.txt. - -package net - -import ( - "context" - "testing" - - "github.com/libp2p/go-libp2p/core/peer" - badger "github.com/sourcenetwork/badger/v4" - "github.com/stretchr/testify/require" - - "github.com/sourcenetwork/defradb/acp" - "github.com/sourcenetwork/defradb/client" - badgerds "github.com/sourcenetwork/defradb/datastore/badger/v4" - "github.com/sourcenetwork/defradb/datastore/memory" - "github.com/sourcenetwork/defradb/internal/db" - netutils "github.com/sourcenetwork/defradb/net/utils" -) - -// Node.Boostrap is not tested because the underlying, *ipfslite.Peer.Bootstrap is a best-effort function. - -func FixtureNewMemoryDBWithBroadcaster(t *testing.T) client.DB { - var database client.DB - ctx := context.Background() - opts := badgerds.Options{Options: badger.DefaultOptions("").WithInMemory(true)} - rootstore, err := badgerds.NewDatastore("", &opts) - require.NoError(t, err) - database, err = db.NewDB(ctx, rootstore, acp.NoACP, nil) - require.NoError(t, err) - return database -} - -func TestNewNode_WithEnableRelay_NoError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - n, err := NewNode( - context.Background(), - db, - WithEnableRelay(true), - ) - require.NoError(t, err) - defer n.Close() -} - -func TestNewNode_WithDBClosed_NoError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - db.Close() - - _, err = NewNode( - context.Background(), - db, - ) - require.ErrorContains(t, err, "datastore closed") -} - -func TestNewNode_NoPubSub_NoError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - n, err := NewNode( - context.Background(), - db, - WithEnablePubSub(false), - ) - require.NoError(t, err) - defer n.Close() - require.Nil(t, n.ps) -} - -func TestNewNode_WithEnablePubSub_NoError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - - n, err := NewNode( - ctx, - db, - WithEnablePubSub(true), - ) - - require.NoError(t, err) - defer n.Close() - // overly simple check of validity of pubsub, avoiding the process of creating a PubSub - require.NotNil(t, n.ps) -} - -func TestNodeClose_NoError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - n, err := NewNode( - context.Background(), - db, - ) - require.NoError(t, err) - n.Close() -} - -func TestNewNode_BootstrapWithNoPeer_NoError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - - n1, err := NewNode( - ctx, - db, - WithListenAddresses("/ip4/0.0.0.0/tcp/0"), - ) - require.NoError(t, err) - defer n1.Close() - n1.Bootstrap([]peer.AddrInfo{}) -} - -func TestNewNode_BootstrapWithOnePeer_NoError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - - n1, err := NewNode( - ctx, - db, - WithListenAddresses("/ip4/0.0.0.0/tcp/0"), - ) - require.NoError(t, err) - defer n1.Close() - n2, err := NewNode( - ctx, - db, - WithListenAddresses("/ip4/0.0.0.0/tcp/0"), - ) - require.NoError(t, err) - defer n2.Close() - addrs, err := netutils.ParsePeers([]string{n1.host.Addrs()[0].String() + "/p2p/" + n1.PeerID().String()}) - if err != nil { - t.Fatal(err) - } - n2.Bootstrap(addrs) -} - -func TestNewNode_BootstrapWithOneValidPeerAndManyInvalidPeers_NoError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - - n1, err := NewNode( - ctx, - db, - WithListenAddresses("/ip4/0.0.0.0/tcp/0"), - ) - require.NoError(t, err) - defer n1.Close() - n2, err := NewNode( - ctx, - db, - WithListenAddresses("/ip4/0.0.0.0/tcp/0"), - ) - require.NoError(t, err) - defer n2.Close() - addrs, err := netutils.ParsePeers([]string{ - n1.host.Addrs()[0].String() + "/p2p/" + n1.PeerID().String(), - "/ip4/0.0.0.0/tcp/1234/p2p/" + "12D3KooWC8YY6Tx3uAeHsdBmoy7PJPwqXAHE4HkCZ5veankKWci6", - "/ip4/0.0.0.0/tcp/1235/p2p/" + "12D3KooWC8YY6Tx3uAeHsdBmoy7PJPwqXAHE4HkCZ5veankKWci5", - "/ip4/0.0.0.0/tcp/1236/p2p/" + "12D3KooWC8YY6Tx3uAeHsdBmoy7PJPwqXAHE4HkCZ5veankKWci4", - }) - require.NoError(t, err) - n2.Bootstrap(addrs) -} - -func TestListenAddrs_WithListenAddresses_NoError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - n, err := NewNode( - context.Background(), - db, - WithListenAddresses("/ip4/0.0.0.0/tcp/0"), - ) - require.NoError(t, err) - defer n.Close() - - require.Contains(t, n.ListenAddrs()[0].String(), "/tcp/") -} diff --git a/net/peer.go b/net/peer.go index adb749de70..00ea8653a0 100644 --- a/net/peer.go +++ b/net/peer.go @@ -14,40 +14,53 @@ package net import ( "context" + "fmt" "sync" + "sync/atomic" "time" "github.com/ipfs/boxo/bitswap" "github.com/ipfs/boxo/bitswap/network" "github.com/ipfs/boxo/blockservice" exchange "github.com/ipfs/boxo/exchange" + "github.com/ipfs/boxo/ipns" "github.com/ipfs/go-cid" ds "github.com/ipfs/go-datastore" + libp2p "github.com/libp2p/go-libp2p" gostream "github.com/libp2p/go-libp2p-gostream" + dht "github.com/libp2p/go-libp2p-kad-dht" + dualdht "github.com/libp2p/go-libp2p-kad-dht/dual" pubsub "github.com/libp2p/go-libp2p-pubsub" + record "github.com/libp2p/go-libp2p-record" + libp2pCrypto "github.com/libp2p/go-libp2p/core/crypto" + libp2pEvent "github.com/libp2p/go-libp2p/core/event" "github.com/libp2p/go-libp2p/core/host" "github.com/libp2p/go-libp2p/core/peer" - peerstore "github.com/libp2p/go-libp2p/core/peerstore" "github.com/libp2p/go-libp2p/core/routing" + + // @TODO: https://github.com/sourcenetwork/defradb/issues/1902 + //nolint:staticcheck + "github.com/libp2p/go-libp2p/p2p/host/peerstore/pstoreds" + "github.com/libp2p/go-libp2p/p2p/net/connmgr" + "github.com/multiformats/go-multiaddr" "github.com/sourcenetwork/corelog" "google.golang.org/grpc" "github.com/sourcenetwork/defradb/client" + "github.com/sourcenetwork/defradb/crypto" "github.com/sourcenetwork/defradb/datastore" "github.com/sourcenetwork/defradb/errors" "github.com/sourcenetwork/defradb/event" - "github.com/sourcenetwork/defradb/internal/core" corenet "github.com/sourcenetwork/defradb/internal/core/net" - "github.com/sourcenetwork/defradb/internal/merkle/clock" pb "github.com/sourcenetwork/defradb/net/pb" ) // Peer is a DefraDB Peer node which exposes all the LibP2P host/peer functionality // to the underlying DefraDB instance. type Peer struct { - //config?? + blockstore datastore.Blockstore - db client.DB + bus *event.Bus updateSub *event.Subscription host host.Host @@ -57,50 +70,156 @@ type Peer struct { server *server p2pRPC *grpc.Server // rpc server over the P2P network - // replicators is a map from collectionName => peerId - replicators map[string]map[peer.ID]struct{} - mu sync.Mutex - // peer DAG service exch exchange.Interface bserv blockservice.BlockService - ctx context.Context - cancel context.CancelFunc + ctx context.Context + cancel context.CancelFunc + dhtClose func() error } // NewPeer creates a new instance of the DefraDB server as a peer-to-peer node. func NewPeer( ctx context.Context, - db client.DB, - h host.Host, - dht routing.Routing, - ps *pubsub.PubSub, - serverOptions []grpc.ServerOption, - dialOptions []grpc.DialOption, -) (*Peer, error) { - if db == nil { + rootstore datastore.Rootstore, + blockstore datastore.Blockstore, + bus *event.Bus, + opts ...NodeOpt, +) (p *Peer, err error) { + if rootstore == nil || blockstore == nil { return nil, ErrNilDB } + options := DefaultOptions() + for _, opt := range opts { + opt(options) + } + + connManager, err := connmgr.NewConnManager(100, 400, connmgr.WithGracePeriod(time.Second*20)) + if err != nil { + return nil, err + } + + var listenAddresses []multiaddr.Multiaddr + for _, addr := range options.ListenAddresses { + listenAddress, err := multiaddr.NewMultiaddr(addr) + if err != nil { + return nil, err + } + listenAddresses = append(listenAddresses, listenAddress) + } + ctx, cancel := context.WithCancel(ctx) - p := &Peer{ - host: h, - dht: dht, - ps: ps, - db: db, - p2pRPC: grpc.NewServer(serverOptions...), - ctx: ctx, - cancel: cancel, - replicators: make(map[string]map[peer.ID]struct{}), - } - var err error - p.server, err = newServer(p, dialOptions...) + defer func() { + if p == nil { + cancel() + } + }() + + peerstore, err := pstoreds.NewPeerstore(ctx, rootstore, pstoreds.DefaultOpts()) if err != nil { return nil, err } - err = p.loadReplicators(p.ctx) + if options.PrivateKey == nil { + // generate an ephemeral private key + key, err := crypto.GenerateEd25519() + if err != nil { + return nil, err + } + options.PrivateKey = key + } + + // unmarshal the private key bytes + privateKey, err := libp2pCrypto.UnmarshalEd25519PrivateKey(options.PrivateKey) + if err != nil { + return nil, err + } + + var ddht *dualdht.DHT + + libp2pOpts := []libp2p.Option{ + libp2p.ConnectionManager(connManager), + libp2p.DefaultTransports, + libp2p.Identity(privateKey), + libp2p.ListenAddrs(listenAddresses...), + libp2p.Peerstore(peerstore), + libp2p.Routing(func(h host.Host) (routing.PeerRouting, error) { + // Delete this line and uncomment the next 6 lines once we remove batchable datastore support. + // var store ds.Batching + // // If `rootstore` doesn't implement `Batching`, `nil` will be passed + // // to newDHT which will cause the DHT to be stored in memory. + // if dsb, isBatching := rootstore.(ds.Batching); isBatching { + // store = dsb + // } + ddht, err = newDHT(ctx, h, rootstore) + return ddht, err + }), + } + if !options.EnableRelay { + libp2pOpts = append(libp2pOpts, libp2p.DisableRelay()) + } + + h, err := libp2p.New(libp2pOpts...) + if err != nil { + return nil, err + } + log.InfoContext( + ctx, + "Created LibP2P host", + corelog.Any("PeerId", h.ID()), + corelog.Any("Address", options.ListenAddresses), + ) + + var ps *pubsub.PubSub + if options.EnablePubSub { + ps, err = pubsub.NewGossipSub( + ctx, + h, + pubsub.WithPeerExchange(true), + pubsub.WithFloodPublish(true), + ) + if err != nil { + return nil, err + } + } + + if err != nil { + return nil, err + } + + sub, err := h.EventBus().Subscribe(&libp2pEvent.EvtPeerConnectednessChanged{}) + if err != nil { + return nil, err + } + // publish subscribed events to the event bus + go func() { + for { + select { + case <-ctx.Done(): + return + case val, isOpen := <-sub.Out(): + if !isOpen { + return + } + bus.Publish(event.NewMessage(event.PeerName, val)) + } + } + }() + + p = &Peer{ + host: h, + dht: ddht, + ps: ps, + blockstore: blockstore, + bus: bus, + p2pRPC: grpc.NewServer(options.GRPCServerOptions...), + ctx: ctx, + cancel: cancel, + } + + p.server, err = newServer(p, options.GRPCDialOptions...) if err != nil { return nil, err } @@ -112,9 +231,6 @@ func NewPeer( // Start all the internal workers/goroutines/loops that manage the P2P state. func (p *Peer) Start() error { - p.mu.Lock() - defer p.mu.Unlock() - // reconnect to known peers var wg sync.WaitGroup for _, id := range p.host.Peerstore().PeersWithAddrs() { @@ -142,13 +258,13 @@ func (p *Peer) Start() error { } if p.ps != nil { - sub, err := p.db.Events().Subscribe(event.UpdateName) + sub, err := p.bus.Subscribe(event.UpdateName, event.P2PTopicName, event.ReplicatorName) if err != nil { return err } p.updateSub = sub log.InfoContext(p.ctx, "Starting internal broadcaster for pubsub network") - go p.handleBroadcastLoop() + go p.handleMessageLoop() } log.InfoContext( @@ -164,6 +280,8 @@ func (p *Peer) Start() error { } }() + p.bus.Publish(event.NewMessage(event.PeerInfoName, event.PeerInfo{Info: p.PeerInfo()})) + return nil } @@ -180,10 +298,9 @@ func (p *Peer) Close() { log.ErrorContextE(p.ctx, "Failed closing server RPC connections", err) } } - stopGRPCServer(p.ctx, p.p2pRPC) if p.updateSub != nil { - p.db.Events().Unsubscribe(p.updateSub) + p.bus.Unsubscribe(p.updateSub) } if err := p.bserv.Close(); err != nil { @@ -194,31 +311,50 @@ func (p *Peer) Close() { log.ErrorContextE(p.ctx, "Error closing host", err) } - p.cancel() + if p.dhtClose != nil { + err := p.dhtClose() + if err != nil { + log.ErrorContextE(p.ctx, "Failed to close DHT", err) + } + } + + stopGRPCServer(p.ctx, p.p2pRPC) + + if p.cancel != nil { + p.cancel() + } } -// handleBroadcast loop manages the transition of messages +// handleMessage loop manages the transition of messages // from the internal broadcaster to the external pubsub network -func (p *Peer) handleBroadcastLoop() { +func (p *Peer) handleMessageLoop() { for { msg, isOpen := <-p.updateSub.Message() if !isOpen { return } - update, ok := msg.Data.(event.Update) - if !ok { - continue // ignore invalid value - } - var err error - if update.IsCreate { - err = p.handleDocCreateLog(update) - } else { - err = p.handleDocUpdateLog(update) - } + switch evt := msg.Data.(type) { + case event.Update: + var err error + if evt.IsCreate { + err = p.handleDocCreateLog(evt) + } else { + err = p.handleDocUpdateLog(evt) + } - if err != nil { - log.ErrorContextE(p.ctx, "Error while handling broadcast log", err) + if err != nil { + log.ErrorContextE(p.ctx, "Error while handling broadcast log", err) + } + + case event.P2PTopic: + p.server.updatePubSubTopics(evt) + + case event.Replicator: + p.server.updateReplicators(evt) + default: + // ignore other events + continue } } } @@ -258,112 +394,6 @@ func (p *Peer) RegisterNewDocument( return p.server.publishLog(p.ctx, schemaRoot, req) } -func (p *Peer) pushToReplicator( - ctx context.Context, - txn datastore.Txn, - collection client.Collection, - docIDsCh <-chan client.DocIDResult, - pid peer.ID, -) { - for docIDResult := range docIDsCh { - if docIDResult.Err != nil { - log.ErrorContextE(ctx, "Key channel error", docIDResult.Err) - continue - } - docID := core.DataStoreKeyFromDocID(docIDResult.ID) - headset := clock.NewHeadSet( - txn.Headstore(), - docID.WithFieldId(core.COMPOSITE_NAMESPACE).ToHeadStoreKey(), - ) - cids, _, err := headset.List(ctx) - if err != nil { - log.ErrorContextE( - ctx, - "Failed to get heads", - err, - corelog.String("DocID", docIDResult.ID.String()), - corelog.Any("PeerID", pid), - corelog.Any("Collection", collection.Name())) - continue - } - // loop over heads, get block, make the required logs, and send - for _, c := range cids { - blk, err := txn.DAGstore().Get(ctx, c) - if err != nil { - log.ErrorContextE(ctx, "Failed to get block", err, - corelog.Any("CID", c), - corelog.Any("PeerID", pid), - corelog.Any("Collection", collection.Name())) - continue - } - - evt := event.Update{ - DocID: docIDResult.ID.String(), - Cid: c, - SchemaRoot: collection.SchemaRoot(), - Block: blk.RawData(), - } - if err := p.server.pushLog(ctx, evt, pid); err != nil { - log.ErrorContextE( - ctx, - "Failed to replicate log", - err, - corelog.Any("CID", c), - corelog.Any("PeerID", pid), - ) - } - } - } -} - -func (p *Peer) loadReplicators(ctx context.Context) error { - reps, err := p.GetAllReplicators(ctx) - if err != nil { - return errors.Wrap("failed to get replicators", err) - } - p.mu.Lock() - defer p.mu.Unlock() - for _, rep := range reps { - for _, schema := range rep.Schemas { - if pReps, exists := p.replicators[schema]; exists { - if _, exists := pReps[rep.Info.ID]; exists { - continue - } - } else { - p.replicators[schema] = make(map[peer.ID]struct{}) - } - - // add to replicators list - p.replicators[schema][rep.Info.ID] = struct{}{} - } - - // Add the destination's peer multiaddress in the peerstore. - // This will be used during connection and stream creation by libp2p. - p.host.Peerstore().AddAddrs(rep.Info.ID, rep.Info.Addrs, peerstore.PermanentAddrTTL) - - log.InfoContext(ctx, "loaded replicators from datastore", corelog.Any("Replicator", rep)) - } - - return nil -} - -func (p *Peer) loadP2PCollections(ctx context.Context) (map[string]struct{}, error) { - collections, err := p.GetAllP2PCollections(ctx) - if err != nil && !errors.Is(err, ds.ErrNotFound) { - return nil, err - } - colMap := make(map[string]struct{}) - for _, col := range collections { - err := p.server.addPubSubTopic(col, true) - if err != nil { - return nil, err - } - colMap[col] = struct{}{} - } - - return colMap, nil -} - func (p *Peer) handleDocCreateLog(evt event.Update) error { docID, err := client.NewDocIDFromString(evt.DocID) if err != nil { @@ -425,9 +455,9 @@ func (p *Peer) pushLogToReplicators(lg event.Update) { peers[peer.String()] = struct{}{} } - p.mu.Lock() - reps, exists := p.replicators[lg.SchemaRoot] - p.mu.Unlock() + p.server.mu.Lock() + reps, exists := p.server.replicators[lg.SchemaRoot] + p.server.mu.Unlock() if exists { for pid := range reps { @@ -453,8 +483,8 @@ func (p *Peer) pushLogToReplicators(lg event.Update) { func (p *Peer) setupBlockService() { bswapnet := network.NewFromIpfsHost(p.host, p.dht) - bswap := bitswap.New(p.ctx, bswapnet, p.db.Blockstore()) - p.bserv = blockservice.New(p.db.Blockstore(), bswap) + bswap := bitswap.New(p.ctx, bswapnet, p.blockstore) + p.bserv = blockservice.New(p.blockstore, bswap) p.exch = bswap } @@ -474,22 +504,63 @@ func stopGRPCServer(ctx context.Context, server *grpc.Server) { } } -// rollbackAddPubSubTopics removes the given topics from the pubsub system. -func (p *Peer) rollbackAddPubSubTopics(topics []string, cause error) error { - for _, topic := range topics { - if err := p.server.removePubSubTopic(topic); err != nil { - return errors.WithStack(err, errors.NewKV("Cause", cause)) - } +// Bootstrap connects to the given peers. +func (p *Peer) Bootstrap(addrs []peer.AddrInfo) { + var connected uint64 + + var wg sync.WaitGroup + for _, pinfo := range addrs { + wg.Add(1) + go func(pinfo peer.AddrInfo) { + defer wg.Done() + err := p.host.Connect(p.ctx, pinfo) + if err != nil { + log.InfoContext(p.ctx, "Cannot connect to peer", corelog.Any("Error", err)) + return + } + log.InfoContext(p.ctx, "Connected", corelog.Any("PeerID", pinfo.ID)) + atomic.AddUint64(&connected, 1) + }(pinfo) + } + + wg.Wait() + + if nPeers := len(addrs); int(connected) < nPeers/2 { + log.InfoContext(p.ctx, fmt.Sprintf("Only connected to %d bootstrap peers out of %d", connected, nPeers)) + } + + err := p.dht.Bootstrap(p.ctx) + if err != nil { + log.ErrorContextE(p.ctx, "Problem bootstraping using DHT", err) + return } - return cause } -// rollbackRemovePubSubTopics adds back the given topics from the pubsub system. -func (p *Peer) rollbackRemovePubSubTopics(topics []string, cause error) error { - for _, topic := range topics { - if err := p.server.addPubSubTopic(topic, true); err != nil { - return errors.WithStack(err, errors.NewKV("Cause", cause)) - } +func (p *Peer) PeerID() peer.ID { + return p.host.ID() +} + +func (p *Peer) ListenAddrs() []multiaddr.Multiaddr { + return p.host.Network().ListenAddresses() +} + +func (p *Peer) PeerInfo() peer.AddrInfo { + return peer.AddrInfo{ + ID: p.host.ID(), + Addrs: p.host.Network().ListenAddresses(), + } +} + +func newDHT(ctx context.Context, h host.Host, dsb ds.Batching) (*dualdht.DHT, error) { + dhtOpts := []dualdht.Option{ + dualdht.DHTOption(dht.NamespacedValidator("pk", record.PublicKeyValidator{})), + dualdht.DHTOption(dht.NamespacedValidator("ipns", ipns.Validator{KeyBook: h.Peerstore()})), + dualdht.DHTOption(dht.Concurrency(10)), + dualdht.DHTOption(dht.Mode(dht.ModeAuto)), } - return cause + if dsb != nil { + dhtOpts = append(dhtOpts, dualdht.DHTOption(dht.Datastore(dsb))) + } + + return dualdht.New(ctx, h, dhtOpts...) } diff --git a/net/peer_replicator.go b/net/peer_replicator.go deleted file mode 100644 index 19accb17c4..0000000000 --- a/net/peer_replicator.go +++ /dev/null @@ -1,230 +0,0 @@ -// Copyright 2023 Democratized Data Foundation -// -// Use of this software is governed by the Business Source License -// included in the file licenses/BSL.txt. -// -// As of the Change Date specified in that file, in accordance with -// the Business Source License, use of this software will be governed -// by the Apache License, Version 2.0, included in the file -// licenses/APL.txt. - -package net - -import ( - "context" - "encoding/json" - - dsq "github.com/ipfs/go-datastore/query" - "github.com/libp2p/go-libp2p/core/peer" - "github.com/libp2p/go-libp2p/core/peerstore" - - "github.com/sourcenetwork/defradb/client" - "github.com/sourcenetwork/defradb/internal/core" - "github.com/sourcenetwork/defradb/internal/db" -) - -func (p *Peer) SetReplicator(ctx context.Context, rep client.Replicator) error { - p.mu.Lock() - defer p.mu.Unlock() - - txn, err := p.db.NewTxn(ctx, false) - if err != nil { - return err - } - defer txn.Discard(ctx) - - if rep.Info.ID == p.host.ID() { - return ErrSelfTargetForReplicator - } - if err := rep.Info.ID.Validate(); err != nil { - return err - } - - // TODO-ACP: Support ACP <> P2P - https://github.com/sourcenetwork/defradb/issues/2366 - // ctx = db.SetContextIdentity(ctx, identity) - ctx = db.SetContextTxn(ctx, txn) - - var collections []client.Collection - switch { - case len(rep.Schemas) > 0: - // if specific collections are chosen get them by name - for _, name := range rep.Schemas { - col, err := p.db.GetCollectionByName(ctx, name) - if err != nil { - return NewErrReplicatorCollections(err) - } - - if col.Description().Policy.HasValue() { - return ErrReplicatorColHasPolicy - } - - collections = append(collections, col) - } - - default: - // default to all collections (unless a collection contains a policy). - // TODO-ACP: default to all collections after resolving https://github.com/sourcenetwork/defradb/issues/2366 - allCollections, err := p.db.GetCollections(ctx, client.CollectionFetchOptions{}) - if err != nil { - return NewErrReplicatorCollections(err) - } - - for _, col := range allCollections { - // Can not default to all collections if any collection has a policy. - // TODO-ACP: remove this check/loop after https://github.com/sourcenetwork/defradb/issues/2366 - if col.Description().Policy.HasValue() { - return ErrReplicatorSomeColsHavePolicy - } - } - collections = allCollections - } - rep.Schemas = nil - - // Add the destination's peer multiaddress in the peerstore. - // This will be used during connection and stream creation by libp2p. - p.host.Peerstore().AddAddrs(rep.Info.ID, rep.Info.Addrs, peerstore.PermanentAddrTTL) - - var added []client.Collection - for _, col := range collections { - reps, exists := p.replicators[col.SchemaRoot()] - if !exists { - p.replicators[col.SchemaRoot()] = make(map[peer.ID]struct{}) - } - if _, exists := reps[rep.Info.ID]; !exists { - // keep track of newly added collections so we don't - // push logs to a replicator peer multiple times. - p.replicators[col.SchemaRoot()][rep.Info.ID] = struct{}{} - added = append(added, col) - } - rep.Schemas = append(rep.Schemas, col.SchemaRoot()) - } - - // persist replicator to the datastore - repBytes, err := json.Marshal(rep) - if err != nil { - return err - } - key := core.NewReplicatorKey(rep.Info.ID.String()) - err = txn.Systemstore().Put(ctx, key.ToDS(), repBytes) - if err != nil { - return err - } - - // push all collection documents to the replicator peer - for _, col := range added { - keysCh, err := col.GetAllDocIDs(ctx) - if err != nil { - return NewErrReplicatorDocID(err, col.Name().Value(), rep.Info.ID) - } - p.pushToReplicator(ctx, txn, col, keysCh, rep.Info.ID) - } - - return txn.Commit(ctx) -} - -func (p *Peer) DeleteReplicator(ctx context.Context, rep client.Replicator) error { - p.mu.Lock() - defer p.mu.Unlock() - - txn, err := p.db.NewTxn(ctx, false) - if err != nil { - return err - } - defer txn.Discard(ctx) - - if rep.Info.ID == p.host.ID() { - return ErrSelfTargetForReplicator - } - if err := rep.Info.ID.Validate(); err != nil { - return err - } - - // set transaction for all operations - ctx = db.SetContextTxn(ctx, txn) - - var collections []client.Collection - switch { - case len(rep.Schemas) > 0: - // if specific collections are chosen get them by name - for _, name := range rep.Schemas { - col, err := p.db.GetCollectionByName(ctx, name) - if err != nil { - return NewErrReplicatorCollections(err) - } - collections = append(collections, col) - } - // make sure the replicator exists in the datastore - key := core.NewReplicatorKey(rep.Info.ID.String()) - _, err = txn.Systemstore().Get(ctx, key.ToDS()) - if err != nil { - return err - } - - default: - // default to all collections - collections, err = p.db.GetCollections(ctx, client.CollectionFetchOptions{}) - if err != nil { - return NewErrReplicatorCollections(err) - } - } - rep.Schemas = nil - - schemaMap := make(map[string]struct{}) - for _, col := range collections { - schemaMap[col.SchemaRoot()] = struct{}{} - } - - // update replicators and add remaining schemas to rep - for key, val := range p.replicators { - if _, exists := val[rep.Info.ID]; exists { - if _, toDelete := schemaMap[key]; toDelete { - delete(p.replicators[key], rep.Info.ID) - } else { - rep.Schemas = append(rep.Schemas, key) - } - } - } - - if len(rep.Schemas) == 0 { - // Remove the destination's peer multiaddress in the peerstore. - p.host.Peerstore().ClearAddrs(rep.Info.ID) - } - - // persist the replicator to the store, deleting it if no schemas remain - key := core.NewReplicatorKey(rep.Info.ID.String()) - if len(rep.Schemas) == 0 { - return txn.Systemstore().Delete(ctx, key.ToDS()) - } - repBytes, err := json.Marshal(rep) - if err != nil { - return err - } - return txn.Systemstore().Put(ctx, key.ToDS(), repBytes) -} - -func (p *Peer) GetAllReplicators(ctx context.Context) ([]client.Replicator, error) { - txn, err := p.db.NewTxn(ctx, true) - if err != nil { - return nil, err - } - defer txn.Discard(ctx) - - // create collection system prefix query - query := dsq.Query{ - Prefix: core.NewReplicatorKey("").ToString(), - } - results, err := txn.Systemstore().Query(ctx, query) - if err != nil { - return nil, err - } - - var reps []client.Replicator - for result := range results.Next() { - var rep client.Replicator - if err = json.Unmarshal(result.Value, &rep); err != nil { - return nil, err - } - reps = append(reps, rep) - } - return reps, nil -} diff --git a/net/peer_test.go b/net/peer_test.go index 6f6fda67ad..cb8c8eab44 100644 --- a/net/peer_test.go +++ b/net/peer_test.go @@ -12,25 +12,20 @@ package net import ( "context" - "encoding/hex" - "fmt" "testing" "time" - "github.com/decred/dcrd/dcrec/secp256k1/v4" "github.com/ipfs/go-cid" - ds "github.com/ipfs/go-datastore" - libp2p "github.com/libp2p/go-libp2p" - pubsub "github.com/libp2p/go-libp2p-pubsub" "github.com/libp2p/go-libp2p/core/peer" mh "github.com/multiformats/go-multihash" + badger "github.com/sourcenetwork/badger/v4" rpc "github.com/sourcenetwork/go-libp2p-pubsub-rpc" "github.com/sourcenetwork/immutable" "github.com/stretchr/testify/require" "github.com/sourcenetwork/defradb/acp" - acpIdentity "github.com/sourcenetwork/defradb/acp/identity" "github.com/sourcenetwork/defradb/client" + badgerds "github.com/sourcenetwork/defradb/datastore/badger/v4" "github.com/sourcenetwork/defradb/datastore/memory" "github.com/sourcenetwork/defradb/event" coreblock "github.com/sourcenetwork/defradb/internal/core/block" @@ -71,16 +66,18 @@ func createCID(doc *client.Document) (cid.Cid, error) { const randomMultiaddr = "/ip4/127.0.0.1/tcp/0" -func newTestNode(ctx context.Context, t *testing.T) (client.DB, *Node) { +func newTestPeer(ctx context.Context, t *testing.T) (client.DB, *Peer) { store := memory.NewDatastore(ctx) acpLocal := acp.NewLocalACP() acpLocal.Init(context.Background(), "") db, err := db.NewDB(ctx, store, immutable.Some[acp.ACP](acpLocal), nil) require.NoError(t, err) - n, err := NewNode( + n, err := NewPeer( ctx, - db, + db.Rootstore(), + db.Blockstore(), + db.Events(), WithListenAddresses(randomMultiaddr), ) require.NoError(t, err) @@ -93,72 +90,26 @@ func TestNewPeer_NoError(t *testing.T) { store := memory.NewDatastore(ctx) db, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) - - h, err := libp2p.New() - require.NoError(t, err) - - _, err = NewPeer(ctx, db, h, nil, nil, nil, nil) + defer db.Close() + p, err := NewPeer(ctx, db.Rootstore(), db.Blockstore(), db.Events()) require.NoError(t, err) + p.Close() } func TestNewPeer_NoDB_NilDBError(t *testing.T) { ctx := context.Background() - - h, err := libp2p.New() - require.NoError(t, err) - - _, err = NewPeer(ctx, nil, h, nil, nil, nil, nil) + _, err := NewPeer(ctx, nil, nil, nil) require.ErrorIs(t, err, ErrNilDB) } -func TestNewPeer_WithExistingTopic_TopicAlreadyExistsError(t *testing.T) { - ctx := context.Background() - store := memory.NewDatastore(ctx) - db, err := db.NewDB(ctx, store, acp.NoACP, nil) - require.NoError(t, err) - - _, err = db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - col, err := db.GetCollectionByName(ctx, "User") - require.NoError(t, err) - - doc, err := client.NewDocFromJSON([]byte(`{"name": "John", "age": 30}`), col.Definition()) - require.NoError(t, err) - - err = col.Create(ctx, doc) - require.NoError(t, err) - - h, err := libp2p.New() - require.NoError(t, err) - - ps, err := pubsub.NewGossipSub( - ctx, - h, - pubsub.WithPeerExchange(true), - pubsub.WithFloodPublish(true), - ) - require.NoError(t, err) - - _, err = rpc.NewTopic(ctx, ps, h.ID(), doc.ID().String(), true) - require.NoError(t, err) - - _, err = NewPeer(ctx, db, h, nil, ps, nil, nil) - require.ErrorContains(t, err, "topic already exists") -} - func TestStartAndClose_NoError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() - err := n.Start() + err := p.Start() require.NoError(t, err) - - db.Close() } func TestStart_WithKnownPeer_NoError(t *testing.T) { @@ -166,23 +117,31 @@ func TestStart_WithKnownPeer_NoError(t *testing.T) { store := memory.NewDatastore(ctx) db1, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) + defer db1.Close() store2 := memory.NewDatastore(ctx) db2, err := db.NewDB(ctx, store2, acp.NoACP, nil) require.NoError(t, err) + defer db2.Close() - n1, err := NewNode( + n1, err := NewPeer( ctx, - db1, + db1.Rootstore(), + db1.Blockstore(), + db1.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) require.NoError(t, err) - n2, err := NewNode( + defer n1.Close() + n2, err := NewPeer( ctx, - db2, + db2.Rootstore(), + db2.Blockstore(), + db2.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) require.NoError(t, err) + defer n2.Close() addrs, err := netutils.ParsePeers([]string{n1.host.Addrs()[0].String() + "/p2p/" + n1.PeerID().String()}) if err != nil { @@ -192,9 +151,6 @@ func TestStart_WithKnownPeer_NoError(t *testing.T) { err = n2.Start() require.NoError(t, err) - - db1.Close() - db2.Close() } func TestStart_WithOfflineKnownPeer_NoError(t *testing.T) { @@ -202,23 +158,31 @@ func TestStart_WithOfflineKnownPeer_NoError(t *testing.T) { store := memory.NewDatastore(ctx) db1, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) + defer db1.Close() store2 := memory.NewDatastore(ctx) db2, err := db.NewDB(ctx, store2, acp.NoACP, nil) require.NoError(t, err) + defer db2.Close() - n1, err := NewNode( + n1, err := NewPeer( ctx, - db1, + db1.Rootstore(), + db1.Blockstore(), + db1.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) require.NoError(t, err) - n2, err := NewNode( + defer n1.Close() + n2, err := NewPeer( ctx, - db2, + db2.Rootstore(), + db2.Blockstore(), + db2.Events(), WithListenAddresses("/ip4/0.0.0.0/tcp/0"), ) require.NoError(t, err) + defer n2.Close() addrs, err := netutils.ParsePeers([]string{n1.host.Addrs()[0].String() + "/p2p/" + n1.PeerID().String()}) if err != nil { @@ -232,15 +196,13 @@ func TestStart_WithOfflineKnownPeer_NoError(t *testing.T) { err = n2.Start() require.NoError(t, err) - - db1.Close() - db2.Close() } func TestRegisterNewDocument_NoError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() _, err := db.AddSchema(ctx, `type User { name: String @@ -257,14 +219,15 @@ func TestRegisterNewDocument_NoError(t *testing.T) { cid, err := createCID(doc) require.NoError(t, err) - err = n.RegisterNewDocument(ctx, doc.ID(), cid, emptyBlock(), col.SchemaRoot()) + err = p.RegisterNewDocument(ctx, doc.ID(), cid, emptyBlock(), col.SchemaRoot()) require.NoError(t, err) } func TestRegisterNewDocument_RPCTopicAlreadyRegisteredError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() _, err := db.AddSchema(ctx, `type User { name: String @@ -278,228 +241,22 @@ func TestRegisterNewDocument_RPCTopicAlreadyRegisteredError(t *testing.T) { doc, err := client.NewDocFromJSON([]byte(`{"name": "John", "age": 30}`), col.Definition()) require.NoError(t, err) - _, err = rpc.NewTopic(ctx, n.Peer.ps, n.Peer.host.ID(), doc.ID().String(), true) + _, err = rpc.NewTopic(ctx, p.ps, p.host.ID(), doc.ID().String(), true) require.NoError(t, err) cid, err := createCID(doc) require.NoError(t, err) - err = n.RegisterNewDocument(ctx, doc.ID(), cid, emptyBlock(), col.SchemaRoot()) + err = p.RegisterNewDocument(ctx, doc.ID(), cid, emptyBlock(), col.SchemaRoot()) require.Equal(t, err.Error(), "creating topic: joining topic: topic already exists") } -func TestSetReplicator_NoError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - info, err := peer.AddrInfoFromString("/ip4/0.0.0.0/tcp/0/p2p/QmYyQSo1c1Ym7orWxLYvCrM2EmxFTANf8wXmmE7DWjhx5N") - require.NoError(t, err) - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: *info, - Schemas: []string{"User"}, - }) - require.NoError(t, err) -} - -// This test documents that we don't allow setting replicator with a collection that has a policy -// until the following is implemented: -// TODO-ACP: ACP <> P2P https://github.com/sourcenetwork/defradb/issues/2366 -func TestSetReplicatorWithACollectionSpecifiedThatHasPolicy_ReturnError(t *testing.T) { - ctx := context.Background() - d, n := newTestNode(ctx, t) - defer n.Close() - - policy := ` - name: test - description: a policy - actor: - name: actor - resources: - user: - permissions: - read: - expr: owner - write: - expr: owner - relations: - owner: - types: - - actor - ` - - privKeyBytes, err := hex.DecodeString("028d53f37a19afb9a0dbc5b4be30c65731479ee8cfa0c9bc8f8bf198cc3c075f") - require.NoError(t, err) - privKey := secp256k1.PrivKeyFromBytes(privKeyBytes) - identity, err := acpIdentity.FromPrivateKey(privKey) - require.NoError(t, err) - - ctx = db.SetContextIdentity(ctx, identity) - policyResult, err := d.AddPolicy(ctx, policy) - policyID := policyResult.PolicyID - require.NoError(t, err) - require.Equal(t, "7b5ed30570e8d9206027ef6d5469879a6c1ea4595625c6ca33a19063a6ed6214", policyID) - - schema := fmt.Sprintf(` - type User @policy(id: "%s", resource: "user") { - name: String - age: Int - } - `, policyID, - ) - _, err = d.AddSchema(ctx, schema) - require.NoError(t, err) - - info, err := peer.AddrInfoFromString("/ip4/0.0.0.0/tcp/0/p2p/QmYyQSo1c1Ym7orWxLYvCrM2EmxFTANf8wXmmE7DWjhx5N") - require.NoError(t, err) - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: *info, - Schemas: []string{"User"}, - }) - require.Error(t, err) - require.ErrorIs(t, err, ErrReplicatorColHasPolicy) -} - -// This test documents that we don't allow setting replicator using default option when any collection has a policy -// until the following is implemented: -// TODO-ACP: ACP <> P2P https://github.com/sourcenetwork/defradb/issues/2366 -func TestSetReplicatorWithSomeCollectionThatHasPolicyUsingAllCollectionsByDefault_ReturnError(t *testing.T) { - ctx := context.Background() - d, n := newTestNode(ctx, t) - defer n.Close() - - policy := ` - name: test - description: a policy - actor: - name: actor - resources: - user: - permissions: - read: - expr: owner - write: - expr: owner - relations: - owner: - types: - - actor - ` - - privKeyBytes, err := hex.DecodeString("028d53f37a19afb9a0dbc5b4be30c65731479ee8cfa0c9bc8f8bf198cc3c075f") - require.NoError(t, err) - privKey := secp256k1.PrivKeyFromBytes(privKeyBytes) - identity, err := acpIdentity.FromPrivateKey(privKey) - require.NoError(t, err) - - ctx = db.SetContextIdentity(ctx, identity) - policyResult, err := d.AddPolicy(ctx, policy) - policyID := policyResult.PolicyID - require.NoError(t, err) - require.Equal(t, "7b5ed30570e8d9206027ef6d5469879a6c1ea4595625c6ca33a19063a6ed6214", policyID) - - schema := fmt.Sprintf(` - type User @policy(id: "%s", resource: "user") { - name: String - age: Int - } - `, policyID, - ) - _, err = d.AddSchema(ctx, schema) - require.NoError(t, err) - - info, err := peer.AddrInfoFromString("/ip4/0.0.0.0/tcp/0/p2p/QmYyQSo1c1Ym7orWxLYvCrM2EmxFTANf8wXmmE7DWjhx5N") - require.NoError(t, err) - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: *info, - // Note: The missing explicit input of schemas here - }) - require.ErrorIs(t, err, ErrReplicatorSomeColsHavePolicy) -} - -func TestSetReplicator_WithInvalidAddress_EmptyPeerIDError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: peer.AddrInfo{}, - Schemas: []string{"User"}, - }) - require.ErrorContains(t, err, "empty peer ID") -} - -func TestSetReplicator_WithDBClosed_DatastoreClosedError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - db.Close() - - info, err := peer.AddrInfoFromString("/ip4/0.0.0.0/tcp/0/p2p/QmYyQSo1c1Ym7orWxLYvCrM2EmxFTANf8wXmmE7DWjhx5N") - require.NoError(t, err) - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: *info, - Schemas: []string{"User"}, - }) - require.ErrorContains(t, err, "datastore closed") -} - -func TestSetReplicator_WithUndefinedCollection_KeyNotFoundError(t *testing.T) { - ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() - - info, err := peer.AddrInfoFromString("/ip4/0.0.0.0/tcp/0/p2p/QmYyQSo1c1Ym7orWxLYvCrM2EmxFTANf8wXmmE7DWjhx5N") - require.NoError(t, err) - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: *info, - Schemas: []string{"User"}, - }) - require.ErrorContains(t, err, "failed to get collections for replicator: datastore: key not found") -} - -func TestSetReplicator_ForAllCollections_NoError(t *testing.T) { +func TestHandleDocCreateLog_NoError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - info, err := peer.AddrInfoFromString("/ip4/0.0.0.0/tcp/0/p2p/QmYyQSo1c1Ym7orWxLYvCrM2EmxFTANf8wXmmE7DWjhx5N") - require.NoError(t, err) - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: *info, - }) - require.NoError(t, err) -} + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() -func TestPushToReplicator_SingleDocumentNoPeer_FailedToReplicateLogError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() _, err := db.AddSchema(ctx, `type User { name: String age: Int @@ -515,313 +272,38 @@ func TestPushToReplicator_SingleDocumentNoPeer_FailedToReplicateLogError(t *test err = col.Create(ctx, doc) require.NoError(t, err) - keysCh, err := col.GetAllDocIDs(ctx) - require.NoError(t, err) - - txn, err := db.NewTxn(ctx, true) - require.NoError(t, err) - - n.pushToReplicator(ctx, txn, col, keysCh, n.PeerID()) -} - -func TestDeleteReplicator_WithDBClosed_DataStoreClosedError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - info := peer.AddrInfo{ - ID: n.PeerID(), - Addrs: n.ListenAddrs(), - } - - db.Close() - - err := n.Peer.DeleteReplicator(ctx, client.Replicator{ - Info: info, - Schemas: []string{"User"}, - }) - require.ErrorContains(t, err, "datastore closed") -} - -func TestDeleteReplicator_WithTargetSelf_SelfTargetForReplicatorError(t *testing.T) { - ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() - - err := n.Peer.DeleteReplicator(ctx, client.Replicator{ - Info: n.PeerInfo(), - Schemas: []string{"User"}, - }) - require.ErrorIs(t, err, ErrSelfTargetForReplicator) -} - -func TestDeleteReplicator_WithInvalidCollection_KeyNotFoundError(t *testing.T) { - ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() - - _, n2 := newTestNode(ctx, t) - defer n2.Close() - - err := n.Peer.DeleteReplicator(ctx, client.Replicator{ - Info: n2.PeerInfo(), - Schemas: []string{"User"}, - }) - require.ErrorContains(t, err, "failed to get collections for replicator: datastore: key not found") -} - -func TestDeleteReplicator_WithCollectionAndPreviouslySetReplicator_NoError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - _, n2 := newTestNode(ctx, t) - defer n2.Close() - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: n2.PeerInfo(), - }) - require.NoError(t, err) - - err = n.Peer.DeleteReplicator(ctx, client.Replicator{ - Info: n2.PeerInfo(), - }) - require.NoError(t, err) -} - -func TestDeleteReplicator_WithNoCollection_NoError(t *testing.T) { - ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() - - _, n2 := newTestNode(ctx, t) - defer n2.Close() - - err := n.Peer.DeleteReplicator(ctx, client.Replicator{ - Info: n2.PeerInfo(), - }) - require.NoError(t, err) -} - -func TestDeleteReplicator_WithNotSetReplicator_KeyNotFoundError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - _, n2 := newTestNode(ctx, t) - defer n2.Close() - - err = n.Peer.DeleteReplicator(ctx, client.Replicator{ - Info: n2.PeerInfo(), - Schemas: []string{"User"}, - }) - require.ErrorContains(t, err, "datastore: key not found") -} - -func TestGetAllReplicator_WithReplicator_NoError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - _, n2 := newTestNode(ctx, t) - defer n2.Close() - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: n2.PeerInfo(), - }) - require.NoError(t, err) - - reps, err := n.Peer.GetAllReplicators(ctx) + headCID, err := getHead(ctx, db, doc.ID()) require.NoError(t, err) - require.Len(t, reps, 1) - require.Equal(t, n2.PeerInfo().ID, reps[0].Info.ID) -} - -func TestGetAllReplicator_WithDBClosed_DatastoreClosedError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - db.Close() - - _, err := n.Peer.GetAllReplicators(ctx) - require.ErrorContains(t, err, "datastore closed") -} - -func TestLoadReplicators_WithDBClosed_DatastoreClosedError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - db.Close() - - err := n.Peer.loadReplicators(ctx) - require.ErrorContains(t, err, "datastore closed") -} - -func TestLoadReplicator_WithReplicator_NoError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) + b, err := db.Blockstore().AsIPLDStorage().Get(ctx, headCID.KeyString()) require.NoError(t, err) - _, n2 := newTestNode(ctx, t) - defer n2.Close() - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: n2.PeerInfo(), + err = p.handleDocCreateLog(event.Update{ + DocID: doc.ID().String(), + Cid: headCID, + SchemaRoot: col.SchemaRoot(), + Block: b, }) require.NoError(t, err) - - err = n.Peer.loadReplicators(ctx) - require.NoError(t, err) } -func TestLoadReplicator_WithReplicatorAndEmptyReplicatorMap_NoError(t *testing.T) { +func TestHandleDocCreateLog_WithInvalidDocID_NoError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - _, n2 := newTestNode(ctx, t) - defer n2.Close() - - err = n.Peer.SetReplicator(ctx, client.Replicator{ - Info: n2.PeerInfo(), + err := p.handleDocCreateLog(event.Update{ + DocID: "some-invalid-key", }) - require.NoError(t, err) - - n.replicators = make(map[string]map[peer.ID]struct{}) - - err = n.Peer.loadReplicators(ctx) - require.NoError(t, err) -} - -func TestAddP2PCollections_WithInvalidCollectionID_NotFoundError(t *testing.T) { - ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() - - err := n.Peer.AddP2PCollections(ctx, []string{"invalid_collection"}) - require.Error(t, err, ds.ErrNotFound) -} - -// This test documents that we don't allow adding p2p collections that have a policy -// until the following is implemented: -// TODO-ACP: ACP <> P2P https://github.com/sourcenetwork/defradb/issues/2366 -func TestAddP2PCollectionsWithPermissionedCollection_Error(t *testing.T) { - ctx := context.Background() - d, n := newTestNode(ctx, t) - defer n.Close() - - policy := ` - name: test - description: a policy - actor: - name: actor - resources: - user: - permissions: - read: - expr: owner - write: - expr: owner - relations: - owner: - types: - - actor - ` - - privKeyBytes, err := hex.DecodeString("028d53f37a19afb9a0dbc5b4be30c65731479ee8cfa0c9bc8f8bf198cc3c075f") - require.NoError(t, err) - privKey := secp256k1.PrivKeyFromBytes(privKeyBytes) - identity, err := acpIdentity.FromPrivateKey(privKey) - require.NoError(t, err) - - ctx = db.SetContextIdentity(ctx, identity) - policyResult, err := d.AddPolicy(ctx, policy) - policyID := policyResult.PolicyID - require.NoError(t, err) - require.Equal(t, "7b5ed30570e8d9206027ef6d5469879a6c1ea4595625c6ca33a19063a6ed6214", policyID) - - schema := fmt.Sprintf(` - type User @policy(id: "%s", resource: "user") { - name: String - age: Int - } - `, policyID, - ) - _, err = d.AddSchema(ctx, schema) - require.NoError(t, err) - - col, err := d.GetCollectionByName(ctx, "User") - require.NoError(t, err) - - err = n.Peer.AddP2PCollections(ctx, []string{col.SchemaRoot()}) - require.Error(t, err) - require.ErrorIs(t, err, ErrP2PColHasPolicy) -} - -func TestAddP2PCollections_NoError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - col, err := db.GetCollectionByName(ctx, "User") - require.NoError(t, err) - - err = n.Peer.AddP2PCollections(ctx, []string{col.SchemaRoot()}) - require.NoError(t, err) -} - -func TestRemoveP2PCollectionsWithInvalidCollectionID(t *testing.T) { - ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() - - err := n.Peer.RemoveP2PCollections(ctx, []string{"invalid_collection"}) - require.Error(t, err, ds.ErrNotFound) + require.ErrorContains(t, err, "failed to get DocID from broadcast message: selected encoding not supported") } -func TestRemoveP2PCollections(t *testing.T) { +func TestHandleDocCreateLog_WithExistingTopic_TopicExistsError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() _, err := db.AddSchema(ctx, `type User { name: String @@ -832,46 +314,27 @@ func TestRemoveP2PCollections(t *testing.T) { col, err := db.GetCollectionByName(ctx, "User") require.NoError(t, err) - err = n.Peer.RemoveP2PCollections(ctx, []string{col.SchemaRoot()}) - require.NoError(t, err) -} - -func TestGetAllP2PCollectionsWithNoCollections(t *testing.T) { - ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() - - cols, err := n.Peer.GetAllP2PCollections(ctx) - require.NoError(t, err) - require.Len(t, cols, 0) -} - -func TestGetAllP2PCollections(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) + doc, err := client.NewDocFromJSON([]byte(`{"name": "John", "age": 30}`), col.Definition()) require.NoError(t, err) - col, err := db.GetCollectionByName(ctx, "User") + err = col.Create(ctx, doc) require.NoError(t, err) - err = n.Peer.AddP2PCollections(ctx, []string{col.SchemaRoot()}) + _, err = rpc.NewTopic(ctx, p.ps, p.host.ID(), doc.ID().String(), true) require.NoError(t, err) - cols, err := n.Peer.GetAllP2PCollections(ctx) - require.NoError(t, err) - require.ElementsMatch(t, []string{col.SchemaRoot()}, cols) + err = p.handleDocCreateLog(event.Update{ + DocID: doc.ID().String(), + SchemaRoot: col.SchemaRoot(), + }) + require.ErrorContains(t, err, "topic already exists") } -func TestHandleDocCreateLog_NoError(t *testing.T) { +func TestHandleDocUpdateLog_NoError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() _, err := db.AddSchema(ctx, `type User { name: String @@ -894,7 +357,7 @@ func TestHandleDocCreateLog_NoError(t *testing.T) { b, err := db.Blockstore().AsIPLDStorage().Get(ctx, headCID.KeyString()) require.NoError(t, err) - err = n.handleDocCreateLog(event.Update{ + err = p.handleDocUpdateLog(event.Update{ DocID: doc.ID().String(), Cid: headCID, SchemaRoot: col.SchemaRoot(), @@ -903,21 +366,23 @@ func TestHandleDocCreateLog_NoError(t *testing.T) { require.NoError(t, err) } -func TestHandleDocCreateLog_WithInvalidDocID_NoError(t *testing.T) { +func TestHandleDoUpdateLog_WithInvalidDocID_NoError(t *testing.T) { ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() - err := n.handleDocCreateLog(event.Update{ + err := p.handleDocUpdateLog(event.Update{ DocID: "some-invalid-key", }) require.ErrorContains(t, err, "failed to get DocID from broadcast message: selected encoding not supported") } -func TestHandleDocCreateLog_WithExistingTopic_TopicExistsError(t *testing.T) { +func TestHandleDocUpdateLog_WithExistingDocIDTopic_TopicExistsError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() _, err := db.AddSchema(ctx, `type User { name: String @@ -934,20 +399,29 @@ func TestHandleDocCreateLog_WithExistingTopic_TopicExistsError(t *testing.T) { err = col.Create(ctx, doc) require.NoError(t, err) - _, err = rpc.NewTopic(ctx, n.ps, n.host.ID(), doc.ID().String(), true) + headCID, err := getHead(ctx, db, doc.ID()) + require.NoError(t, err) + + b, err := db.Blockstore().AsIPLDStorage().Get(ctx, headCID.KeyString()) + require.NoError(t, err) + + _, err = rpc.NewTopic(ctx, p.ps, p.host.ID(), doc.ID().String(), true) require.NoError(t, err) - err = n.handleDocCreateLog(event.Update{ + err = p.handleDocUpdateLog(event.Update{ DocID: doc.ID().String(), + Cid: headCID, SchemaRoot: col.SchemaRoot(), + Block: b, }) require.ErrorContains(t, err, "topic already exists") } -func TestHandleDocUpdateLog_NoError(t *testing.T) { +func TestHandleDocUpdateLog_WithExistingSchemaTopic_TopicExistsError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() _, err := db.AddSchema(ctx, `type User { name: String @@ -970,98 +444,219 @@ func TestHandleDocUpdateLog_NoError(t *testing.T) { b, err := db.Blockstore().AsIPLDStorage().Get(ctx, headCID.KeyString()) require.NoError(t, err) - err = n.handleDocUpdateLog(event.Update{ + _, err = rpc.NewTopic(ctx, p.ps, p.host.ID(), col.SchemaRoot(), true) + require.NoError(t, err) + + err = p.handleDocUpdateLog(event.Update{ DocID: doc.ID().String(), Cid: headCID, SchemaRoot: col.SchemaRoot(), Block: b, }) - require.NoError(t, err) + require.ErrorContains(t, err, "topic already exists") } -func TestHandleDoUpdateLog_WithInvalidDocID_NoError(t *testing.T) { +func FixtureNewMemoryDBWithBroadcaster(t *testing.T) client.DB { + var database client.DB ctx := context.Background() - _, n := newTestNode(ctx, t) - defer n.Close() + opts := badgerds.Options{Options: badger.DefaultOptions("").WithInMemory(true)} + rootstore, err := badgerds.NewDatastore("", &opts) + require.NoError(t, err) + database, err = db.NewDB(ctx, rootstore, acp.NoACP, nil) + require.NoError(t, err) + return database +} - err := n.handleDocUpdateLog(event.Update{ - DocID: "some-invalid-key", - }) - require.ErrorContains(t, err, "failed to get DocID from broadcast message: selected encoding not supported") +func TestNewPeer_WithEnableRelay_NoError(t *testing.T) { + ctx := context.Background() + store := memory.NewDatastore(ctx) + db, err := db.NewDB(ctx, store, acp.NoACP, nil) + require.NoError(t, err) + defer db.Close() + n, err := NewPeer( + context.Background(), + db.Rootstore(), + db.Blockstore(), + db.Events(), + WithEnableRelay(true), + ) + require.NoError(t, err) + n.Close() } -func TestHandleDocUpdateLog_WithExistingDocIDTopic_TopicExistsError(t *testing.T) { +func TestNewPeer_WithDBClosed_NoError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() + store := memory.NewDatastore(ctx) - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) + db, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) + db.Close() - col, err := db.GetCollectionByName(ctx, "User") - require.NoError(t, err) + _, err = NewPeer( + context.Background(), + db.Rootstore(), + db.Blockstore(), + db.Events(), + ) + require.ErrorContains(t, err, "datastore closed") +} - doc, err := client.NewDocFromJSON([]byte(`{"name": "John", "age": 30}`), col.Definition()) +func TestNewPeer_NoPubSub_NoError(t *testing.T) { + ctx := context.Background() + store := memory.NewDatastore(ctx) + db, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) + defer db.Close() - err = col.Create(ctx, doc) + n, err := NewPeer( + context.Background(), + db.Rootstore(), + db.Blockstore(), + db.Events(), + WithEnablePubSub(false), + ) require.NoError(t, err) + require.Nil(t, n.ps) + n.Close() +} - headCID, err := getHead(ctx, db, doc.ID()) +func TestNewPeer_WithEnablePubSub_NoError(t *testing.T) { + ctx := context.Background() + store := memory.NewDatastore(ctx) + db, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) + defer db.Close() - b, err := db.Blockstore().AsIPLDStorage().Get(ctx, headCID.KeyString()) - require.NoError(t, err) + n, err := NewPeer( + ctx, + db.Rootstore(), + db.Blockstore(), + db.Events(), + WithEnablePubSub(true), + ) - _, err = rpc.NewTopic(ctx, n.ps, n.host.ID(), doc.ID().String(), true) require.NoError(t, err) - - err = n.handleDocUpdateLog(event.Update{ - DocID: doc.ID().String(), - Cid: headCID, - SchemaRoot: col.SchemaRoot(), - Block: b, - }) - require.ErrorContains(t, err, "topic already exists") + // overly simple check of validity of pubsub, avoiding the process of creating a PubSub + require.NotNil(t, n.ps) + n.Close() } -func TestHandleDocUpdateLog_WithExistingSchemaTopic_TopicExistsError(t *testing.T) { +func TestNodeClose_NoError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - defer n.Close() - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) + store := memory.NewDatastore(ctx) + db, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) + defer db.Close() + n, err := NewPeer( + context.Background(), + db.Rootstore(), + db.Blockstore(), + db.Events(), + ) + require.NoError(t, err) + n.Close() +} - col, err := db.GetCollectionByName(ctx, "User") +func TestNewPeer_BootstrapWithNoPeer_NoError(t *testing.T) { + ctx := context.Background() + store := memory.NewDatastore(ctx) + db, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) + defer db.Close() - doc, err := client.NewDocFromJSON([]byte(`{"name": "John", "age": 30}`), col.Definition()) + n1, err := NewPeer( + ctx, + db.Rootstore(), + db.Blockstore(), + db.Events(), + WithListenAddresses("/ip4/0.0.0.0/tcp/0"), + ) require.NoError(t, err) + n1.Bootstrap([]peer.AddrInfo{}) + n1.Close() +} - err = col.Create(ctx, doc) +func TestNewPeer_BootstrapWithOnePeer_NoError(t *testing.T) { + ctx := context.Background() + store := memory.NewDatastore(ctx) + db, err := db.NewDB(ctx, store, acp.NoACP, nil) + require.NoError(t, err) + defer db.Close() + n1, err := NewPeer( + ctx, + db.Rootstore(), + db.Blockstore(), + db.Events(), + WithListenAddresses("/ip4/0.0.0.0/tcp/0"), + ) require.NoError(t, err) + defer n1.Close() + n2, err := NewPeer( + ctx, + db.Rootstore(), + db.Blockstore(), + db.Events(), + WithListenAddresses("/ip4/0.0.0.0/tcp/0"), + ) + require.NoError(t, err) + defer n2.Close() + addrs, err := netutils.ParsePeers([]string{n1.host.Addrs()[0].String() + "/p2p/" + n1.PeerID().String()}) + if err != nil { + t.Fatal(err) + } + n2.Bootstrap(addrs) +} - headCID, err := getHead(ctx, db, doc.ID()) +func TestNewPeer_BootstrapWithOneValidPeerAndManyInvalidPeers_NoError(t *testing.T) { + ctx := context.Background() + store := memory.NewDatastore(ctx) + db, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) + defer db.Close() - b, err := db.Blockstore().AsIPLDStorage().Get(ctx, headCID.KeyString()) + n1, err := NewPeer( + ctx, + db.Rootstore(), + db.Blockstore(), + db.Events(), + WithListenAddresses("/ip4/0.0.0.0/tcp/0"), + ) + require.NoError(t, err) + defer n1.Close() + n2, err := NewPeer( + ctx, + db.Rootstore(), + db.Blockstore(), + db.Events(), + WithListenAddresses("/ip4/0.0.0.0/tcp/0"), + ) require.NoError(t, err) + defer n2.Close() + addrs, err := netutils.ParsePeers([]string{ + n1.host.Addrs()[0].String() + "/p2p/" + n1.PeerID().String(), + "/ip4/0.0.0.0/tcp/1234/p2p/" + "12D3KooWC8YY6Tx3uAeHsdBmoy7PJPwqXAHE4HkCZ5veankKWci6", + "/ip4/0.0.0.0/tcp/1235/p2p/" + "12D3KooWC8YY6Tx3uAeHsdBmoy7PJPwqXAHE4HkCZ5veankKWci5", + "/ip4/0.0.0.0/tcp/1236/p2p/" + "12D3KooWC8YY6Tx3uAeHsdBmoy7PJPwqXAHE4HkCZ5veankKWci4", + }) + require.NoError(t, err) + n2.Bootstrap(addrs) +} - _, err = rpc.NewTopic(ctx, n.ps, n.host.ID(), col.SchemaRoot(), true) +func TestListenAddrs_WithListenAddresses_NoError(t *testing.T) { + ctx := context.Background() + store := memory.NewDatastore(ctx) + db, err := db.NewDB(ctx, store, acp.NoACP, nil) require.NoError(t, err) + defer db.Close() - err = n.handleDocUpdateLog(event.Update{ - DocID: doc.ID().String(), - Cid: headCID, - SchemaRoot: col.SchemaRoot(), - Block: b, - }) - require.ErrorContains(t, err, "topic already exists") + n, err := NewPeer( + context.Background(), + db.Rootstore(), + db.Blockstore(), + db.Events(), + WithListenAddresses("/ip4/0.0.0.0/tcp/0"), + ) + require.NoError(t, err) + require.Contains(t, n.ListenAddrs()[0].String(), "/tcp/") + n.Close() } diff --git a/net/server.go b/net/server.go index 413f391064..3b4922fe5e 100644 --- a/net/server.go +++ b/net/server.go @@ -18,7 +18,9 @@ import ( "sync" cid "github.com/ipfs/go-cid" + "github.com/libp2p/go-libp2p/core/peer" libpeer "github.com/libp2p/go-libp2p/core/peer" + "github.com/libp2p/go-libp2p/core/peerstore" "github.com/sourcenetwork/corelog" rpc "github.com/sourcenetwork/go-libp2p-pubsub-rpc" "google.golang.org/grpc" @@ -43,7 +45,9 @@ type server struct { opts []grpc.DialOption topics map[string]pubsubTopic - mu sync.Mutex + // replicators is a map from collectionName => peerId + replicators map[string]map[peer.ID]struct{} + mu sync.Mutex conns map[libpeer.ID]*grpc.ClientConn @@ -61,9 +65,10 @@ type pubsubTopic struct { // underlying DB instance. func newServer(p *Peer, opts ...grpc.DialOption) (*server, error) { s := &server{ - peer: p, - conns: make(map[libpeer.ID]*grpc.ClientConn), - topics: make(map[string]pubsubTopic), + peer: p, + conns: make(map[libpeer.ID]*grpc.ClientConn), + topics: make(map[string]pubsubTopic), + replicators: make(map[string]map[peer.ID]struct{}), } cred := insecure.NewCredentials() @@ -73,38 +78,6 @@ func newServer(p *Peer, opts ...grpc.DialOption) (*server, error) { } s.opts = append(defaultOpts, opts...) - if s.peer.ps != nil { - colMap, err := p.loadP2PCollections(p.ctx) - if err != nil { - return nil, err - } - - // Get all DocIDs across all collections in the DB - cols, err := s.peer.db.GetCollections(s.peer.ctx, client.CollectionFetchOptions{}) - if err != nil { - return nil, err - } - - i := 0 - for _, col := range cols { - // If we subscribed to the collection, we skip subscribing to the collection's docIDs. - if _, ok := colMap[col.SchemaRoot()]; ok { - continue - } - // TODO-ACP: Support ACP <> P2P - https://github.com/sourcenetwork/defradb/issues/2366 - docIDChan, err := col.GetAllDocIDs(p.ctx) - if err != nil { - return nil, err - } - - for docID := range docIDChan { - if err := s.addPubSubTopic(docID.ID.String(), true); err != nil { - return nil, err - } - i++ - } - } - } return s, nil } @@ -157,7 +130,7 @@ func (s *server) PushLog(ctx context.Context, req *pb.PushLogRequest) (*pb.PushL return nil, err } - s.peer.db.Events().Publish(event.NewMessage(event.MergeName, event.Merge{ + s.peer.bus.Publish(event.NewMessage(event.MergeName, event.Merge{ DocID: docID.String(), ByPeer: byPeer, FromPeer: pid, @@ -313,7 +286,7 @@ func (s *server) pubSubEventHandler(from libpeer.ID, topic string, msg []byte) { evt := event.NewMessage(event.PubSubName, event.PubSub{ Peer: from, }) - s.peer.db.Events().Publish(evt) + s.peer.bus.Publish(evt) } // addr implements net.Addr and holds a libp2p peer ID. @@ -337,3 +310,64 @@ func peerIDFromContext(ctx context.Context) (libpeer.ID, error) { } return pid, nil } + +func (s *server) updatePubSubTopics(evt event.P2PTopic) { + for _, topic := range evt.ToAdd { + err := s.addPubSubTopic(topic, true) + if err != nil { + log.ErrorContextE(s.peer.ctx, "Failed to add pubsub topic.", err) + } + } + + for _, topic := range evt.ToRemove { + err := s.removePubSubTopic(topic) + if err != nil { + log.ErrorContextE(s.peer.ctx, "Failed to remove pubsub topic.", err) + } + } + s.peer.bus.Publish(event.NewMessage(event.P2PTopicCompletedName, nil)) +} + +func (s *server) updateReplicators(evt event.Replicator) { + isDeleteRep := len(evt.Schemas) == 0 + // update the cached replicators + s.mu.Lock() + for schema, peers := range s.replicators { + if _, hasSchema := evt.Schemas[schema]; hasSchema { + s.replicators[schema][evt.Info.ID] = struct{}{} + delete(evt.Schemas, schema) + } else { + if _, exists := peers[evt.Info.ID]; exists { + delete(s.replicators[schema], evt.Info.ID) + } + } + } + for schema := range evt.Schemas { + if _, exists := s.replicators[schema]; !exists { + s.replicators[schema] = make(map[peer.ID]struct{}) + } + s.replicators[schema][evt.Info.ID] = struct{}{} + } + s.mu.Unlock() + + if isDeleteRep { + s.peer.host.Peerstore().ClearAddrs(evt.Info.ID) + } else { + s.peer.host.Peerstore().AddAddrs(evt.Info.ID, evt.Info.Addrs, peerstore.PermanentAddrTTL) + } + + if evt.Docs != nil { + for update := range evt.Docs { + if err := s.pushLog(s.peer.ctx, update, evt.Info.ID); err != nil { + log.ErrorContextE( + s.peer.ctx, + "Failed to replicate log", + err, + corelog.Any("CID", update.Cid), + corelog.Any("PeerID", evt.Info.ID), + ) + } + } + } + s.peer.bus.Publish(event.NewMessage(event.ReplicatorCompletedName, nil)) +} diff --git a/net/server_test.go b/net/server_test.go index d17705b404..1ac178a2d1 100644 --- a/net/server_test.go +++ b/net/server_test.go @@ -18,12 +18,10 @@ import ( "github.com/ipfs/go-datastore/query" "github.com/libp2p/go-libp2p/core/event" "github.com/libp2p/go-libp2p/core/host" - rpc "github.com/sourcenetwork/go-libp2p-pubsub-rpc" "github.com/stretchr/testify/require" grpcpeer "google.golang.org/grpc/peer" "github.com/sourcenetwork/defradb/client" - "github.com/sourcenetwork/defradb/datastore/memory" "github.com/sourcenetwork/defradb/errors" "github.com/sourcenetwork/defradb/internal/core" net_pb "github.com/sourcenetwork/defradb/net/pb" @@ -31,124 +29,15 @@ import ( func TestNewServerSimple(t *testing.T) { ctx := context.Background() - _, n := newTestNode(ctx, t) - _, err := newServer(n.Peer) + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() + _, err := newServer(p) require.NoError(t, err) } -func TestNewServerWithDBClosed(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - db.Close() - - _, err := newServer(n.Peer) - require.ErrorIs(t, err, memory.ErrClosed) -} - var mockError = errors.New("mock error") -type mockDBColError struct { - client.DB -} - -func (mDB *mockDBColError) GetCollections(context.Context, client.CollectionFetchOptions) ([]client.Collection, error) { - return nil, mockError -} - -func TestNewServerWithGetAllCollectionError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - mDB := mockDBColError{db} - n.Peer.db = &mDB - _, err := newServer(n.Peer) - require.ErrorIs(t, err, mockError) -} - -func TestNewServerWithCollectionSubscribed(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - col, err := db.GetCollectionByName(ctx, "User") - require.NoError(t, err) - - err = n.AddP2PCollections(ctx, []string{col.SchemaRoot()}) - require.NoError(t, err) - - _, err = newServer(n.Peer) - require.NoError(t, err) -} - -type mockDBDocIDsError struct { - client.DB -} - -func (mDB *mockDBDocIDsError) GetCollections(context.Context, client.CollectionFetchOptions) ([]client.Collection, error) { - return []client.Collection{ - &mockCollection{}, - }, nil -} - -type mockCollection struct { - client.Collection -} - -func (mCol *mockCollection) SchemaRoot() string { - return "mockColID" -} -func (mCol *mockCollection) GetAllDocIDs( - ctx context.Context, -) (<-chan client.DocIDResult, error) { - return nil, mockError -} - -func TestNewServerWithGetAllDocIDsError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - mDB := mockDBDocIDsError{db} - n.Peer.db = &mDB - _, err = newServer(n.Peer) - require.ErrorIs(t, err, mockError) -} - -func TestNewServerWithAddTopicError(t *testing.T) { - ctx := context.Background() - db, n := newTestNode(ctx, t) - - _, err := db.AddSchema(ctx, `type User { - name: String - age: Int - }`) - require.NoError(t, err) - - col, err := db.GetCollectionByName(ctx, "User") - require.NoError(t, err) - - doc, err := client.NewDocFromJSON([]byte(`{"name": "John", "age": 30}`), col.Definition()) - require.NoError(t, err) - - err = col.Create(ctx, doc) - require.NoError(t, err) - - _, err = rpc.NewTopic(ctx, n.Peer.ps, n.Peer.host.ID(), doc.ID().String(), true) - require.NoError(t, err) - - _, err = newServer(n.Peer) - require.ErrorContains(t, err, "topic already exists") -} - type mockHost struct { host.Host } @@ -171,7 +60,9 @@ func (mB *mockBus) Subscribe(eventType any, opts ...event.SubscriptionOpt) (even func TestNewServerWithEmitterError(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() _, err := db.AddSchema(ctx, `type User { name: String @@ -188,40 +79,48 @@ func TestNewServerWithEmitterError(t *testing.T) { err = col.Create(ctx, doc) require.NoError(t, err) - n.Peer.host = &mockHost{n.Peer.host} + p.host = &mockHost{p.host} - _, err = newServer(n.Peer) + _, err = newServer(p) require.NoError(t, err) } func TestGetDocGraph(t *testing.T) { ctx := context.Background() - _, n := newTestNode(ctx, t) - r, err := n.server.GetDocGraph(ctx, &net_pb.GetDocGraphRequest{}) + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() + r, err := p.server.GetDocGraph(ctx, &net_pb.GetDocGraphRequest{}) require.Nil(t, r) require.Nil(t, err) } func TestPushDocGraph(t *testing.T) { ctx := context.Background() - _, n := newTestNode(ctx, t) - r, err := n.server.PushDocGraph(ctx, &net_pb.PushDocGraphRequest{}) + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() + r, err := p.server.PushDocGraph(ctx, &net_pb.PushDocGraphRequest{}) require.Nil(t, r) require.Nil(t, err) } func TestGetLog(t *testing.T) { ctx := context.Background() - _, n := newTestNode(ctx, t) - r, err := n.server.GetLog(ctx, &net_pb.GetLogRequest{}) + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() + r, err := p.server.GetLog(ctx, &net_pb.GetLogRequest{}) require.Nil(t, r) require.Nil(t, err) } func TestGetHeadLog(t *testing.T) { ctx := context.Background() - _, n := newTestNode(ctx, t) - r, err := n.server.GetHeadLog(ctx, &net_pb.GetHeadLogRequest{}) + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() + r, err := p.server.GetHeadLog(ctx, &net_pb.GetHeadLogRequest{}) require.Nil(t, r) require.Nil(t, err) } @@ -249,8 +148,10 @@ func getHead(ctx context.Context, db client.DB, docID client.DocID) (cid.Cid, er func TestPushLog(t *testing.T) { ctx := context.Background() - db, n := newTestNode(ctx, t) - err := n.Start() + db, p := newTestPeer(ctx, t) + defer db.Close() + defer p.Close() + err := p.Start() require.NoError(t, err) _, err = db.AddSchema(ctx, `type User { @@ -266,7 +167,7 @@ func TestPushLog(t *testing.T) { require.NoError(t, err) ctx = grpcpeer.NewContext(ctx, &grpcpeer.Peer{ - Addr: addr{n.PeerID()}, + Addr: addr{p.PeerID()}, }) err = col.Create(ctx, doc) @@ -278,12 +179,12 @@ func TestPushLog(t *testing.T) { b, err := db.Blockstore().AsIPLDStorage().Get(ctx, headCID.KeyString()) require.NoError(t, err) - _, err = n.server.PushLog(ctx, &net_pb.PushLogRequest{ + _, err = p.server.PushLog(ctx, &net_pb.PushLogRequest{ Body: &net_pb.PushLogRequest_Body{ DocID: []byte(doc.ID().String()), Cid: headCID.Bytes(), SchemaRoot: []byte(col.SchemaRoot()), - Creator: n.PeerID().String(), + Creator: p.PeerID().String(), Log: &net_pb.Document_Log{ Block: b, }, diff --git a/node/node.go b/node/node.go index 215cf05fc7..6dad35f593 100644 --- a/node/node.go +++ b/node/node.go @@ -77,7 +77,7 @@ func WithPeers(peers ...peer.AddrInfo) NodeOpt { // Node is a DefraDB instance with optional sub-systems. type Node struct { DB client.DB - Node *net.Node + Peer *net.Peer Server *http.Server } @@ -138,27 +138,22 @@ func NewNode(ctx context.Context, opts ...Option) (*Node, error) { return nil, err } - var node *net.Node + var peer *net.Peer if !options.disableP2P { // setup net node - node, err = net.NewNode(ctx, db, netOpts...) + peer, err = net.NewPeer(ctx, db.Rootstore(), db.Blockstore(), db.Events(), netOpts...) if err != nil { return nil, err } if len(options.peers) > 0 { - node.Bootstrap(options.peers) + peer.Bootstrap(options.peers) } } var server *http.Server if !options.disableAPI { // setup http server - var handler *http.Handler - if node != nil { - handler, err = http.NewHandler(node) - } else { - handler, err = http.NewHandler(db) - } + handler, err := http.NewHandler(db) if err != nil { return nil, err } @@ -170,15 +165,15 @@ func NewNode(ctx context.Context, opts ...Option) (*Node, error) { return &Node{ DB: db, - Node: node, + Peer: peer, Server: server, }, nil } // Start starts the node sub-systems. func (n *Node) Start(ctx context.Context) error { - if n.Node != nil { - if err := n.Node.Start(); err != nil { + if n.Peer != nil { + if err := n.Peer.Start(); err != nil { return err } } @@ -203,9 +198,10 @@ func (n *Node) Close(ctx context.Context) error { if n.Server != nil { err = n.Server.Shutdown(ctx) } - if n.Node != nil { - n.Node.Close() - } else { + if n.Peer != nil { + n.Peer.Close() + } + if n.DB != nil { n.DB.Close() } return err diff --git a/node/store.go b/node/store.go index 8354c0f7df..1a5b46f8e0 100644 --- a/node/store.go +++ b/node/store.go @@ -77,7 +77,7 @@ func WithEncryptionKey(encryptionKey []byte) StoreOpt { } // NewStore returns a new store with the given options. -func NewStore(ctx context.Context, opts ...StoreOpt) (datastore.RootStore, error) { +func NewStore(ctx context.Context, opts ...StoreOpt) (datastore.Rootstore, error) { options := DefaultStoreOptions() for _, opt := range opts { opt(options) diff --git a/tests/bench/bench_util.go b/tests/bench/bench_util.go index 5993ee50f8..a574088148 100644 --- a/tests/bench/bench_util.go +++ b/tests/bench/bench_util.go @@ -216,7 +216,7 @@ func NewTestDB(ctx context.Context, t testing.TB) (client.DB, error) { func NewTestStorage(ctx context.Context, t testing.TB) (ds.Batching, error) { dbi, err := newBenchStoreInfo(ctx, t) - return dbi.Root(), err + return dbi.Rootstore(), err } func newBenchStoreInfo(ctx context.Context, t testing.TB) (client.DB, error) { diff --git a/tests/bench/query/planner/utils.go b/tests/bench/query/planner/utils.go index b91b0aa2a3..5a842222f5 100644 --- a/tests/bench/query/planner/utils.go +++ b/tests/bench/query/planner/utils.go @@ -136,7 +136,7 @@ func (*dummyTxn) Rootstore() datastore.DSReaderWriter { return nil } func (*dummyTxn) Datastore() datastore.DSReaderWriter { return nil } func (*dummyTxn) Headstore() datastore.DSReaderWriter { return nil } func (*dummyTxn) Peerstore() datastore.DSBatching { return nil } -func (*dummyTxn) DAGstore() datastore.DAGStore { return nil } +func (*dummyTxn) Blockstore() datastore.Blockstore { return nil } func (*dummyTxn) Systemstore() datastore.DSReaderWriter { return nil } func (*dummyTxn) Commit(ctx context.Context) error { return nil } func (*dummyTxn) Discard(ctx context.Context) {} diff --git a/tests/bench/storage/utils.go b/tests/bench/storage/utils.go index 5c550f25db..c413f0bd42 100644 --- a/tests/bench/storage/utils.go +++ b/tests/bench/storage/utils.go @@ -69,7 +69,7 @@ func runStorageBenchTxnGet( if err != nil { return err } - defer db.Root().Close() //nolint:errcheck + defer db.Rootstore().Close() //nolint:errcheck keys, err := backfillBenchmarkTxn(ctx, db, objCount, valueSize) if err != nil { @@ -109,7 +109,7 @@ func runStorageBenchTxnIterator( if err != nil { return err } - defer db.Root().Close() //nolint:errcheck + defer db.Rootstore().Close() //nolint:errcheck keys, err := backfillBenchmarkTxn(ctx, db, objCount, valueSize) if err != nil { diff --git a/tests/clients/cli/wrapper.go b/tests/clients/cli/wrapper.go index 18e306c0f4..a6f3feef4c 100644 --- a/tests/clients/cli/wrapper.go +++ b/tests/clients/cli/wrapper.go @@ -31,20 +31,20 @@ import ( "github.com/sourcenetwork/defradb/datastore" "github.com/sourcenetwork/defradb/event" "github.com/sourcenetwork/defradb/http" - "github.com/sourcenetwork/defradb/net" + "github.com/sourcenetwork/defradb/node" ) -var _ client.P2P = (*Wrapper)(nil) +var _ client.DB = (*Wrapper)(nil) type Wrapper struct { - node *net.Node + node *node.Node cmd *cliWrapper handler *http.Handler httpServer *httptest.Server } -func NewWrapper(node *net.Node) (*Wrapper, error) { - handler, err := http.NewHandler(node) +func NewWrapper(node *node.Node) (*Wrapper, error) { + handler, err := http.NewHandler(node.DB) if err != nil { return nil, err } @@ -503,40 +503,40 @@ func (w *Wrapper) NewConcurrentTxn(ctx context.Context, readOnly bool) (datastor return &Transaction{tx, w.cmd}, nil } -func (w *Wrapper) Root() datastore.RootStore { - return w.node.Root() +func (w *Wrapper) Rootstore() datastore.Rootstore { + return w.node.DB.Rootstore() } -func (w *Wrapper) Blockstore() datastore.DAGStore { - return w.node.Blockstore() +func (w *Wrapper) Blockstore() datastore.Blockstore { + return w.node.DB.Blockstore() } func (w *Wrapper) Headstore() ds.Read { - return w.node.Headstore() + return w.node.DB.Headstore() } func (w *Wrapper) Peerstore() datastore.DSBatching { - return w.node.Peerstore() + return w.node.DB.Peerstore() } func (w *Wrapper) Close() { w.httpServer.CloseClientConnections() w.httpServer.Close() - w.node.Close() + _ = w.node.Close(context.Background()) } func (w *Wrapper) Events() *event.Bus { - return w.node.Events() + return w.node.DB.Events() } func (w *Wrapper) MaxTxnRetries() int { - return w.node.MaxTxnRetries() + return w.node.DB.MaxTxnRetries() } func (w *Wrapper) PrintDump(ctx context.Context) error { - return w.node.PrintDump(ctx) + return w.node.DB.PrintDump(ctx) } func (w *Wrapper) Bootstrap(addrs []peer.AddrInfo) { - w.node.Bootstrap(addrs) + w.node.Peer.Bootstrap(addrs) } diff --git a/tests/clients/cli/wrapper_tx.go b/tests/clients/cli/wrapper_tx.go index 5b5b2c3ea7..0330b8d47e 100644 --- a/tests/clients/cli/wrapper_tx.go +++ b/tests/clients/cli/wrapper_tx.go @@ -83,8 +83,8 @@ func (w *Transaction) Peerstore() datastore.DSBatching { return w.tx.Peerstore() } -func (w *Transaction) DAGstore() datastore.DAGStore { - return w.tx.DAGstore() +func (w *Transaction) Blockstore() datastore.Blockstore { + return w.tx.Blockstore() } func (w *Transaction) Systemstore() datastore.DSReaderWriter { diff --git a/tests/clients/clients.go b/tests/clients/clients.go index 249b1e767f..f5d822ab39 100644 --- a/tests/clients/clients.go +++ b/tests/clients/clients.go @@ -19,6 +19,6 @@ import ( // Client implements the P2P interface along with a few other methods // required for testing. type Client interface { - client.P2P + client.DB Bootstrap([]peer.AddrInfo) } diff --git a/tests/clients/http/wrapper.go b/tests/clients/http/wrapper.go index 89b5bce5e7..f183a4671a 100644 --- a/tests/clients/http/wrapper.go +++ b/tests/clients/http/wrapper.go @@ -24,22 +24,22 @@ import ( "github.com/sourcenetwork/defradb/datastore" "github.com/sourcenetwork/defradb/event" "github.com/sourcenetwork/defradb/http" - "github.com/sourcenetwork/defradb/net" + "github.com/sourcenetwork/defradb/node" ) -var _ client.P2P = (*Wrapper)(nil) +var _ client.DB = (*Wrapper)(nil) // Wrapper combines an HTTP client and server into a // single struct that implements the client.DB interface. type Wrapper struct { - node *net.Node + node *node.Node handler *http.Handler client *http.Client httpServer *httptest.Server } -func NewWrapper(node *net.Node) (*Wrapper, error) { - handler, err := http.NewHandler(node) +func NewWrapper(node *node.Node) (*Wrapper, error) { + handler, err := http.NewHandler(node.DB) if err != nil { return nil, err } @@ -199,40 +199,40 @@ func (w *Wrapper) NewConcurrentTxn(ctx context.Context, readOnly bool) (datastor return &TxWrapper{server, client}, nil } -func (w *Wrapper) Root() datastore.RootStore { - return w.node.Root() +func (w *Wrapper) Rootstore() datastore.Rootstore { + return w.node.DB.Rootstore() } -func (w *Wrapper) Blockstore() datastore.DAGStore { - return w.node.Blockstore() +func (w *Wrapper) Blockstore() datastore.Blockstore { + return w.node.DB.Blockstore() } func (w *Wrapper) Headstore() ds.Read { - return w.node.Headstore() + return w.node.DB.Headstore() } func (w *Wrapper) Peerstore() datastore.DSBatching { - return w.node.Peerstore() + return w.node.DB.Peerstore() } func (w *Wrapper) Close() { w.httpServer.CloseClientConnections() w.httpServer.Close() - w.node.Close() + _ = w.node.Close(context.Background()) } func (w *Wrapper) Events() *event.Bus { - return w.node.Events() + return w.node.DB.Events() } func (w *Wrapper) MaxTxnRetries() int { - return w.node.MaxTxnRetries() + return w.node.DB.MaxTxnRetries() } func (w *Wrapper) PrintDump(ctx context.Context) error { - return w.node.PrintDump(ctx) + return w.node.DB.PrintDump(ctx) } func (w *Wrapper) Bootstrap(addrs []peer.AddrInfo) { - w.node.Bootstrap(addrs) + w.node.Peer.Bootstrap(addrs) } diff --git a/tests/clients/http/wrapper_tx.go b/tests/clients/http/wrapper_tx.go index d53d967b3b..133d3bc1d3 100644 --- a/tests/clients/http/wrapper_tx.go +++ b/tests/clients/http/wrapper_tx.go @@ -77,8 +77,8 @@ func (w *TxWrapper) Peerstore() datastore.DSBatching { return w.server.Peerstore() } -func (w *TxWrapper) DAGstore() datastore.DAGStore { - return w.server.DAGstore() +func (w *TxWrapper) Blockstore() datastore.Blockstore { + return w.server.Blockstore() } func (w *TxWrapper) Systemstore() datastore.DSReaderWriter { diff --git a/tests/integration/client.go b/tests/integration/client.go index 1d06bfc744..dee5a3f0c9 100644 --- a/tests/integration/client.go +++ b/tests/integration/client.go @@ -15,7 +15,11 @@ import ( "os" "strconv" + "github.com/libp2p/go-libp2p/core/peer" + + "github.com/sourcenetwork/defradb/client" "github.com/sourcenetwork/defradb/net" + "github.com/sourcenetwork/defradb/node" "github.com/sourcenetwork/defradb/tests/clients" "github.com/sourcenetwork/defradb/tests/clients/cli" "github.com/sourcenetwork/defradb/tests/clients/http" @@ -63,7 +67,7 @@ func init() { // setupClient returns the client implementation for the current // testing state. The client type on the test state is used to // select the client implementation to use. -func setupClient(s *state, node *net.Node) (impl clients.Client, err error) { +func setupClient(s *state, node *node.Node) (impl clients.Client, err error) { switch s.clientType { case HTTPClientType: impl, err = http.NewWrapper(node) @@ -72,7 +76,7 @@ func setupClient(s *state, node *net.Node) (impl clients.Client, err error) { impl, err = cli.NewWrapper(node) case GoClientType: - impl = node + impl = newGoClientWrapper(node) default: err = fmt.Errorf("invalid client type: %v", s.dbt) @@ -83,3 +87,28 @@ func setupClient(s *state, node *net.Node) (impl clients.Client, err error) { } return } + +type goClientWrapper struct { + client.DB + peer *net.Peer +} + +func newGoClientWrapper(n *node.Node) *goClientWrapper { + return &goClientWrapper{ + DB: n.DB, + peer: n.Peer, + } +} + +func (w *goClientWrapper) Bootstrap(addrs []peer.AddrInfo) { + if w.peer != nil { + w.peer.Bootstrap(addrs) + } +} + +func (w *goClientWrapper) Close() { + if w.peer != nil { + w.peer.Close() + } + w.DB.Close() +} diff --git a/tests/integration/db.go b/tests/integration/db.go index ab15e2d5fc..dbbbed7ffb 100644 --- a/tests/integration/db.go +++ b/tests/integration/db.go @@ -97,10 +97,10 @@ func NewBadgerFileDB(ctx context.Context, t testing.TB) (client.DB, error) { return node.DB, err } -// setupDatabase returns the database implementation for the current +// setupNode returns the database implementation for the current // testing state. The database type on the test state is used to // select the datastore implementation to use. -func setupDatabase(s *state) (client.DB, string, error) { +func setupNode(s *state) (*node.Node, string, error) { opts := []node.Option{ node.WithLensPoolSize(lensPoolSize), // The test framework sets this up elsewhere when required so that it may be wrapped @@ -158,5 +158,5 @@ func setupDatabase(s *state) (client.DB, string, error) { return nil, "", err } - return node.DB, path, nil + return node, path, nil } diff --git a/tests/integration/p2p.go b/tests/integration/p2p.go index d990a3d322..4a57ac7a1a 100644 --- a/tests/integration/p2p.go +++ b/tests/integration/p2p.go @@ -288,9 +288,15 @@ func configureReplicator( sourceNode := s.nodes[cfg.SourceNodeID] targetNode := s.nodes[cfg.TargetNodeID] - err := sourceNode.SetReplicator(s.ctx, client.Replicator{ + sub, err := sourceNode.Events().Subscribe(event.ReplicatorCompletedName) + require.NoError(s.t, err) + err = sourceNode.SetReplicator(s.ctx, client.Replicator{ Info: targetNode.PeerInfo(), }) + if err == nil { + // wait for the replicator setup to complete + <-sub.Message() + } expectedErrorRaised := AssertError(s.t, s.testCase.Description, err, cfg.ExpectedError) assertExpectedErrorRaised(s.t, s.testCase.Description, cfg.ExpectedError, expectedErrorRaised) @@ -306,9 +312,15 @@ func deleteReplicator( sourceNode := s.nodes[cfg.SourceNodeID] targetNode := s.nodes[cfg.TargetNodeID] - err := sourceNode.DeleteReplicator(s.ctx, client.Replicator{ + sub, err := sourceNode.Events().Subscribe(event.ReplicatorCompletedName) + require.NoError(s.t, err) + err = sourceNode.DeleteReplicator(s.ctx, client.Replicator{ Info: targetNode.PeerInfo(), }) + if err == nil { + // wait for the replicator setup to complete + <-sub.Message() + } require.NoError(s.t, err) } @@ -390,7 +402,15 @@ func subscribeToCollection( schemaRoots = append(schemaRoots, col.SchemaRoot()) } - err := n.AddP2PCollections(s.ctx, schemaRoots) + sub, err := n.Events().Subscribe(event.P2PTopicCompletedName) + require.NoError(s.t, err) + + err = n.AddP2PCollections(s.ctx, schemaRoots) + if err == nil { + // wait for the p2p collection setup to complete + <-sub.Message() + } + expectedErrorRaised := AssertError(s.t, s.testCase.Description, err, action.ExpectedError) assertExpectedErrorRaised(s.t, s.testCase.Description, action.ExpectedError, expectedErrorRaised) @@ -420,7 +440,15 @@ func unsubscribeToCollection( schemaRoots = append(schemaRoots, col.SchemaRoot()) } - err := n.RemoveP2PCollections(s.ctx, schemaRoots) + sub, err := n.Events().Subscribe(event.P2PTopicCompletedName) + require.NoError(s.t, err) + + err = n.RemoveP2PCollections(s.ctx, schemaRoots) + if err == nil { + // wait for the p2p collection setup to complete + <-sub.Message() + } + expectedErrorRaised := AssertError(s.t, s.testCase.Description, err, action.ExpectedError) assertExpectedErrorRaised(s.t, s.testCase.Description, action.ExpectedError, expectedErrorRaised) diff --git a/tests/integration/utils2.go b/tests/integration/utils2.go index 42ab28c04c..fab8cc5ed9 100644 --- a/tests/integration/utils2.go +++ b/tests/integration/utils2.go @@ -649,10 +649,10 @@ func setStartingNodes( // If nodes have not been explicitly configured via actions, setup a default one. if !hasExplicitNode { - db, path, err := setupDatabase(s) + node, path, err := setupNode(s) require.Nil(s.t, err) - c, err := setupClient(s, &net.Node{DB: db}) + c, err := setupClient(s, node) require.Nil(s.t, err) s.nodes = append(s.nodes, c) @@ -673,19 +673,35 @@ func restartNodes( for i := len(s.nodes) - 1; i >= 0; i-- { originalPath := databaseDir databaseDir = s.dbPaths[i] - db, _, err := setupDatabase(s) + node, _, err := setupNode(s) require.Nil(s.t, err) databaseDir = originalPath if len(s.nodeConfigs) == 0 { // If there are no explicit node configuration actions the node will be // basic (i.e. no P2P stuff) and can be yielded now. - c, err := setupClient(s, &net.Node{DB: db}) + c, err := setupClient(s, node) require.NoError(s.t, err) s.nodes[i] = c continue } + // We need to ensure that on restart, the node pubsub is configured before + // we continue with the test. Otherwise, we may miss update events. + readySub, err := node.DB.Events().Subscribe(event.P2PTopicCompletedName, event.ReplicatorCompletedName) + require.NoError(s.t, err) + waitLen := 0 + cols, err := node.DB.GetAllP2PCollections(s.ctx) + require.NoError(s.t, err) + if len(cols) > 0 { + // there is only one message for loading of P2P collections + waitLen++ + } + reps, err := node.DB.GetAllReplicators(s.ctx) + require.NoError(s.t, err) + // there is one message per replicator + waitLen += len(reps) + // We need to make sure the node is configured with its old address, otherwise // a new one may be selected and reconnnection to it will fail. var addresses []string @@ -696,16 +712,16 @@ func restartNodes( nodeOpts := s.nodeConfigs[i] nodeOpts = append(nodeOpts, net.WithListenAddresses(addresses...)) - var n *net.Node - n, err = net.NewNode(s.ctx, db, nodeOpts...) + p, err := net.NewPeer(s.ctx, node.DB.Rootstore(), node.DB.Blockstore(), node.DB.Events(), nodeOpts...) require.NoError(s.t, err) - if err := n.Start(); err != nil { - n.Close() + if err := p.Start(); err != nil { + p.Close() require.NoError(s.t, err) } + node.Peer = p - c, err := setupClient(s, n) + c, err := setupClient(s, node) require.NoError(s.t, err) s.nodes[i] = c @@ -713,6 +729,14 @@ func restartNodes( sub, err := c.Events().Subscribe(event.MergeCompleteName) require.NoError(s.t, err) s.eventSubs[i] = sub + for waitLen > 0 { + select { + case <-readySub.Message(): + waitLen-- + case <-time.After(10 * time.Second): + s.t.Fatalf("timeout waiting for node to be ready") + } + } } // The index of the action after the last wait action before the current restart action. @@ -787,7 +811,7 @@ func configureNode( return } - db, path, err := setupDatabase(s) //disable change dector, or allow it? + node, path, err := setupNode(s) //disable change dector, or allow it? require.NoError(s.t, err) privateKey, err := crypto.GenerateEd25519() @@ -796,20 +820,21 @@ func configureNode( nodeOpts := action() nodeOpts = append(nodeOpts, net.WithPrivateKey(privateKey)) - var n *net.Node - n, err = net.NewNode(s.ctx, db, nodeOpts...) + p, err := net.NewPeer(s.ctx, node.DB.Rootstore(), node.DB.Blockstore(), node.DB.Events(), nodeOpts...) require.NoError(s.t, err) - log.InfoContext(s.ctx, "Starting P2P node", corelog.Any("P2P address", n.PeerInfo())) - if err := n.Start(); err != nil { - n.Close() + log.InfoContext(s.ctx, "Starting P2P node", corelog.Any("P2P address", p.PeerInfo())) + if err := p.Start(); err != nil { + p.Close() require.NoError(s.t, err) } - s.nodeAddresses = append(s.nodeAddresses, n.PeerInfo()) + s.nodeAddresses = append(s.nodeAddresses, p.PeerInfo()) s.nodeConfigs = append(s.nodeConfigs, nodeOpts) - c, err := setupClient(s, n) + node.Peer = p + + c, err := setupClient(s, node) require.NoError(s.t, err) s.nodes = append(s.nodes, c) @@ -1144,7 +1169,7 @@ func createDoc( substituteRelations(s, action) } - var mutation func(*state, CreateDoc, client.P2P, []client.Collection) (*client.Document, error) + var mutation func(*state, CreateDoc, client.DB, []client.Collection) (*client.Document, error) switch mutationType { case CollectionSaveMutationType: @@ -1185,7 +1210,7 @@ func createDoc( func createDocViaColSave( s *state, action CreateDoc, - node client.P2P, + node client.DB, collections []client.Collection, ) (*client.Document, error) { var err error @@ -1210,7 +1235,7 @@ func createDocViaColSave( func createDocViaColCreate( s *state, action CreateDoc, - node client.P2P, + node client.DB, collections []client.Collection, ) (*client.Document, error) { var err error @@ -1235,7 +1260,7 @@ func createDocViaColCreate( func createDocViaGQL( s *state, action CreateDoc, - node client.P2P, + node client.DB, collections []client.Collection, ) (*client.Document, error) { collection := collections[action.CollectionID] @@ -1337,7 +1362,7 @@ func updateDoc( s *state, action UpdateDoc, ) { - var mutation func(*state, UpdateDoc, client.P2P, []client.Collection) error + var mutation func(*state, UpdateDoc, client.DB, []client.Collection) error switch mutationType { case CollectionSaveMutationType: @@ -1367,7 +1392,7 @@ func updateDoc( func updateDocViaColSave( s *state, action UpdateDoc, - node client.P2P, + node client.DB, collections []client.Collection, ) error { cachedDoc := s.documents[action.CollectionID][action.DocID] @@ -1394,7 +1419,7 @@ func updateDocViaColSave( func updateDocViaColUpdate( s *state, action UpdateDoc, - node client.P2P, + node client.DB, collections []client.Collection, ) error { cachedDoc := s.documents[action.CollectionID][action.DocID] @@ -1418,7 +1443,7 @@ func updateDocViaColUpdate( func updateDocViaGQL( s *state, action UpdateDoc, - node client.P2P, + node client.DB, collections []client.Collection, ) error { doc := s.documents[action.CollectionID][action.DocID]