2020-10-28 14:49:30 +00:00
|
|
|
package meta
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/rand"
|
|
|
|
"crypto/sha256"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/nspcc-dev/neofs-api-go/pkg"
|
|
|
|
"github.com/nspcc-dev/neofs-api-go/pkg/container"
|
|
|
|
objectSDK "github.com/nspcc-dev/neofs-api-go/pkg/object"
|
|
|
|
"github.com/nspcc-dev/neofs-api-go/pkg/owner"
|
2020-10-29 16:12:38 +00:00
|
|
|
v2object "github.com/nspcc-dev/neofs-api-go/v2/object"
|
2020-10-28 14:49:30 +00:00
|
|
|
"github.com/nspcc-dev/neofs-node/pkg/core/object"
|
|
|
|
"github.com/nspcc-dev/neofs-node/pkg/util/test"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"go.etcd.io/bbolt"
|
|
|
|
)
|
|
|
|
|
|
|
|
func testSelect(t *testing.T, db *DB, fs objectSDK.SearchFilters, exp ...*objectSDK.Address) {
|
|
|
|
res, err := db.Select(fs)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Len(t, res, len(exp))
|
|
|
|
|
|
|
|
for i := range exp {
|
|
|
|
require.Contains(t, res, exp[i])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-29 12:03:55 +00:00
|
|
|
func testCID() *container.ID {
|
|
|
|
cs := [sha256.Size]byte{}
|
|
|
|
rand.Read(cs[:])
|
|
|
|
|
|
|
|
id := container.NewID()
|
|
|
|
id.SetSHA256(cs)
|
|
|
|
|
|
|
|
return id
|
|
|
|
}
|
|
|
|
|
|
|
|
func testOID() *objectSDK.ID {
|
|
|
|
cs := [sha256.Size]byte{}
|
|
|
|
rand.Read(cs[:])
|
|
|
|
|
|
|
|
id := objectSDK.NewID()
|
|
|
|
id.SetSHA256(cs)
|
|
|
|
|
|
|
|
return id
|
|
|
|
}
|
|
|
|
|
2020-10-28 14:49:30 +00:00
|
|
|
func TestDB(t *testing.T) {
|
|
|
|
version := pkg.NewVersion()
|
|
|
|
version.SetMajor(2)
|
|
|
|
version.SetMinor(1)
|
|
|
|
|
2020-10-29 12:03:55 +00:00
|
|
|
cid := testCID()
|
2020-10-28 14:49:30 +00:00
|
|
|
|
|
|
|
w, err := owner.NEO3WalletFromPublicKey(&test.DecodeKey(-1).PublicKey)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
ownerID := owner.NewID()
|
|
|
|
ownerID.SetNeo3Wallet(w)
|
|
|
|
|
2020-10-29 12:03:55 +00:00
|
|
|
oid := testOID()
|
2020-10-28 14:49:30 +00:00
|
|
|
|
|
|
|
obj := object.NewRaw()
|
|
|
|
obj.SetID(oid)
|
|
|
|
obj.SetOwnerID(ownerID)
|
|
|
|
obj.SetContainerID(cid)
|
|
|
|
obj.SetVersion(version)
|
|
|
|
|
|
|
|
k, v := "key", "value"
|
|
|
|
|
|
|
|
a := objectSDK.NewAttribute()
|
|
|
|
a.SetKey(k)
|
|
|
|
a.SetValue(v)
|
|
|
|
|
|
|
|
obj.SetAttributes(a)
|
|
|
|
|
|
|
|
path := "test.db"
|
|
|
|
|
|
|
|
bdb, err := bbolt.Open(path, 0600, nil)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
bdb.Close()
|
|
|
|
os.Remove(path)
|
|
|
|
}()
|
|
|
|
|
|
|
|
db := NewDB(bdb)
|
|
|
|
|
|
|
|
o := obj.Object()
|
|
|
|
|
|
|
|
require.NoError(t, db.Put(o))
|
|
|
|
|
|
|
|
o2, err := db.Get(o.Address())
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
require.Equal(t, o, o2)
|
|
|
|
|
|
|
|
fs := objectSDK.SearchFilters{}
|
|
|
|
|
|
|
|
// filter container ID
|
|
|
|
fs.AddObjectContainerIDFilter(objectSDK.MatchStringEqual, cid)
|
|
|
|
testSelect(t, db, fs, o.Address())
|
|
|
|
|
|
|
|
// filter owner ID
|
|
|
|
fs.AddObjectOwnerIDFilter(objectSDK.MatchStringEqual, ownerID)
|
|
|
|
testSelect(t, db, fs, o.Address())
|
|
|
|
|
|
|
|
// filter attribute
|
|
|
|
fs.AddFilter(k, v, objectSDK.MatchStringEqual)
|
|
|
|
testSelect(t, db, fs, o.Address())
|
|
|
|
|
|
|
|
// filter mismatch
|
|
|
|
fs.AddFilter(k, v+"1", objectSDK.MatchStringEqual)
|
|
|
|
testSelect(t, db, fs)
|
|
|
|
}
|
2020-10-29 12:03:55 +00:00
|
|
|
|
|
|
|
func TestDB_Delete(t *testing.T) {
|
|
|
|
path := "test.db"
|
|
|
|
|
|
|
|
bdb, err := bbolt.Open(path, 0600, nil)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
bdb.Close()
|
|
|
|
os.Remove(path)
|
|
|
|
}()
|
|
|
|
|
|
|
|
db := NewDB(bdb)
|
|
|
|
|
|
|
|
obj := object.NewRaw()
|
|
|
|
obj.SetContainerID(testCID())
|
|
|
|
obj.SetID(testOID())
|
|
|
|
|
|
|
|
o := obj.Object()
|
|
|
|
|
|
|
|
require.NoError(t, db.Put(o))
|
|
|
|
|
|
|
|
addr := o.Address()
|
|
|
|
|
|
|
|
_, err = db.Get(addr)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
fs := objectSDK.SearchFilters{}
|
|
|
|
fs.AddObjectContainerIDFilter(objectSDK.MatchStringEqual, o.GetContainerID())
|
|
|
|
|
|
|
|
testSelect(t, db, fs, o.Address())
|
|
|
|
|
|
|
|
require.NoError(t, db.Delete(addr))
|
|
|
|
|
|
|
|
_, err = db.Get(addr)
|
|
|
|
require.Error(t, err)
|
|
|
|
|
|
|
|
testSelect(t, db, fs)
|
|
|
|
}
|
2020-10-29 16:12:38 +00:00
|
|
|
|
|
|
|
func TestDB_SelectProperties(t *testing.T) {
|
|
|
|
path := "test.db"
|
|
|
|
|
|
|
|
bdb, err := bbolt.Open(path, 0600, nil)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
bdb.Close()
|
|
|
|
os.Remove(path)
|
|
|
|
}()
|
|
|
|
|
|
|
|
db := NewDB(bdb)
|
|
|
|
|
|
|
|
parent := object.NewRaw()
|
|
|
|
parent.SetContainerID(testCID())
|
|
|
|
parent.SetID(testOID())
|
|
|
|
|
|
|
|
child := object.NewRaw()
|
|
|
|
child.SetContainerID(testCID())
|
|
|
|
child.SetID(testOID())
|
|
|
|
child.SetParent(parent.Object().SDK())
|
|
|
|
|
|
|
|
parAddr := parent.Object().Address()
|
|
|
|
childAddr := child.Object().Address()
|
|
|
|
|
|
|
|
require.NoError(t, db.Put(child.Object()))
|
|
|
|
|
|
|
|
// root filter
|
|
|
|
fs := objectSDK.SearchFilters{}
|
|
|
|
fs.AddRootFilter()
|
|
|
|
testSelect(t, db, fs, parAddr)
|
|
|
|
|
|
|
|
// non-root filter
|
|
|
|
fs = fs[:0]
|
|
|
|
fs.AddNonRootFilter()
|
|
|
|
testSelect(t, db, fs, childAddr)
|
|
|
|
|
|
|
|
// root filter (with random false value)
|
|
|
|
fs = fs[:0]
|
|
|
|
fs.AddFilter(v2object.FilterPropertyRoot, "some false value", objectSDK.MatchStringEqual)
|
|
|
|
testSelect(t, db, fs, childAddr)
|
|
|
|
|
|
|
|
// leaf filter
|
|
|
|
fs = fs[:0]
|
|
|
|
fs.AddLeafFilter()
|
|
|
|
testSelect(t, db, fs, childAddr)
|
|
|
|
|
|
|
|
// non-leaf filter
|
|
|
|
fs = fs[:0]
|
|
|
|
fs.AddNonLeafFilter()
|
|
|
|
testSelect(t, db, fs, parAddr)
|
|
|
|
|
|
|
|
// leaf filter (with random false value)
|
|
|
|
fs = fs[:0]
|
|
|
|
fs.AddFilter(v2object.FilterPropertyLeaf, "some false value", objectSDK.MatchStringEqual)
|
|
|
|
testSelect(t, db, fs, parAddr)
|
|
|
|
|
|
|
|
lnk := object.NewRaw()
|
|
|
|
lnk.SetContainerID(testCID())
|
|
|
|
lnk.SetID(testOID())
|
|
|
|
lnk.SetChildren(testOID())
|
|
|
|
|
|
|
|
lnkAddr := lnk.Object().Address()
|
|
|
|
|
|
|
|
require.NoError(t, db.Put(lnk.Object()))
|
|
|
|
|
|
|
|
// childfree filter
|
|
|
|
fs = fs[:0]
|
|
|
|
fs.AddChildfreeFilter()
|
|
|
|
testSelect(t, db, fs, childAddr, parAddr)
|
|
|
|
|
|
|
|
// non-childfree filter
|
|
|
|
fs = fs[:0]
|
|
|
|
fs.AddNonChildfreeFilter()
|
|
|
|
testSelect(t, db, fs, lnkAddr)
|
|
|
|
|
|
|
|
// childfree filter (with random false value)
|
|
|
|
fs = fs[:0]
|
|
|
|
fs.AddFilter(v2object.FilterPropertyChildfree, "some false value", objectSDK.MatchStringEqual)
|
|
|
|
testSelect(t, db, fs, lnkAddr)
|
|
|
|
}
|