2024-10-15 12:27:28 +00:00
package util
import (
"context"
"fmt"
"strconv"
"sync"
"time"
"github.com/nspcc-dev/neo-go/cli/cmdargs"
"github.com/nspcc-dev/neo-go/cli/options"
"github.com/nspcc-dev/neo-go/pkg/core/block"
2024-12-13 13:29:17 +00:00
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
2024-10-15 12:27:28 +00:00
"github.com/nspcc-dev/neo-go/pkg/io"
2024-10-30 09:08:40 +00:00
"github.com/nspcc-dev/neo-go/pkg/rpcclient"
2024-12-12 15:46:42 +00:00
"github.com/nspcc-dev/neo-go/pkg/services/helpers/neofs"
2024-10-15 12:27:28 +00:00
"github.com/nspcc-dev/neofs-sdk-go/client"
2024-10-21 20:49:09 +00:00
"github.com/nspcc-dev/neofs-sdk-go/container"
2024-10-15 12:27:28 +00:00
cid "github.com/nspcc-dev/neofs-sdk-go/container/id"
"github.com/nspcc-dev/neofs-sdk-go/object"
oid "github.com/nspcc-dev/neofs-sdk-go/object/id"
"github.com/nspcc-dev/neofs-sdk-go/pool"
"github.com/nspcc-dev/neofs-sdk-go/user"
"github.com/urfave/cli/v2"
)
2024-11-25 20:04:28 +00:00
// poolWrapper wraps a NeoFS pool to adapt its Close method to return an error.
type poolWrapper struct {
* pool . Pool
}
// Close closes the pool and returns nil.
func ( p poolWrapper ) Close ( ) error {
p . Pool . Close ( )
return nil
}
2024-10-15 12:27:28 +00:00
func uploadBin ( ctx * cli . Context ) error {
if err := cmdargs . EnsureNone ( ctx ) ; err != nil {
return err
}
rpcNeoFS := ctx . StringSlice ( "fs-rpc-endpoint" )
containerIDStr := ctx . String ( "container" )
attr := ctx . String ( "block-attribute" )
2024-12-07 14:59:05 +00:00
numWorkers := ctx . Uint ( "workers" )
maxParallelSearches := ctx . Uint ( "searchers" )
maxRetries := ctx . Uint ( "retries" )
2024-11-15 12:35:37 +00:00
debug := ctx . Bool ( "debug" )
2024-12-07 14:59:05 +00:00
indexFileSize := ctx . Uint ( "index-file-size" )
indexAttrKey := ctx . String ( "index-attribute" )
2024-10-15 12:27:28 +00:00
acc , _ , err := options . GetAccFromContext ( ctx )
if err != nil {
return cli . Exit ( fmt . Sprintf ( "failed to load account: %v" , err ) , 1 )
}
var containerID cid . ID
if err = containerID . DecodeString ( containerIDStr ) ; err != nil {
return cli . Exit ( fmt . Sprintf ( "failed to decode container ID: %v" , err ) , 1 )
}
gctx , cancel := options . GetTimeoutContext ( ctx )
defer cancel ( )
rpc , err := options . GetRPCClient ( gctx , ctx )
if err != nil {
return cli . Exit ( fmt . Sprintf ( "failed to create RPC client: %v" , err ) , 1 )
}
signer := user . NewAutoIDSignerRFC6979 ( acc . PrivateKey ( ) . PrivateKey )
2024-10-17 22:00:20 +00:00
params := pool . DefaultOptions ( )
2024-12-12 15:46:42 +00:00
params . SetHealthcheckTimeout ( neofs . DefaultHealthcheckTimeout )
params . SetNodeDialTimeout ( neofs . DefaultDialTimeout )
params . SetNodeStreamTimeout ( neofs . DefaultStreamTimeout )
2024-10-17 22:00:20 +00:00
p , err := pool . New ( pool . NewFlatNodeParams ( rpcNeoFS ) , signer , params )
2024-10-15 12:27:28 +00:00
if err != nil {
return cli . Exit ( fmt . Sprintf ( "failed to create NeoFS pool: %v" , err ) , 1 )
}
2024-11-25 20:04:28 +00:00
pWrapper := poolWrapper { p }
if err = pWrapper . Dial ( context . Background ( ) ) ; err != nil {
2024-10-15 12:27:28 +00:00
return cli . Exit ( fmt . Sprintf ( "failed to dial NeoFS pool: %v" , err ) , 1 )
}
defer p . Close ( )
2024-10-23 10:21:32 +00:00
2024-10-21 20:49:09 +00:00
var containerObj container . Container
err = retry ( func ( ) error {
containerObj , err = p . ContainerGet ( ctx . Context , containerID , client . PrmContainerGet { } )
return err
2024-12-13 09:48:04 +00:00
} , maxRetries , debug )
2024-10-21 20:49:09 +00:00
if err != nil {
return cli . Exit ( fmt . Errorf ( "failed to get container with ID %s: %w" , containerID , err ) , 1 )
}
containerMagic := containerObj . Attribute ( "Magic" )
v , err := rpc . GetVersion ( )
if err != nil {
return cli . Exit ( fmt . Sprintf ( "failed to get version from RPC: %v" , err ) , 1 )
}
magic := strconv . Itoa ( int ( v . Protocol . Network ) )
if containerMagic != magic {
2024-10-31 15:13:35 +00:00
return cli . Exit ( fmt . Sprintf ( "container magic %s does not match the network magic %s" , containerMagic , magic ) , 1 )
2024-10-21 20:49:09 +00:00
}
currentBlockHeight , err := rpc . GetBlockCount ( )
if err != nil {
return cli . Exit ( fmt . Sprintf ( "failed to get current block height from RPC: %v" , err ) , 1 )
}
fmt . Fprintln ( ctx . App . Writer , "Chain block height:" , currentBlockHeight )
2024-12-13 13:29:17 +00:00
i , buf , err := searchIndexFile ( ctx , pWrapper , containerID , acc . PrivateKey ( ) , signer , indexFileSize , attr , indexAttrKey , maxParallelSearches , maxRetries , debug )
2024-12-07 14:59:05 +00:00
if err != nil {
return cli . Exit ( fmt . Errorf ( "failed to find objects: %w" , err ) , 1 )
2024-10-15 12:27:28 +00:00
}
2024-12-13 16:00:34 +00:00
err = uploadBlocksAndIndexFiles ( ctx , pWrapper , rpc , signer , containerID , attr , indexAttrKey , buf , i , indexFileSize , uint ( currentBlockHeight ) , numWorkers , maxRetries , debug )
2024-10-15 12:27:28 +00:00
if err != nil {
2024-12-07 14:59:05 +00:00
return cli . Exit ( fmt . Errorf ( "failed to upload objects: %w" , err ) , 1 )
2024-10-15 12:27:28 +00:00
}
return nil
}
// retry function with exponential backoff.
2024-12-13 09:48:04 +00:00
func retry ( action func ( ) error , maxRetries uint , debug bool ) error {
2024-10-15 12:27:28 +00:00
var err error
2024-12-12 15:46:42 +00:00
backoff := neofs . InitialBackoff
2024-12-13 09:48:04 +00:00
for i := range maxRetries {
2024-10-15 12:27:28 +00:00
if err = action ( ) ; err == nil {
return nil // Success, no retry needed.
}
2024-12-13 09:48:04 +00:00
if debug {
fmt . Printf ( "Retry %d: %v\n" , i , err )
}
2024-10-15 12:27:28 +00:00
time . Sleep ( backoff ) // Backoff before retrying.
2024-12-12 15:46:42 +00:00
backoff *= time . Duration ( neofs . BackoffFactor )
if backoff > neofs . MaxBackoff {
backoff = neofs . MaxBackoff
2024-10-15 12:27:28 +00:00
}
}
return err // Return the last error after exhausting retries.
}
2024-12-07 14:59:05 +00:00
// uploadBlocksAndIndexFiles uploads the blocks and index files to the container using the pool.
2024-12-13 16:00:34 +00:00
func uploadBlocksAndIndexFiles ( ctx * cli . Context , p poolWrapper , rpc * rpcclient . Client , signer user . Signer , containerID cid . ID , attr , indexAttributeKey string , buf [ ] byte , currentIndexFileID , indexFileSize , currentBlockHeight uint , numWorkers , maxRetries uint , debug bool ) error {
2024-12-07 14:59:05 +00:00
if currentIndexFileID * indexFileSize >= currentBlockHeight {
fmt . Fprintf ( ctx . App . Writer , "No new blocks to upload. Need to upload starting from %d, current height %d\n" , currentIndexFileID * indexFileSize , currentBlockHeight )
2024-10-30 09:08:40 +00:00
return nil
}
2024-12-07 14:59:05 +00:00
fmt . Fprintln ( ctx . App . Writer , "Uploading blocks and index files..." )
for indexFileStart := currentIndexFileID * indexFileSize ; indexFileStart < currentBlockHeight ; indexFileStart += indexFileSize {
2024-10-30 09:08:40 +00:00
var (
2024-12-07 14:59:05 +00:00
indexFileEnd = min ( indexFileStart + indexFileSize , currentBlockHeight )
errCh = make ( chan error )
doneCh = make ( chan struct { } )
wg sync . WaitGroup
2024-10-30 09:08:40 +00:00
)
2024-12-07 14:59:05 +00:00
fmt . Fprintf ( ctx . App . Writer , "Processing batch from %d to %d\n" , indexFileStart , indexFileEnd - 1 )
wg . Add ( int ( numWorkers ) )
2024-10-30 09:08:40 +00:00
for i := range numWorkers {
2024-12-07 14:59:05 +00:00
go func ( i uint ) {
2024-10-30 09:08:40 +00:00
defer wg . Done ( )
2024-12-07 14:59:05 +00:00
for blockIndex := indexFileStart + i ; blockIndex < indexFileEnd ; blockIndex += numWorkers {
2024-12-13 13:29:17 +00:00
if ! oid . ID ( buf [ blockIndex % indexFileSize * oid . Size : blockIndex % indexFileSize * oid . Size + oid . Size ] ) . IsZero ( ) {
2024-12-07 14:59:05 +00:00
if debug {
fmt . Fprintf ( ctx . App . Writer , "Block %d is already uploaded\n" , blockIndex )
}
continue
}
2024-10-30 09:08:40 +00:00
var blk * block . Block
errGet := retry ( func ( ) error {
var errGetBlock error
blk , errGetBlock = rpc . GetBlockByIndex ( uint32 ( blockIndex ) )
if errGetBlock != nil {
return fmt . Errorf ( "failed to fetch block %d: %w" , blockIndex , errGetBlock )
}
return nil
2024-12-13 09:48:04 +00:00
} , maxRetries , debug )
2024-10-30 09:08:40 +00:00
if errGet != nil {
select {
case errCh <- errGet :
default :
}
return
}
bw := io . NewBufBinWriter ( )
blk . EncodeBinary ( bw . BinWriter )
if bw . Err != nil {
select {
case errCh <- fmt . Errorf ( "failed to encode block %d: %w" , blockIndex , bw . Err ) :
default :
}
return
}
attrs := [ ] object . Attribute {
* object . NewAttribute ( attr , strconv . Itoa ( int ( blk . Index ) ) ) ,
* object . NewAttribute ( "Primary" , strconv . Itoa ( int ( blk . PrimaryIndex ) ) ) ,
* object . NewAttribute ( "Hash" , blk . Hash ( ) . StringLE ( ) ) ,
* object . NewAttribute ( "PrevHash" , blk . PrevHash . StringLE ( ) ) ,
2024-12-12 17:03:24 +00:00
* object . NewAttribute ( "BlockTime" , strconv . FormatUint ( blk . Timestamp , 10 ) ) ,
* object . NewAttribute ( "Timestamp" , strconv . FormatInt ( time . Now ( ) . Unix ( ) , 10 ) ) ,
2024-10-30 09:08:40 +00:00
}
2024-12-07 14:59:05 +00:00
var (
objBytes = bw . Bytes ( )
resOid oid . ID
)
2024-10-30 09:08:40 +00:00
errRetr := retry ( func ( ) error {
2024-12-07 14:59:05 +00:00
var errUpload error
2024-12-13 16:00:34 +00:00
resOid , errUpload = uploadObj ( ctx . Context , p , signer , containerID , objBytes , attrs )
2024-11-15 12:35:37 +00:00
if errUpload != nil {
return errUpload
}
if debug {
fmt . Fprintf ( ctx . App . Writer , "Uploaded block %d with object ID: %s\n" , blockIndex , resOid . String ( ) )
}
return errUpload
2024-12-13 09:48:04 +00:00
} , maxRetries , debug )
2024-10-30 09:08:40 +00:00
if errRetr != nil {
select {
case errCh <- errRetr :
default :
}
return
}
2024-12-13 13:29:17 +00:00
copy ( buf [ blockIndex % indexFileSize * oid . Size : ] , resOid [ : ] )
2024-10-30 09:08:40 +00:00
}
} ( i )
}
go func ( ) {
wg . Wait ( )
close ( doneCh )
} ( )
select {
case err := <- errCh :
return fmt . Errorf ( "upload error: %w" , err )
case <- doneCh :
}
2024-12-07 14:59:05 +00:00
fmt . Fprintf ( ctx . App . Writer , "Successfully processed batch of blocks: from %d to %d\n" , indexFileStart , indexFileEnd - 1 )
2024-10-30 09:08:40 +00:00
2024-12-07 14:59:05 +00:00
// Additional check for empty OIDs in the buffer.
2024-12-13 13:29:17 +00:00
for k := uint ( 0 ) ; k < ( indexFileEnd - indexFileStart ) * oid . Size ; k += oid . Size {
if oid . ID ( buf [ k : k + oid . Size ] ) . IsZero ( ) {
return fmt . Errorf ( "empty OID found in index file %d at position %d (block index %d)" , indexFileStart / indexFileSize , k / oid . Size , indexFileStart / indexFileSize * indexFileSize + k / oid . Size )
2024-12-07 14:59:05 +00:00
}
}
if indexFileEnd - indexFileStart == indexFileSize {
attrs := [ ] object . Attribute {
* object . NewAttribute ( indexAttributeKey , strconv . Itoa ( int ( indexFileStart / indexFileSize ) ) ) ,
* object . NewAttribute ( "IndexSize" , strconv . Itoa ( int ( indexFileSize ) ) ) ,
2024-12-12 17:03:24 +00:00
* object . NewAttribute ( "Timestamp" , strconv . FormatInt ( time . Now ( ) . Unix ( ) , 10 ) ) ,
2024-12-07 14:59:05 +00:00
}
err := retry ( func ( ) error {
var errUpload error
2024-12-13 16:00:34 +00:00
_ , errUpload = uploadObj ( ctx . Context , p , signer , containerID , buf , attrs )
2024-12-07 14:59:05 +00:00
return errUpload
2024-12-13 09:48:04 +00:00
} , maxRetries , debug )
2024-12-07 14:59:05 +00:00
if err != nil {
return fmt . Errorf ( "failed to upload index file: %w" , err )
}
fmt . Println ( "Successfully uploaded index file " , indexFileStart / indexFileSize )
}
clear ( buf )
2024-10-30 09:08:40 +00:00
}
return nil
}
2024-12-07 14:59:05 +00:00
// searchIndexFile returns the ID and buffer for the next index file to be uploaded.
2024-12-13 13:29:17 +00:00
func searchIndexFile ( ctx * cli . Context , p poolWrapper , containerID cid . ID , privKeys * keys . PrivateKey , signer user . Signer , indexFileSize uint , blockAttributeKey , attributeKey string , maxParallelSearches , maxRetries uint , debug bool ) ( uint , [ ] byte , error ) {
2024-10-15 12:27:28 +00:00
var (
2024-12-07 14:59:05 +00:00
// buf is used to store OIDs of the uploaded blocks.
2024-12-13 13:29:17 +00:00
buf = make ( [ ] byte , indexFileSize * oid . Size )
2024-12-07 14:59:05 +00:00
doneCh = make ( chan struct { } )
errCh = make ( chan error )
2024-11-21 14:54:25 +00:00
existingIndexCount = uint ( 0 )
filters = object . NewSearchFilters ( )
2024-10-15 12:27:28 +00:00
)
2024-11-01 16:50:05 +00:00
go func ( ) {
defer close ( doneCh )
2024-11-21 14:54:25 +00:00
// Search for existing index files.
filters . AddFilter ( "IndexSize" , fmt . Sprintf ( "%d" , indexFileSize ) , object . MatchStringEqual )
2024-12-07 14:59:05 +00:00
for i := 0 ; ; i ++ {
2024-12-13 13:29:17 +00:00
indexIDs := searchObjects ( ctx . Context , p , containerID , privKeys , attributeKey , uint ( i ) , uint ( i + 1 ) , 1 , maxRetries , debug , errCh , filters )
2024-12-17 10:53:19 +00:00
resOIDs := make ( [ ] oid . ID , 0 , 1 )
for id := range indexIDs {
resOIDs = append ( resOIDs , id )
2024-12-07 14:59:05 +00:00
}
2024-12-17 10:53:19 +00:00
if len ( resOIDs ) == 0 {
2024-12-07 14:59:05 +00:00
break
}
2024-12-17 10:53:19 +00:00
if len ( resOIDs ) > 1 {
fmt . Fprintf ( ctx . App . Writer , "WARN: %d duplicated index files with index %d found: %s\n" , len ( resOIDs ) , i , resOIDs )
}
2024-11-21 14:54:25 +00:00
existingIndexCount ++
}
2024-12-07 14:59:05 +00:00
fmt . Fprintf ( ctx . App . Writer , "Current index files count: %d\n" , existingIndexCount )
2024-11-21 14:54:25 +00:00
2024-12-07 14:59:05 +00:00
// Start block parsing goroutines.
var (
// processedIndices is a mapping from position in buffer to the block index.
// It prevents duplicates.
processedIndices sync . Map
wg sync . WaitGroup
oidCh = make ( chan oid . ID , 2 * maxParallelSearches )
)
wg . Add ( int ( maxParallelSearches ) )
for range maxParallelSearches {
go func ( ) {
defer wg . Done ( )
for id := range oidCh {
var obj object . Object
errRetr := retry ( func ( ) error {
var errGet error
obj , _ , errGet = p . ObjectGetInit ( ctx . Context , containerID , id , signer , client . PrmObjectGet { } )
return errGet
2024-12-13 09:48:04 +00:00
} , maxRetries , debug )
2024-12-07 14:59:05 +00:00
if errRetr != nil {
select {
case errCh <- fmt . Errorf ( "failed to fetch object %s: %w" , id . String ( ) , errRetr ) :
default :
2024-11-01 16:50:05 +00:00
}
2024-12-07 14:59:05 +00:00
return
2024-10-15 12:27:28 +00:00
}
2024-12-07 14:59:05 +00:00
blockIndex , err := getBlockIndex ( obj , blockAttributeKey )
if err != nil {
2024-11-01 16:50:05 +00:00
select {
2024-12-07 14:59:05 +00:00
case errCh <- fmt . Errorf ( "failed to get block index from object %s: %w" , id . String ( ) , err ) :
2024-11-01 16:50:05 +00:00
default :
}
return
}
2024-12-07 14:59:05 +00:00
pos := uint ( blockIndex ) % indexFileSize
if _ , ok := processedIndices . LoadOrStore ( pos , blockIndex ) ; ! ok {
2024-12-13 13:29:17 +00:00
copy ( buf [ pos * oid . Size : ] , id [ : ] )
2024-10-23 10:12:21 +00:00
}
2024-10-15 12:27:28 +00:00
}
2024-12-07 14:59:05 +00:00
} ( )
}
2024-10-15 12:27:28 +00:00
2024-12-07 14:59:05 +00:00
// Search for blocks within the index file range.
2024-12-13 13:29:17 +00:00
objIDs := searchObjects ( ctx . Context , p , containerID , privKeys , blockAttributeKey , existingIndexCount * indexFileSize , ( existingIndexCount + 1 ) * indexFileSize , maxParallelSearches , maxRetries , debug , errCh )
2024-12-07 14:59:05 +00:00
for id := range objIDs {
oidCh <- id
2024-10-23 12:27:45 +00:00
}
2024-12-07 14:59:05 +00:00
close ( oidCh )
wg . Wait ( )
2024-11-01 16:50:05 +00:00
} ( )
select {
case err := <- errCh :
2024-12-07 14:59:05 +00:00
return existingIndexCount , nil , err
2024-11-01 16:50:05 +00:00
case <- doneCh :
2024-12-07 14:59:05 +00:00
return existingIndexCount , buf , nil
2024-10-15 12:27:28 +00:00
}
}
2024-11-01 16:50:05 +00:00
// searchObjects searches in parallel for objects with attribute GE startIndex and LT
// endIndex. It returns a buffered channel of resulting object IDs and closes it once
// OID search is finished. Errors are sent to errCh in a non-blocking way.
2024-12-13 13:29:17 +00:00
func searchObjects ( ctx context . Context , p poolWrapper , containerID cid . ID , privKeys * keys . PrivateKey , blockAttributeKey string , startIndex , endIndex , maxParallelSearches , maxRetries uint , debug bool , errCh chan error , additionalFilters ... object . SearchFilters ) chan oid . ID {
2024-12-12 15:46:42 +00:00
var res = make ( chan oid . ID , 2 * neofs . DefaultSearchBatchSize )
2024-11-01 16:50:05 +00:00
go func ( ) {
var wg sync . WaitGroup
defer close ( res )
2024-12-12 15:46:42 +00:00
for i := startIndex ; i < endIndex ; i += neofs . DefaultSearchBatchSize * maxParallelSearches {
2024-11-01 16:50:05 +00:00
for j := range maxParallelSearches {
2024-12-12 15:46:42 +00:00
start := i + j * neofs . DefaultSearchBatchSize
end := start + neofs . DefaultSearchBatchSize
2024-11-01 16:50:05 +00:00
2024-12-07 14:59:05 +00:00
if start >= endIndex {
2024-11-01 16:50:05 +00:00
break
}
2024-12-07 14:59:05 +00:00
if end > endIndex {
end = endIndex
2024-11-01 16:50:05 +00:00
}
wg . Add ( 1 )
2024-12-07 14:59:05 +00:00
go func ( start , end uint ) {
2024-11-01 16:50:05 +00:00
defer wg . Done ( )
prm := client . PrmObjectSearch { }
filters := object . NewSearchFilters ( )
2024-11-21 14:54:25 +00:00
if len ( additionalFilters ) != 0 {
filters = additionalFilters [ 0 ]
}
2024-11-21 14:56:55 +00:00
if end == start + 1 {
filters . AddFilter ( blockAttributeKey , fmt . Sprintf ( "%d" , start ) , object . MatchStringEqual )
} else {
filters . AddFilter ( blockAttributeKey , fmt . Sprintf ( "%d" , start ) , object . MatchNumGE )
filters . AddFilter ( blockAttributeKey , fmt . Sprintf ( "%d" , end ) , object . MatchNumLT )
}
2024-11-01 16:50:05 +00:00
prm . SetFilters ( filters )
var objIDs [ ] oid . ID
err := retry ( func ( ) error {
var errBlockSearch error
2024-12-13 13:29:17 +00:00
objIDs , errBlockSearch = neofs . ObjectSearch ( ctx , p , privKeys , containerID . String ( ) , prm )
2024-11-01 16:50:05 +00:00
return errBlockSearch
2024-12-13 09:48:04 +00:00
} , maxRetries , debug )
2024-11-01 16:50:05 +00:00
if err != nil {
select {
case errCh <- fmt . Errorf ( "failed to search for block(s) from %d to %d: %w" , start , end , err ) :
default :
}
return
}
for _ , id := range objIDs {
res <- id
}
} ( start , end )
}
wg . Wait ( )
}
} ( )
return res
}
2024-10-31 15:13:35 +00:00
// uploadObj uploads object to the container using provided settings.
2024-12-13 16:00:34 +00:00
func uploadObj ( ctx context . Context , p poolWrapper , signer user . Signer , containerID cid . ID , objData [ ] byte , attrs [ ] object . Attribute ) ( oid . ID , error ) {
2024-10-15 12:27:28 +00:00
var (
hdr object . Object
prmObjectPutInit client . PrmObjectPutInit
2024-11-15 12:35:37 +00:00
resOID = oid . ID { }
2024-10-15 12:27:28 +00:00
)
hdr . SetContainerID ( containerID )
2024-12-13 16:00:34 +00:00
hdr . SetOwner ( signer . UserID ( ) )
2024-10-15 12:27:28 +00:00
hdr . SetAttributes ( attrs ... )
writer , err := p . ObjectPutInit ( ctx , hdr , signer , prmObjectPutInit )
if err != nil {
2024-11-15 12:35:37 +00:00
return resOID , fmt . Errorf ( "failed to initiate object upload: %w" , err )
2024-10-15 12:27:28 +00:00
}
_ , err = writer . Write ( objData )
if err != nil {
2024-11-02 18:43:20 +00:00
_ = writer . Close ( )
2024-11-15 12:35:37 +00:00
return resOID , fmt . Errorf ( "failed to write object data: %w" , err )
2024-10-15 12:27:28 +00:00
}
2024-11-02 18:43:20 +00:00
err = writer . Close ( )
if err != nil {
2024-11-15 12:35:37 +00:00
return resOID , fmt . Errorf ( "failed to close object writer: %w" , err )
2024-11-02 18:43:20 +00:00
}
res := writer . GetResult ( )
2024-11-15 12:35:37 +00:00
resOID = res . StoredObjectID ( )
return resOID , nil
2024-10-15 12:27:28 +00:00
}
2024-11-02 09:04:02 +00:00
func getBlockIndex ( header object . Object , attribute string ) ( int , error ) {
2024-10-15 12:27:28 +00:00
for _ , attr := range header . UserAttributes ( ) {
if attr . Key ( ) == attribute {
2024-10-23 10:14:35 +00:00
value := attr . Value ( )
blockIndex , err := strconv . Atoi ( value )
if err != nil {
return - 1 , fmt . Errorf ( "attribute %s has invalid value: %s, error: %w" , attribute , value , err )
}
return blockIndex , nil
2024-10-15 12:27:28 +00:00
}
}
return - 1 , fmt . Errorf ( "attribute %s not found" , attribute )
}