| 
									
										
										
										
											2019-02-07 18:40:26 +01: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 localstore | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	"testing" | 
					
						
							|  |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/swarm/chunk" | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TestModeGetRequest validates ModeGetRequest index values on the provided DB. | 
					
						
							|  |  |  | func TestModeGetRequest(t *testing.T) { | 
					
						
							|  |  |  | 	db, cleanupFunc := newTestDB(t, nil) | 
					
						
							|  |  |  | 	defer cleanupFunc() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	uploadTimestamp := time.Now().UTC().UnixNano() | 
					
						
							|  |  |  | 	defer setNow(func() (t int64) { | 
					
						
							|  |  |  | 		return uploadTimestamp | 
					
						
							|  |  |  | 	})() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	ch := generateTestRandomChunk() | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	_, err := db.Put(context.Background(), chunk.ModePutUpload, ch) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// set update gc test hook to signal when | 
					
						
							|  |  |  | 	// update gc goroutine is done by sending to | 
					
						
							|  |  |  | 	// testHookUpdateGCChan channel, which is | 
					
						
							|  |  |  | 	// used to wait for garbage colletion index | 
					
						
							|  |  |  | 	// changes | 
					
						
							|  |  |  | 	testHookUpdateGCChan := make(chan struct{}) | 
					
						
							|  |  |  | 	defer setTestHookUpdateGC(func() { | 
					
						
							|  |  |  | 		testHookUpdateGCChan <- struct{}{} | 
					
						
							|  |  |  | 	})() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t.Run("get unsynced", func(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		got, err := db.Get(context.Background(), chunk.ModeGetRequest, ch.Address()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// wait for update gc goroutine to be done | 
					
						
							|  |  |  | 		<-testHookUpdateGCChan | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		if !bytes.Equal(got.Address(), ch.Address()) { | 
					
						
							|  |  |  | 			t.Errorf("got chunk address %x, want %x", got.Address(), ch.Address()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		if !bytes.Equal(got.Data(), ch.Data()) { | 
					
						
							|  |  |  | 			t.Errorf("got chunk data %x, want %x", got.Data(), ch.Data()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		t.Run("retrieve indexes", newRetrieveIndexesTestWithAccess(db, ch, uploadTimestamp, 0)) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		t.Run("gc index count", newItemsCountTest(db.gcIndex, 0)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t.Run("gc size", newIndexGCSizeTest(db)) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// set chunk to synced state | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	err = db.Set(context.Background(), chunk.ModeSetSync, ch.Address()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t.Run("first get", func(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		got, err := db.Get(context.Background(), chunk.ModeGetRequest, ch.Address()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// wait for update gc goroutine to be done | 
					
						
							|  |  |  | 		<-testHookUpdateGCChan | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		if !bytes.Equal(got.Address(), ch.Address()) { | 
					
						
							|  |  |  | 			t.Errorf("got chunk address %x, want %x", got.Address(), ch.Address()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		if !bytes.Equal(got.Data(), ch.Data()) { | 
					
						
							|  |  |  | 			t.Errorf("got chunk data %x, want %x", got.Data(), ch.Data()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		t.Run("retrieve indexes", newRetrieveIndexesTestWithAccess(db, ch, uploadTimestamp, uploadTimestamp)) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		t.Run("gc index", newGCIndexTest(db, ch, uploadTimestamp, uploadTimestamp, 1)) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		t.Run("gc index count", newItemsCountTest(db.gcIndex, 1)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t.Run("gc size", newIndexGCSizeTest(db)) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t.Run("second get", func(t *testing.T) { | 
					
						
							|  |  |  | 		accessTimestamp := time.Now().UTC().UnixNano() | 
					
						
							|  |  |  | 		defer setNow(func() (t int64) { | 
					
						
							|  |  |  | 			return accessTimestamp | 
					
						
							|  |  |  | 		})() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		got, err := db.Get(context.Background(), chunk.ModeGetRequest, ch.Address()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// wait for update gc goroutine to be done | 
					
						
							|  |  |  | 		<-testHookUpdateGCChan | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		if !bytes.Equal(got.Address(), ch.Address()) { | 
					
						
							|  |  |  | 			t.Errorf("got chunk address %x, want %x", got.Address(), ch.Address()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		if !bytes.Equal(got.Data(), ch.Data()) { | 
					
						
							|  |  |  | 			t.Errorf("got chunk data %x, want %x", got.Data(), ch.Data()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		t.Run("retrieve indexes", newRetrieveIndexesTestWithAccess(db, ch, uploadTimestamp, accessTimestamp)) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 		t.Run("gc index", newGCIndexTest(db, ch, uploadTimestamp, accessTimestamp, 1)) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		t.Run("gc index count", newItemsCountTest(db.gcIndex, 1)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t.Run("gc size", newIndexGCSizeTest(db)) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TestModeGetSync validates ModeGetSync index values on the provided DB. | 
					
						
							|  |  |  | func TestModeGetSync(t *testing.T) { | 
					
						
							|  |  |  | 	db, cleanupFunc := newTestDB(t, nil) | 
					
						
							|  |  |  | 	defer cleanupFunc() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	uploadTimestamp := time.Now().UTC().UnixNano() | 
					
						
							|  |  |  | 	defer setNow(func() (t int64) { | 
					
						
							|  |  |  | 		return uploadTimestamp | 
					
						
							|  |  |  | 	})() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	ch := generateTestRandomChunk() | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	_, err := db.Put(context.Background(), chunk.ModePutUpload, ch) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	got, err := db.Get(context.Background(), chunk.ModeGetSync, ch.Address()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	if !bytes.Equal(got.Address(), ch.Address()) { | 
					
						
							|  |  |  | 		t.Errorf("got chunk address %x, want %x", got.Address(), ch.Address()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	if !bytes.Equal(got.Data(), ch.Data()) { | 
					
						
							|  |  |  | 		t.Errorf("got chunk data %x, want %x", got.Data(), ch.Data()) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	t.Run("retrieve indexes", newRetrieveIndexesTestWithAccess(db, ch, uploadTimestamp, 0)) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	t.Run("gc index count", newItemsCountTest(db.gcIndex, 0)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t.Run("gc size", newIndexGCSizeTest(db)) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // setTestHookUpdateGC sets testHookUpdateGC and | 
					
						
							|  |  |  | // returns a function that will reset it to the | 
					
						
							|  |  |  | // value before the change. | 
					
						
							|  |  |  | func setTestHookUpdateGC(h func()) (reset func()) { | 
					
						
							|  |  |  | 	current := testHookUpdateGC | 
					
						
							|  |  |  | 	reset = func() { testHookUpdateGC = current } | 
					
						
							|  |  |  | 	testHookUpdateGC = h | 
					
						
							|  |  |  | 	return reset | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TestSetTestHookUpdateGC tests if setTestHookUpdateGC changes | 
					
						
							|  |  |  | // testHookUpdateGC function correctly and if its reset function | 
					
						
							|  |  |  | // resets the original function. | 
					
						
							|  |  |  | func TestSetTestHookUpdateGC(t *testing.T) { | 
					
						
							|  |  |  | 	// Set the current function after the test finishes. | 
					
						
							|  |  |  | 	defer func(h func()) { testHookUpdateGC = h }(testHookUpdateGC) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// expected value for the unchanged function | 
					
						
							|  |  |  | 	original := 1 | 
					
						
							|  |  |  | 	// expected value for the changed function | 
					
						
							|  |  |  | 	changed := 2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// this variable will be set with two different functions | 
					
						
							|  |  |  | 	var got int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// define the original (unchanged) functions | 
					
						
							|  |  |  | 	testHookUpdateGC = func() { | 
					
						
							|  |  |  | 		got = original | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// set got variable | 
					
						
							|  |  |  | 	testHookUpdateGC() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// test if got variable is set correctly | 
					
						
							|  |  |  | 	if got != original { | 
					
						
							|  |  |  | 		t.Errorf("got hook value %v, want %v", got, original) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// set the new function | 
					
						
							|  |  |  | 	reset := setTestHookUpdateGC(func() { | 
					
						
							|  |  |  | 		got = changed | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// set got variable | 
					
						
							|  |  |  | 	testHookUpdateGC() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// test if got variable is set correctly to changed value | 
					
						
							|  |  |  | 	if got != changed { | 
					
						
							|  |  |  | 		t.Errorf("got hook value %v, want %v", got, changed) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// set the function to the original one | 
					
						
							|  |  |  | 	reset() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// set got variable | 
					
						
							|  |  |  | 	testHookUpdateGC() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// test if got variable is set correctly to original value | 
					
						
							|  |  |  | 	if got != original { | 
					
						
							|  |  |  | 		t.Errorf("got hook value %v, want %v", got, original) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |