From fb412109b825ff3de7ba6755652a9db7c81a82c9 Mon Sep 17 00:00:00 2001 From: Acha Bill <57879913+acha-bill@users.noreply.github.com> Date: Wed, 24 Jan 2024 00:47:44 +0100 Subject: [PATCH 1/2] feat: stamper index assignment rule update (#4552) --- pkg/postage/stamper.go | 24 +++++++++++++----------- pkg/postage/stamper_test.go | 30 +++++++++++++++++++++--------- pkg/postage/stampissuer.go | 8 -------- 3 files changed, 34 insertions(+), 28 deletions(-) diff --git a/pkg/postage/stamper.go b/pkg/postage/stamper.go index 5f32fa8ecd3..e7b8d51b4c5 100644 --- a/pkg/postage/stamper.go +++ b/pkg/postage/stamper.go @@ -11,12 +11,12 @@ import ( "github.com/ethersphere/bee/pkg/crypto" "github.com/ethersphere/bee/pkg/storage" "github.com/ethersphere/bee/pkg/swarm" + "resenje.org/multex" ) var ( // ErrBucketFull is the error when a collision bucket is full. - ErrBucketFull = errors.New("bucket full") - ErrOverwriteImmutableIndex = errors.New("immutable batch old index overwrite due to previous faulty save") + ErrBucketFull = errors.New("bucket full") ) // Stamper can issue stamps from the given address of chunk. @@ -30,31 +30,33 @@ type stamper struct { store storage.Store issuer *StampIssuer signer crypto.Signer + mu *multex.Multex } // NewStamper constructs a Stamper. func NewStamper(store storage.Store, issuer *StampIssuer, signer crypto.Signer) Stamper { - return &stamper{store, issuer, signer} + return &stamper{store, issuer, signer, multex.New()} } -// Stamp takes chunk, see if the chunk can included in the batch and +// Stamp takes chunk, see if the chunk can be included in the batch and // signs it with the owner of the batch of this Stamp issuer. func (st *stamper) Stamp(addr swarm.Address) (*Stamp, error) { + st.mu.Lock(addr.ByteString()) + defer st.mu.Unlock(addr.ByteString()) + item := &StampItem{ BatchID: st.issuer.data.BatchID, chunkAddress: addr, } switch err := st.store.Get(item); { case err == nil: - // The index should be in the past. It could happen that we encountered - // some error after assigning this index and did not save the issuer data. In - // this case we should assign a new index and update it. - if st.issuer.assigned(item.BatchIndex) { + if st.issuer.ImmutableFlag() { break - } else if st.issuer.ImmutableFlag() { - return nil, ErrOverwriteImmutableIndex } - fallthrough + item.BatchTimestamp = unixTime() + if err = st.store.Put(item); err != nil { + return nil, err + } case errors.Is(err, storage.ErrNotFound): item.BatchIndex, item.BatchTimestamp, err = st.issuer.increment(addr) if err != nil { diff --git a/pkg/postage/stamper_test.go b/pkg/postage/stamper_test.go index 1fc3b3d5c34..7fbc845e4de 100644 --- a/pkg/postage/stamper_test.go +++ b/pkg/postage/stamper_test.go @@ -106,11 +106,11 @@ func TestStamperStamping(t *testing.T) { } }) - t.Run("incorrect old index", func(t *testing.T) { + t.Run("reuse index but get new timestamp for mutable batch", func(t *testing.T) { st := newTestStampIssuerMutability(t, 1000, false) chunkAddr := swarm.RandAddress(t) bIdx := postage.ToBucket(st.BucketDepth(), chunkAddr) - index := postage.IndexToBytes(bIdx, 100) + index := postage.IndexToBytes(bIdx, 4) testItem := postage.NewStampItem(). WithBatchID(st.ID()). WithChunkAddress(chunkAddr). @@ -124,25 +124,37 @@ func TestStamperStamping(t *testing.T) { if err := stamp.Valid(chunkAddr, owner, 12, 8, true); err != nil { t.Fatalf("expected no error, got %v", err) } - if bytes.Equal(stamp.Index(), testItem.BatchIndex) { - t.Fatalf("expected index to be different, got %x", stamp.Index()) + if bytes.Equal(stamp.Timestamp(), testItem.BatchTimestamp) { + t.Fatalf("expected timestamp to be different, got %x", stamp.Index()) + } + if !bytes.Equal(stamp.Index(), testItem.BatchIndex) { + t.Fatalf("expected index to be the same, got %x", stamp.Index()) } }) - t.Run("incorrect old index immutable", func(t *testing.T) { + t.Run("reuse same index and timestamp for immutable batch", func(t *testing.T) { st := newTestStampIssuerMutability(t, 1000, true) chunkAddr := swarm.RandAddress(t) bIdx := postage.ToBucket(st.BucketDepth(), chunkAddr) - index := postage.IndexToBytes(bIdx, 100) + index := postage.IndexToBytes(bIdx, 4) testItem := postage.NewStampItem(). WithBatchID(st.ID()). WithChunkAddress(chunkAddr). WithBatchIndex(index) testSt := &testStore{Store: inmemstore.New(), stampItem: testItem} stamper := postage.NewStamper(testSt, st, signer) - _, err := stamper.Stamp(chunkAddr) - if !errors.Is(err, postage.ErrOverwriteImmutableIndex) { - t.Fatalf("got err %v, wanted %v", err, postage.ErrOverwriteImmutableIndex) + stamp, err := stamper.Stamp(chunkAddr) + if err != nil { + t.Fatal(err) + } + if err := stamp.Valid(chunkAddr, owner, 12, 8, true); err != nil { + t.Fatalf("expected no error, got %v", err) + } + if !bytes.Equal(stamp.Timestamp(), testItem.BatchTimestamp) { + t.Fatalf("expected timestamp to be the same, got %x", stamp.Index()) + } + if !bytes.Equal(stamp.Index(), testItem.BatchIndex) { + t.Fatalf("expected index to be the same, got %x", stamp.Index()) } }) diff --git a/pkg/postage/stampissuer.go b/pkg/postage/stampissuer.go index 2606feb36eb..84b5292528a 100644 --- a/pkg/postage/stampissuer.go +++ b/pkg/postage/stampissuer.go @@ -201,14 +201,6 @@ func (si *StampIssuer) increment(addr swarm.Address) (batchIndex []byte, batchTi return indexToBytes(bIdx, bCnt), unixTime(), nil } -// check if this stamp index has already been assigned -func (si *StampIssuer) assigned(stampIdx []byte) bool { - si.bucketMtx.Lock() - defer si.bucketMtx.Unlock() - b, idx := BucketIndexFromBytes(stampIdx) - return idx < si.data.Buckets[b] -} - // Label returns the label of the issuer. func (si *StampIssuer) Label() string { return si.data.Label From c49f24294d2a9647263b089af11efe36868f41ec Mon Sep 17 00:00:00 2001 From: Acha Bill Date: Wed, 31 Jan 2024 01:25:48 +0100 Subject: [PATCH 2/2] fix: review comments --- pkg/postage/stamper.go | 3 --- pkg/postage/stamper_test.go | 46 +++++++++---------------------------- 2 files changed, 11 insertions(+), 38 deletions(-) diff --git a/pkg/postage/stamper.go b/pkg/postage/stamper.go index e7b8d51b4c5..1ee16a8a133 100644 --- a/pkg/postage/stamper.go +++ b/pkg/postage/stamper.go @@ -50,9 +50,6 @@ func (st *stamper) Stamp(addr swarm.Address) (*Stamp, error) { } switch err := st.store.Get(item); { case err == nil: - if st.issuer.ImmutableFlag() { - break - } item.BatchTimestamp = unixTime() if err = st.store.Put(item); err != nil { return nil, err diff --git a/pkg/postage/stamper_test.go b/pkg/postage/stamper_test.go index 7fbc845e4de..520d6da0faf 100644 --- a/pkg/postage/stamper_test.go +++ b/pkg/postage/stamper_test.go @@ -106,7 +106,7 @@ func TestStamperStamping(t *testing.T) { } }) - t.Run("reuse index but get new timestamp for mutable batch", func(t *testing.T) { + t.Run("reuse index but get new timestamp for mutable or immutable batch", func(t *testing.T) { st := newTestStampIssuerMutability(t, 1000, false) chunkAddr := swarm.RandAddress(t) bIdx := postage.ToBucket(st.BucketDepth(), chunkAddr) @@ -121,40 +121,16 @@ func TestStamperStamping(t *testing.T) { if err != nil { t.Fatal(err) } - if err := stamp.Valid(chunkAddr, owner, 12, 8, true); err != nil { - t.Fatalf("expected no error, got %v", err) - } - if bytes.Equal(stamp.Timestamp(), testItem.BatchTimestamp) { - t.Fatalf("expected timestamp to be different, got %x", stamp.Index()) - } - if !bytes.Equal(stamp.Index(), testItem.BatchIndex) { - t.Fatalf("expected index to be the same, got %x", stamp.Index()) - } - }) - - t.Run("reuse same index and timestamp for immutable batch", func(t *testing.T) { - st := newTestStampIssuerMutability(t, 1000, true) - chunkAddr := swarm.RandAddress(t) - bIdx := postage.ToBucket(st.BucketDepth(), chunkAddr) - index := postage.IndexToBytes(bIdx, 4) - testItem := postage.NewStampItem(). - WithBatchID(st.ID()). - WithChunkAddress(chunkAddr). - WithBatchIndex(index) - testSt := &testStore{Store: inmemstore.New(), stampItem: testItem} - stamper := postage.NewStamper(testSt, st, signer) - stamp, err := stamper.Stamp(chunkAddr) - if err != nil { - t.Fatal(err) - } - if err := stamp.Valid(chunkAddr, owner, 12, 8, true); err != nil { - t.Fatalf("expected no error, got %v", err) - } - if !bytes.Equal(stamp.Timestamp(), testItem.BatchTimestamp) { - t.Fatalf("expected timestamp to be the same, got %x", stamp.Index()) - } - if !bytes.Equal(stamp.Index(), testItem.BatchIndex) { - t.Fatalf("expected index to be the same, got %x", stamp.Index()) + for _, mutability := range []bool{true, false} { + if err := stamp.Valid(chunkAddr, owner, 12, 8, mutability); err != nil { + t.Fatalf("expected no error, got %v", err) + } + if bytes.Equal(stamp.Timestamp(), testItem.BatchTimestamp) { + t.Fatalf("expected timestamp to be different, got %x", stamp.Index()) + } + if !bytes.Equal(stamp.Index(), testItem.BatchIndex) { + t.Fatalf("expected index to be the same, got %x", stamp.Index()) + } } })