swarm/storage/feed: add context handling/cancellation to Swarm Feeds lookup, fix bad hint lookup bug (#19353)
* swarm/storage/feed/lookup: Add context handling/forwarding * swarm/storage/feed/lookup: Add test to catch bad hint * swarm/storage/feed/lookup: Added context cancellation test
This commit is contained in:
parent
1528b791ac
commit
73fc65bb5b
@ -176,21 +176,25 @@ func (h *Handler) Lookup(ctx context.Context, query *Query) (*cacheEntry, error)
|
|||||||
return nil, NewError(ErrInit, "Call Handler.SetStore() before performing lookups")
|
return nil, NewError(ErrInit, "Call Handler.SetStore() before performing lookups")
|
||||||
}
|
}
|
||||||
|
|
||||||
var id ID
|
|
||||||
id.Feed = query.Feed
|
|
||||||
var readCount int
|
var readCount int
|
||||||
|
|
||||||
// Invoke the lookup engine.
|
// Invoke the lookup engine.
|
||||||
// The callback will be called every time the lookup algorithm needs to guess
|
// The callback will be called every time the lookup algorithm needs to guess
|
||||||
requestPtr, err := lookup.Lookup(timeLimit, query.Hint, func(epoch lookup.Epoch, now uint64) (interface{}, error) {
|
requestPtr, err := lookup.Lookup(ctx, timeLimit, query.Hint, func(ctx context.Context, epoch lookup.Epoch, now uint64) (interface{}, error) {
|
||||||
readCount++
|
readCount++
|
||||||
id.Epoch = epoch
|
id := ID{
|
||||||
|
Feed: query.Feed,
|
||||||
|
Epoch: epoch,
|
||||||
|
}
|
||||||
ctx, cancel := context.WithTimeout(ctx, defaultRetrieveTimeout)
|
ctx, cancel := context.WithTimeout(ctx, defaultRetrieveTimeout)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
chunk, err := h.chunkStore.Get(ctx, id.Addr())
|
chunk, err := h.chunkStore.Get(ctx, id.Addr())
|
||||||
if err != nil { // TODO: check for catastrophic errors other than chunk not found
|
if err != nil {
|
||||||
return nil, nil
|
if err == context.DeadlineExceeded { // chunk not found
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
return nil, err //something else happened or context was cancelled.
|
||||||
}
|
}
|
||||||
|
|
||||||
var request Request
|
var request Request
|
||||||
|
@ -20,6 +20,8 @@ so they can be found
|
|||||||
*/
|
*/
|
||||||
package lookup
|
package lookup
|
||||||
|
|
||||||
|
import "context"
|
||||||
|
|
||||||
const maxuint64 = ^uint64(0)
|
const maxuint64 = ^uint64(0)
|
||||||
|
|
||||||
// LowestLevel establishes the frequency resolution of the lookup algorithm as a power of 2.
|
// LowestLevel establishes the frequency resolution of the lookup algorithm as a power of 2.
|
||||||
@ -33,7 +35,7 @@ const HighestLevel = 25 // default is 25 (~1 year)
|
|||||||
const DefaultLevel = HighestLevel
|
const DefaultLevel = HighestLevel
|
||||||
|
|
||||||
//Algorithm is the function signature of a lookup algorithm
|
//Algorithm is the function signature of a lookup algorithm
|
||||||
type Algorithm func(now uint64, hint Epoch, read ReadFunc) (value interface{}, err error)
|
type Algorithm func(ctx context.Context, now uint64, hint Epoch, read ReadFunc) (value interface{}, err error)
|
||||||
|
|
||||||
// Lookup finds the update with the highest timestamp that is smaller or equal than 'now'
|
// Lookup finds the update with the highest timestamp that is smaller or equal than 'now'
|
||||||
// It takes a hint which should be the epoch where the last known update was
|
// It takes a hint which should be the epoch where the last known update was
|
||||||
@ -48,7 +50,7 @@ var Lookup Algorithm = FluzCapacitorAlgorithm
|
|||||||
// It should return <nil> if a value is found, but its timestamp is higher than "now"
|
// It should return <nil> if a value is found, but its timestamp is higher than "now"
|
||||||
// It should only return an error in case the handler wants to stop the
|
// It should only return an error in case the handler wants to stop the
|
||||||
// lookup process entirely.
|
// lookup process entirely.
|
||||||
type ReadFunc func(epoch Epoch, now uint64) (interface{}, error)
|
type ReadFunc func(ctx context.Context, epoch Epoch, now uint64) (interface{}, error)
|
||||||
|
|
||||||
// NoClue is a hint that can be provided when the Lookup caller does not have
|
// NoClue is a hint that can be provided when the Lookup caller does not have
|
||||||
// a clue about where the last update may be
|
// a clue about where the last update may be
|
||||||
@ -128,7 +130,7 @@ var worstHint = Epoch{Time: 0, Level: 63}
|
|||||||
// or the epochs right below. If however, that lookup succeeds, then the update must be
|
// or the epochs right below. If however, that lookup succeeds, then the update must be
|
||||||
// that one or within the epochs right below.
|
// that one or within the epochs right below.
|
||||||
// see the guide for a more graphical representation
|
// see the guide for a more graphical representation
|
||||||
func FluzCapacitorAlgorithm(now uint64, hint Epoch, read ReadFunc) (value interface{}, err error) {
|
func FluzCapacitorAlgorithm(ctx context.Context, now uint64, hint Epoch, read ReadFunc) (value interface{}, err error) {
|
||||||
var lastFound interface{}
|
var lastFound interface{}
|
||||||
var epoch Epoch
|
var epoch Epoch
|
||||||
if hint == NoClue {
|
if hint == NoClue {
|
||||||
@ -139,7 +141,7 @@ func FluzCapacitorAlgorithm(now uint64, hint Epoch, read ReadFunc) (value interf
|
|||||||
|
|
||||||
for {
|
for {
|
||||||
epoch = GetNextEpoch(hint, t)
|
epoch = GetNextEpoch(hint, t)
|
||||||
value, err = read(epoch, now)
|
value, err = read(ctx, epoch, now)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -160,7 +162,7 @@ func FluzCapacitorAlgorithm(now uint64, hint Epoch, read ReadFunc) (value interf
|
|||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
// check it out
|
// check it out
|
||||||
value, err = read(hint, now)
|
value, err = read(ctx, hint, now)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -168,8 +170,9 @@ func FluzCapacitorAlgorithm(now uint64, hint Epoch, read ReadFunc) (value interf
|
|||||||
return value, nil
|
return value, nil
|
||||||
}
|
}
|
||||||
// bad hint.
|
// bad hint.
|
||||||
epoch = hint
|
t = hint.Base()
|
||||||
hint = worstHint
|
hint = worstHint
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
base := epoch.Base()
|
base := epoch.Base()
|
||||||
if base == 0 {
|
if base == 0 {
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
package lookup_test
|
package lookup_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"testing"
|
"testing"
|
||||||
@ -50,7 +51,7 @@ const Year = Day * 365
|
|||||||
const Month = Day * 30
|
const Month = Day * 30
|
||||||
|
|
||||||
func makeReadFunc(store Store, counter *int) lookup.ReadFunc {
|
func makeReadFunc(store Store, counter *int) lookup.ReadFunc {
|
||||||
return func(epoch lookup.Epoch, now uint64) (interface{}, error) {
|
return func(ctx context.Context, epoch lookup.Epoch, now uint64) (interface{}, error) {
|
||||||
*counter++
|
*counter++
|
||||||
data := store[epoch.ID()]
|
data := store[epoch.ID()]
|
||||||
var valueStr string
|
var valueStr string
|
||||||
@ -88,7 +89,7 @@ func TestLookup(t *testing.T) {
|
|||||||
|
|
||||||
// try to get the last value
|
// try to get the last value
|
||||||
|
|
||||||
value, err := lookup.Lookup(now, lookup.NoClue, readFunc)
|
value, err := lookup.Lookup(context.Background(), now, lookup.NoClue, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -102,7 +103,7 @@ func TestLookup(t *testing.T) {
|
|||||||
// reset the read count for the next test
|
// reset the read count for the next test
|
||||||
readCount = 0
|
readCount = 0
|
||||||
// Provide a hint to get a faster lookup. In particular, we give the exact location of the last update
|
// Provide a hint to get a faster lookup. In particular, we give the exact location of the last update
|
||||||
value, err = lookup.Lookup(now, epoch, readFunc)
|
value, err = lookup.Lookup(context.Background(), now, epoch, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -121,7 +122,7 @@ func TestLookup(t *testing.T) {
|
|||||||
|
|
||||||
expectedTime := now - Year*3 + 6*Month
|
expectedTime := now - Year*3 + 6*Month
|
||||||
|
|
||||||
value, err = lookup.Lookup(expectedTime, lookup.NoClue, readFunc)
|
value, err = lookup.Lookup(context.Background(), expectedTime, lookup.NoClue, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -153,7 +154,7 @@ func TestOneUpdateAt0(t *testing.T) {
|
|||||||
}
|
}
|
||||||
update(store, epoch, 0, &data)
|
update(store, epoch, 0, &data)
|
||||||
|
|
||||||
value, err := lookup.Lookup(now, lookup.NoClue, readFunc)
|
value, err := lookup.Lookup(context.Background(), now, lookup.NoClue, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -186,7 +187,7 @@ func TestBadHint(t *testing.T) {
|
|||||||
Time: 1200000000,
|
Time: 1200000000,
|
||||||
}
|
}
|
||||||
|
|
||||||
value, err := lookup.Lookup(now, badHint, readFunc)
|
value, err := lookup.Lookup(context.Background(), now, badHint, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -195,6 +196,106 @@ func TestBadHint(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Tests whether the update is found when the bad hint is exactly below the last update
|
||||||
|
func TestBadHintNextToUpdate(t *testing.T) {
|
||||||
|
store := make(Store)
|
||||||
|
readCount := 0
|
||||||
|
|
||||||
|
readFunc := makeReadFunc(store, &readCount)
|
||||||
|
now := uint64(1533903729)
|
||||||
|
var last *Data
|
||||||
|
|
||||||
|
/* the following loop places updates in the following epochs:
|
||||||
|
Update# Time Base Level
|
||||||
|
0 1200000000 1174405120 25
|
||||||
|
1 1200000001 1191182336 24
|
||||||
|
2 1200000002 1199570944 23
|
||||||
|
3 1200000003 1199570944 22
|
||||||
|
4 1200000004 1199570944 21
|
||||||
|
|
||||||
|
The situation we want to trigger is to give a bad hint exactly
|
||||||
|
in T=1200000005, B=1199570944 and L=20, which is where the next
|
||||||
|
update would have logically been.
|
||||||
|
This affects only when the bad hint's base == previous update's base,
|
||||||
|
in this case 1199570944
|
||||||
|
|
||||||
|
*/
|
||||||
|
var epoch lookup.Epoch
|
||||||
|
for i := uint64(0); i < 5; i++ {
|
||||||
|
data := Data{
|
||||||
|
Payload: i,
|
||||||
|
Time: 0,
|
||||||
|
}
|
||||||
|
last = &data
|
||||||
|
epoch = update(store, epoch, 1200000000+i, &data)
|
||||||
|
}
|
||||||
|
|
||||||
|
// come up with some evil hint:
|
||||||
|
// put it where the next update would have been
|
||||||
|
badHint := lookup.Epoch{
|
||||||
|
Level: 20,
|
||||||
|
Time: 1200000005,
|
||||||
|
}
|
||||||
|
|
||||||
|
value, err := lookup.Lookup(context.Background(), now, badHint, readFunc)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if value != last {
|
||||||
|
t.Fatalf("Expected lookup to return the last written value: %v. Got %v", last, value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestContextCancellation(t *testing.T) {
|
||||||
|
|
||||||
|
readFunc := func(ctx context.Context, epoch lookup.Epoch, now uint64) (interface{}, error) {
|
||||||
|
<-ctx.Done()
|
||||||
|
return nil, ctx.Err()
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
|
||||||
|
errc := make(chan error)
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
_, err := lookup.Lookup(ctx, 1200000000, lookup.NoClue, readFunc)
|
||||||
|
errc <- err
|
||||||
|
}()
|
||||||
|
|
||||||
|
cancel()
|
||||||
|
|
||||||
|
if err := <-errc; err != context.Canceled {
|
||||||
|
t.Fatalf("Expected lookup to return a context Cancelled error, got %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// text context cancellation during hint lookup:
|
||||||
|
ctx, cancel = context.WithCancel(context.Background())
|
||||||
|
errc = make(chan error)
|
||||||
|
someHint := lookup.Epoch{
|
||||||
|
Level: 25,
|
||||||
|
Time: 300,
|
||||||
|
}
|
||||||
|
|
||||||
|
readFunc = func(ctx context.Context, epoch lookup.Epoch, now uint64) (interface{}, error) {
|
||||||
|
if epoch == someHint {
|
||||||
|
go cancel()
|
||||||
|
<-ctx.Done()
|
||||||
|
return nil, ctx.Err()
|
||||||
|
}
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
_, err := lookup.Lookup(ctx, 301, someHint, readFunc)
|
||||||
|
errc <- err
|
||||||
|
}()
|
||||||
|
|
||||||
|
if err := <-errc; err != context.Canceled {
|
||||||
|
t.Fatalf("Expected lookup to return a context Cancelled error, got %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
func TestLookupFail(t *testing.T) {
|
func TestLookupFail(t *testing.T) {
|
||||||
|
|
||||||
store := make(Store)
|
store := make(Store)
|
||||||
@ -206,7 +307,7 @@ func TestLookupFail(t *testing.T) {
|
|||||||
// don't write anything and try to look up.
|
// don't write anything and try to look up.
|
||||||
// we're testing we don't get stuck in a loop
|
// we're testing we don't get stuck in a loop
|
||||||
|
|
||||||
value, err := lookup.Lookup(now, lookup.NoClue, readFunc)
|
value, err := lookup.Lookup(context.Background(), now, lookup.NoClue, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -242,7 +343,7 @@ func TestHighFreqUpdates(t *testing.T) {
|
|||||||
lastData = &data
|
lastData = &data
|
||||||
}
|
}
|
||||||
|
|
||||||
value, err := lookup.Lookup(lastData.Time, lookup.NoClue, readFunc)
|
value, err := lookup.Lookup(context.Background(), lastData.Time, lookup.NoClue, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -255,7 +356,7 @@ func TestHighFreqUpdates(t *testing.T) {
|
|||||||
// reset the read count for the next test
|
// reset the read count for the next test
|
||||||
readCount = 0
|
readCount = 0
|
||||||
// Provide a hint to get a faster lookup. In particular, we give the exact location of the last update
|
// Provide a hint to get a faster lookup. In particular, we give the exact location of the last update
|
||||||
value, err = lookup.Lookup(now, epoch, readFunc)
|
value, err = lookup.Lookup(context.Background(), now, epoch, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -270,7 +371,7 @@ func TestHighFreqUpdates(t *testing.T) {
|
|||||||
|
|
||||||
for i := uint64(0); i <= 994; i++ {
|
for i := uint64(0); i <= 994; i++ {
|
||||||
T := uint64(now - 1000 + i) // update every second for the last 1000 seconds
|
T := uint64(now - 1000 + i) // update every second for the last 1000 seconds
|
||||||
value, err := lookup.Lookup(T, lookup.NoClue, readFunc)
|
value, err := lookup.Lookup(context.Background(), T, lookup.NoClue, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -308,7 +409,7 @@ func TestSparseUpdates(t *testing.T) {
|
|||||||
|
|
||||||
// try to get the last value
|
// try to get the last value
|
||||||
|
|
||||||
value, err := lookup.Lookup(now, lookup.NoClue, readFunc)
|
value, err := lookup.Lookup(context.Background(), now, lookup.NoClue, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -322,7 +423,7 @@ func TestSparseUpdates(t *testing.T) {
|
|||||||
// reset the read count for the next test
|
// reset the read count for the next test
|
||||||
readCount = 0
|
readCount = 0
|
||||||
// Provide a hint to get a faster lookup. In particular, we give the exact location of the last update
|
// Provide a hint to get a faster lookup. In particular, we give the exact location of the last update
|
||||||
value, err = lookup.Lookup(now, epoch, readFunc)
|
value, err = lookup.Lookup(context.Background(), now, epoch, readFunc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user