From fbeebfca79bd09e09146740eb9923a8cecae93bf Mon Sep 17 00:00:00 2001 From: Adin Schmahmann Date: Tue, 17 Jan 2023 01:15:43 -0500 Subject: [PATCH 1/4] feat(experiment): add a parallel HAMT traversal function --- hamt.go | 203 +++++++++++++++++++++++++++++++++++++++++++++++++++++--- 1 file changed, 194 insertions(+), 9 deletions(-) diff --git a/hamt.go b/hamt.go index e5891ff..1c0d291 100644 --- a/hamt.go +++ b/hamt.go @@ -4,8 +4,10 @@ import ( "bytes" "context" "fmt" + "golang.org/x/sync/errgroup" "math/big" "sort" + "sync" cid "github.com/ipfs/go-cid" cbor "github.com/ipfs/go-ipld-cbor" @@ -372,16 +374,27 @@ func loadNode( out.bitWidth = bitWidth out.hash = hashFunction + if err := validateNode(out, isRoot); err != nil { + return nil, err + } + return &out, nil +} + +// validates a node +func validateNode( + out Node, + isRoot bool, +) error { // Validation // too many elements in the data array for the configured bitWidth? if len(out.Pointers) > 1< bucketSize { - return nil, ErrMalformedHamt + return ErrMalformedHamt } for i := 1; i < len(ch.KVs); i++ { if bytes.Compare(ch.KVs[i-1].Key, ch.KVs[i].Key) >= 0 { - return nil, ErrMalformedHamt + return ErrMalformedHamt } } } @@ -410,16 +423,16 @@ func loadNode( if !isRoot { // the only valid empty node is a root node if len(out.Pointers) == 0 { - return nil, ErrMalformedHamt + return ErrMalformedHamt } // a non-root node that contains <=bucketSize direct elements should not // exist under compaction rules if out.directChildCount() == 0 && out.directKVCount() <= bucketSize { - return nil, ErrMalformedHamt + return ErrMalformedHamt } } - return &out, nil + return nil } // checkSize computes the total serialized size of the entire HAMT. @@ -877,3 +890,175 @@ func (n *Node) ForEach(ctx context.Context, f func(k string, val *cbg.Deferred) } return nil } + +// ForEachParallel calls function f on each k / val pair found in the HAMT. +// This performs a full traversal of the graph and for large HAMTs can cause +// a large number of loads from the underlying store. +// The values are returned as raw bytes, not decoded. +// Unlike ForEach this runs in parallel so passed callbacks should not conflict with each other +func (n *Node) ForEachParallel(ctx context.Context, f func(k string, val *cbg.Deferred) error) error { + return parallelShardWalk(ctx, n, f) +} + +type OptionalInteger struct { + Value int + Error error +} + +// TODO: This interface is obviously wrong, but we may want some GetMany-style grouping or even a "session" object +// we can push CIDs into to leverage efficiencies without tons of goroutines +type getManyIPLDStore interface { + GetMany(ctx context.Context, cids []cid.Cid, outs []interface{}) <-chan *OptionalInteger +} + +type listCidsAndShards struct { + cids []cid.Cid + shards []*Node +} + +func (n *Node) walkChildren(f func(k string, val *cbg.Deferred) error) (*listCidsAndShards, error) { + res := &listCidsAndShards{} + + for _, p := range n.Pointers { + if p.isShard() { + if p.cache != nil { + res.shards = append(res.shards, p.cache) + } else { + res.cids = append(res.cids, p.Link) + } + } else { + for _, kv := range p.KVs { + if err := f(string(kv.Key), kv.Value); err != nil { + return nil, err + } + } + } + } + + return res, nil +} + +// parallelShardWalk walks the HAMT concurrently processing callbacks upon encountering leaf nodes +func parallelShardWalk(ctx context.Context, root *Node, processShardValues func(k string, val *cbg.Deferred) error) error { + const concurrency = 16 // TODO: should be an option, also this number was basically made up with a bit of empirical testing/usage + + var visitlk sync.Mutex + visitSet := cid.NewSet() + visit := visitSet.Visit + + // Setup synchronization + grp, errGrpCtx := errgroup.WithContext(ctx) + + // Input and output queues for workers. + feed := make(chan *listCidsAndShards) + out := make(chan *listCidsAndShards) + done := make(chan struct{}) + + for i := 0; i < concurrency; i++ { + grp.Go(func() error { + for feedChildren := range feed { + for _, nextShard := range feedChildren.shards { + nextChildren, err := nextShard.walkChildren(processShardValues) + if err != nil { + return err + } + + select { + case out <- nextChildren: + case <-errGrpCtx.Done(): + return nil + } + } + + var linksToVisit []cid.Cid + for _, nextCid := range feedChildren.cids { + var shouldVisit bool + + visitlk.Lock() + shouldVisit = visit(nextCid) + visitlk.Unlock() + + if shouldVisit { + linksToVisit = append(linksToVisit, nextCid) + } + } + + // TODO: allow for Pointer caching + dserv := root.store.(getManyIPLDStore) + nodes := make([]interface{}, len(linksToVisit)) + for i := 0; i < len(linksToVisit); i++ { + nodes[i] = new(Node) + } + chNodes := dserv.GetMany(errGrpCtx, linksToVisit, nodes) + for optNode := range chNodes { + if optNode.Error != nil { + return optNode.Error + } + nextShard := nodes[optNode.Value].(*Node) + nextShard.store = root.store + nextShard.bitWidth = root.bitWidth + nextShard.hash = root.hash + if err := validateNode(*nextShard, false); err != nil { + return err + } + + nextChildren, err := nextShard.walkChildren(processShardValues) + if err != nil { + return err + } + + select { + case out <- nextChildren: + case <-errGrpCtx.Done(): + return nil + } + } + + select { + case done <- struct{}{}: + case <-errGrpCtx.Done(): + } + } + return nil + }) + } + + send := feed + var todoQueue []*listCidsAndShards + var inProgress int + + next := &listCidsAndShards{ + shards: []*Node{root}, + } + +dispatcherLoop: + for { + select { + case send <- next: + inProgress++ + if len(todoQueue) > 0 { + next = todoQueue[0] + todoQueue = todoQueue[1:] + } else { + next = nil + send = nil + } + case <-done: + inProgress-- + if inProgress == 0 && next == nil { + break dispatcherLoop + } + case nextNodes := <-out: + if next == nil { + next = nextNodes + send = feed + } else { + todoQueue = append(todoQueue, nextNodes) + } + case <-errGrpCtx.Done(): + break dispatcherLoop + } + } + close(feed) + return grp.Wait() +} From 043dd12291125f86cec1d689fabe127114de2e3e Mon Sep 17 00:00:00 2001 From: i-norden Date: Tue, 17 Oct 2023 17:02:31 -0500 Subject: [PATCH 2/4] update parallel ForEach method --- hamt.go | 112 +++++++++++++++++++++++++++++++------------------------- 1 file changed, 62 insertions(+), 50 deletions(-) diff --git a/hamt.go b/hamt.go index 1c0d291..fc30358 100644 --- a/hamt.go +++ b/hamt.go @@ -4,11 +4,12 @@ import ( "bytes" "context" "fmt" - "golang.org/x/sync/errgroup" "math/big" "sort" "sync" + "golang.org/x/sync/errgroup" + cid "github.com/ipfs/go-cid" cbor "github.com/ipfs/go-ipld-cbor" cbg "github.com/whyrusleeping/cbor-gen" @@ -896,35 +897,35 @@ func (n *Node) ForEach(ctx context.Context, f func(k string, val *cbg.Deferred) // a large number of loads from the underlying store. // The values are returned as raw bytes, not decoded. // Unlike ForEach this runs in parallel so passed callbacks should not conflict with each other -func (n *Node) ForEachParallel(ctx context.Context, f func(k string, val *cbg.Deferred) error) error { - return parallelShardWalk(ctx, n, f) +func (n *Node) ForEachParallel(ctx context.Context, f func(k string, val *cbg.Deferred) error, concurrency int) error { + return parallelShardWalk(ctx, n, f, concurrency) } -type OptionalInteger struct { - Value int - Error error -} - -// TODO: This interface is obviously wrong, but we may want some GetMany-style grouping or even a "session" object -// we can push CIDs into to leverage efficiencies without tons of goroutines -type getManyIPLDStore interface { - GetMany(ctx context.Context, cids []cid.Cid, outs []interface{}) <-chan *OptionalInteger +type child struct { + cid cid.Cid + shard *Node } type listCidsAndShards struct { - cids []cid.Cid - shards []*Node + children []child } func (n *Node) walkChildren(f func(k string, val *cbg.Deferred) error) (*listCidsAndShards, error) { res := &listCidsAndShards{} + res.children = make([]child, 0, len(n.Pointers)) for _, p := range n.Pointers { if p.isShard() { if p.cache != nil { - res.shards = append(res.shards, p.cache) + res.children = append(res.children, child{ + shard: p.cache, + }) + } else if p.Link != cid.Undef { + res.children = append(res.children, child{ + cid: p.Link, + }) } else { - res.cids = append(res.cids, p.Link) + continue } } else { for _, kv := range p.KVs { @@ -939,9 +940,7 @@ func (n *Node) walkChildren(f func(k string, val *cbg.Deferred) error) (*listCid } // parallelShardWalk walks the HAMT concurrently processing callbacks upon encountering leaf nodes -func parallelShardWalk(ctx context.Context, root *Node, processShardValues func(k string, val *cbg.Deferred) error) error { - const concurrency = 16 // TODO: should be an option, also this number was basically made up with a bit of empirical testing/usage - +func parallelShardWalk(ctx context.Context, root *Node, processShardValues func(k string, val *cbg.Deferred) error, concurrency int) error { var visitlk sync.Mutex visitSet := cid.NewSet() visit := visitSet.Visit @@ -957,44 +956,51 @@ func parallelShardWalk(ctx context.Context, root *Node, processShardValues func( for i := 0; i < concurrency; i++ { grp.Go(func() error { for feedChildren := range feed { - for _, nextShard := range feedChildren.shards { - nextChildren, err := nextShard.walkChildren(processShardValues) - if err != nil { - return err - } - - select { - case out <- nextChildren: - case <-errGrpCtx.Done(): - return nil - } - } - - var linksToVisit []cid.Cid - for _, nextCid := range feedChildren.cids { - var shouldVisit bool - - visitlk.Lock() - shouldVisit = visit(nextCid) - visitlk.Unlock() - - if shouldVisit { - linksToVisit = append(linksToVisit, nextCid) + linksToVisit := make([]cid.Cid, 0, len(feedChildren.children)) + for _, nextChild := range feedChildren.children { + if nextChild.shard != nil { + nextChildren, err := nextChild.shard.walkChildren(processShardValues) + if err != nil { + return err + } + select { + case out <- nextChildren: + case <-errGrpCtx.Done(): + return nil + } + } else if nextChild.cid != cid.Undef { + var shouldVisit bool + + visitlk.Lock() + shouldVisit = visit(nextChild.cid) + visitlk.Unlock() + + if shouldVisit { + linksToVisit = append(linksToVisit, nextChild.cid) + } + } else { + return fmt.Errorf("invalid child") } } // TODO: allow for Pointer caching - dserv := root.store.(getManyIPLDStore) + dserv := root.store.(cbor.IpldBatchOpStore) nodes := make([]interface{}, len(linksToVisit)) for i := 0; i < len(linksToVisit); i++ { nodes[i] = new(Node) } - chNodes := dserv.GetMany(errGrpCtx, linksToVisit, nodes) - for optNode := range chNodes { - if optNode.Error != nil { - return optNode.Error + cursorChan, missingCIDs, err := dserv.GetMany(errGrpCtx, linksToVisit, nodes) + if err != nil { + return err + } + if len(missingCIDs) != 0 { + return fmt.Errorf("GetMany returned an incomplete result set. The set is missing these CIDs: %+v", missingCIDs) + } + for cursor := range cursorChan { + if cursor.Err != nil { + return cursor.Err } - nextShard := nodes[optNode.Value].(*Node) + nextShard := nodes[cursor.Index].(*Node) nextShard.store = root.store nextShard.bitWidth = root.bitWidth nextShard.hash = root.hash @@ -1027,9 +1033,15 @@ func parallelShardWalk(ctx context.Context, root *Node, processShardValues func( var todoQueue []*listCidsAndShards var inProgress int - next := &listCidsAndShards{ - shards: []*Node{root}, + // start the walk + children, err := root.walkChildren(processShardValues) + // if we hit an error or there are no children, then we're done + if err != nil || children == nil { + close(feed) + grp.Wait() + return err } + next := children dispatcherLoop: for { From 2e2ad36fcd86e6f494ae57937e46f8f5f209320a Mon Sep 17 00:00:00 2001 From: i-norden Date: Tue, 17 Oct 2023 17:02:45 -0500 Subject: [PATCH 3/4] parallel ForEach unit test --- hamt_test.go | 158 ++++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 156 insertions(+), 2 deletions(-) diff --git a/hamt_test.go b/hamt_test.go index b6f810a..d00f0b5 100644 --- a/hamt_test.go +++ b/hamt_test.go @@ -12,6 +12,7 @@ import ( "strconv" "strings" "sync" + "sync/atomic" "testing" "time" @@ -33,7 +34,7 @@ func newMockBlocks() *mockBlocks { return &mockBlocks{make(map[cid.Cid]block.Block), sync.Mutex{}, blockstoreStats{}} } -func (mb *mockBlocks) Get(c cid.Cid) (block.Block, error) { +func (mb *mockBlocks) Get(ctx context.Context, c cid.Cid) (block.Block, error) { mb.dataMu.Lock() defer mb.dataMu.Unlock() mb.stats.evtcntGet++ @@ -44,7 +45,24 @@ func (mb *mockBlocks) Get(c cid.Cid) (block.Block, error) { return nil, fmt.Errorf("Not Found") } -func (mb *mockBlocks) Put(b block.Block) error { +func (mb *mockBlocks) GetMany(ctx context.Context, cs []cid.Cid) ([]block.Block, []cid.Cid, error) { + mb.dataMu.Lock() + defer mb.dataMu.Unlock() + blocks := make([]block.Block, 0, len(cs)) + missingCIDs := make([]cid.Cid, 0, len(cs)) + for _, c := range cs { + mb.stats.evtcntGet++ + d, ok := mb.data[c] + if !ok { + missingCIDs = append(missingCIDs, c) + } else { + blocks = append(blocks, d) + } + } + return blocks, missingCIDs, nil +} + +func (mb *mockBlocks) Put(ctx context.Context, b block.Block) error { mb.dataMu.Lock() defer mb.dataMu.Unlock() mb.stats.evtcntPut++ @@ -55,6 +73,19 @@ func (mb *mockBlocks) Put(b block.Block) error { return nil } +func (mb *mockBlocks) PutMany(ctx context.Context, bs []block.Block) error { + mb.dataMu.Lock() + defer mb.dataMu.Unlock() + for _, b := range bs { + mb.stats.evtcntPut++ + if _, exists := mb.data[b.Cid()]; exists { + mb.stats.evtcntPutDup++ + } + mb.data[b.Cid()] = b + } + return nil +} + type blockstoreStats struct { evtcntGet int evtcntPut int @@ -562,6 +593,129 @@ func testBasic(t *testing.T, options ...Option) { } } +func TestForEach(t *testing.T) { + testForEach(t) + testForEachParallel(t) +} + +func testForEach(t *testing.T, options ...Option) { + ctx := context.Background() + cs := cbor.NewCborStore(newMockBlocks()) + begn, err := NewNode(cs, options...) + require.NoError(t, err) + + val := cborstr("cat dog bear") + valueBuf := new(bytes.Buffer) + err = val.MarshalCBOR(valueBuf) + require.NoError(t, err) + err = begn.Set(ctx, "foo", val) + require.NoError(t, err) + + kvs := make(map[string][]byte, 1001) + kvs["foo"] = valueBuf.Bytes() + + for i := 0; i < 1000; i++ { + k := randKey() + v := randValue() + valueBuf := new(bytes.Buffer) + err := v.MarshalCBOR(valueBuf) + require.NoError(t, err) + err = begn.Set(ctx, k, v) + require.NoError(t, err) + kvs[k] = valueBuf.Bytes() + } + require.NoError(t, begn.Flush(ctx)) + + called := 0 + f := func(key string, val *cbg.Deferred) error { + called++ + expectedVal, ok := kvs[key] + require.True(t, ok) + require.Equal(t, expectedVal, val.Raw) + return nil + } + + err = begn.ForEach(ctx, f) + require.NoError(t, err) + require.Equal(t, 1001, called) +} + +func testForEachParallel(t *testing.T, options ...Option) { + ctx := context.Background() + cs := cbor.NewGetManyCborStore(newMockBlocks()) + begn, err := NewNode(cs, options...) + require.NoError(t, err) + + val := cborstr("cat dog bear") + valueBuf := new(bytes.Buffer) + err = val.MarshalCBOR(valueBuf) + require.NoError(t, err) + err = begn.Set(ctx, "foo", val) + require.NoError(t, err) + + kvs := make(map[string][]byte, 10001) + kvs["foo"] = valueBuf.Bytes() + + for i := 0; i < 10000; i++ { + k := randKey() + v := randValue() + valueBuf := new(bytes.Buffer) + err := v.MarshalCBOR(valueBuf) + require.NoError(t, err) + err = begn.Set(ctx, k, v) + require.NoError(t, err) + kvs[k] = valueBuf.Bytes() + } + // test before flushing + var called uint64 = 0 + f := func(key string, val *cbg.Deferred) error { + atomic.AddUint64(&called, 1) + expectedVal, ok := kvs[key] + require.True(t, ok) + require.Equal(t, expectedVal, val.Raw) + return nil + } + + err = begn.ForEachParallel(ctx, f, 16) + require.NoError(t, err) + require.Equal(t, uint64(10001), called) + + require.NoError(t, begn.Flush(ctx)) + c, err := cs.Put(ctx, begn) + require.NoError(t, err) + + // test after flushing + called = 0 + f = func(key string, val *cbg.Deferred) error { + atomic.AddUint64(&called, 1) + expectedVal, ok := kvs[key] + require.True(t, ok) + require.Equal(t, expectedVal, val.Raw) + return nil + } + + err = begn.ForEachParallel(ctx, f, 16) + require.NoError(t, err) + require.Equal(t, uint64(10001), called) + + loadedRoot, err := LoadNode(ctx, cs, c) + require.NoError(t, err) + + // test with loaded root + called = 0 + f = func(key string, val *cbg.Deferred) error { + atomic.AddUint64(&called, 1) + expectedVal, ok := kvs[key] + require.True(t, ok) + require.Equal(t, expectedVal, val.Raw) + return nil + } + + err = loadedRoot.ForEachParallel(ctx, f, 16) + require.NoError(t, err) + require.Equal(t, uint64(10001), called) +} + func TestSetIfAbsent(t *testing.T) { ctx := context.Background() cs := cbor.NewCborStore(newMockBlocks()) From 5ad168969dcb2cfd4737bf13f43c595a53840bce Mon Sep 17 00:00:00 2001 From: i-norden Date: Tue, 17 Oct 2023 17:05:05 -0500 Subject: [PATCH 4/4] go mod after rebase --- go.mod | 40 ++++++++++--------- go.sum | 120 +++++++++++++++++++++++++++++++++------------------------ 2 files changed, 92 insertions(+), 68 deletions(-) diff --git a/go.mod b/go.mod index 1a0bd9e..b6aa560 100644 --- a/go.mod +++ b/go.mod @@ -1,35 +1,39 @@ module github.com/filecoin-project/go-hamt-ipld/v3 require ( - github.com/ipfs/go-block-format v0.0.2 - github.com/ipfs/go-cid v0.0.6 - github.com/ipfs/go-ipld-cbor v0.0.4 + github.com/ipfs/go-block-format v0.1.2 + github.com/ipfs/go-cid v0.4.1 + github.com/ipfs/go-ipld-cbor v0.1.0 github.com/spaolacci/murmur3 v1.1.0 github.com/stretchr/testify v1.7.0 - github.com/whyrusleeping/cbor-gen v0.0.0-20220323183124-98fa8256a799 + github.com/whyrusleeping/cbor-gen v0.0.0-20230818171029-f91ae536ca25 golang.org/x/sync v0.0.0-20210220032951-036812b2e83c - golang.org/x/sys v0.1.0 // indirect - golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 + golang.org/x/sys v0.12.0 // indirect + golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 // indirect ) require ( github.com/davecgh/go-spew v1.1.1 // indirect - github.com/ipfs/go-ipfs-util v0.0.1 // indirect - github.com/ipfs/go-ipld-format v0.0.2 // indirect + github.com/gopherjs/gopherjs v1.17.2 // indirect + github.com/ipfs/go-ipfs-util v0.0.2 // indirect + github.com/ipfs/go-ipld-format v0.5.0 // indirect + github.com/klauspost/cpuid/v2 v2.2.5 // indirect github.com/kr/pretty v0.1.0 // indirect - github.com/minio/blake2b-simd v0.0.0-20160723061019-3f5f724cb5b1 // indirect - github.com/minio/sha256-simd v0.1.1-0.20190913151208-6de447530771 // indirect - github.com/mr-tron/base58 v1.1.3 // indirect - github.com/multiformats/go-base32 v0.0.3 // indirect - github.com/multiformats/go-base36 v0.1.0 // indirect - github.com/multiformats/go-multibase v0.0.3 // indirect - github.com/multiformats/go-multihash v0.0.13 // indirect - github.com/multiformats/go-varint v0.0.5 // indirect + github.com/minio/sha256-simd v1.0.1 // indirect + github.com/mr-tron/base58 v1.2.0 // indirect + github.com/multiformats/go-base32 v0.1.0 // indirect + github.com/multiformats/go-base36 v0.2.0 // indirect + github.com/multiformats/go-multibase v0.2.0 // indirect + github.com/multiformats/go-multihash v0.2.3 // indirect + github.com/multiformats/go-varint v0.0.7 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect - github.com/polydawn/refmt v0.0.0-20190221155625-df39d6c2d992 // indirect - golang.org/x/crypto v0.1.0 // indirect + github.com/polydawn/refmt v0.89.0 // indirect + golang.org/x/crypto v0.12.0 // indirect gopkg.in/yaml.v3 v3.0.0 // indirect + lukechampine.com/blake3 v1.2.1 // indirect ) go 1.20 + +replace github.com/ipfs/go-ipld-cbor => github.com/vulcanize/go-ipld-cbor v0.1.1-internal-0.0.1 diff --git a/go.sum b/go.sum index 2409d50..ecb7190 100644 --- a/go.sum +++ b/go.sum @@ -1,92 +1,112 @@ +github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= +github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/go-yaml/yaml v2.1.0+incompatible/go.mod h1:w2MrLa16VYP0jy6N7M5kHaCkaLENm+P+Tv+MfurjSw0= github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1 h1:EGx4pi6eqNxGaHF6qqu48+N2wcFQ5qg5FXgOdqsJ5d8= +github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= -github.com/gxed/hashland/keccakpg v0.0.1/go.mod h1:kRzw3HkwxFU1mpmPP8v1WyQzwdGfmKFJ6tItnhQ67kU= -github.com/gxed/hashland/murmur3 v0.0.1/go.mod h1:KjXop02n4/ckmZSnY2+HKcLud/tcmvhST0bie/0lS48= -github.com/ipfs/go-block-format v0.0.2 h1:qPDvcP19izTjU8rgo6p7gTXZlkMkF5bz5G3fqIsSCPE= -github.com/ipfs/go-block-format v0.0.2/go.mod h1:AWR46JfpcObNfg3ok2JHDUfdiHRgWhJgCQF+KIgOPJY= -github.com/ipfs/go-cid v0.0.1/go.mod h1:GHWU/WuQdMPmIosc4Yn1bcCT7dSeX4lBafM7iqUPQvM= -github.com/ipfs/go-cid v0.0.2/go.mod h1:GHWU/WuQdMPmIosc4Yn1bcCT7dSeX4lBafM7iqUPQvM= -github.com/ipfs/go-cid v0.0.3/go.mod h1:GHWU/WuQdMPmIosc4Yn1bcCT7dSeX4lBafM7iqUPQvM= -github.com/ipfs/go-cid v0.0.6 h1:go0y+GcDOGeJIV01FeBsta4FHngoA4Wz7KMeLkXAhMs= +github.com/gopherjs/gopherjs v1.17.2 h1:fQnZVsXk8uxXIStYb0N4bGk7jeyTalG/wsZjQ25dO0g= +github.com/gopherjs/gopherjs v1.17.2/go.mod h1:pRRIvn/QzFLrKfvEz3qUuEhtE/zLCWfreZ6J5gM2i+k= +github.com/hashicorp/golang-lru/v2 v2.0.5 h1:wW7h1TG88eUIJ2i69gaE3uNVtEPIagzhGvHgwfx2Vm4= +github.com/ipfs/bbloom v0.0.4 h1:Gi+8EGJ2y5qiD5FbsbpX/TMNcJw8gSqr7eyjHa4Fhvs= +github.com/ipfs/boxo v0.13.1 h1:nQ5oQzcMZR3oL41REJDcTbrvDvuZh3J9ckc9+ILeRQI= +github.com/ipfs/go-block-format v0.1.2 h1:GAjkfhVx1f4YTODS6Esrj1wt2HhrtwTnhEr+DyPUaJo= +github.com/ipfs/go-block-format v0.1.2/go.mod h1:mACVcrxarQKstUU3Yf/RdwbC4DzPV6++rO2a3d+a/KE= github.com/ipfs/go-cid v0.0.6/go.mod h1:6Ux9z5e+HpkQdckYoX1PG/6xqKspzlEIR5SDmgqgC/I= -github.com/ipfs/go-ipfs-util v0.0.1 h1:Wz9bL2wB2YBJqggkA4dD7oSmqB4cAnpNbGrlHJulv50= -github.com/ipfs/go-ipfs-util v0.0.1/go.mod h1:spsl5z8KUnrve+73pOhSVZND1SIxPW5RyBCNzQxlJBc= -github.com/ipfs/go-ipld-cbor v0.0.4 h1:Aw3KPOKXjvrm6VjwJvFf1F1ekR/BH3jdof3Bk7OTiSA= -github.com/ipfs/go-ipld-cbor v0.0.4/go.mod h1:BkCduEx3XBCO6t2Sfo5BaHzuok7hbhdMm9Oh8B2Ftq4= -github.com/ipfs/go-ipld-format v0.0.1/go.mod h1:kyJtbkDALmFHv3QR6et67i35QzO3S0dCDnkOJhcZkms= -github.com/ipfs/go-ipld-format v0.0.2 h1:OVAGlyYT6JPZ0pEfGntFPS40lfrDmaDbQwNHEY2G9Zs= -github.com/ipfs/go-ipld-format v0.0.2/go.mod h1:4B6+FM2u9OJ9zCV+kSbgFAZlOrv1Hqbf0INGQgiKf9k= -github.com/jtolds/gls v4.2.1+incompatible h1:fSuqC+Gmlu6l/ZYAoZzx2pyucC8Xza35fpRVWLVmUEE= -github.com/jtolds/gls v4.2.1+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= +github.com/ipfs/go-cid v0.4.1 h1:A/T3qGvxi4kpKWWcPC/PgbvDA2bjVLO7n4UeVwnbs/s= +github.com/ipfs/go-cid v0.4.1/go.mod h1:uQHwDeX4c6CtyrFwdqyhpNcxVewur1M7l7fNU7LKwZk= +github.com/ipfs/go-datastore v0.6.0 h1:JKyz+Gvz1QEZw0LsX1IBn+JFCJQH4SJVFtM4uWU0Myk= +github.com/ipfs/go-detect-race v0.0.1 h1:qX/xay2W3E4Q1U7d9lNs1sU9nvguX0a7319XbyQ6cOk= +github.com/ipfs/go-ipfs-util v0.0.2 h1:59Sswnk1MFaiq+VcaknX7aYEyGyGDAA73ilhEK2POp8= +github.com/ipfs/go-ipfs-util v0.0.2/go.mod h1:CbPtkWJzjLdEcezDns2XYaehFVNXG9zrdrtMecczcsQ= +github.com/ipfs/go-ipld-format v0.5.0 h1:WyEle9K96MSrvr47zZHKKcDxJ/vlpET6PSiQsAFO+Ds= +github.com/ipfs/go-ipld-format v0.5.0/go.mod h1:ImdZqJQaEouMjCvqCe0ORUS+uoBmf7Hf+EO/jh+nk3M= +github.com/ipfs/go-log/v2 v2.5.1 h1:1XdUzF7048prq4aBjDQQ4SL5RxftpRGdXhNRwKSAlcY= +github.com/ipfs/go-metrics-interface v0.0.1 h1:j+cpbjYvu4R8zbleSs36gvB7jR+wsL2fGD6n0jO4kdg= +github.com/jbenet/goprocess v0.1.4 h1:DRGOFReOMqqDNXwW70QkacFW0YN9QnwLV0Vqk+3oU0o= +github.com/jtolds/gls v4.20.0+incompatible h1:xdiiI2gbIgH/gLH7ADydsJ1uDOEzR8yvV7C0MuV77Wo= +github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= +github.com/klauspost/cpuid/v2 v2.2.5 h1:0E5MSMDEoAulmXNFquVs//DdoomxaoTY1kUhbc/qbZg= +github.com/klauspost/cpuid/v2 v2.2.5/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws= github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= -github.com/minio/blake2b-simd v0.0.0-20160723061019-3f5f724cb5b1 h1:lYpkrQH5ajf0OXOcUbGjvZxxijuBwbbmlSxLiuofa+g= +github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA= github.com/minio/blake2b-simd v0.0.0-20160723061019-3f5f724cb5b1/go.mod h1:pD8RvIylQ358TN4wwqatJ8rNavkEINozVn9DtGI3dfQ= -github.com/minio/sha256-simd v0.0.0-20190131020904-2d45a736cd16/go.mod h1:2FMWW+8GMoPweT6+pI63m9YE3Lmw4J71hV56Chs1E/U= -github.com/minio/sha256-simd v0.1.1-0.20190913151208-6de447530771 h1:MHkK1uRtFbVqvAgvWxafZe54+5uBxLluGylDiKgdhwo= github.com/minio/sha256-simd v0.1.1-0.20190913151208-6de447530771/go.mod h1:B5e1o+1/KgNmWrSQK08Y6Z1Vb5pwIktudl0J58iy0KM= +github.com/minio/sha256-simd v1.0.1 h1:6kaan5IFmwTNynnKKpDHe6FWHohJOHhCPchzK49dzMM= +github.com/minio/sha256-simd v1.0.1/go.mod h1:Pz6AKMiUdngCLpeTL/RJY1M9rUuPMYujV5xJjtbRSN8= github.com/mr-tron/base58 v1.1.0/go.mod h1:xcD2VGqlgYjBdcBLw+TuYLr8afG+Hj8g2eTVqeSzSU8= -github.com/mr-tron/base58 v1.1.2/go.mod h1:BinMc/sQntlIE1frQmRFPUoPA1Zkr8VRgBdjWI2mNwc= -github.com/mr-tron/base58 v1.1.3 h1:v+sk57XuaCKGXpWtVBX8YJzO7hMGx4Aajh4TQbdEFdc= github.com/mr-tron/base58 v1.1.3/go.mod h1:BinMc/sQntlIE1frQmRFPUoPA1Zkr8VRgBdjWI2mNwc= -github.com/multiformats/go-base32 v0.0.3 h1:tw5+NhuwaOjJCC5Pp82QuXbrmLzWg7uxlMFp8Nq/kkI= +github.com/mr-tron/base58 v1.2.0 h1:T/HDJBh4ZCPbU39/+c3rRvE0uKBQlU27+QI8LJ4t64o= +github.com/mr-tron/base58 v1.2.0/go.mod h1:BinMc/sQntlIE1frQmRFPUoPA1Zkr8VRgBdjWI2mNwc= github.com/multiformats/go-base32 v0.0.3/go.mod h1:pLiuGC8y0QR3Ue4Zug5UzK9LjgbkL8NSQj0zQ5Nz/AA= -github.com/multiformats/go-base36 v0.1.0 h1:JR6TyF7JjGd3m6FbLU2cOxhC0Li8z8dLNGQ89tUg4F4= +github.com/multiformats/go-base32 v0.1.0 h1:pVx9xoSPqEIQG8o+UbAe7DNi51oej1NtK+aGkbLYxPE= +github.com/multiformats/go-base32 v0.1.0/go.mod h1:Kj3tFY6zNr+ABYMqeUNeGvkIC/UYgtWibDcT0rExnbI= github.com/multiformats/go-base36 v0.1.0/go.mod h1:kFGE83c6s80PklsHO9sRn2NCoffoRdUUOENyW/Vv6sM= -github.com/multiformats/go-multibase v0.0.1/go.mod h1:bja2MqRZ3ggyXtZSEDKpl0uO/gviWFaSteVbWT51qgs= -github.com/multiformats/go-multibase v0.0.3 h1:l/B6bJDQjvQ5G52jw4QGSYeOTZoAwIO77RblWplfIqk= +github.com/multiformats/go-base36 v0.2.0 h1:lFsAbNOGeKtuKozrtBsAkSVhv1p9D0/qedU9rQyccr0= +github.com/multiformats/go-base36 v0.2.0/go.mod h1:qvnKE++v+2MWCfePClUEjE78Z7P2a1UV0xHgWc0hkp4= github.com/multiformats/go-multibase v0.0.3/go.mod h1:5+1R4eQrT3PkYZ24C3W2Ue2tPwIdYQD509ZjSb5y9Oc= -github.com/multiformats/go-multihash v0.0.1/go.mod h1:w/5tugSrLEbWqlcgJabL3oHFKTwfvkofsjW2Qa1ct4U= -github.com/multiformats/go-multihash v0.0.10/go.mod h1:YSLudS+Pi8NHE7o6tb3D8vrpKa63epEDmG8nTduyAew= -github.com/multiformats/go-multihash v0.0.13 h1:06x+mk/zj1FoMsgNejLpy6QTvJqlSt/BhLEy87zidlc= +github.com/multiformats/go-multibase v0.2.0 h1:isdYCVLvksgWlMW9OZRYJEa9pZETFivncJHmHnnd87g= +github.com/multiformats/go-multibase v0.2.0/go.mod h1:bFBZX4lKCA/2lyOFSAoKH5SS6oPyjtnzK/XTFDPkNuk= github.com/multiformats/go-multihash v0.0.13/go.mod h1:VdAWLKTwram9oKAatUcLxBNUjdtcVwxObEQBtRfuyjc= -github.com/multiformats/go-varint v0.0.5 h1:XVZwSo04Cs3j/jS0uAEPpT3JY6DzMcVLLoWOSnCxOjg= +github.com/multiformats/go-multihash v0.2.3 h1:7Lyc8XfX/IY2jWb/gI7JP+o7JEq9hOa7BFvVU9RSh+U= +github.com/multiformats/go-multihash v0.2.3/go.mod h1:dXgKXCXjBzdscBLk9JkjINiEsCKRVch90MdaGiKsvSM= github.com/multiformats/go-varint v0.0.5/go.mod h1:3Ls8CIEsrijN6+B7PbrXRPxHRPuXSrVKRY101jdMZYE= +github.com/multiformats/go-varint v0.0.7 h1:sWSGR+f/eu5ABZA2ZpYKBILXTTs9JWpdEM/nEGOHFS8= +github.com/multiformats/go-varint v0.0.7/go.mod h1:r8PUYw/fD/SjBCiKOoDlGF6QawOELpZAu9eioSos/OU= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= -github.com/polydawn/refmt v0.0.0-20190221155625-df39d6c2d992 h1:bzMe+2coZJYHnhGgVlcQKuRy4FSny4ds8dLQjw5P1XE= -github.com/polydawn/refmt v0.0.0-20190221155625-df39d6c2d992/go.mod h1:uIp+gprXxxrWSjjklXD+mN4wed/tMfjMMmN/9+JsA9o= -github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d h1:zE9ykElWQ6/NYmHa3jpm/yHnI4xSofP+UP6SpjHcSeM= -github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= -github.com/smartystreets/goconvey v0.0.0-20190222223459-a17d461953aa h1:E+gaaifzi2xF65PbDmuKI3PhLWY6G5opMLniFq8vmXA= -github.com/smartystreets/goconvey v0.0.0-20190222223459-a17d461953aa/go.mod h1:2RVY1rIf+2J2o/IM9+vPq9RzmHDSseB7FoXiSNIUsoU= +github.com/polydawn/refmt v0.89.0 h1:ADJTApkvkeBZsN0tBTx8QjpD9JkmxbKp0cxfr9qszm4= +github.com/polydawn/refmt v0.89.0/go.mod h1:/zvteZs/GwLtCgZ4BL6CBsk9IKIlexP43ObX9AxTqTw= +github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= +github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= +github.com/smartystreets/assertions v1.2.0 h1:42S6lae5dvLc7BrLu/0ugRtcFVjoJNMC/N3yZFZkDFs= +github.com/smartystreets/assertions v1.2.0/go.mod h1:tcbTF8ujkAEcZ8TElKY+i30BzYlVhC/LOxJk7iOWnoo= +github.com/smartystreets/goconvey v1.7.2 h1:9RBaZCeXEQ3UselpuwUQHltGVXvdwm6cv1hgR6gDIPg= +github.com/smartystreets/goconvey v1.7.2/go.mod h1:Vw0tHAZW6lzCRk3xgdin6fKYcG+G3Pg9vgXWeJpQFMM= github.com/spaolacci/murmur3 v1.1.0 h1:7c1g84S4BPRrfL5Xrdp6fOJ206sU9y293DDHaoy0bLI= github.com/spaolacci/murmur3 v1.1.0/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/warpfork/go-wish v0.0.0-20180510122957-5ad1f5abf436 h1:qOpVTI+BrstcjTZLm2Yz/3sOnqkzj3FQoh0g+E5s3Gc= -github.com/warpfork/go-wish v0.0.0-20180510122957-5ad1f5abf436/go.mod h1:x6AKhvSSexNrVSrViXSHUEbICjmGXhtgABaHIySUSGw= -github.com/whyrusleeping/cbor-gen v0.0.0-20200123233031-1cdf64d27158/go.mod h1:Xj/M2wWU+QdTdRbu/L/1dIZY8/Wb2K9pAhtroQuxJJI= -github.com/whyrusleeping/cbor-gen v0.0.0-20220323183124-98fa8256a799 h1:DOOT2B85S0tHoLGTzV+FakaSSihgRCVwZkjqKQP5L/w= -github.com/whyrusleeping/cbor-gen v0.0.0-20220323183124-98fa8256a799/go.mod h1:fgkXqYy7bV2cFeIEOkVTZS/WjXARfBqSH6Q2qHL33hQ= -golang.org/x/crypto v0.0.0-20190211182817-74369b46fc67/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +github.com/urfave/cli v1.22.10/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= +github.com/vulcanize/go-ipld-cbor v0.1.1-internal-0.0.1 h1:Y9NZaJQ1rBrhNz95W1HcJJbrqs6ZlFbYMwls8q5qLlU= +github.com/vulcanize/go-ipld-cbor v0.1.1-internal-0.0.1/go.mod h1:ikVQT4Jo3JB0Nks0Mr/Pyy7fz25ypxbdEoZHeBrsOo4= +github.com/warpfork/go-wish v0.0.0-20220906213052-39a1cc7a02d0 h1:GDDkbFiaK8jsSDJfjId/PEGEShv6ugrt4kYsC5UIDaQ= +github.com/warpfork/go-wish v0.0.0-20220906213052-39a1cc7a02d0/go.mod h1:x6AKhvSSexNrVSrViXSHUEbICjmGXhtgABaHIySUSGw= +github.com/whyrusleeping/cbor-gen v0.0.0-20230818171029-f91ae536ca25 h1:yVYDLoN2gmB3OdBXFW8e1UwgVbmCvNlnAKhvHPaNARI= +github.com/whyrusleeping/cbor-gen v0.0.0-20230818171029-f91ae536ca25/go.mod h1:fgkXqYy7bV2cFeIEOkVTZS/WjXARfBqSH6Q2qHL33hQ= +go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= +go.uber.org/zap v1.25.0 h1:4Hvk6GtkucQ790dqmj7l1eEnRdKm3k3ZUrUMS2d5+5c= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20190611184440-5c40567a22f8/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.1.0 h1:MDRAIl0xIo9Io2xV565hzXHw3zVseKrJKodhohM5CjU= -golang.org/x/crypto v0.1.0/go.mod h1:RecgLatLF4+eUMCP1PoPZQb+cVrJcOPbHkTkbkB9sbw= +golang.org/x/crypto v0.12.0 h1:tFM/ta59kqch6LlvYnPa0yx5a83cL2nHflFhYKvv9Yk= +golang.org/x/crypto v0.12.0/go.mod h1:NF0Gs7EO5K4qLn+Ylc+fih8BSTeIjAP05siRnAh98yw= +golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ= golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20190219092855-153ac476189d/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.1.0 h1:kunALQeHf1/185U1i0GOB/fy1IPRDDpuoOOqRReG57U= -golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.12.0 h1:CM0HF96J0hcLAwsHPJZjfdNzs0gftsLfgKt57wWHJ0o= +golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= -golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 h1:H2TDz8ibqkAF6YGhCdN3jS9O0/s90v0rJh3X/OLHEUk= +golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0 h1:hjy8E9ON/egN1tAYqKb61G10WtihqetD4sz2H+8nIeA= gopkg.in/yaml.v3 v3.0.0/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +lukechampine.com/blake3 v1.2.1 h1:YuqqRuaqsGV71BV/nm9xlI0MKUv4QC54jQnBChWbGnI= +lukechampine.com/blake3 v1.2.1/go.mod h1:0OFRp7fBtAylGVCO40o87sbupkyIGgbpv1+M1k1LM6k=