Skip to content

Commit d13d712

Browse files
committed
feat(pinning): pinning existing CID with different name updates pin
1 parent a391d02 commit d13d712

File tree

4 files changed

+91
-17
lines changed

4 files changed

+91
-17
lines changed

CHANGELOG.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ The following emojis are used to highlight certain changes:
1616

1717
### Added
1818

19-
* 🛠 `pinning/pinner`: you can now give a custom name when pinning a CID. To reflect this, the `Pinner` has been adjusted.
19+
* 🛠 `pinning/pinner`: you can now give a custom name when pinning a CID. To reflect this, the `Pinner` has been adjusted. Note that calling `Pin` for the same CID with a different name will replace its current name by the newly given name.
2020

2121
### Changed
2222

pinning/pinner/dspinner/pin.go

+25-2
Original file line numberDiff line numberDiff line change
@@ -197,8 +197,15 @@ func (p *pinner) doPinRecursive(ctx context.Context, c cid.Cid, fetch bool, name
197197
if err != nil {
198198
return err
199199
}
200+
// Do not return immediately! Just remove the recursive pins for the current CID.
201+
// This allows the process to continue and the pin to be re-added with a new name.
202+
//
203+
// TODO: remove this to support multiple pins per CID
200204
if found {
201-
return nil
205+
_, err = p.removePinsForCid(ctx, c, ipfspinner.Recursive)
206+
if err != nil {
207+
return err
208+
}
202209
}
203210

204211
dirtyBefore := p.dirty
@@ -222,7 +229,7 @@ func (p *pinner) doPinRecursive(ctx context.Context, c cid.Cid, fetch bool, name
222229

223230
// Only look again if something has changed.
224231
if p.dirty != dirtyBefore {
225-
found, err = p.cidRIndex.HasAny(ctx, cidKey)
232+
found, err := p.cidRIndex.HasAny(ctx, cidKey)
226233
if err != nil {
227234
return err
228235
}
@@ -264,6 +271,22 @@ func (p *pinner) doPinDirect(ctx context.Context, c cid.Cid, name string) error
264271
return fmt.Errorf("%s already pinned recursively", c.String())
265272
}
266273

274+
// Remove existing direct pins for this CID. This ensures that the pin will be
275+
// re-saved with the new name and that there aren't clashing pins for the same
276+
// CID.
277+
//
278+
// TODO: remove this to support multiple pins per CID.
279+
found, err = p.cidDIndex.HasAny(ctx, cidKey)
280+
if err != nil {
281+
return err
282+
}
283+
if found {
284+
_, err = p.removePinsForCid(ctx, c, ipfspinner.Direct)
285+
if err != nil {
286+
return err
287+
}
288+
}
289+
267290
_, err = p.addPin(ctx, c, ipfspinner.Direct, name)
268291
if err != nil {
269292
return err

pinning/pinner/dspinner/pin_test.go

+63-13
Original file line numberDiff line numberDiff line change
@@ -99,6 +99,16 @@ func assertUnpinned(t *testing.T, p ipfspin.Pinner, c cid.Cid, failmsg string) {
9999
}
100100
}
101101

102+
func allPins(t *testing.T, ch <-chan ipfspin.StreamedPin) (pins []ipfspin.Pinned) {
103+
for val := range ch {
104+
if val.Err != nil {
105+
t.Fatal(val.Err)
106+
}
107+
pins = append(pins, val.Pin)
108+
}
109+
return pins
110+
}
111+
102112
func TestPinnerBasic(t *testing.T) {
103113
ctx, cancel := context.WithCancel(context.Background())
104114
defer cancel()
@@ -201,17 +211,7 @@ func TestPinnerBasic(t *testing.T) {
201211
dk := d.Cid()
202212
assertPinned(t, p, dk, "pinned node not found.")
203213

204-
allPins := func(ch <-chan ipfspin.StreamedPin) (pins []ipfspin.Pinned) {
205-
for val := range ch {
206-
if val.Err != nil {
207-
t.Fatal(val.Err)
208-
}
209-
pins = append(pins, val.Pin)
210-
}
211-
return pins
212-
}
213-
214-
pins := allPins(p.RecursiveKeys(ctx, true))
214+
pins := allPins(t, p.RecursiveKeys(ctx, true))
215215
if len(pins) != 2 {
216216
t.Error("expected 2 recursive pins")
217217
}
@@ -256,15 +256,15 @@ func TestPinnerBasic(t *testing.T) {
256256
}
257257
}
258258

259-
pins = allPins(p.DirectKeys(ctx, false))
259+
pins = allPins(t, p.DirectKeys(ctx, false))
260260
if len(pins) != 1 {
261261
t.Error("expected 1 direct pin")
262262
}
263263
if pins[0].Key != ak {
264264
t.Error("wrong direct pin")
265265
}
266266

267-
pins = allPins(p.InternalPins(ctx, false))
267+
pins = allPins(t, p.InternalPins(ctx, false))
268268
if len(pins) != 0 {
269269
t.Error("should not have internal keys")
270270
}
@@ -385,6 +385,56 @@ func TestAddLoadPin(t *testing.T) {
385385
}
386386
}
387387

388+
func TestPinAddOverwriteName(t *testing.T) {
389+
makeTest := func(recursive bool) func(t *testing.T) {
390+
return func(t *testing.T) {
391+
ctx, cancel := context.WithCancel(context.Background())
392+
defer cancel()
393+
dstore := dssync.MutexWrap(ds.NewMapDatastore())
394+
bstore := blockstore.NewBlockstore(dstore)
395+
bserv := bs.New(bstore, offline.Exchange(bstore))
396+
397+
dserv := mdag.NewDAGService(bserv)
398+
399+
p, err := New(ctx, dstore, dserv)
400+
require.NoError(t, err)
401+
402+
a, aCid := randNode()
403+
err = dserv.Add(ctx, a)
404+
require.NoError(t, err)
405+
406+
var (
407+
getPins func(ctx context.Context, detailed bool) <-chan ipfspin.StreamedPin
408+
mode ipfspin.Mode
409+
)
410+
411+
if recursive {
412+
getPins = p.RecursiveKeys
413+
mode = ipfspin.Recursive
414+
} else {
415+
getPins = p.DirectKeys
416+
mode = ipfspin.Direct
417+
}
418+
419+
for _, name := range []string{"", "pin label", "yet another pin label"} {
420+
err = p.Pin(ctx, a, recursive, name)
421+
require.NoError(t, err)
422+
423+
err = p.Flush(ctx)
424+
require.NoError(t, err)
425+
pins := allPins(t, getPins(ctx, true))
426+
require.Len(t, pins, 1)
427+
require.Equal(t, aCid, pins[0].Key)
428+
require.Equal(t, mode, pins[0].Mode)
429+
require.Equal(t, name, pins[0].Name)
430+
}
431+
}
432+
}
433+
434+
t.Run("Direct", makeTest(false))
435+
t.Run("Recursive", makeTest(true))
436+
}
437+
388438
func TestIsPinnedLookup(t *testing.T) {
389439
// Test that lookups work in pins which share
390440
// the same branches. For that construct this tree:

pinning/pinner/pin.go

+2-1
Original file line numberDiff line numberDiff line change
@@ -94,7 +94,8 @@ type Pinner interface {
9494

9595
// Pin the given node, optionally recursively.
9696
// Pin will make sure that the given node and its children if recursive is set
97-
// are stored locally.
97+
// are stored locally. Pinning with a different name for an already existing
98+
// pin must replace the existing name.
9899
Pin(ctx context.Context, node ipld.Node, recursive bool, name string) error
99100

100101
// Unpin the given cid. If recursive is true, removes either a recursive or

0 commit comments

Comments
 (0)