ipfs-cluster/adder/sharding/adder.go
Hector Sanjuan f8ff9883b0 wip
License: MIT
Signed-off-by: Hector Sanjuan <code@hector.link>
2018-08-07 20:12:05 +02:00

102 lines
2.2 KiB
Go

// package sharding implements a sharding adder that chunks and
// shards content while it's added, creating Cluster DAGs and
// pinning them.
package sharding
import (
"context"
"fmt"
"mime/multipart"
cid "github.com/ipfs/go-cid"
"github.com/ipfs/ipfs-cluster/adder"
"github.com/ipfs/ipfs-cluster/api"
rpc "github.com/hsanjuan/go-libp2p-gorpc"
"github.com/ipfs/go-ipfs-cmdkit/files"
logging "github.com/ipfs/go-log"
)
var logger = logging.Logger("addshard")
type Adder struct {
rpcClient *rpc.Client
}
func New(rpc *rpc.Client) *Adder {
return &Adder{
rpcClient: rpc,
}
}
func (a *Adder) FromMultipart(ctx context.Context, r *multipart.Reader, p *adder.Params) (*cid.Cid, error) {
logger.Debugf("adding from multipart with params: %+v", p)
f := &files.MultipartFile{
Mediatype: "multipart/form-data",
Reader: r,
}
pinOpts := api.PinOptions{
ReplicationFactorMin: p.ReplicationFactorMin,
ReplicationFactorMax: p.ReplicationFactorMax,
Name: p.Name,
ShardSize: p.ShardSize,
}
dagBuilder := newClusterDAGBuilder(a.rpcClient, pinOpts)
// Always stop the builder
defer dagBuilder.Cancel()
blockSink := dagBuilder.Blocks()
blockHandle := func(ctx context.Context, n *api.NodeWithMeta) (string, error) {
logger.Debugf("handling block %s (size %d)", n.Cid, n.Size())
select {
case <-dagBuilder.Done():
return "", dagBuilder.Err()
case <-ctx.Done():
return "", ctx.Err()
case dagBuilder.Blocks() <- n:
return n.Cid, nil
}
}
logger.Debug("creating importer")
importer, err := adder.NewImporter(f, p)
if err != nil {
return nil, err
}
logger.Infof("importing file to Cluster (name '%s')", p.Name)
rootCidStr, err := importer.Run(ctx, blockHandle)
if err != nil {
logger.Error("Importing aborted: ", err)
return nil, err
}
// Trigger shard finalize
close(blockSink)
select {
case <-dagBuilder.Done(): // wait for the builder to finish
err = dagBuilder.Err()
case <-ctx.Done():
err = ctx.Err()
}
if err != nil {
logger.Info("import process finished with error: ", err)
return nil, err
}
rootCid, err := cid.Decode(rootCidStr)
if err != nil {
return nil, fmt.Errorf("bad root cid: %s", err)
}
logger.Info("import process finished successfully")
return rootCid, nil
}