Dmitrii Stepanov
fb928616cc
All checks were successful
DCO action / DCO (pull_request) Successful in 35s
Vulncheck / Vulncheck (pull_request) Successful in 1m6s
Build / Build Components (pull_request) Successful in 1m45s
Pre-commit hooks / Pre-commit (pull_request) Successful in 1m44s
Tests and linters / Staticcheck (pull_request) Successful in 1m59s
Tests and linters / gopls check (pull_request) Successful in 2m19s
Tests and linters / Run gofumpt (pull_request) Successful in 2m35s
Tests and linters / Lint (pull_request) Successful in 2m56s
Tests and linters / Tests with -race (pull_request) Successful in 3m48s
Tests and linters / Tests (pull_request) Successful in 3m58s
To drop unnecessary parameters and return values. Signed-off-by: Dmitrii Stepanov <d.stepanov@yadro.com>
151 lines
2.9 KiB
Go
151 lines
2.9 KiB
Go
package tui
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"go.etcd.io/bbolt"
|
|
)
|
|
|
|
type Item[T any] struct {
|
|
val T
|
|
err error
|
|
}
|
|
|
|
func resolvePath(tx *bbolt.Tx, path [][]byte) (*bbolt.Bucket, error) {
|
|
if len(path) == 0 {
|
|
return nil, errors.New("can't find bucket without path")
|
|
}
|
|
|
|
name := path[0]
|
|
bucket := tx.Bucket(name)
|
|
if bucket == nil {
|
|
return nil, fmt.Errorf("no bucket with name %s", name)
|
|
}
|
|
for _, name := range path[1:] {
|
|
bucket = bucket.Bucket(name)
|
|
if bucket == nil {
|
|
return nil, fmt.Errorf("no bucket with name %s", name)
|
|
}
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func load[T any](
|
|
ctx context.Context, db *bbolt.DB, path [][]byte, bufferSize int,
|
|
filter func(key, value []byte) bool, transform func(key, value []byte) T,
|
|
) <-chan Item[T] {
|
|
buffer := make(chan Item[T], bufferSize)
|
|
|
|
go func() {
|
|
defer close(buffer)
|
|
|
|
err := db.View(func(tx *bbolt.Tx) error {
|
|
var cursor *bbolt.Cursor
|
|
if len(path) == 0 {
|
|
cursor = tx.Cursor()
|
|
} else {
|
|
bucket, err := resolvePath(tx, path)
|
|
if err != nil {
|
|
buffer <- Item[T]{err: fmt.Errorf("can't find bucket: %w", err)}
|
|
return nil
|
|
}
|
|
cursor = bucket.Cursor()
|
|
}
|
|
|
|
key, value := cursor.First()
|
|
for {
|
|
if key == nil {
|
|
return nil
|
|
}
|
|
if filter != nil && !filter(key, value) {
|
|
key, value = cursor.Next()
|
|
continue
|
|
}
|
|
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil
|
|
case buffer <- Item[T]{val: transform(key, value)}:
|
|
key, value = cursor.Next()
|
|
}
|
|
}
|
|
})
|
|
if err != nil {
|
|
buffer <- Item[T]{err: err}
|
|
}
|
|
}()
|
|
|
|
return buffer
|
|
}
|
|
|
|
func LoadBuckets(
|
|
ctx context.Context, db *bbolt.DB, path [][]byte, bufferSize int,
|
|
) <-chan Item[*Bucket] {
|
|
buffer := load(
|
|
ctx, db, path, bufferSize,
|
|
func(_, value []byte) bool {
|
|
return value == nil
|
|
},
|
|
func(key, _ []byte) *Bucket {
|
|
base := make([][]byte, 0, len(path))
|
|
base = append(base, path...)
|
|
|
|
return &Bucket{
|
|
Name: key,
|
|
Path: append(base, key),
|
|
}
|
|
},
|
|
)
|
|
|
|
return buffer
|
|
}
|
|
|
|
func LoadRecords(
|
|
ctx context.Context, db *bbolt.DB, path [][]byte, bufferSize int,
|
|
) <-chan Item[*Record] {
|
|
buffer := load(
|
|
ctx, db, path, bufferSize,
|
|
func(_, value []byte) bool {
|
|
return value != nil
|
|
},
|
|
func(key, value []byte) *Record {
|
|
base := make([][]byte, 0, len(path))
|
|
base = append(base, path...)
|
|
|
|
return &Record{
|
|
Key: key,
|
|
Value: value,
|
|
Path: append(base, key),
|
|
}
|
|
},
|
|
)
|
|
|
|
return buffer
|
|
}
|
|
|
|
// HasBuckets checks if a bucket has nested buckets. It relies on assumption
|
|
// that a bucket can have either nested buckets or records but not both.
|
|
func HasBuckets(ctx context.Context, db *bbolt.DB, path [][]byte) (bool, error) {
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
defer cancel()
|
|
|
|
buffer := load(
|
|
ctx, db, path, 1,
|
|
nil,
|
|
func(_, value []byte) []byte { return value },
|
|
)
|
|
|
|
x, ok := <-buffer
|
|
if !ok {
|
|
return false, nil
|
|
}
|
|
if x.err != nil {
|
|
return false, x.err
|
|
}
|
|
if x.val != nil {
|
|
return false, nil
|
|
}
|
|
return true, nil
|
|
}
|