691 lines
20 KiB
Go
691 lines
20 KiB
Go
package netmap
|
|
|
|
import (
|
|
"cmp"
|
|
"crypto/rand"
|
|
"encoding/binary"
|
|
"fmt"
|
|
mrand "math/rand"
|
|
"reflect"
|
|
"slices"
|
|
"strconv"
|
|
"strings"
|
|
"testing"
|
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/netmap"
|
|
"git.frostfs.info/TrueCloudLab/hrw"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func BenchmarkHRWSort(b *testing.B) {
|
|
const netmapSize = 1000
|
|
|
|
vectors := make([]nodes, netmapSize)
|
|
weights := make([]float64, netmapSize)
|
|
for i := range vectors {
|
|
key := make([]byte, 33)
|
|
rand.Read(key)
|
|
|
|
var node NodeInfo
|
|
node.SetPrice(1)
|
|
node.SetCapacity(100)
|
|
node.SetPublicKey(key)
|
|
|
|
vectors[i] = nodes{node}
|
|
weights[i] = float64(mrand.Uint32()%10) / 10.0
|
|
}
|
|
|
|
pivot := mrand.Uint64()
|
|
b.Run("sort by index, no weight", func(b *testing.B) {
|
|
realNodes := make([]nodes, netmapSize)
|
|
b.ResetTimer()
|
|
for range b.N {
|
|
b.StopTimer()
|
|
copy(realNodes, vectors)
|
|
b.StartTimer()
|
|
|
|
hrw.SortSliceByIndex(realNodes, pivot)
|
|
}
|
|
})
|
|
b.Run("sort by value, no weight", func(b *testing.B) {
|
|
realNodes := make([]nodes, netmapSize)
|
|
b.ResetTimer()
|
|
for range b.N {
|
|
b.StopTimer()
|
|
copy(realNodes, vectors)
|
|
b.StartTimer()
|
|
|
|
hrw.SortHasherSliceByValue(realNodes, pivot)
|
|
}
|
|
})
|
|
b.Run("only sort by index", func(b *testing.B) {
|
|
realNodes := make([]nodes, netmapSize)
|
|
b.ResetTimer()
|
|
for range b.N {
|
|
b.StopTimer()
|
|
copy(realNodes, vectors)
|
|
b.StartTimer()
|
|
|
|
hrw.SortSliceByWeightIndex(realNodes, weights, pivot)
|
|
}
|
|
})
|
|
b.Run("sort by value", func(b *testing.B) {
|
|
realNodes := make([]nodes, netmapSize)
|
|
b.ResetTimer()
|
|
for range b.N {
|
|
b.StopTimer()
|
|
copy(realNodes, vectors)
|
|
b.StartTimer()
|
|
|
|
hrw.SortHasherSliceByWeightValue(realNodes, weights, pivot)
|
|
}
|
|
})
|
|
b.Run("sort by ID, then by index (deterministic)", func(b *testing.B) {
|
|
realNodes := make([]nodes, netmapSize)
|
|
b.ResetTimer()
|
|
for range b.N {
|
|
b.StopTimer()
|
|
copy(realNodes, vectors)
|
|
b.StartTimer()
|
|
|
|
slices.SortFunc(vectors, func(vi, vj nodes) int {
|
|
return cmp.Compare(vi[0].Hash(), vj[0].Hash())
|
|
})
|
|
hrw.SortSliceByWeightIndex(realNodes, weights, pivot)
|
|
}
|
|
})
|
|
}
|
|
|
|
func BenchmarkPolicyHRWType(b *testing.B) {
|
|
const netmapSize = 100
|
|
|
|
p := newPlacementPolicy(1,
|
|
[]ReplicaDescriptor{
|
|
newReplica(1, "loc1"),
|
|
newReplica(1, "loc2"),
|
|
},
|
|
[]Selector{
|
|
newSelector("loc1", "Location", 1, "loc1", (*Selector).SelectSame),
|
|
newSelector("loc2", "Location", 1, "loc2", (*Selector).SelectSame),
|
|
},
|
|
[]Filter{
|
|
newFilter("loc1", "Location", "Shanghai", netmap.EQ),
|
|
newFilter("loc2", "Location", "Shanghai", netmap.NE),
|
|
})
|
|
|
|
nodes := make([]NodeInfo, netmapSize)
|
|
for i := range nodes {
|
|
var loc string
|
|
switch i % 20 {
|
|
case 0:
|
|
loc = "Shanghai"
|
|
default:
|
|
loc = strconv.Itoa(i % 20)
|
|
}
|
|
|
|
// Having the same price and capacity ensures equal weights for all nodes.
|
|
// This way placement is more dependent on the initial order.
|
|
nodes[i] = nodeInfoFromAttributes("Location", loc, "Price", "1", "Capacity", "10")
|
|
pub := make([]byte, 33)
|
|
pub[0] = byte(i)
|
|
nodes[i].SetPublicKey(pub)
|
|
}
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodes)
|
|
|
|
b.ResetTimer()
|
|
for range b.N {
|
|
_, err := nm.ContainerNodes(p, []byte{1})
|
|
if err != nil {
|
|
b.Fatal()
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestPlacementPolicy_DeterministicOrder(t *testing.T) {
|
|
const netmapSize = 100
|
|
|
|
p := newPlacementPolicy(1,
|
|
[]ReplicaDescriptor{
|
|
newReplica(1, "loc1"),
|
|
newReplica(1, "loc2"),
|
|
},
|
|
[]Selector{
|
|
newSelector("loc1", "Location", 1, "loc1", (*Selector).SelectSame),
|
|
newSelector("loc2", "Location", 1, "loc2", (*Selector).SelectSame),
|
|
},
|
|
[]Filter{
|
|
newFilter("loc1", "Location", "Shanghai", netmap.EQ),
|
|
newFilter("loc2", "Location", "Shanghai", netmap.NE),
|
|
})
|
|
|
|
nodeList := make([]NodeInfo, netmapSize)
|
|
for i := range nodeList {
|
|
var loc string
|
|
switch i % 20 {
|
|
case 0:
|
|
loc = "Shanghai"
|
|
default:
|
|
loc = strconv.Itoa(i % 20)
|
|
}
|
|
|
|
// Having the same price and capacity ensures equal weights for all nodes.
|
|
// This way placement is more dependent on the initial order.
|
|
nodeList[i] = nodeInfoFromAttributes("Location", loc, "Price", "1", "Capacity", "10")
|
|
pub := make([]byte, 33)
|
|
pub[0] = byte(i)
|
|
nodeList[i].SetPublicKey(pub)
|
|
}
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodeList)
|
|
|
|
getIndices := func(t *testing.T) (uint64, uint64) {
|
|
v, err := nm.ContainerNodes(p, []byte{1})
|
|
require.NoError(t, err)
|
|
|
|
nss := make([]nodes, len(v))
|
|
for i := range v {
|
|
nss[i] = v[i]
|
|
}
|
|
|
|
ns := flattenNodes(nss)
|
|
require.Equal(t, 2, len(ns))
|
|
return ns[0].Hash(), ns[1].Hash()
|
|
}
|
|
|
|
a, b := getIndices(t)
|
|
for range 10 {
|
|
x, y := getIndices(t)
|
|
require.Equal(t, a, x)
|
|
require.Equal(t, b, y)
|
|
}
|
|
}
|
|
|
|
func TestPlacementPolicy_ProcessSelectors(t *testing.T) {
|
|
p := newPlacementPolicy(2, nil,
|
|
[]Selector{
|
|
newSelector("SameRU", "City", 2, "FromRU", (*Selector).SelectSame),
|
|
newSelector("DistinctRU", "City", 2, "FromRU", (*Selector).SelectDistinct),
|
|
newSelector("Good", "Country", 2, "Good", (*Selector).SelectDistinct),
|
|
newSelector("Main", "Country", 3, "*", (*Selector).SelectDistinct),
|
|
},
|
|
[]Filter{
|
|
newFilter("FromRU", "Country", "Russia", netmap.EQ),
|
|
newFilter("Good", "Rating", "4", netmap.GE),
|
|
})
|
|
nodes := []NodeInfo{
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "1", "City", "SPB"),
|
|
nodeInfoFromAttributes("Country", "Germany", "Rating", "5", "City", "Berlin"),
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "6", "City", "Moscow"),
|
|
nodeInfoFromAttributes("Country", "France", "Rating", "4", "City", "Paris"),
|
|
nodeInfoFromAttributes("Country", "France", "Rating", "1", "City", "Lyon"),
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "5", "City", "SPB"),
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "7", "City", "Moscow"),
|
|
nodeInfoFromAttributes("Country", "Germany", "Rating", "3", "City", "Darmstadt"),
|
|
nodeInfoFromAttributes("Country", "Germany", "Rating", "7", "City", "Frankfurt"),
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "9", "City", "SPB"),
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "9", "City", "SPB"),
|
|
}
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodes)
|
|
c := newContext(nm)
|
|
c.setCBF(p.backupFactor)
|
|
require.NoError(t, c.processFilters(p))
|
|
require.NoError(t, c.processSelectors(p))
|
|
|
|
for _, s := range p.selectors {
|
|
sel := c.selections[s.GetName()]
|
|
s := c.processedSelectors[s.GetName()]
|
|
bucketCount, nodesInBucket := calcNodesCount(*s)
|
|
nodesInBucket *= int(c.cbf)
|
|
targ := fmt.Sprintf("selector '%s'", s.GetName())
|
|
require.Equal(t, bucketCount, len(sel), targ)
|
|
fName := s.GetFilter()
|
|
for _, res := range sel {
|
|
require.Equal(t, nodesInBucket, len(res), targ)
|
|
for j := range res {
|
|
require.True(t, fName == mainFilterName || c.match(c.processedFilters[fName], res[j]), targ)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestPlacementPolicy_PartiallyMissingContainer(t *testing.T) {
|
|
s := `REP 2 IN D1OBJ
|
|
REP 2 IN D2OBJ
|
|
CBF 2
|
|
SELECT 2 FROM PRIMARY AS D1OBJ
|
|
SELECT 2 FROM SECONDARY AS D2OBJ
|
|
FILTER ClusterName EQ D1OBJ AS PRIMARY
|
|
FILTER ClusterName EQ D2OBJ AS SECONDARY`
|
|
|
|
var p PlacementPolicy
|
|
require.NoError(t, p.DecodeString(s))
|
|
|
|
nodes := []NodeInfo{
|
|
nodeInfoFromAttributes("ClusterName", "D1OBJ", "ID", "1"),
|
|
nodeInfoFromAttributes("ClusterName", "D1OBJ", "ID", "2"),
|
|
nodeInfoFromAttributes("ClusterName", "D1OBJ", "ID", "3"),
|
|
nodeInfoFromAttributes("ClusterName", "D1OBJ", "ID", "4"),
|
|
nodeInfoFromAttributes("ClusterName", "D1OBJ", "ID", "5"),
|
|
nodeInfoFromAttributes("ClusterName", "D1OBJ", "ID", "6"),
|
|
nodeInfoFromAttributes("ClusterName", "D1OBJ", "ID", "7"),
|
|
nodeInfoFromAttributes("ClusterName", "D1OBJ", "ID", "8"),
|
|
}
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodes)
|
|
|
|
res, err := nm.ContainerNodes(p, nil)
|
|
require.NoError(t, err)
|
|
require.Len(t, res, 2)
|
|
require.Len(t, res[0], 4)
|
|
require.Len(t, res[1], 0)
|
|
}
|
|
|
|
func TestPlacementPolicy_Like(t *testing.T) {
|
|
nodes := []NodeInfo{
|
|
nodeInfoFromAttributes("Country", "Russia"),
|
|
nodeInfoFromAttributes("Country", "Germany"),
|
|
nodeInfoFromAttributes("Country", "Belarus"),
|
|
}
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodes)
|
|
|
|
t.Run("LIKE all", func(t *testing.T) {
|
|
ssNamed := []Selector{newSelector("X", "Country", 4, "FromRU", (*Selector).SelectDistinct)}
|
|
fsNamed := []Filter{newFilter("FromRU", "Country", "*", netmap.LIKE)}
|
|
rsNamed := []ReplicaDescriptor{newReplica(4, "X")}
|
|
pNamed := newPlacementPolicy(0, rsNamed, ssNamed, fsNamed)
|
|
|
|
n, err := nm.ContainerNodes(pNamed, []byte{1})
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 3, len(n[0]))
|
|
for _, n := range n[0] {
|
|
require.True(t, strings.Contains("GermanyRussiaBelarus", n.Attribute("Country")))
|
|
}
|
|
})
|
|
|
|
t.Run("LIKE no wildcard", func(t *testing.T) {
|
|
ssNamed := []Selector{newSelector("X", "Country", 1, "FromRU", (*Selector).SelectDistinct)}
|
|
fsNamed := []Filter{newFilter("FromRU", "Country", "Russia", netmap.LIKE)}
|
|
rsNamed := []ReplicaDescriptor{newReplica(1, "X")}
|
|
pNamed := newPlacementPolicy(0, rsNamed, ssNamed, fsNamed)
|
|
|
|
n, err := nm.ContainerNodes(pNamed, []byte{1})
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 1, len(n[0]))
|
|
for _, n := range n[0] {
|
|
require.True(t, n.Attribute("Country") == "Russia")
|
|
}
|
|
})
|
|
|
|
t.Run("LIKE prefix", func(t *testing.T) {
|
|
ssNamed := []Selector{newSelector("X", "Country", 1, "FromRU", (*Selector).SelectDistinct)}
|
|
fsNamed := []Filter{newFilter("FromRU", "Country", "Ge*", netmap.LIKE)}
|
|
rsNamed := []ReplicaDescriptor{newReplica(1, "X")}
|
|
pNamed := newPlacementPolicy(0, rsNamed, ssNamed, fsNamed)
|
|
|
|
n, err := nm.ContainerNodes(pNamed, []byte{1})
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 1, len(n[0]))
|
|
for _, n := range n[0] {
|
|
require.True(t, n.Attribute("Country") == "Germany")
|
|
}
|
|
})
|
|
|
|
t.Run("LIKE suffix", func(t *testing.T) {
|
|
ssNamed := []Selector{newSelector("X", "Country", 1, "FromRU", (*Selector).SelectDistinct)}
|
|
fsNamed := []Filter{newFilter("FromRU", "Country", "*sia", netmap.LIKE)}
|
|
rsNamed := []ReplicaDescriptor{newReplica(1, "X")}
|
|
pNamed := newPlacementPolicy(0, rsNamed, ssNamed, fsNamed)
|
|
|
|
n, err := nm.ContainerNodes(pNamed, []byte{1})
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 1, len(n[0]))
|
|
for _, n := range n[0] {
|
|
require.True(t, n.Attribute("Country") == "Russia")
|
|
}
|
|
})
|
|
|
|
t.Run("LIKE middle", func(t *testing.T) {
|
|
ssNamed := []Selector{newSelector("X", "Country", 2, "FromRU", (*Selector).SelectDistinct)}
|
|
fsNamed := []Filter{newFilter("FromRU", "Country", "*us*", netmap.LIKE)}
|
|
rsNamed := []ReplicaDescriptor{newReplica(2, "X")}
|
|
pNamed := newPlacementPolicy(0, rsNamed, ssNamed, fsNamed)
|
|
|
|
n, err := nm.ContainerNodes(pNamed, []byte{1})
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 2, len(n[0]))
|
|
for _, n := range n[0] {
|
|
require.True(t, strings.Contains("RussiaBelarus", n.Attribute("Country")))
|
|
}
|
|
})
|
|
}
|
|
|
|
func TestPlacementPolicy_Unique(t *testing.T) {
|
|
p := newPlacementPolicy(2,
|
|
[]ReplicaDescriptor{
|
|
newReplica(1, "S"),
|
|
newReplica(1, "S"),
|
|
},
|
|
[]Selector{
|
|
newSelector("S", "City", 1, "*", (*Selector).SelectSame),
|
|
},
|
|
[]Filter{})
|
|
p.unique = true
|
|
|
|
var nodes []NodeInfo
|
|
for i, city := range []string{"Moscow", "Berlin", "Shenzhen"} {
|
|
for j := range 3 {
|
|
node := nodeInfoFromAttributes("City", city)
|
|
node.SetPublicKey(binary.BigEndian.AppendUint16(nil, uint16(i*4+j)))
|
|
nodes = append(nodes, node)
|
|
}
|
|
}
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodes)
|
|
|
|
v, err := nm.ContainerNodes(p, nil)
|
|
require.NoError(t, err)
|
|
for i, vi := range v {
|
|
for _, ni := range vi {
|
|
for j := range i {
|
|
for _, nj := range v[j] {
|
|
require.NotEqual(t, ni.hash, nj.hash)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestPlacementPolicy_SingleOmitNames(t *testing.T) {
|
|
nodes := []NodeInfo{
|
|
nodeInfoFromAttributes("ID", "1", "Country", "Russia", "City", "SPB"),
|
|
nodeInfoFromAttributes("ID", "2", "Country", "Germany", "City", "Berlin"),
|
|
nodeInfoFromAttributes("ID", "3", "Country", "Russia", "City", "Moscow"),
|
|
nodeInfoFromAttributes("ID", "4", "Country", "France", "City", "Paris"),
|
|
nodeInfoFromAttributes("ID", "5", "Country", "France", "City", "Lyon"),
|
|
nodeInfoFromAttributes("ID", "6", "Country", "Russia", "City", "SPB"),
|
|
nodeInfoFromAttributes("ID", "7", "Country", "Russia", "City", "Moscow"),
|
|
nodeInfoFromAttributes("ID", "8", "Country", "Germany", "City", "Darmstadt"),
|
|
nodeInfoFromAttributes("ID", "9", "Country", "Germany", "City", "Frankfurt"),
|
|
nodeInfoFromAttributes("ID", "10", "Country", "Russia", "City", "SPB"),
|
|
nodeInfoFromAttributes("ID", "11", "Country", "Russia", "City", "Moscow"),
|
|
nodeInfoFromAttributes("ID", "12", "Country", "Germany", "City", "London"),
|
|
}
|
|
for i := range nodes {
|
|
pub := make([]byte, 33)
|
|
rand.Read(pub)
|
|
nodes[i].SetPublicKey(pub)
|
|
}
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodes)
|
|
|
|
for _, unique := range []bool{false, true} {
|
|
t.Run(fmt.Sprintf("unique=%t", unique), func(t *testing.T) {
|
|
ssNamed := []Selector{newSelector("X", "City", 2, "FromRU", (*Selector).SelectDistinct)}
|
|
fsNamed := []Filter{newFilter("FromRU", "Country", "Russia", netmap.EQ)}
|
|
rsNamed := []ReplicaDescriptor{newReplica(1, "X")}
|
|
pNamed := newPlacementPolicy(3, rsNamed, ssNamed, fsNamed)
|
|
pNamed.unique = unique
|
|
|
|
vNamed, err := nm.ContainerNodes(pNamed, []byte{1})
|
|
require.NoError(t, err)
|
|
|
|
ssUnnamed := []Selector{newSelector("", "City", 2, "FromRU", (*Selector).SelectDistinct)}
|
|
fsUnnamed := []Filter{newFilter("FromRU", "Country", "Russia", netmap.EQ)}
|
|
rsUnnamed := []ReplicaDescriptor{newReplica(1, "")}
|
|
pUnnamed := newPlacementPolicy(3, rsUnnamed, ssUnnamed, fsUnnamed)
|
|
pUnnamed.unique = unique
|
|
|
|
vUnnamed, err := nm.ContainerNodes(pUnnamed, []byte{1})
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, vNamed, vUnnamed)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestPlacementPolicy_MultiREP(t *testing.T) {
|
|
nodes := []NodeInfo{
|
|
nodeInfoFromAttributes("ID", "1", "Country", "Russia", "City", "SPB"),
|
|
nodeInfoFromAttributes("ID", "2", "Country", "Germany", "City", "Berlin"),
|
|
nodeInfoFromAttributes("ID", "3", "Country", "Russia", "City", "Moscow"),
|
|
nodeInfoFromAttributes("ID", "4", "Country", "France", "City", "Paris"),
|
|
nodeInfoFromAttributes("ID", "5", "Country", "France", "City", "Lyon"),
|
|
nodeInfoFromAttributes("ID", "6", "Country", "Russia", "City", "SPB"),
|
|
nodeInfoFromAttributes("ID", "7", "Country", "Russia", "City", "Moscow"),
|
|
nodeInfoFromAttributes("ID", "8", "Country", "Germany", "City", "Darmstadt"),
|
|
nodeInfoFromAttributes("ID", "9", "Country", "Germany", "City", "Frankfurt"),
|
|
nodeInfoFromAttributes("ID", "10", "Country", "Russia", "City", "SPB"),
|
|
nodeInfoFromAttributes("ID", "11", "Country", "Russia", "City", "Moscow"),
|
|
nodeInfoFromAttributes("ID", "12", "Country", "Germany", "City", "London"),
|
|
}
|
|
for i := range nodes {
|
|
pub := make([]byte, 33)
|
|
rand.Read(pub)
|
|
nodes[i].SetPublicKey(pub)
|
|
}
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodes)
|
|
|
|
ss := []Selector{newSelector("SameRU", "City", 2, "FromRU", (*Selector).SelectDistinct)}
|
|
fs := []Filter{newFilter("FromRU", "Country", "Russia", netmap.EQ)}
|
|
|
|
for _, unique := range []bool{false, true} {
|
|
for _, additional := range []int{0, 1, 2} {
|
|
t.Run(fmt.Sprintf("unique=%t, additional=%d", unique, additional), func(t *testing.T) {
|
|
rs := []ReplicaDescriptor{newReplica(1, "SameRU")}
|
|
for range additional {
|
|
rs = append(rs, newReplica(1, ""))
|
|
}
|
|
|
|
p := newPlacementPolicy(3, rs, ss, fs)
|
|
p.unique = unique
|
|
|
|
v, err := nm.ContainerNodes(p, []byte{1})
|
|
require.NoError(t, err)
|
|
require.Equal(t, 1+additional, len(v))
|
|
require.Equal(t, 6, len(v[0]))
|
|
|
|
for i := 1; i < additional; i++ {
|
|
require.Equal(t, 3, len(v[i]))
|
|
if !unique {
|
|
require.Equal(t, v[1], v[i])
|
|
}
|
|
}
|
|
|
|
if unique {
|
|
seen := make(map[string]bool)
|
|
for i := range v {
|
|
for j := range v[i] {
|
|
attr := v[i][j].Attribute("ID")
|
|
require.NotEmpty(t, attr)
|
|
require.False(t, seen[attr])
|
|
|
|
seen[attr] = true
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestPlacementPolicy_ProcessSelectorsExceptForNodes(t *testing.T) {
|
|
p := newPlacementPolicy(1, nil,
|
|
[]Selector{
|
|
newSelector("ExceptRU", "City", 2, "ExceptRU", (*Selector).SelectSame),
|
|
},
|
|
[]Filter{
|
|
newFilter("ExceptRU", "", "", netmap.NOT,
|
|
newFilter("", "", "", netmap.AND,
|
|
newFilter("", "City", "Lyon", netmap.EQ),
|
|
newFilter("", "Rating", "10", netmap.LE),
|
|
),
|
|
),
|
|
})
|
|
nodes := []NodeInfo{
|
|
nodeInfoFromAttributes("Country", "Germany", "Rating", "1", "City", "Berlin"),
|
|
nodeInfoFromAttributes("Country", "Germany", "Rating", "5", "City", "Berlin"),
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "6", "City", "Moscow"),
|
|
nodeInfoFromAttributes("Country", "France", "Rating", "4", "City", "Paris"),
|
|
nodeInfoFromAttributes("Country", "France", "Rating", "1", "City", "Lyon"),
|
|
nodeInfoFromAttributes("Country", "France", "Rating", "5", "City", "Lyon"),
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "7", "City", "Moscow"),
|
|
nodeInfoFromAttributes("Country", "Germany", "Rating", "3", "City", "Darmstadt"),
|
|
nodeInfoFromAttributes("Country", "Germany", "Rating", "7", "City", "Frankfurt"),
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "9", "City", "SPB"),
|
|
nodeInfoFromAttributes("Country", "Russia", "Rating", "9", "City", "SPB"),
|
|
}
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodes)
|
|
c := newContext(nm)
|
|
c.setCBF(p.backupFactor)
|
|
require.NoError(t, c.processFilters(p))
|
|
require.NoError(t, c.processSelectors(p))
|
|
|
|
for _, s := range p.selectors {
|
|
sel := c.selections[s.GetName()]
|
|
s := c.processedSelectors[s.GetName()]
|
|
bucketCount, nodesInBucket := calcNodesCount(*s)
|
|
nodesInBucket *= int(c.cbf)
|
|
targ := fmt.Sprintf("selector '%s'", s.GetName())
|
|
require.Equal(t, bucketCount, len(sel), targ)
|
|
fName := s.GetFilter()
|
|
for _, res := range sel {
|
|
require.Equal(t, nodesInBucket, len(res), targ)
|
|
for j := range res {
|
|
require.True(t, fName == mainFilterName || c.match(c.processedFilters[fName], res[j]), targ)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestPlacementPolicy_NonAsciiAttributes(t *testing.T) {
|
|
p := newPlacementPolicy(
|
|
1,
|
|
[]ReplicaDescriptor{
|
|
newReplica(2, "Nodes"),
|
|
newReplica(2, "Nodes"),
|
|
},
|
|
[]Selector{
|
|
newSelector("Nodes", "Цвет", 2, "Colorful", (*Selector).SelectSame),
|
|
},
|
|
[]Filter{
|
|
newFilter("Colorful", "", "", netmap.OR,
|
|
newFilter("", "Цвет", "Красный", netmap.EQ),
|
|
newFilter("", "Цвет", "Синий", netmap.EQ),
|
|
),
|
|
},
|
|
)
|
|
p.SetUnique(true)
|
|
|
|
nodes := []NodeInfo{
|
|
nodeInfoFromAttributes("Цвет", "Красный", "Форма", "Треугольник"),
|
|
nodeInfoFromAttributes("Цвет", "Красный", "Форма", "Круг"),
|
|
nodeInfoFromAttributes("Цвет", "Синий", "Форма", "Треугольник"),
|
|
nodeInfoFromAttributes("Цвет", "Синий", "Форма", "Круг"),
|
|
nodeInfoFromAttributes("Свойство", "Мягкий", "Форма", "Треугольник"),
|
|
nodeInfoFromAttributes("Свойство", "Теплый", "Форма", "Круг"),
|
|
}
|
|
for i := range nodes {
|
|
nodes[i].SetPublicKey([]byte{byte(i)})
|
|
}
|
|
|
|
redNodes := nodes[:2]
|
|
blueNodes := nodes[2:4]
|
|
|
|
var nm NetMap
|
|
nm.SetNodes(nodes)
|
|
|
|
pivot := make([]byte, 42)
|
|
_, _ = rand.Read(pivot)
|
|
|
|
nodesPerReplica, err := nm.ContainerNodes(p, pivot)
|
|
require.NoError(t, err)
|
|
require.Len(t, nodesPerReplica, 2)
|
|
|
|
for i := range nodesPerReplica {
|
|
slices.SortFunc(nodesPerReplica[i], func(n1, n2 NodeInfo) int {
|
|
pk1, pk2 := string(n1.PublicKey()), string(n2.PublicKey())
|
|
return cmp.Compare(pk1, pk2)
|
|
})
|
|
}
|
|
|
|
redMatchFirst := reflect.DeepEqual(redNodes, nodesPerReplica[0])
|
|
blueMatchFirst := reflect.DeepEqual(blueNodes, nodesPerReplica[0])
|
|
|
|
redMatchSecond := reflect.DeepEqual(redNodes, nodesPerReplica[1])
|
|
blueMatchSecond := reflect.DeepEqual(blueNodes, nodesPerReplica[1])
|
|
|
|
assert.True(t, redMatchFirst && blueMatchSecond || blueMatchFirst && redMatchSecond)
|
|
}
|
|
|
|
func TestSelector_SetName(t *testing.T) {
|
|
const name = "some name"
|
|
var s Selector
|
|
|
|
require.Zero(t, s.m.GetName())
|
|
|
|
s.SetName(name)
|
|
require.Equal(t, name, s.m.GetName())
|
|
}
|
|
|
|
func TestSelector_SetNumberOfNodes(t *testing.T) {
|
|
const num = 3
|
|
var s Selector
|
|
|
|
require.Zero(t, s.m.GetCount())
|
|
|
|
s.SetNumberOfNodes(num)
|
|
|
|
require.EqualValues(t, num, s.m.GetCount())
|
|
}
|
|
|
|
func TestSelectorClauses(t *testing.T) {
|
|
var s Selector
|
|
|
|
require.Equal(t, netmap.UnspecifiedClause, s.m.GetClause())
|
|
|
|
s.SelectDistinct()
|
|
require.Equal(t, netmap.Distinct, s.m.GetClause())
|
|
|
|
s.SelectSame()
|
|
require.Equal(t, netmap.Same, s.m.GetClause())
|
|
}
|
|
|
|
func TestSelector_SelectByBucketAttribute(t *testing.T) {
|
|
const attr = "some attribute"
|
|
var s Selector
|
|
|
|
require.Zero(t, s.m.GetAttribute())
|
|
|
|
s.SelectByBucketAttribute(attr)
|
|
require.Equal(t, attr, s.m.GetAttribute())
|
|
}
|
|
|
|
func TestSelector_SetFilterName(t *testing.T) {
|
|
const fName = "some filter"
|
|
var s Selector
|
|
|
|
require.Zero(t, s.m.GetFilter())
|
|
|
|
s.SetFilterName(fName)
|
|
require.Equal(t, fName, s.m.GetFilter())
|
|
}
|