mirror of
https://github.com/ipfs/kubo.git
synced 2026-02-23 03:17:43 +08:00
270 lines
6.7 KiB
Go
270 lines
6.7 KiB
Go
// package blockservice implements a BlockService interface that provides
|
|
// a single GetBlock/AddBlock interface that seamlessly retrieves data either
|
|
// locally or from a remote peer through the exchange.
|
|
package blockservice
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/ipfs/go-ipfs/blocks/blockstore"
|
|
exchange "github.com/ipfs/go-ipfs/exchange"
|
|
bitswap "github.com/ipfs/go-ipfs/exchange/bitswap"
|
|
|
|
cid "gx/ipfs/QmNp85zy9RLrQ5oQD4hPyS39ezrrXpcaa7R4Y9kxdWQLLQ/go-cid"
|
|
blocks "gx/ipfs/QmSn9Td7xgxm9EV7iEjTckpUWmWApggzPxu7eFGWkkpwin/go-block-format"
|
|
logging "gx/ipfs/QmSpJByNKFX1sCsHBEp3R73FL4NF6FnQTEGyNAXHm2GS52/go-log"
|
|
)
|
|
|
|
var log = logging.Logger("blockservice")
|
|
|
|
var ErrNotFound = errors.New("blockservice: key not found")
|
|
|
|
// BlockService is a hybrid block datastore. It stores data in a local
|
|
// datastore and may retrieve data from a remote Exchange.
|
|
// It uses an internal `datastore.Datastore` instance to store values.
|
|
type BlockService interface {
|
|
Blockstore() blockstore.Blockstore
|
|
Exchange() exchange.Interface
|
|
AddBlock(o blocks.Block) (*cid.Cid, error)
|
|
AddBlocks(bs []blocks.Block) ([]*cid.Cid, error)
|
|
GetBlock(ctx context.Context, c *cid.Cid) (blocks.Block, error)
|
|
GetBlocks(ctx context.Context, ks []*cid.Cid) <-chan blocks.Block
|
|
DeleteBlock(o blocks.Block) error
|
|
Close() error
|
|
}
|
|
|
|
type blockService struct {
|
|
blockstore blockstore.Blockstore
|
|
exchange exchange.Interface
|
|
// If checkFirst is true then first check that a block doesn't
|
|
// already exist to avoid republishing the block on the exchange.
|
|
checkFirst bool
|
|
}
|
|
|
|
// NewBlockService creates a BlockService with given datastore instance.
|
|
func New(bs blockstore.Blockstore, rem exchange.Interface) BlockService {
|
|
if rem == nil {
|
|
log.Warning("blockservice running in local (offline) mode.")
|
|
}
|
|
|
|
return &blockService{
|
|
blockstore: bs,
|
|
exchange: rem,
|
|
checkFirst: true,
|
|
}
|
|
}
|
|
|
|
// NewWriteThrough ceates a BlockService that guarantees writes will go
|
|
// through to the blockstore and are not skipped by cache checks.
|
|
func NewWriteThrough(bs blockstore.Blockstore, rem exchange.Interface) BlockService {
|
|
if rem == nil {
|
|
log.Warning("blockservice running in local (offline) mode.")
|
|
}
|
|
|
|
return &blockService{
|
|
blockstore: bs,
|
|
exchange: rem,
|
|
checkFirst: false,
|
|
}
|
|
}
|
|
|
|
func (bs *blockService) Blockstore() blockstore.Blockstore {
|
|
return bs.blockstore
|
|
}
|
|
|
|
func (bs *blockService) Exchange() exchange.Interface {
|
|
return bs.exchange
|
|
}
|
|
|
|
// NewSession creates a bitswap session that allows for controlled exchange of
|
|
// wantlists to decrease the bandwidth overhead.
|
|
func NewSession(ctx context.Context, bs BlockService) *Session {
|
|
exchange := bs.Exchange()
|
|
if bswap, ok := exchange.(*bitswap.Bitswap); ok {
|
|
ses := bswap.NewSession(ctx)
|
|
return &Session{
|
|
ses: ses,
|
|
bs: bs.Blockstore(),
|
|
}
|
|
}
|
|
return &Session{
|
|
ses: exchange,
|
|
bs: bs.Blockstore(),
|
|
}
|
|
}
|
|
|
|
// AddBlock adds a particular block to the service, Putting it into the datastore.
|
|
// TODO pass a context into this if the remote.HasBlock is going to remain here.
|
|
func (s *blockService) AddBlock(o blocks.Block) (*cid.Cid, error) {
|
|
c := o.Cid()
|
|
if s.checkFirst {
|
|
has, err := s.blockstore.Has(c)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if has {
|
|
return c, nil
|
|
}
|
|
}
|
|
|
|
err := s.blockstore.Put(o)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := s.exchange.HasBlock(o); err != nil {
|
|
return nil, errors.New("blockservice is closed")
|
|
}
|
|
|
|
return c, nil
|
|
}
|
|
|
|
func (s *blockService) AddBlocks(bs []blocks.Block) ([]*cid.Cid, error) {
|
|
var toput []blocks.Block
|
|
if s.checkFirst {
|
|
for _, b := range bs {
|
|
has, err := s.blockstore.Has(b.Cid())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if !has {
|
|
toput = append(toput, b)
|
|
}
|
|
}
|
|
} else {
|
|
toput = bs
|
|
}
|
|
|
|
err := s.blockstore.PutMany(toput)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var ks []*cid.Cid
|
|
for _, o := range toput {
|
|
if err := s.exchange.HasBlock(o); err != nil {
|
|
return nil, fmt.Errorf("blockservice is closed (%s)", err)
|
|
}
|
|
|
|
ks = append(ks, o.Cid())
|
|
}
|
|
return ks, nil
|
|
}
|
|
|
|
// GetBlock retrieves a particular block from the service,
|
|
// Getting it from the datastore using the key (hash).
|
|
func (s *blockService) GetBlock(ctx context.Context, c *cid.Cid) (blocks.Block, error) {
|
|
log.Debugf("BlockService GetBlock: '%s'", c)
|
|
|
|
var f exchange.Fetcher
|
|
if s.exchange != nil {
|
|
f = s.exchange
|
|
}
|
|
|
|
return getBlock(ctx, c, s.blockstore, f)
|
|
}
|
|
|
|
func getBlock(ctx context.Context, c *cid.Cid, bs blockstore.Blockstore, f exchange.Fetcher) (blocks.Block, error) {
|
|
block, err := bs.Get(c)
|
|
if err == nil {
|
|
return block, nil
|
|
}
|
|
|
|
if err == blockstore.ErrNotFound && f != nil {
|
|
// TODO be careful checking ErrNotFound. If the underlying
|
|
// implementation changes, this will break.
|
|
log.Debug("Blockservice: Searching bitswap")
|
|
blk, err := f.GetBlock(ctx, c)
|
|
if err != nil {
|
|
if err == blockstore.ErrNotFound {
|
|
return nil, ErrNotFound
|
|
}
|
|
return nil, err
|
|
}
|
|
return blk, nil
|
|
}
|
|
|
|
log.Debug("Blockservice GetBlock: Not found")
|
|
if err == blockstore.ErrNotFound {
|
|
return nil, ErrNotFound
|
|
}
|
|
|
|
return nil, err
|
|
}
|
|
|
|
// GetBlocks gets a list of blocks asynchronously and returns through
|
|
// the returned channel.
|
|
// NB: No guarantees are made about order.
|
|
func (s *blockService) GetBlocks(ctx context.Context, ks []*cid.Cid) <-chan blocks.Block {
|
|
return getBlocks(ctx, ks, s.blockstore, s.exchange)
|
|
}
|
|
|
|
func getBlocks(ctx context.Context, ks []*cid.Cid, bs blockstore.Blockstore, f exchange.Fetcher) <-chan blocks.Block {
|
|
out := make(chan blocks.Block)
|
|
go func() {
|
|
defer close(out)
|
|
var misses []*cid.Cid
|
|
for _, c := range ks {
|
|
hit, err := bs.Get(c)
|
|
if err != nil {
|
|
misses = append(misses, c)
|
|
continue
|
|
}
|
|
log.Debug("Blockservice: Got data in datastore")
|
|
select {
|
|
case out <- hit:
|
|
case <-ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
|
|
if len(misses) == 0 {
|
|
return
|
|
}
|
|
|
|
rblocks, err := f.GetBlocks(ctx, misses)
|
|
if err != nil {
|
|
log.Debugf("Error with GetBlocks: %s", err)
|
|
return
|
|
}
|
|
|
|
for b := range rblocks {
|
|
select {
|
|
case out <- b:
|
|
case <-ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
return out
|
|
}
|
|
|
|
// DeleteBlock deletes a block in the blockservice from the datastore
|
|
func (s *blockService) DeleteBlock(o blocks.Block) error {
|
|
return s.blockstore.DeleteBlock(o.Cid())
|
|
}
|
|
|
|
func (s *blockService) Close() error {
|
|
log.Debug("blockservice is shutting down...")
|
|
return s.exchange.Close()
|
|
}
|
|
|
|
// Session is a helper type to provide higher level access to bitswap sessions
|
|
type Session struct {
|
|
bs blockstore.Blockstore
|
|
ses exchange.Fetcher
|
|
}
|
|
|
|
// GetBlock gets a block in the context of a request session
|
|
func (s *Session) GetBlock(ctx context.Context, c *cid.Cid) (blocks.Block, error) {
|
|
return getBlock(ctx, c, s.bs, s.ses)
|
|
}
|
|
|
|
// GetBlocks gets blocks in the context of a request session
|
|
func (s *Session) GetBlocks(ctx context.Context, ks []*cid.Cid) <-chan blocks.Block {
|
|
return getBlocks(ctx, ks, s.bs, s.ses)
|
|
}
|