2020-09-27 19:10:05 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"log"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/fatih/color"
|
|
|
|
blocks "github.com/ipfs/go-block-format"
|
|
|
|
"github.com/ipfs/go-blockservice"
|
|
|
|
"github.com/ipfs/go-cid"
|
|
|
|
ds "github.com/ipfs/go-datastore"
|
2022-06-14 15:00:51 +00:00
|
|
|
dssync "github.com/ipfs/go-datastore/sync"
|
2020-09-27 19:10:05 +00:00
|
|
|
exchange "github.com/ipfs/go-ipfs-exchange-interface"
|
|
|
|
offline "github.com/ipfs/go-ipfs-exchange-offline"
|
|
|
|
cbor "github.com/ipfs/go-ipld-cbor"
|
|
|
|
format "github.com/ipfs/go-ipld-format"
|
|
|
|
"github.com/ipfs/go-merkledag"
|
2022-11-07 21:50:55 +00:00
|
|
|
"golang.org/x/xerrors"
|
2022-06-14 15:00:51 +00:00
|
|
|
|
|
|
|
"github.com/filecoin-project/lotus/api/v0api"
|
|
|
|
"github.com/filecoin-project/lotus/blockstore"
|
|
|
|
"github.com/filecoin-project/lotus/chain/actors/adt"
|
2020-09-27 19:10:05 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Stores is a collection of the different stores and services that are needed
|
|
|
|
// to deal with the data layer of Filecoin, conveniently interlinked with one
|
|
|
|
// another.
|
|
|
|
type Stores struct {
|
|
|
|
CBORStore cbor.IpldStore
|
|
|
|
ADTStore adt.Store
|
|
|
|
Datastore ds.Batching
|
|
|
|
Blockstore blockstore.Blockstore
|
|
|
|
BlockService blockservice.BlockService
|
|
|
|
Exchange exchange.Interface
|
|
|
|
DAGService format.DAGService
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewProxyingStores is a set of Stores backed by a proxying Blockstore that
|
|
|
|
// proxies Get requests for unknown CIDs to a Filecoin node, via the
|
|
|
|
// ChainReadObj RPC.
|
2021-04-05 18:12:47 +00:00
|
|
|
func NewProxyingStores(ctx context.Context, api v0api.FullNode) *Stores {
|
2020-09-27 20:06:07 +00:00
|
|
|
ds := dssync.MutexWrap(ds.NewMapDatastore())
|
2020-09-27 19:10:05 +00:00
|
|
|
bs := &proxyingBlockstore{
|
|
|
|
ctx: ctx,
|
|
|
|
api: api,
|
2021-01-29 20:01:00 +00:00
|
|
|
Blockstore: blockstore.FromDatastore(ds),
|
2020-09-27 19:10:05 +00:00
|
|
|
}
|
|
|
|
return NewStores(ctx, ds, bs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewStores creates a non-proxying set of Stores.
|
|
|
|
func NewStores(ctx context.Context, ds ds.Batching, bs blockstore.Blockstore) *Stores {
|
|
|
|
var (
|
|
|
|
cborstore = cbor.NewCborStore(bs)
|
|
|
|
offl = offline.Exchange(bs)
|
|
|
|
blkserv = blockservice.New(bs, offl)
|
|
|
|
dserv = merkledag.NewDAGService(blkserv)
|
|
|
|
)
|
|
|
|
|
|
|
|
return &Stores{
|
|
|
|
CBORStore: cborstore,
|
|
|
|
ADTStore: adt.WrapStore(ctx, cborstore),
|
|
|
|
Datastore: ds,
|
|
|
|
Blockstore: bs,
|
|
|
|
Exchange: offl,
|
|
|
|
BlockService: blkserv,
|
|
|
|
DAGService: dserv,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TracingBlockstore is a Blockstore trait that records CIDs that were accessed
|
|
|
|
// through Get.
|
|
|
|
type TracingBlockstore interface {
|
|
|
|
// StartTracing starts tracing CIDs accessed through the this Blockstore.
|
|
|
|
StartTracing()
|
|
|
|
|
|
|
|
// FinishTracing finishes tracing accessed CIDs, and returns a map of the
|
|
|
|
// CIDs that were traced.
|
|
|
|
FinishTracing() map[cid.Cid]struct{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// proxyingBlockstore is a Blockstore wrapper that fetches unknown CIDs from
|
|
|
|
// a Filecoin node via JSON-RPC.
|
|
|
|
type proxyingBlockstore struct {
|
|
|
|
ctx context.Context
|
2021-04-05 18:12:47 +00:00
|
|
|
api v0api.FullNode
|
2020-09-27 19:10:05 +00:00
|
|
|
|
2020-10-09 21:20:15 +00:00
|
|
|
lk sync.Mutex
|
2020-09-27 19:10:05 +00:00
|
|
|
tracing bool
|
|
|
|
traced map[cid.Cid]struct{}
|
|
|
|
|
|
|
|
blockstore.Blockstore
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ TracingBlockstore = (*proxyingBlockstore)(nil)
|
|
|
|
|
|
|
|
func (pb *proxyingBlockstore) StartTracing() {
|
|
|
|
pb.lk.Lock()
|
|
|
|
pb.tracing = true
|
|
|
|
pb.traced = map[cid.Cid]struct{}{}
|
|
|
|
pb.lk.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pb *proxyingBlockstore) FinishTracing() map[cid.Cid]struct{} {
|
|
|
|
pb.lk.Lock()
|
|
|
|
ret := pb.traced
|
|
|
|
pb.tracing = false
|
|
|
|
pb.traced = map[cid.Cid]struct{}{}
|
|
|
|
pb.lk.Unlock()
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
2021-12-13 13:03:54 +00:00
|
|
|
func (pb *proxyingBlockstore) Get(ctx context.Context, cid cid.Cid) (blocks.Block, error) {
|
2020-10-09 21:20:15 +00:00
|
|
|
pb.lk.Lock()
|
2020-09-27 19:10:05 +00:00
|
|
|
if pb.tracing {
|
|
|
|
pb.traced[cid] = struct{}{}
|
|
|
|
}
|
2020-10-09 21:20:15 +00:00
|
|
|
pb.lk.Unlock()
|
2020-09-27 19:10:05 +00:00
|
|
|
|
2021-12-13 13:03:54 +00:00
|
|
|
if block, err := pb.Blockstore.Get(ctx, cid); err == nil {
|
2020-09-27 19:10:05 +00:00
|
|
|
return block, err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Println(color.CyanString("fetching cid via rpc: %v", cid))
|
|
|
|
item, err := pb.api.ChainReadObj(pb.ctx, cid)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
block, err := blocks.NewBlockWithCid(item, cid)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-12-13 13:03:54 +00:00
|
|
|
err = pb.Blockstore.Put(ctx, block)
|
2020-09-27 19:10:05 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return block, nil
|
|
|
|
}
|
2020-10-09 21:20:15 +00:00
|
|
|
|
2021-12-13 13:03:54 +00:00
|
|
|
func (pb *proxyingBlockstore) Put(ctx context.Context, block blocks.Block) error {
|
2020-10-09 21:20:15 +00:00
|
|
|
pb.lk.Lock()
|
|
|
|
if pb.tracing {
|
|
|
|
pb.traced[block.Cid()] = struct{}{}
|
|
|
|
}
|
|
|
|
pb.lk.Unlock()
|
2021-12-13 13:03:54 +00:00
|
|
|
return pb.Blockstore.Put(ctx, block)
|
2020-10-09 21:20:15 +00:00
|
|
|
}
|
2020-12-16 21:40:37 +00:00
|
|
|
|
2021-12-13 13:03:54 +00:00
|
|
|
func (pb *proxyingBlockstore) PutMany(ctx context.Context, blocks []blocks.Block) error {
|
2020-12-16 21:40:37 +00:00
|
|
|
pb.lk.Lock()
|
|
|
|
if pb.tracing {
|
|
|
|
for _, b := range blocks {
|
|
|
|
pb.traced[b.Cid()] = struct{}{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pb.lk.Unlock()
|
2021-12-13 13:03:54 +00:00
|
|
|
return pb.Blockstore.PutMany(ctx, blocks)
|
2020-12-16 21:40:37 +00:00
|
|
|
}
|
2022-11-07 21:38:50 +00:00
|
|
|
|
|
|
|
func (pb *proxyingBlockstore) View(ctx context.Context, c cid.Cid, callback func([]byte) error) error {
|
|
|
|
blk, err := pb.Get(ctx, c)
|
|
|
|
if err != nil {
|
|
|
|
return xerrors.Errorf("failed to Get cid %s: %w", c, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return callback(blk.RawData())
|
|
|
|
}
|