forked from TrueCloudLab/distribution
1025 lines
25 KiB
Go
1025 lines
25 KiB
Go
|
package frostfs
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"crypto/ecdsa"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"net/http"
|
||
|
"path/filepath"
|
||
|
"sort"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"time"
|
||
|
|
||
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/container"
|
||
|
cid "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/container/id"
|
||
|
resolver "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/ns"
|
||
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/object"
|
||
|
oid "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/object/id"
|
||
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/object/transformer"
|
||
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/pool"
|
||
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/user"
|
||
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/version"
|
||
|
"github.com/distribution/distribution/v3/internal/dcontext"
|
||
|
storagedriver "github.com/distribution/distribution/v3/registry/storage/driver"
|
||
|
"github.com/distribution/distribution/v3/registry/storage/driver/base"
|
||
|
"github.com/distribution/distribution/v3/registry/storage/driver/factory"
|
||
|
"github.com/nspcc-dev/neo-go/cli/flags"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
driverName = "frostfs"
|
||
|
|
||
|
attributeFilePath = "FilePath"
|
||
|
attributeSHAState = "sha256state"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
paramPeers = "peers"
|
||
|
paramAddress = "address"
|
||
|
paramWeight = "weight"
|
||
|
paramPriority = "priority"
|
||
|
paramWallet = "wallet"
|
||
|
paramPath = "path"
|
||
|
paramPassword = "password"
|
||
|
paramContainer = "container"
|
||
|
paramConnectionTimeout = "connection_timeout"
|
||
|
paramRequestTimeout = "request_timeout"
|
||
|
paramRebalanceInterval = "rebalance_interval"
|
||
|
paramSessionExpirationDuration = "session_expiration_duration"
|
||
|
paramRPCEndpoint = "rpc_endpoint"
|
||
|
|
||
|
defaultConnectionTimeout = 4 * time.Second
|
||
|
defaultRequestTimeout = 4 * time.Second
|
||
|
defaultRebalanceInterval = 20 * time.Second
|
||
|
defaultSessionExpirationDuration = 100 // in epoch
|
||
|
)
|
||
|
|
||
|
// DriverParameters is a struct that encapsulates all of the driver parameters after all values have been set.
|
||
|
type DriverParameters struct {
|
||
|
ContainerID string
|
||
|
Peers []*PeerInfo
|
||
|
Wallet *Wallet
|
||
|
ConnectionTimeout time.Duration
|
||
|
RequestTimeout time.Duration
|
||
|
RebalanceInterval time.Duration
|
||
|
SessionExpirationDuration uint64
|
||
|
RPCEndpoint string
|
||
|
}
|
||
|
|
||
|
// Wallet contains params to get key from wallet.
|
||
|
type Wallet struct {
|
||
|
Path string
|
||
|
Password string
|
||
|
Address string
|
||
|
}
|
||
|
|
||
|
// PeerInfo contains node params.
|
||
|
type PeerInfo struct {
|
||
|
Address string
|
||
|
Weight float64
|
||
|
Priority int
|
||
|
}
|
||
|
|
||
|
func init() {
|
||
|
factory.Register(driverName, &frostfsDriverFactory{})
|
||
|
}
|
||
|
|
||
|
type frostfsDriverFactory struct{}
|
||
|
|
||
|
func (n *frostfsDriverFactory) Create(_ context.Context, parameters map[string]interface{}) (storagedriver.StorageDriver, error) {
|
||
|
return FromParameters(parameters)
|
||
|
}
|
||
|
|
||
|
type driver struct {
|
||
|
sdkPool *pool.Pool
|
||
|
owner *user.ID
|
||
|
key *ecdsa.PrivateKey
|
||
|
containerID cid.ID
|
||
|
maxSize uint64
|
||
|
}
|
||
|
|
||
|
type baseEmbed struct {
|
||
|
base.Base
|
||
|
}
|
||
|
|
||
|
// Driver is a storagedriver.StorageDriver implementation backed by FrostFS
|
||
|
// Objects are stored at absolute keys in the provided container.
|
||
|
type Driver struct {
|
||
|
baseEmbed
|
||
|
}
|
||
|
|
||
|
// FromParameters constructs a new Driver with a given parameters map
|
||
|
// Required parameters:
|
||
|
// - peers
|
||
|
// - wallet
|
||
|
// Optional Parameters:
|
||
|
// - connection_timeout
|
||
|
// - request_timeout
|
||
|
// - rebalance_interval
|
||
|
// - session_expiration_duration
|
||
|
// - rpc_endpoint
|
||
|
func FromParameters(parameters map[string]interface{}) (storagedriver.StorageDriver, error) {
|
||
|
peers, err := parsePeers(parameters)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
walletInfo, err := parseWallet(parameters)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
containerID, ok := parameters[paramContainer].(string)
|
||
|
if !ok {
|
||
|
return nil, fmt.Errorf("no container provided")
|
||
|
}
|
||
|
|
||
|
var rpcEndpoint string
|
||
|
rpcEndpointParam := parameters[paramRPCEndpoint]
|
||
|
if rpcEndpointParam != nil {
|
||
|
if rpcEndpoint, ok = rpcEndpointParam.(string); !ok {
|
||
|
return nil, fmt.Errorf("invalid rpc_endpoint param")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
connectionTimeout, err := parseTimeout(parameters, paramConnectionTimeout, defaultConnectionTimeout)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
requestTimeout, err := parseTimeout(parameters, paramRequestTimeout, defaultRequestTimeout)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
rebalanceInterval, err := parseTimeout(parameters, paramRebalanceInterval, defaultRebalanceInterval)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
expiration, err := parseUInt64(parameters, paramSessionExpirationDuration, defaultSessionExpirationDuration)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
params := DriverParameters{
|
||
|
Peers: peers,
|
||
|
ContainerID: containerID,
|
||
|
Wallet: walletInfo,
|
||
|
ConnectionTimeout: connectionTimeout,
|
||
|
RequestTimeout: requestTimeout,
|
||
|
RebalanceInterval: rebalanceInterval,
|
||
|
SessionExpirationDuration: expiration,
|
||
|
RPCEndpoint: rpcEndpoint,
|
||
|
}
|
||
|
|
||
|
return New(params)
|
||
|
}
|
||
|
|
||
|
func parseWallet(parameters map[string]interface{}) (*Wallet, error) {
|
||
|
walletInfo := new(Wallet)
|
||
|
|
||
|
walletParams, ok := parameters[paramWallet].(map[interface{}]interface{})
|
||
|
if !ok {
|
||
|
return nil, fmt.Errorf("no wallet params provided")
|
||
|
}
|
||
|
|
||
|
walletInfo.Path, ok = walletParams[paramPath].(string)
|
||
|
if !ok {
|
||
|
return nil, fmt.Errorf("no path provided")
|
||
|
}
|
||
|
|
||
|
walletInfo.Password, ok = walletParams[paramPassword].(string)
|
||
|
if !ok {
|
||
|
return nil, fmt.Errorf("no password provided")
|
||
|
}
|
||
|
|
||
|
addressParam := walletParams[paramAddress]
|
||
|
if addressParam != nil {
|
||
|
if walletInfo.Address, ok = addressParam.(string); !ok {
|
||
|
return nil, fmt.Errorf("invalid address param")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return walletInfo, nil
|
||
|
}
|
||
|
|
||
|
func parsePeers(parameters map[string]interface{}) ([]*PeerInfo, error) {
|
||
|
poolParams, ok := parameters[paramPeers].(map[interface{}]interface{})
|
||
|
if !ok {
|
||
|
return nil, fmt.Errorf("no peers params provided")
|
||
|
}
|
||
|
|
||
|
var peers []*PeerInfo
|
||
|
for _, val := range poolParams {
|
||
|
peerInfo, ok := val.(map[interface{}]interface{})
|
||
|
if !ok {
|
||
|
return nil, fmt.Errorf("invalid peers params")
|
||
|
}
|
||
|
|
||
|
peer := new(PeerInfo)
|
||
|
|
||
|
peer.Address, ok = peerInfo[paramAddress].(string)
|
||
|
if !ok {
|
||
|
return nil, fmt.Errorf("invalid peer address")
|
||
|
}
|
||
|
|
||
|
weightParam := peerInfo[paramWeight]
|
||
|
if weightParam != nil {
|
||
|
switch weight := weightParam.(type) {
|
||
|
case int:
|
||
|
peer.Weight = float64(weight)
|
||
|
case float64:
|
||
|
peer.Weight = weight
|
||
|
default:
|
||
|
return nil, fmt.Errorf("invalid weight param")
|
||
|
}
|
||
|
if peer.Weight <= 0 {
|
||
|
peer.Weight = 1
|
||
|
}
|
||
|
}
|
||
|
|
||
|
priorityParam := peerInfo[paramPriority]
|
||
|
if priorityParam != nil {
|
||
|
if peer.Priority, ok = priorityParam.(int); !ok {
|
||
|
return nil, fmt.Errorf("invalid priority param")
|
||
|
} else if peer.Priority <= 0 {
|
||
|
peer.Priority = 1
|
||
|
}
|
||
|
}
|
||
|
|
||
|
peers = append(peers, peer)
|
||
|
}
|
||
|
|
||
|
return peers, nil
|
||
|
}
|
||
|
|
||
|
func parseTimeout(parameters map[string]interface{}, name string, defaultValue time.Duration) (time.Duration, error) {
|
||
|
timeoutValue := parameters[name]
|
||
|
if timeoutValue == nil {
|
||
|
return defaultValue, nil
|
||
|
}
|
||
|
|
||
|
switch val := timeoutValue.(type) {
|
||
|
case int:
|
||
|
return time.Duration(val), nil
|
||
|
case int64:
|
||
|
return time.Duration(val), nil
|
||
|
case string:
|
||
|
timeout, err := time.ParseDuration(val)
|
||
|
if err != nil {
|
||
|
return 0, fmt.Errorf("couldn't parse duration '%s': %w", val, err)
|
||
|
}
|
||
|
return timeout, nil
|
||
|
}
|
||
|
|
||
|
return 0, fmt.Errorf("invalid %s", name)
|
||
|
}
|
||
|
|
||
|
func parseUInt64(parameters map[string]interface{}, name string, defaultValue uint64) (uint64, error) {
|
||
|
expirationValue := parameters[name]
|
||
|
if expirationValue == nil {
|
||
|
return defaultValue, nil
|
||
|
}
|
||
|
|
||
|
switch val := expirationValue.(type) {
|
||
|
case int:
|
||
|
return uint64(val), nil
|
||
|
case int64:
|
||
|
return uint64(val), nil
|
||
|
}
|
||
|
|
||
|
return 0, fmt.Errorf("invalid %s", name)
|
||
|
}
|
||
|
|
||
|
// New constructs a new Driver with the given FrostFS params
|
||
|
func New(params DriverParameters) (*Driver, error) {
|
||
|
ctx := context.Background()
|
||
|
|
||
|
acc, err := getAccount(params.Wallet)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
var owner user.ID
|
||
|
user.IDFromKey(&owner, acc.PrivateKey().PrivateKey.PublicKey)
|
||
|
|
||
|
sdkPool, err := createPool(ctx, acc, params)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't create sdk pool: %w", err)
|
||
|
}
|
||
|
|
||
|
maxObjectSize, err := getMaxObjectSize(ctx, sdkPool)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't get max object size: %w", err)
|
||
|
}
|
||
|
|
||
|
cnrID, err := getContainerID(params)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't get container id: %w", err)
|
||
|
}
|
||
|
|
||
|
d := &driver{
|
||
|
sdkPool: sdkPool,
|
||
|
owner: &owner,
|
||
|
key: &acc.PrivateKey().PrivateKey,
|
||
|
containerID: cnrID,
|
||
|
maxSize: maxObjectSize,
|
||
|
}
|
||
|
|
||
|
return &Driver{
|
||
|
baseEmbed: baseEmbed{
|
||
|
Base: base.Base{
|
||
|
StorageDriver: d,
|
||
|
},
|
||
|
},
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
func getMaxObjectSize(ctx context.Context, sdkPool *pool.Pool) (uint64, error) {
|
||
|
networkInfo, err := sdkPool.NetworkInfo(ctx)
|
||
|
if err != nil {
|
||
|
return 0, fmt.Errorf("couldn't get connection: %w", err)
|
||
|
}
|
||
|
|
||
|
maxObjectSize := networkInfo.MaxObjectSize()
|
||
|
|
||
|
if maxObjectSize == 0 {
|
||
|
return 0, fmt.Errorf("max object size must not be zero")
|
||
|
}
|
||
|
|
||
|
return maxObjectSize, nil
|
||
|
}
|
||
|
|
||
|
func getContainerID(params DriverParameters) (cid.ID, error) {
|
||
|
var (
|
||
|
cnrID cid.ID
|
||
|
domain container.Domain
|
||
|
)
|
||
|
if err := cnrID.DecodeString(params.ContainerID); err == nil {
|
||
|
return cnrID, nil
|
||
|
}
|
||
|
|
||
|
nnsResolver, err := createNnsResolver(params)
|
||
|
if err != nil {
|
||
|
return cid.ID{}, fmt.Errorf("couldn't create nns resolver: %w", err)
|
||
|
}
|
||
|
|
||
|
domain.SetName(params.ContainerID)
|
||
|
|
||
|
if cnrID, err = nnsResolver.ResolveContainerDomain(domain); err != nil {
|
||
|
return cid.ID{}, fmt.Errorf("couldn't resolve container name '%s': %w", params.ContainerID, err)
|
||
|
}
|
||
|
|
||
|
return cnrID, nil
|
||
|
}
|
||
|
|
||
|
func createPool(ctx context.Context, acc *wallet.Account, param DriverParameters) (*pool.Pool, error) {
|
||
|
var prm pool.InitParameters
|
||
|
prm.SetKey(&acc.PrivateKey().PrivateKey)
|
||
|
prm.SetNodeDialTimeout(param.ConnectionTimeout)
|
||
|
prm.SetHealthcheckTimeout(param.RequestTimeout)
|
||
|
prm.SetClientRebalanceInterval(param.RebalanceInterval)
|
||
|
prm.SetSessionExpirationDuration(param.SessionExpirationDuration)
|
||
|
|
||
|
for _, peer := range param.Peers {
|
||
|
prm.AddNode(pool.NewNodeParam(peer.Priority, peer.Address, peer.Weight))
|
||
|
}
|
||
|
|
||
|
p, err := pool.NewPool(prm)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("create pool: %w", err)
|
||
|
}
|
||
|
|
||
|
if err = p.Dial(ctx); err != nil {
|
||
|
return nil, fmt.Errorf("dial pool: %w", err)
|
||
|
}
|
||
|
|
||
|
return p, nil
|
||
|
}
|
||
|
|
||
|
func createNnsResolver(params DriverParameters) (*resolver.NNS, error) {
|
||
|
if params.RPCEndpoint == "" {
|
||
|
return nil, fmt.Errorf("empty rpc endpoind")
|
||
|
}
|
||
|
|
||
|
var nns resolver.NNS
|
||
|
if err := nns.Dial(params.RPCEndpoint); err != nil {
|
||
|
return nil, fmt.Errorf("dial nns resolver: %w", err)
|
||
|
}
|
||
|
|
||
|
return &nns, nil
|
||
|
}
|
||
|
|
||
|
func getAccount(walletInfo *Wallet) (*wallet.Account, error) {
|
||
|
w, err := wallet.NewWalletFromFile(walletInfo.Path)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
addr := w.GetChangeAddress()
|
||
|
if walletInfo.Address != "" {
|
||
|
addr, err = flags.ParseAddress(walletInfo.Address)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("invalid address")
|
||
|
}
|
||
|
}
|
||
|
acc := w.GetAccount(addr)
|
||
|
err = acc.Decrypt(walletInfo.Password, w.Scrypt)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return acc, nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) objectAddress(objID oid.ID) oid.Address {
|
||
|
var addr oid.Address
|
||
|
addr.SetContainer(d.containerID)
|
||
|
addr.SetObject(objID)
|
||
|
return addr
|
||
|
}
|
||
|
|
||
|
func (d *driver) formObject(path string) *object.Object {
|
||
|
attrFilePath := object.NewAttribute()
|
||
|
attrFilePath.SetKey(attributeFilePath)
|
||
|
attrFilePath.SetValue(path)
|
||
|
|
||
|
attrFileName := object.NewAttribute()
|
||
|
attrFileName.SetKey(object.AttributeFileName)
|
||
|
attrFileName.SetValue(filepath.Base(path))
|
||
|
|
||
|
attrTimestamp := object.NewAttribute()
|
||
|
attrTimestamp.SetKey(object.AttributeTimestamp)
|
||
|
attrTimestamp.SetValue(strconv.FormatInt(time.Now().UTC().Unix(), 10))
|
||
|
|
||
|
obj := object.New()
|
||
|
obj.SetOwnerID(d.owner)
|
||
|
obj.SetContainerID(d.containerID)
|
||
|
obj.SetAttributes(*attrFilePath, *attrFileName, *attrTimestamp)
|
||
|
|
||
|
return obj
|
||
|
}
|
||
|
|
||
|
func (d *driver) Name() string {
|
||
|
return driverName
|
||
|
}
|
||
|
|
||
|
func (d *driver) GetContent(ctx context.Context, path string) ([]byte, error) {
|
||
|
id, err := d.searchOne(ctx, path)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
var prm pool.PrmObjectGet
|
||
|
prm.SetAddress(d.objectAddress(id))
|
||
|
|
||
|
obj, err := d.sdkPool.GetObject(ctx, prm)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't get object '%s': %w", id, err)
|
||
|
}
|
||
|
|
||
|
return io.ReadAll(obj.Payload)
|
||
|
}
|
||
|
|
||
|
func (d *driver) PutContent(ctx context.Context, path string, content []byte) error {
|
||
|
if err := d.Delete(ctx, path); err != nil {
|
||
|
return fmt.Errorf("couldn't delete '%s': %s", path, err)
|
||
|
}
|
||
|
|
||
|
obj := d.formObject(path)
|
||
|
obj.SetPayload(content)
|
||
|
|
||
|
var prm pool.PrmObjectPut
|
||
|
prm.SetHeader(*obj)
|
||
|
|
||
|
if _, err := d.sdkPool.PutObject(ctx, prm); err != nil {
|
||
|
return fmt.Errorf("couldn't put object '%s': %w", path, err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) Reader(ctx context.Context, path string, offset int64) (io.ReadCloser, error) {
|
||
|
id, err := d.searchOne(ctx, path)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
addr := d.objectAddress(id)
|
||
|
|
||
|
var prmHead pool.PrmObjectHead
|
||
|
prmHead.SetAddress(addr)
|
||
|
|
||
|
obj, err := d.sdkPool.HeadObject(ctx, prmHead)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't head object '%s', id '%s': %w", path, id, err)
|
||
|
}
|
||
|
|
||
|
if uint64(offset) >= obj.PayloadSize() {
|
||
|
return nil, fmt.Errorf("invalid offset %d for object length %d", offset, obj.PayloadSize())
|
||
|
}
|
||
|
|
||
|
length := obj.PayloadSize() - uint64(offset)
|
||
|
|
||
|
var prmRange pool.PrmObjectRange
|
||
|
prmRange.SetAddress(addr)
|
||
|
prmRange.SetOffset(uint64(offset))
|
||
|
prmRange.SetLength(length)
|
||
|
|
||
|
res, err := d.sdkPool.ObjectRange(ctx, prmRange)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't get payload range of object '%s', offset %d, length %d, id '%s': %w",
|
||
|
path, offset, length, id, err)
|
||
|
}
|
||
|
|
||
|
return &res, nil
|
||
|
}
|
||
|
|
||
|
func getUploadUUID(ctx context.Context) (uuid string) {
|
||
|
return dcontext.GetStringValue(ctx, "vars.uuid")
|
||
|
}
|
||
|
|
||
|
func (d *driver) Writer(ctx context.Context, path string, append bool) (storagedriver.FileWriter, error) {
|
||
|
splitID := object.NewSplitID()
|
||
|
uploadUUID := getUploadUUID(ctx)
|
||
|
if err := splitID.Parse(uploadUUID); err != nil {
|
||
|
return nil, fmt.Errorf("couldn't parse split id as upload uuid '%s': %w", uploadUUID, err)
|
||
|
}
|
||
|
|
||
|
parts, noChild, err := d.headSplitParts(ctx, splitID, path, append)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't search split parts '%s': %w", path, err)
|
||
|
}
|
||
|
|
||
|
splitInfo := object.NewSplitInfo()
|
||
|
splitInfo.SetSplitID(splitID)
|
||
|
|
||
|
for _, obj := range parts {
|
||
|
if obj.Parent() != nil {
|
||
|
return nil, fmt.Errorf("object already exist '%s'", path)
|
||
|
}
|
||
|
|
||
|
prevID, _ := obj.PreviousID()
|
||
|
delete(noChild, prevID)
|
||
|
}
|
||
|
|
||
|
if len(noChild) > 1 {
|
||
|
return nil, fmt.Errorf("couldn't find last part '%s'", path)
|
||
|
}
|
||
|
|
||
|
for lastPartID := range noChild {
|
||
|
splitInfo.SetLastPart(lastPartID)
|
||
|
break
|
||
|
}
|
||
|
|
||
|
wrtr, err := newSizeLimiterWriter(ctx, d, path, splitInfo, parts)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't init size limiter writer: %w", err)
|
||
|
}
|
||
|
|
||
|
return wrtr, nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) Stat(ctx context.Context, path string) (storagedriver.FileInfo, error) {
|
||
|
if path == "/" { // healthcheck
|
||
|
if _, err := d.sdkPool.NetworkInfo(ctx); err != nil {
|
||
|
return nil, fmt.Errorf("healthcheck failed: %w", err)
|
||
|
}
|
||
|
return newFileInfoDir(path), nil
|
||
|
}
|
||
|
|
||
|
id, ok, err := d.searchOneBase(ctx, path, true)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
// assume there is not object with directory name
|
||
|
// e.g. if file '/a/b/c' exists, files '/a/b' and '/a' don't
|
||
|
if !ok {
|
||
|
return newFileInfoDir(path), nil
|
||
|
}
|
||
|
|
||
|
var prm pool.PrmObjectHead
|
||
|
prm.SetAddress(d.objectAddress(id))
|
||
|
|
||
|
obj, err := d.sdkPool.HeadObject(ctx, prm)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't get head object '%s': %w", id, err)
|
||
|
}
|
||
|
|
||
|
fileInfo := newFileInfo(ctx, obj, "")
|
||
|
|
||
|
// e.g. search '/a/b' but because of prefix search we found '/a/b/c'
|
||
|
// so we should return directory
|
||
|
if fileInfo.Path() != path {
|
||
|
return newFileInfoDir(path), nil
|
||
|
}
|
||
|
|
||
|
return fileInfo, nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) List(ctx context.Context, path string) ([]string, error) {
|
||
|
ids, err := d.searchByPrefix(ctx, path)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't search by prefix '%s': %w", path, err)
|
||
|
}
|
||
|
|
||
|
added := make(map[string]bool)
|
||
|
|
||
|
result := make([]string, 0, len(ids))
|
||
|
for _, id := range ids {
|
||
|
var prm pool.PrmObjectHead
|
||
|
prm.SetAddress(d.objectAddress(id))
|
||
|
|
||
|
obj, err := d.sdkPool.HeadObject(ctx, prm)
|
||
|
if err != nil {
|
||
|
dcontext.GetLogger(ctx).Warnf("couldn't get list object '%s' in path '%s': %s", id, path, err)
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
fileInf := newFileInfo(ctx, obj, path)
|
||
|
if !added[fileInf.Path()] {
|
||
|
result = append(result, fileInf.Path())
|
||
|
added[fileInf.Path()] = true
|
||
|
}
|
||
|
}
|
||
|
|
||
|
sort.Strings(result)
|
||
|
return result, nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) Move(ctx context.Context, sourcePath string, destPath string) error {
|
||
|
sourceID, err := d.searchOne(ctx, sourcePath)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if err = d.Delete(ctx, destPath); err != nil {
|
||
|
return fmt.Errorf("couldn't delete '%s' object: %w", destPath, err)
|
||
|
}
|
||
|
|
||
|
sourceAddr := d.objectAddress(sourceID)
|
||
|
|
||
|
var prmGet pool.PrmObjectGet
|
||
|
prmGet.SetAddress(sourceAddr)
|
||
|
|
||
|
obj, err := d.sdkPool.GetObject(ctx, prmGet)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("could not get source object '%s' by oid '%s': %w", sourcePath, sourceID, err)
|
||
|
}
|
||
|
defer func() {
|
||
|
if err = obj.Payload.Close(); err != nil {
|
||
|
dcontext.GetLogger(ctx).Errorf("couldn't close object payload reader, path '%s' by oid '%s': %s",
|
||
|
sourcePath, sourceID, err.Error())
|
||
|
}
|
||
|
}()
|
||
|
|
||
|
objHeader := d.formObject(destPath)
|
||
|
var prmPut pool.PrmObjectPut
|
||
|
prmPut.SetHeader(*objHeader)
|
||
|
prmPut.SetPayload(obj.Payload)
|
||
|
|
||
|
if _, err = d.sdkPool.PutObject(ctx, prmPut); err != nil {
|
||
|
return fmt.Errorf("couldn't put object '%s': %w", destPath, err)
|
||
|
}
|
||
|
|
||
|
var prmDelete pool.PrmObjectDelete
|
||
|
prmDelete.SetAddress(sourceAddr)
|
||
|
|
||
|
if err = d.sdkPool.DeleteObject(ctx, prmDelete); err != nil {
|
||
|
return fmt.Errorf("couldn't remove source file '%s', id '%s': %w", sourcePath, sourceID, err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) Delete(ctx context.Context, path string) error {
|
||
|
filters := object.NewSearchFilters()
|
||
|
filters.AddRootFilter()
|
||
|
filters.AddFilter(attributeFilePath, path, object.MatchStringEqual)
|
||
|
|
||
|
var prmSearch pool.PrmObjectSearch
|
||
|
prmSearch.SetContainerID(d.containerID)
|
||
|
prmSearch.SetFilters(filters)
|
||
|
|
||
|
res, err := d.sdkPool.SearchObjects(ctx, prmSearch)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("init searching using client: %w", err)
|
||
|
}
|
||
|
|
||
|
defer res.Close()
|
||
|
|
||
|
var inErr error
|
||
|
err = res.Iterate(func(id oid.ID) bool {
|
||
|
if err = d.delete(ctx, id); err != nil {
|
||
|
inErr = fmt.Errorf("couldn't delete object by path '%s': %w", path, err)
|
||
|
return true
|
||
|
}
|
||
|
return false
|
||
|
})
|
||
|
if err == nil {
|
||
|
err = inErr
|
||
|
}
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("iterate objects: %w", err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) delete(ctx context.Context, id oid.ID) error {
|
||
|
var prm pool.PrmObjectDelete
|
||
|
prm.SetAddress(d.objectAddress(id))
|
||
|
if err := d.sdkPool.DeleteObject(ctx, prm); err != nil {
|
||
|
return fmt.Errorf("couldn't delete object '%s': %w", id, err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) RedirectURL(_ *http.Request, _ string) (string, error) {
|
||
|
return "", nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) Walk(ctx context.Context, path string, f storagedriver.WalkFn, options ...func(*storagedriver.WalkOptions)) error {
|
||
|
return storagedriver.WalkFallback(ctx, d, path, f, options...)
|
||
|
}
|
||
|
|
||
|
func (d *driver) searchByPrefix(ctx context.Context, prefix string) ([]oid.ID, error) {
|
||
|
filters := object.NewSearchFilters()
|
||
|
filters.AddRootFilter()
|
||
|
filters.AddFilter(attributeFilePath, prefix, object.MatchCommonPrefix)
|
||
|
|
||
|
return d.baseSearch(ctx, filters)
|
||
|
}
|
||
|
|
||
|
func (d *driver) headSplitParts(ctx context.Context, splitID *object.SplitID, path string, isAppend bool) ([]*object.Object, map[oid.ID]struct{}, error) {
|
||
|
filters := object.NewSearchFilters()
|
||
|
filters.AddPhyFilter()
|
||
|
filters.AddSplitIDFilter(object.MatchStringEqual, splitID)
|
||
|
|
||
|
var prm pool.PrmObjectSearch
|
||
|
prm.SetContainerID(d.containerID)
|
||
|
prm.SetFilters(filters)
|
||
|
|
||
|
res, err := d.sdkPool.SearchObjects(ctx, prm)
|
||
|
if err != nil {
|
||
|
return nil, nil, fmt.Errorf("init searching using client: %w", err)
|
||
|
}
|
||
|
|
||
|
defer res.Close()
|
||
|
|
||
|
var addr oid.Address
|
||
|
addr.SetContainer(d.containerID)
|
||
|
|
||
|
var inErr error
|
||
|
var prmHead pool.PrmObjectHead
|
||
|
|
||
|
var objects []*object.Object
|
||
|
noChild := make(map[oid.ID]struct{})
|
||
|
|
||
|
err = res.Iterate(func(id oid.ID) bool {
|
||
|
addr.SetObject(id)
|
||
|
prmHead.SetAddress(addr)
|
||
|
|
||
|
obj, err := d.sdkPool.HeadObject(ctx, prmHead)
|
||
|
if err != nil {
|
||
|
inErr = fmt.Errorf("couldn't head object part '%s', id '%s', splitID '%s': %w", path, id, splitID, err)
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
if isAppend {
|
||
|
objects = append(objects, &obj)
|
||
|
noChild[id] = struct{}{}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
inErr = fmt.Errorf("init upload part '%s' already exist, splitID '%s'", path, splitID)
|
||
|
return true
|
||
|
})
|
||
|
if err == nil {
|
||
|
err = inErr
|
||
|
}
|
||
|
if err != nil {
|
||
|
return nil, nil, fmt.Errorf("iterate objects: %w", err)
|
||
|
}
|
||
|
|
||
|
return objects, noChild, nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) baseSearch(ctx context.Context, filters object.SearchFilters) ([]oid.ID, error) {
|
||
|
var prm pool.PrmObjectSearch
|
||
|
prm.SetContainerID(d.containerID)
|
||
|
prm.SetFilters(filters)
|
||
|
|
||
|
res, err := d.sdkPool.SearchObjects(ctx, prm)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("init searching using client: %w", err)
|
||
|
}
|
||
|
|
||
|
defer res.Close()
|
||
|
|
||
|
var buf []oid.ID
|
||
|
|
||
|
err = res.Iterate(func(id oid.ID) bool {
|
||
|
buf = append(buf, id)
|
||
|
return false
|
||
|
})
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("iterate objects: %w", err)
|
||
|
}
|
||
|
|
||
|
return buf, nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) searchOne(ctx context.Context, path string) (oid.ID, error) {
|
||
|
id, ok, err := d.searchOneBase(ctx, path, false)
|
||
|
if err != nil {
|
||
|
return oid.ID{}, err
|
||
|
}
|
||
|
|
||
|
if !ok {
|
||
|
return oid.ID{}, fmt.Errorf("found more than one object by path '%s'", path)
|
||
|
}
|
||
|
|
||
|
return id, nil
|
||
|
}
|
||
|
|
||
|
func (d *driver) searchOneBase(ctx context.Context, path string, byPrefix bool) (resID oid.ID, ok bool, err error) {
|
||
|
filters := object.NewSearchFilters()
|
||
|
filters.AddRootFilter()
|
||
|
if byPrefix {
|
||
|
filters.AddFilter(attributeFilePath, path, object.MatchCommonPrefix)
|
||
|
} else {
|
||
|
filters.AddFilter(attributeFilePath, path, object.MatchStringEqual)
|
||
|
}
|
||
|
|
||
|
var prm pool.PrmObjectSearch
|
||
|
prm.SetContainerID(d.containerID)
|
||
|
prm.SetFilters(filters)
|
||
|
|
||
|
res, err := d.sdkPool.SearchObjects(ctx, prm)
|
||
|
if err != nil {
|
||
|
return oid.ID{}, false, fmt.Errorf("init searching using client: %w", err)
|
||
|
}
|
||
|
|
||
|
defer res.Close()
|
||
|
|
||
|
var found bool
|
||
|
err = res.Iterate(func(id oid.ID) bool {
|
||
|
if found {
|
||
|
ok = false
|
||
|
return true
|
||
|
}
|
||
|
found = true
|
||
|
resID = id
|
||
|
ok = true
|
||
|
return false
|
||
|
})
|
||
|
if err != nil {
|
||
|
return oid.ID{}, false, fmt.Errorf("iterate objects by path '%s': %w", path, err)
|
||
|
}
|
||
|
|
||
|
if !found {
|
||
|
return oid.ID{}, false, storagedriver.PathNotFoundError{Path: path, DriverName: driverName}
|
||
|
}
|
||
|
|
||
|
return resID, ok, nil
|
||
|
}
|
||
|
|
||
|
func newFileInfo(ctx context.Context, obj object.Object, prefix string) storagedriver.FileInfo {
|
||
|
fileInfoFields := storagedriver.FileInfoFields{
|
||
|
Size: int64(obj.PayloadSize()),
|
||
|
}
|
||
|
|
||
|
for _, attr := range obj.Attributes() {
|
||
|
switch attr.Key() {
|
||
|
case attributeFilePath:
|
||
|
fileInfoFields.Path = attr.Value()
|
||
|
case object.AttributeTimestamp:
|
||
|
timestamp, err := strconv.ParseInt(attr.Value(), 10, 64)
|
||
|
if err != nil {
|
||
|
objID, _ := obj.ID()
|
||
|
dcontext.GetLogger(ctx).Warnf("object '%s' has invalid timestamp '%s'", objID.EncodeToString(), attr.Value())
|
||
|
continue
|
||
|
}
|
||
|
fileInfoFields.ModTime = time.Unix(timestamp, 0)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if len(prefix) > 0 {
|
||
|
tail := strings.TrimPrefix(fileInfoFields.Path, prefix)
|
||
|
if len(tail) > 0 {
|
||
|
index := strings.Index(tail[1:], "/")
|
||
|
if index >= 0 {
|
||
|
fileInfoFields.IsDir = true
|
||
|
fileInfoFields.Path = prefix + tail[:index+1]
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return storagedriver.FileInfoInternal{FileInfoFields: fileInfoFields}
|
||
|
}
|
||
|
|
||
|
func newFileInfoDir(path string) storagedriver.FileInfo {
|
||
|
return storagedriver.FileInfoInternal{
|
||
|
FileInfoFields: storagedriver.FileInfoFields{
|
||
|
Path: path,
|
||
|
ModTime: time.Now(),
|
||
|
IsDir: true,
|
||
|
},
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (d *driver) newObjTarget() transformer.ChunkedObjectWriter {
|
||
|
return &objTarget{
|
||
|
sdkPool: d.sdkPool,
|
||
|
key: d.key,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type objTarget struct {
|
||
|
sdkPool *pool.Pool
|
||
|
key *ecdsa.PrivateKey
|
||
|
obj *object.Object
|
||
|
chunks [][]byte
|
||
|
}
|
||
|
|
||
|
func (t *objTarget) WriteHeader(_ context.Context, obj *object.Object) error {
|
||
|
t.obj = obj
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (t *objTarget) Write(_ context.Context, p []byte) (n int, err error) {
|
||
|
t.chunks = append(t.chunks, p)
|
||
|
return len(p), nil
|
||
|
}
|
||
|
|
||
|
func (t *objTarget) Close(ctx context.Context) (*transformer.AccessIdentifiers, error) {
|
||
|
networkInfo, err := t.sdkPool.NetworkInfo(ctx)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't get network info: %w", err)
|
||
|
}
|
||
|
|
||
|
sz := 0
|
||
|
for i := range t.chunks {
|
||
|
sz += len(t.chunks[i])
|
||
|
}
|
||
|
|
||
|
ver := version.Current()
|
||
|
currEpoch := networkInfo.CurrentEpoch()
|
||
|
|
||
|
t.obj.SetPayloadSize(uint64(sz))
|
||
|
t.obj.SetVersion(&ver)
|
||
|
t.obj.SetCreationEpoch(currEpoch)
|
||
|
|
||
|
var (
|
||
|
parID oid.ID
|
||
|
parHdr *object.Object
|
||
|
)
|
||
|
|
||
|
if par := t.obj.Parent(); par != nil && par.Signature() == nil {
|
||
|
objPar := object.NewFromV2(par.ToV2())
|
||
|
|
||
|
objPar.SetCreationEpoch(currEpoch)
|
||
|
|
||
|
if err := object.SetIDWithSignature(*t.key, objPar); err != nil {
|
||
|
return nil, fmt.Errorf("could not finalize parent object: %w", err)
|
||
|
}
|
||
|
|
||
|
parID, _ = objPar.ID()
|
||
|
|
||
|
t.obj.SetParent(objPar)
|
||
|
}
|
||
|
|
||
|
if err = object.SetIDWithSignature(*t.key, t.obj); err != nil {
|
||
|
return nil, fmt.Errorf("could not finalize object: %w", err)
|
||
|
}
|
||
|
|
||
|
payload := make([]byte, 0, sz)
|
||
|
for i := range t.chunks {
|
||
|
payload = append(payload, t.chunks[i]...)
|
||
|
}
|
||
|
t.obj.SetPayload(payload)
|
||
|
|
||
|
var prm pool.PrmObjectPut
|
||
|
prm.SetHeader(*t.obj)
|
||
|
|
||
|
_, err = t.sdkPool.PutObject(ctx, prm)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("couldn't put part: %w", err)
|
||
|
}
|
||
|
|
||
|
objID, _ := t.obj.ID()
|
||
|
|
||
|
return &transformer.AccessIdentifiers{
|
||
|
ParentID: &parID,
|
||
|
SelfID: objID,
|
||
|
ParentHeader: parHdr,
|
||
|
}, nil
|
||
|
}
|