e3e303754b
Optimizing collecting monkit metrics: * initialize metrics once at the begining * avoid using string in map for getting stats structs per redundancy Benchmark results (compared against part 1 optimization change): name old time/op new time/op delta RemoteSegment/Cockroach/healthy_segment-8 31.4µs ± 6% 21.7µs ± 8% -30.73% (p=0.008 n=5+5) name old alloc/op new alloc/op delta RemoteSegment/healthy_segment-8 10.2kB ± 0% 7.4kB ± 0% -27.03% (p=0.008 n=5+5) name old allocs/op new allocs/op delta RemoteSegment/healthy_segment-8 250 ± 0% 150 ± 0% -40.00% (p=0.008 n=5+5) Change-Id: Ie09476eb469a4d6c09e52550c8ba92b3b4b34271
178 lines
4.7 KiB
Go
178 lines
4.7 KiB
Go
// Copyright (C) 2023 Storj Labs, Inc.
|
|
// See LICENSE for copying information.
|
|
|
|
package checker
|
|
|
|
import (
|
|
"fmt"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
"go.uber.org/zap/zaptest"
|
|
|
|
"storj.io/common/identity/testidentity"
|
|
"storj.io/common/storj"
|
|
"storj.io/common/storj/location"
|
|
"storj.io/common/testcontext"
|
|
"storj.io/storj/satellite/metabase"
|
|
"storj.io/storj/satellite/metabase/rangedloop"
|
|
"storj.io/storj/satellite/nodeselection"
|
|
"storj.io/storj/satellite/overlay"
|
|
"storj.io/storj/satellite/repair/queue"
|
|
)
|
|
|
|
func TestObserverForkProcess(t *testing.T) {
|
|
|
|
nodes := func() (res []nodeselection.SelectedNode) {
|
|
for i := 0; i < 10; i++ {
|
|
res = append(res, nodeselection.SelectedNode{
|
|
ID: testidentity.MustPregeneratedIdentity(i, storj.LatestIDVersion()).ID,
|
|
Online: true,
|
|
CountryCode: location.Germany,
|
|
LastNet: "127.0.0.0",
|
|
})
|
|
}
|
|
return res
|
|
}()
|
|
|
|
mapNodes := func(nodes []nodeselection.SelectedNode, include func(node nodeselection.SelectedNode) bool) map[storj.NodeID]nodeselection.SelectedNode {
|
|
res := map[storj.NodeID]nodeselection.SelectedNode{}
|
|
for _, node := range nodes {
|
|
if include(node) {
|
|
res[node.ID] = node
|
|
}
|
|
}
|
|
return res
|
|
}
|
|
|
|
ctx := testcontext.New(t)
|
|
placementRules := overlay.ConfigurablePlacementRule{}
|
|
parsed, err := placementRules.Parse()
|
|
require.NoError(t, err)
|
|
createDefaultObserver := func() *Observer {
|
|
o := &Observer{
|
|
statsCollector: make(map[storj.RedundancyScheme]*observerRSStats),
|
|
nodesCache: &ReliabilityCache{
|
|
staleness: time.Hour,
|
|
},
|
|
placementRules: parsed.CreateFilters,
|
|
}
|
|
|
|
o.nodesCache.state.Store(&reliabilityState{
|
|
nodeByID: mapNodes(nodes, func(node nodeselection.SelectedNode) bool {
|
|
return true
|
|
}),
|
|
created: time.Now(),
|
|
})
|
|
return o
|
|
}
|
|
|
|
createFork := func(o *Observer, q queue.RepairQueue) *observerFork {
|
|
return &observerFork{
|
|
log: zaptest.NewLogger(t),
|
|
getObserverStats: o.getObserverStats,
|
|
rsStats: make(map[storj.RedundancyScheme]*partialRSStats),
|
|
doDeclumping: o.doDeclumping,
|
|
doPlacementCheck: o.doPlacementCheck,
|
|
placementRules: o.placementRules,
|
|
getNodesEstimate: o.getNodesEstimate,
|
|
nodesCache: o.nodesCache,
|
|
repairQueue: queue.NewInsertBuffer(q, 1000),
|
|
}
|
|
}
|
|
|
|
createPieces := func(nodes []nodeselection.SelectedNode, selected ...int) metabase.Pieces {
|
|
pieces := make(metabase.Pieces, len(selected))
|
|
for ix, s := range selected {
|
|
pieces[ix] = metabase.Piece{
|
|
Number: uint16(ix),
|
|
StorageNode: nodes[s].ID,
|
|
}
|
|
}
|
|
return pieces
|
|
}
|
|
|
|
t.Run("all healthy", func(t *testing.T) {
|
|
o := createDefaultObserver()
|
|
q := queue.MockRepairQueue{}
|
|
fork := createFork(o, &q)
|
|
err := fork.process(ctx, &rangedloop.Segment{
|
|
Pieces: createPieces(nodes, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9),
|
|
Redundancy: storj.RedundancyScheme{
|
|
Algorithm: storj.ReedSolomon,
|
|
ShareSize: 256,
|
|
RepairShares: 4,
|
|
RequiredShares: 6,
|
|
OptimalShares: 8,
|
|
TotalShares: 10,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
err = fork.repairQueue.Flush(ctx)
|
|
require.NoError(t, err)
|
|
|
|
require.Len(t, q.Segments, 0)
|
|
|
|
})
|
|
|
|
t.Run("declumping", func(t *testing.T) {
|
|
o := createDefaultObserver()
|
|
o.doDeclumping = true
|
|
q := queue.MockRepairQueue{}
|
|
fork := createFork(o, &q)
|
|
err := fork.process(ctx, &rangedloop.Segment{
|
|
Pieces: createPieces(nodes, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9),
|
|
Redundancy: storj.RedundancyScheme{
|
|
Algorithm: storj.ReedSolomon,
|
|
ShareSize: 256,
|
|
RepairShares: 4,
|
|
RequiredShares: 6,
|
|
OptimalShares: 8,
|
|
TotalShares: 10,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
err = fork.repairQueue.Flush(ctx)
|
|
require.NoError(t, err)
|
|
|
|
// as all test nodes are in the same subnet...
|
|
require.Len(t, q.Segments, 1)
|
|
})
|
|
|
|
t.Run("declumping is ignored by annotation", func(t *testing.T) {
|
|
o := createDefaultObserver()
|
|
o.doDeclumping = true
|
|
|
|
placements := overlay.ConfigurablePlacementRule{}
|
|
require.NoError(t, placements.Set(fmt.Sprintf(`10:annotated(country("DE"),annotation("%s","%s"))`, nodeselection.AutoExcludeSubnet, nodeselection.AutoExcludeSubnetOFF)))
|
|
parsed, err := placements.Parse()
|
|
require.NoError(t, err)
|
|
o.placementRules = parsed.CreateFilters
|
|
|
|
q := queue.MockRepairQueue{}
|
|
fork := createFork(o, &q)
|
|
err = fork.process(ctx, &rangedloop.Segment{
|
|
Placement: 10,
|
|
Pieces: createPieces(nodes, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9),
|
|
Redundancy: storj.RedundancyScheme{
|
|
Algorithm: storj.ReedSolomon,
|
|
ShareSize: 256,
|
|
RepairShares: 4,
|
|
RequiredShares: 6,
|
|
OptimalShares: 8,
|
|
TotalShares: 10,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
err = fork.repairQueue.Flush(ctx)
|
|
require.NoError(t, err)
|
|
|
|
require.Len(t, q.Segments, 0)
|
|
})
|
|
|
|
}
|