[#957] treesvc: Abstract out Heap interface for listing
``` goos: linux goarch: amd64 pkg: git.frostfs.info/TrueCloudLab/frostfs-node/pkg/services/tree cpu: 11th Gen Intel(R) Core(TM) i5-1135G7 @ 2.40GHz │ old │ new │ │ sec/op │ sec/op vs base │ GetSubTree/latency-8 3.542m ± 12% 4.059m ± 20% +14.59% (p=0.005 n=10) GetSubTree/total_time-8 73.19m ± 9% 78.82m ± 4% +7.69% (p=0.043 n=10) geomean 16.10m 17.89m +11.09% │ old │ new │ │ B/op │ B/op vs base │ GetSubTree/latency-8 28.23Mi ± 0% 32.81Mi ± 0% +16.22% (p=0.000 n=10) GetSubTree/total_time-8 28.23Mi ± 0% 32.81Mi ± 0% +16.22% (p=0.000 n=10) geomean 28.23Mi 32.81Mi +16.22% │ old │ new │ │ allocs/op │ allocs/op vs base │ GetSubTree/latency-8 400.0k ± 0% 400.0k ± 0% +0.00% (p=0.000 n=10) GetSubTree/total_time-8 400.0k ± 0% 400.0k ± 0% +0.00% (p=0.000 n=10) geomean 400.0k 400.0k +0.00% ``` Signed-off-by: Evgenii Stratonikov <e.stratonikov@yadro.com>
This commit is contained in:
parent
7eab9fcede
commit
3601feab81
3 changed files with 104 additions and 22 deletions
47
pkg/services/tree/heap/ordered_slice.go
Normal file
47
pkg/services/tree/heap/ordered_slice.go
Normal file
|
@ -0,0 +1,47 @@
|
|||
package heap
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"sort"
|
||||
|
||||
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/local_object_storage/pilorama"
|
||||
)
|
||||
|
||||
type OrderedSlice struct {
|
||||
nodes []pilorama.NodeInfo
|
||||
sorted bool
|
||||
}
|
||||
|
||||
func NewOrderedSlice() *OrderedSlice {
|
||||
return new(OrderedSlice)
|
||||
}
|
||||
|
||||
func (s *OrderedSlice) Insert(infos ...pilorama.NodeInfo) {
|
||||
s.sorted = false
|
||||
s.nodes = append(s.nodes, infos...)
|
||||
}
|
||||
|
||||
func (s *OrderedSlice) IsEmpty() bool {
|
||||
return len(s.nodes) == 0
|
||||
}
|
||||
|
||||
func (s *OrderedSlice) ExtractMin() pilorama.NodeInfo {
|
||||
if !s.sorted {
|
||||
sortByFilename(s.nodes)
|
||||
s.sorted = true
|
||||
}
|
||||
|
||||
node := s.nodes[0]
|
||||
s.nodes = s.nodes[1:]
|
||||
return node
|
||||
}
|
||||
|
||||
func sortByFilename(nodes []pilorama.NodeInfo) {
|
||||
if len(nodes) == 0 {
|
||||
return
|
||||
}
|
||||
less := func(i, j int) bool {
|
||||
return bytes.Compare(nodes[i].Meta.GetAttr(pilorama.AttributeFilename), nodes[j].Meta.GetAttr(pilorama.AttributeFilename)) < 0
|
||||
}
|
||||
sort.Slice(nodes, less)
|
||||
}
|
25
pkg/services/tree/heap/unordered_slice.go
Normal file
25
pkg/services/tree/heap/unordered_slice.go
Normal file
|
@ -0,0 +1,25 @@
|
|||
package heap
|
||||
|
||||
import (
|
||||
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/local_object_storage/pilorama"
|
||||
)
|
||||
|
||||
type UnorderedSlice []pilorama.NodeInfo
|
||||
|
||||
func NewUnorderedSlice() *UnorderedSlice {
|
||||
return &UnorderedSlice{}
|
||||
}
|
||||
|
||||
func (s *UnorderedSlice) Insert(infos ...pilorama.NodeInfo) {
|
||||
*s = append(*s, infos...)
|
||||
}
|
||||
|
||||
func (s *UnorderedSlice) IsEmpty() bool {
|
||||
return len(*s) == 0
|
||||
}
|
||||
|
||||
func (s *UnorderedSlice) ExtractMin() pilorama.NodeInfo {
|
||||
node := (*s)[0]
|
||||
*s = (*s)[1:]
|
||||
return node
|
||||
}
|
|
@ -5,11 +5,11 @@ import (
|
|||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"sort"
|
||||
"sync"
|
||||
"sync/atomic"
|
||||
|
||||
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/local_object_storage/pilorama"
|
||||
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/services/tree/heap"
|
||||
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/util/logger"
|
||||
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/container/acl"
|
||||
cidSDK "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/container/id"
|
||||
|
@ -440,29 +440,48 @@ func (s *Service) GetSubTree(req *GetSubTreeRequest, srv TreeService_GetSubTreeS
|
|||
return getSubTree(srv.Context(), srv, cid, b, s.forest)
|
||||
}
|
||||
|
||||
type Heap interface {
|
||||
Insert(...pilorama.NodeInfo)
|
||||
IsEmpty() bool
|
||||
ExtractMin() pilorama.NodeInfo
|
||||
}
|
||||
|
||||
func makeHeap(ordered bool) Heap {
|
||||
if ordered {
|
||||
return heap.NewOrderedSlice()
|
||||
}
|
||||
return heap.NewUnorderedSlice()
|
||||
}
|
||||
|
||||
func getSubTree(ctx context.Context, srv TreeService_GetSubTreeServer, cid cidSDK.ID, b *GetSubTreeRequest_Body, forest pilorama.Forest) error {
|
||||
ordered, err := needOrder(b.GetOrderBy().GetDirection())
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// Traverse the tree in a DFS manner. Because we need to support arbitrary depth,
|
||||
// recursive implementation is not suitable here, so we maintain explicit stack.
|
||||
m, p, err := forest.TreeGetMeta(ctx, cid, b.GetTreeId(), b.GetRootId())
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
stack := [][]pilorama.NodeInfo{{{
|
||||
|
||||
stack := []Heap{makeHeap(ordered)}
|
||||
stack[0].Insert(pilorama.NodeInfo{
|
||||
ID: b.GetRootId(),
|
||||
Meta: m,
|
||||
ParentID: p,
|
||||
}}}
|
||||
})
|
||||
|
||||
for {
|
||||
if len(stack) == 0 {
|
||||
break
|
||||
} else if len(stack[len(stack)-1]) == 0 {
|
||||
} else if stack[len(stack)-1].IsEmpty() {
|
||||
stack = stack[:len(stack)-1]
|
||||
continue
|
||||
}
|
||||
|
||||
node := stack[len(stack)-1][0]
|
||||
stack[len(stack)-1] = stack[len(stack)-1][1:]
|
||||
node := stack[len(stack)-1].ExtractMin()
|
||||
|
||||
err = srv.Send(&GetSubTreeResponse{
|
||||
Body: &GetSubTreeResponse_Body{
|
||||
|
@ -481,33 +500,24 @@ func getSubTree(ctx context.Context, srv TreeService_GetSubTreeServer, cid cidSD
|
|||
if err != nil {
|
||||
return err
|
||||
}
|
||||
children, err = sortByFilename(children, b.GetOrderBy().GetDirection())
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if len(children) != 0 {
|
||||
stack = append(stack, children)
|
||||
h := makeHeap(ordered)
|
||||
h.Insert(children...)
|
||||
stack = append(stack, h)
|
||||
}
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func sortByFilename(nodes []pilorama.NodeInfo, d GetSubTreeRequest_Body_Order_Direction) ([]pilorama.NodeInfo, error) {
|
||||
func needOrder(d GetSubTreeRequest_Body_Order_Direction) (bool, error) {
|
||||
switch d {
|
||||
case GetSubTreeRequest_Body_Order_None:
|
||||
return nodes, nil
|
||||
return false, nil
|
||||
case GetSubTreeRequest_Body_Order_Asc:
|
||||
if len(nodes) == 0 {
|
||||
return nodes, nil
|
||||
}
|
||||
less := func(i, j int) bool {
|
||||
return bytes.Compare(nodes[i].Meta.GetAttr(pilorama.AttributeFilename), nodes[j].Meta.GetAttr(pilorama.AttributeFilename)) < 0
|
||||
}
|
||||
sort.Slice(nodes, less)
|
||||
return nodes, nil
|
||||
return true, nil
|
||||
default:
|
||||
return nil, fmt.Errorf("unsupported order direction: %s", d.String())
|
||||
return false, fmt.Errorf("unsupported order direction: %s", d.String())
|
||||
}
|
||||
}
|
||||
|
||||
|
|
Loading…
Add table
Reference in a new issue