mirror of
https://github.com/ipfs/kubo.git
synced 2026-03-07 01:08:08 +08:00
94 lines
1.9 KiB
Go
94 lines
1.9 KiB
Go
package reprovide
|
|
|
|
import (
|
|
"context"
|
|
|
|
blocks "github.com/ipfs/go-ipfs/blocks/blockstore"
|
|
merkledag "github.com/ipfs/go-ipfs/merkledag"
|
|
pin "github.com/ipfs/go-ipfs/pin"
|
|
|
|
cid "gx/ipfs/QmNp85zy9RLrQ5oQD4hPyS39ezrrXpcaa7R4Y9kxdWQLLQ/go-cid"
|
|
)
|
|
|
|
// NewBlockstoreProvider returns key provider using bstore.AllKeysChan
|
|
func NewBlockstoreProvider(bstore blocks.Blockstore) KeyChanFunc {
|
|
return func(ctx context.Context) (<-chan *cid.Cid, error) {
|
|
return bstore.AllKeysChan(ctx)
|
|
}
|
|
}
|
|
|
|
// NewPinnedProvider returns provider supplying pinned keys
|
|
func NewPinnedProvider(pinning pin.Pinner, dag merkledag.DAGService, onlyRoots bool) KeyChanFunc {
|
|
return func(ctx context.Context) (<-chan *cid.Cid, error) {
|
|
set, err := pinSet(ctx, pinning, dag, onlyRoots)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
outCh := make(chan *cid.Cid)
|
|
go func() {
|
|
defer close(outCh)
|
|
for c := range set.new {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case outCh <- c:
|
|
}
|
|
}
|
|
|
|
}()
|
|
|
|
return outCh, nil
|
|
}
|
|
}
|
|
|
|
func pinSet(ctx context.Context, pinning pin.Pinner, dag merkledag.DAGService, onlyRoots bool) (*streamingSet, error) {
|
|
set := newStreamingSet()
|
|
|
|
go func() {
|
|
defer close(set.new)
|
|
|
|
for _, key := range pinning.DirectKeys() {
|
|
set.add(key)
|
|
}
|
|
|
|
for _, key := range pinning.RecursiveKeys() {
|
|
set.add(key)
|
|
|
|
if !onlyRoots {
|
|
err := merkledag.EnumerateChildren(ctx, dag.GetLinks, key, set.add)
|
|
if err != nil {
|
|
log.Errorf("reprovide indirect pins: %s", err)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}()
|
|
|
|
return set, nil
|
|
}
|
|
|
|
type streamingSet struct {
|
|
set *cid.Set
|
|
new chan *cid.Cid
|
|
}
|
|
|
|
// NewSet initializes and returns a new Set.
|
|
func newStreamingSet() *streamingSet {
|
|
return &streamingSet{
|
|
set: cid.NewSet(),
|
|
new: make(chan *cid.Cid),
|
|
}
|
|
}
|
|
|
|
// add adds a Cid to the set only if it is
|
|
// not in it already.
|
|
func (s *streamingSet) add(c *cid.Cid) bool {
|
|
if s.set.Visit(c) {
|
|
s.new <- c
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|