| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | // Copyright 2018 The go-ethereum Authors | 
					
						
							|  |  |  | // This file is part of the go-ethereum library. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							|  |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package storage | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							| 
									
										
										
										
											2018-11-13 07:41:01 +01:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2018-08-14 16:03:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-26 16:09:32 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/swarm/chunk" | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	hashfunc = MakeHashFunc(DefaultHash) | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // tests that the content address validator correctly checks the data | 
					
						
							| 
									
										
										
										
											2018-10-02 09:32:46 +02:00
										 |  |  | // tests that feed update chunks are passed through content address validator | 
					
						
							| 
									
										
										
										
											2018-09-30 09:43:10 +02:00
										 |  |  | // the test checking the resouce update validator internal correctness is found in storage/feeds/handler_test.go | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | func TestValidator(t *testing.T) { | 
					
						
							|  |  |  | 	// set up localstore | 
					
						
							|  |  |  | 	datadir, err := ioutil.TempDir("", "storage-testvalidator") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer os.RemoveAll(datadir) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	params := NewDefaultLocalStoreParams() | 
					
						
							|  |  |  | 	params.Init(datadir) | 
					
						
							|  |  |  | 	store, err := NewLocalStore(params, nil) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// check puts with no validators, both succeed | 
					
						
							|  |  |  | 	chunks := GenerateRandomChunks(259, 2) | 
					
						
							|  |  |  | 	goodChunk := chunks[0] | 
					
						
							|  |  |  | 	badChunk := chunks[1] | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	copy(badChunk.Data(), goodChunk.Data()) | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	errs := putChunks(store, goodChunk, badChunk) | 
					
						
							|  |  |  | 	if errs[0] != nil { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 		t.Fatalf("expected no error on good content address chunk in spite of no validation, but got: %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	if errs[1] != nil { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 		t.Fatalf("expected no error on bad content address chunk in spite of no validation, but got: %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// add content address validator and check puts | 
					
						
							|  |  |  | 	// bad should fail, good should pass | 
					
						
							|  |  |  | 	store.Validators = append(store.Validators, NewContentAddressValidator(hashfunc)) | 
					
						
							| 
									
										
										
										
											2019-02-26 16:09:32 +01:00
										 |  |  | 	chunks = GenerateRandomChunks(chunk.DefaultSize, 2) | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 	goodChunk = chunks[0] | 
					
						
							|  |  |  | 	badChunk = chunks[1] | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	copy(badChunk.Data(), goodChunk.Data()) | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	errs = putChunks(store, goodChunk, badChunk) | 
					
						
							|  |  |  | 	if errs[0] != nil { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 		t.Fatalf("expected no error on good content address chunk with content address validator only, but got: %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	if errs[1] == nil { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 		t.Fatal("expected error on bad content address chunk with content address validator only, but got nil") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// append a validator that always denies | 
					
						
							|  |  |  | 	// bad should fail, good should pass, | 
					
						
							|  |  |  | 	var negV boolTestValidator | 
					
						
							|  |  |  | 	store.Validators = append(store.Validators, negV) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-26 16:09:32 +01:00
										 |  |  | 	chunks = GenerateRandomChunks(chunk.DefaultSize, 2) | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 	goodChunk = chunks[0] | 
					
						
							|  |  |  | 	badChunk = chunks[1] | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	copy(badChunk.Data(), goodChunk.Data()) | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	errs = putChunks(store, goodChunk, badChunk) | 
					
						
							|  |  |  | 	if errs[0] != nil { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 		t.Fatalf("expected no error on good content address chunk with content address validator only, but got: %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	if errs[1] == nil { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 		t.Fatal("expected error on bad content address chunk with content address validator only, but got nil") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// append a validator that always approves | 
					
						
							|  |  |  | 	// all shall pass | 
					
						
							|  |  |  | 	var posV boolTestValidator = true | 
					
						
							|  |  |  | 	store.Validators = append(store.Validators, posV) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-26 16:09:32 +01:00
										 |  |  | 	chunks = GenerateRandomChunks(chunk.DefaultSize, 2) | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 	goodChunk = chunks[0] | 
					
						
							|  |  |  | 	badChunk = chunks[1] | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	copy(badChunk.Data(), goodChunk.Data()) | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	errs = putChunks(store, goodChunk, badChunk) | 
					
						
							|  |  |  | 	if errs[0] != nil { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 		t.Fatalf("expected no error on good content address chunk with content address validator only, but got: %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 	if errs[1] != nil { | 
					
						
							|  |  |  | 		t.Fatalf("expected no error on bad content address chunk in spite of no validation, but got: %s", err) | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type boolTestValidator bool | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-12 16:22:17 +01:00
										 |  |  | func (self boolTestValidator) Validate(chunk Chunk) bool { | 
					
						
							| 
									
										
										
										
											2018-06-20 14:06:27 +02:00
										 |  |  | 	return bool(self) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // putChunks adds chunks  to localstore | 
					
						
							|  |  |  | // It waits for receive on the stored channel | 
					
						
							|  |  |  | // It logs but does not fail on delivery error | 
					
						
							|  |  |  | func putChunks(store *LocalStore, chunks ...Chunk) []error { | 
					
						
							|  |  |  | 	i := 0 | 
					
						
							|  |  |  | 	f := func(n int64) Chunk { | 
					
						
							|  |  |  | 		chunk := chunks[i] | 
					
						
							|  |  |  | 		i++ | 
					
						
							|  |  |  | 		return chunk | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	_, errs := put(store, len(chunks), f) | 
					
						
							|  |  |  | 	return errs | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func put(store *LocalStore, n int, f func(i int64) Chunk) (hs []Address, errs []error) { | 
					
						
							|  |  |  | 	for i := int64(0); i < int64(n); i++ { | 
					
						
							| 
									
										
										
										
											2019-02-26 16:09:32 +01:00
										 |  |  | 		chunk := f(chunk.DefaultSize) | 
					
						
							| 
									
										
										
										
											2018-09-13 11:42:19 +02:00
										 |  |  | 		err := store.Put(context.TODO(), chunk) | 
					
						
							|  |  |  | 		errs = append(errs, err) | 
					
						
							|  |  |  | 		hs = append(hs, chunk.Address()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return hs, errs | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-11-13 07:41:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // TestGetFrequentlyAccessedChunkWontGetGarbageCollected tests that the most | 
					
						
							|  |  |  | // frequently accessed chunk is not garbage collected from LDBStore, i.e., | 
					
						
							|  |  |  | // from disk when we are at the capacity and garbage collector runs. For that | 
					
						
							|  |  |  | // we start putting random chunks into the DB while continuously accessing the | 
					
						
							|  |  |  | // chunk we care about then check if we can still retrieve it from disk. | 
					
						
							|  |  |  | func TestGetFrequentlyAccessedChunkWontGetGarbageCollected(t *testing.T) { | 
					
						
							|  |  |  | 	ldbCap := defaultGCRatio | 
					
						
							|  |  |  | 	store, cleanup := setupLocalStore(t, ldbCap) | 
					
						
							|  |  |  | 	defer cleanup() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var chunks []Chunk | 
					
						
							|  |  |  | 	for i := 0; i < ldbCap; i++ { | 
					
						
							| 
									
										
										
										
											2019-02-26 16:09:32 +01:00
										 |  |  | 		chunks = append(chunks, GenerateRandomChunk(chunk.DefaultSize)) | 
					
						
							| 
									
										
										
										
											2018-11-13 07:41:01 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mostAccessed := chunks[0].Address() | 
					
						
							|  |  |  | 	for _, chunk := range chunks { | 
					
						
							|  |  |  | 		if err := store.Put(context.Background(), chunk); err != nil { | 
					
						
							|  |  |  | 			t.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if _, err := store.Get(context.Background(), mostAccessed); err != nil { | 
					
						
							|  |  |  | 			t.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Add time for MarkAccessed() to be able to finish in a separate Goroutine | 
					
						
							|  |  |  | 		time.Sleep(1 * time.Millisecond) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	store.DbStore.collectGarbage() | 
					
						
							|  |  |  | 	if _, err := store.DbStore.Get(context.Background(), mostAccessed); err != nil { | 
					
						
							|  |  |  | 		t.Logf("most frequntly accessed chunk not found on disk (key: %v)", mostAccessed) | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func setupLocalStore(t *testing.T, ldbCap int) (ls *LocalStore, cleanup func()) { | 
					
						
							|  |  |  | 	t.Helper() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var err error | 
					
						
							|  |  |  | 	datadir, err := ioutil.TempDir("", "storage") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	params := &LocalStoreParams{ | 
					
						
							|  |  |  | 		StoreParams: NewStoreParams(uint64(ldbCap), uint(ldbCap), nil, nil), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	params.Init(datadir) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	store, err := NewLocalStore(params, nil) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		_ = os.RemoveAll(datadir) | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cleanup = func() { | 
					
						
							|  |  |  | 		store.Close() | 
					
						
							|  |  |  | 		_ = os.RemoveAll(datadir) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return store, cleanup | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-02-07 09:49:19 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestHas(t *testing.T) { | 
					
						
							|  |  |  | 	ldbCap := defaultGCRatio | 
					
						
							|  |  |  | 	store, cleanup := setupLocalStore(t, ldbCap) | 
					
						
							|  |  |  | 	defer cleanup() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	nonStoredAddr := GenerateRandomChunk(128).Address() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	has := store.Has(context.Background(), nonStoredAddr) | 
					
						
							|  |  |  | 	if has { | 
					
						
							|  |  |  | 		t.Fatal("Expected Has() to return false, but returned true!") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	storeChunks := GenerateRandomChunks(128, 3) | 
					
						
							|  |  |  | 	for _, ch := range storeChunks { | 
					
						
							|  |  |  | 		err := store.Put(context.Background(), ch) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatalf("Expected store to store chunk, but it failed: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		has := store.Has(context.Background(), ch.Address()) | 
					
						
							|  |  |  | 		if !has { | 
					
						
							|  |  |  | 			t.Fatal("Expected Has() to return true, but returned false!") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//let's be paranoic and test again that the non-existent chunk returns false | 
					
						
							|  |  |  | 	has = store.Has(context.Background(), nonStoredAddr) | 
					
						
							|  |  |  | 	if has { | 
					
						
							|  |  |  | 		t.Fatal("Expected Has() to return false, but returned true!") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |