* swarm/storage/mock: implement listings methods for mem and rpc stores * swarm/storage/mock/rpc: add comments and newTestStore helper function * swarm/storage/mock/mem: add missing comments * swarm/storage/mock: add comments to new types and constants * swarm/storage/mock/db: implement listings for mock/db global store * swarm/storage/mock/test: add comments for MockStoreListings * swarm/storage/mock/explorer: initial implementation * cmd/swarm/global-store: add chunk explorer * cmd/swarm/global-store: add chunk explorer tests * swarm/storage/mock/explorer: add tests * swarm/storage/mock/explorer: add swagger api definition * swarm/storage/mock/explorer: not-zero test values for invalid addr and key * swarm/storage/mock/explorer: test wildcard cors origin * swarm/storage/mock/db: renames based on Fabio's suggestions * swarm/storage/mock/explorer: add more comments to testHandler function * cmd/swarm/global-store: terminate subprocess with Kill in tests
		
			
				
	
	
		
			472 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			472 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2019 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 explorer
 | |
| 
 | |
| import (
 | |
| 	"encoding/binary"
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"io/ioutil"
 | |
| 	"net/http"
 | |
| 	"net/http/httptest"
 | |
| 	"net/url"
 | |
| 	"os"
 | |
| 	"sort"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"github.com/ethereum/go-ethereum/common"
 | |
| 	"github.com/ethereum/go-ethereum/swarm/storage/mock"
 | |
| 	"github.com/ethereum/go-ethereum/swarm/storage/mock/db"
 | |
| 	"github.com/ethereum/go-ethereum/swarm/storage/mock/mem"
 | |
| )
 | |
| 
 | |
| // TestHandler_memGlobalStore runs a set of tests
 | |
| // to validate handler with mem global store.
 | |
| func TestHandler_memGlobalStore(t *testing.T) {
 | |
| 	t.Parallel()
 | |
| 
 | |
| 	globalStore := mem.NewGlobalStore()
 | |
| 
 | |
| 	testHandler(t, globalStore)
 | |
| }
 | |
| 
 | |
| // TestHandler_dbGlobalStore runs a set of tests
 | |
| // to validate handler with database global store.
 | |
| func TestHandler_dbGlobalStore(t *testing.T) {
 | |
| 	t.Parallel()
 | |
| 
 | |
| 	dir, err := ioutil.TempDir("", "swarm-mock-explorer-db-")
 | |
| 	if err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 	defer os.RemoveAll(dir)
 | |
| 
 | |
| 	globalStore, err := db.NewGlobalStore(dir)
 | |
| 	if err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 	defer globalStore.Close()
 | |
| 
 | |
| 	testHandler(t, globalStore)
 | |
| }
 | |
| 
 | |
| // testHandler stores data distributed by node addresses
 | |
| // and validates if this data is correctly retrievable
 | |
| // by using the http.Handler returned by NewHandler function.
 | |
| // This test covers all HTTP routes and various get parameters
 | |
| // on them to check paginated results.
 | |
| func testHandler(t *testing.T, globalStore mock.GlobalStorer) {
 | |
| 	const (
 | |
| 		nodeCount       = 350
 | |
| 		keyCount        = 250
 | |
| 		keysOnNodeCount = 150
 | |
| 	)
 | |
| 
 | |
| 	// keys for every node
 | |
| 	nodeKeys := make(map[string][]string)
 | |
| 
 | |
| 	// a node address that is not present in global store
 | |
| 	invalidAddr := "0x7b8b72938c254cf002c4e1e714d27e022be88d93"
 | |
| 
 | |
| 	// a key that is not present in global store
 | |
| 	invalidKey := "f9824192fb515cfb"
 | |
| 
 | |
| 	for i := 1; i <= nodeCount; i++ {
 | |
| 		b := make([]byte, 8)
 | |
| 		binary.BigEndian.PutUint64(b, uint64(i))
 | |
| 		addr := common.BytesToAddress(b).Hex()
 | |
| 		nodeKeys[addr] = make([]string, 0)
 | |
| 	}
 | |
| 
 | |
| 	for i := 1; i <= keyCount; i++ {
 | |
| 		b := make([]byte, 8)
 | |
| 		binary.BigEndian.PutUint64(b, uint64(i))
 | |
| 
 | |
| 		key := common.Bytes2Hex(b)
 | |
| 
 | |
| 		var c int
 | |
| 		for addr := range nodeKeys {
 | |
| 			nodeKeys[addr] = append(nodeKeys[addr], key)
 | |
| 			c++
 | |
| 			if c >= keysOnNodeCount {
 | |
| 				break
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// sort keys for every node as they are expected to be
 | |
| 	// sorted in HTTP responses
 | |
| 	for _, keys := range nodeKeys {
 | |
| 		sort.Strings(keys)
 | |
| 	}
 | |
| 
 | |
| 	// nodes for every key
 | |
| 	keyNodes := make(map[string][]string)
 | |
| 
 | |
| 	// construct a reverse mapping of nodes for every key
 | |
| 	for addr, keys := range nodeKeys {
 | |
| 		for _, key := range keys {
 | |
| 			keyNodes[key] = append(keyNodes[key], addr)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// sort node addresses with case insensitive sort,
 | |
| 	// as hex letters in node addresses are in mixed caps
 | |
| 	for _, addrs := range keyNodes {
 | |
| 		sortCaseInsensitive(addrs)
 | |
| 	}
 | |
| 
 | |
| 	// find a key that is not stored at the address
 | |
| 	var (
 | |
| 		unmatchedAddr string
 | |
| 		unmatchedKey  string
 | |
| 	)
 | |
| 	for addr, keys := range nodeKeys {
 | |
| 		for key := range keyNodes {
 | |
| 			var found bool
 | |
| 			for _, k := range keys {
 | |
| 				if k == key {
 | |
| 					found = true
 | |
| 					break
 | |
| 				}
 | |
| 			}
 | |
| 			if !found {
 | |
| 				unmatchedAddr = addr
 | |
| 				unmatchedKey = key
 | |
| 			}
 | |
| 			break
 | |
| 		}
 | |
| 		if unmatchedAddr != "" {
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 	// check if unmatched key/address pair is found
 | |
| 	if unmatchedAddr == "" || unmatchedKey == "" {
 | |
| 		t.Fatalf("could not find a key that is not associated with a node")
 | |
| 	}
 | |
| 
 | |
| 	// store the data
 | |
| 	for addr, keys := range nodeKeys {
 | |
| 		for _, key := range keys {
 | |
| 			err := globalStore.Put(common.HexToAddress(addr), common.Hex2Bytes(key), []byte("data"))
 | |
| 			if err != nil {
 | |
| 				t.Fatal(err)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	handler := NewHandler(globalStore, nil)
 | |
| 
 | |
| 	// this subtest confirms that it has uploaded key and that it does not have invalid keys
 | |
| 	t.Run("has key", func(t *testing.T) {
 | |
| 		for addr, keys := range nodeKeys {
 | |
| 			for _, key := range keys {
 | |
| 				testStatusResponse(t, handler, "/api/has-key/"+addr+"/"+key, http.StatusOK)
 | |
| 				testStatusResponse(t, handler, "/api/has-key/"+invalidAddr+"/"+key, http.StatusNotFound)
 | |
| 			}
 | |
| 			testStatusResponse(t, handler, "/api/has-key/"+addr+"/"+invalidKey, http.StatusNotFound)
 | |
| 		}
 | |
| 		testStatusResponse(t, handler, "/api/has-key/"+invalidAddr+"/"+invalidKey, http.StatusNotFound)
 | |
| 		testStatusResponse(t, handler, "/api/has-key/"+unmatchedAddr+"/"+unmatchedKey, http.StatusNotFound)
 | |
| 	})
 | |
| 
 | |
| 	// this subtest confirms that all keys are are listed in correct order with expected pagination
 | |
| 	t.Run("keys", func(t *testing.T) {
 | |
| 		var allKeys []string
 | |
| 		for key := range keyNodes {
 | |
| 			allKeys = append(allKeys, key)
 | |
| 		}
 | |
| 		sort.Strings(allKeys)
 | |
| 
 | |
| 		t.Run("limit 0", testKeys(handler, allKeys, 0, ""))
 | |
| 		t.Run("limit default", testKeys(handler, allKeys, mock.DefaultLimit, ""))
 | |
| 		t.Run("limit 2x default", testKeys(handler, allKeys, 2*mock.DefaultLimit, ""))
 | |
| 		t.Run("limit 0.5x default", testKeys(handler, allKeys, mock.DefaultLimit/2, ""))
 | |
| 		t.Run("limit max", testKeys(handler, allKeys, mock.MaxLimit, ""))
 | |
| 		t.Run("limit 2x max", testKeys(handler, allKeys, 2*mock.MaxLimit, ""))
 | |
| 		t.Run("limit negative", testKeys(handler, allKeys, -10, ""))
 | |
| 	})
 | |
| 
 | |
| 	// this subtest confirms that all keys are are listed for every node in correct order
 | |
| 	// and that for one node different pagination options are correct
 | |
| 	t.Run("node keys", func(t *testing.T) {
 | |
| 		var limitCheckAddr string
 | |
| 
 | |
| 		for addr, keys := range nodeKeys {
 | |
| 			testKeys(handler, keys, 0, addr)(t)
 | |
| 			if limitCheckAddr == "" {
 | |
| 				limitCheckAddr = addr
 | |
| 			}
 | |
| 		}
 | |
| 		testKeys(handler, nil, 0, invalidAddr)(t)
 | |
| 
 | |
| 		limitCheckKeys := nodeKeys[limitCheckAddr]
 | |
| 		t.Run("limit 0", testKeys(handler, limitCheckKeys, 0, limitCheckAddr))
 | |
| 		t.Run("limit default", testKeys(handler, limitCheckKeys, mock.DefaultLimit, limitCheckAddr))
 | |
| 		t.Run("limit 2x default", testKeys(handler, limitCheckKeys, 2*mock.DefaultLimit, limitCheckAddr))
 | |
| 		t.Run("limit 0.5x default", testKeys(handler, limitCheckKeys, mock.DefaultLimit/2, limitCheckAddr))
 | |
| 		t.Run("limit max", testKeys(handler, limitCheckKeys, mock.MaxLimit, limitCheckAddr))
 | |
| 		t.Run("limit 2x max", testKeys(handler, limitCheckKeys, 2*mock.MaxLimit, limitCheckAddr))
 | |
| 		t.Run("limit negative", testKeys(handler, limitCheckKeys, -10, limitCheckAddr))
 | |
| 	})
 | |
| 
 | |
| 	// this subtest confirms that all nodes are are listed in correct order with expected pagination
 | |
| 	t.Run("nodes", func(t *testing.T) {
 | |
| 		var allNodes []string
 | |
| 		for addr := range nodeKeys {
 | |
| 			allNodes = append(allNodes, addr)
 | |
| 		}
 | |
| 		sortCaseInsensitive(allNodes)
 | |
| 
 | |
| 		t.Run("limit 0", testNodes(handler, allNodes, 0, ""))
 | |
| 		t.Run("limit default", testNodes(handler, allNodes, mock.DefaultLimit, ""))
 | |
| 		t.Run("limit 2x default", testNodes(handler, allNodes, 2*mock.DefaultLimit, ""))
 | |
| 		t.Run("limit 0.5x default", testNodes(handler, allNodes, mock.DefaultLimit/2, ""))
 | |
| 		t.Run("limit max", testNodes(handler, allNodes, mock.MaxLimit, ""))
 | |
| 		t.Run("limit 2x max", testNodes(handler, allNodes, 2*mock.MaxLimit, ""))
 | |
| 		t.Run("limit negative", testNodes(handler, allNodes, -10, ""))
 | |
| 	})
 | |
| 
 | |
| 	// this subtest confirms that all nodes are are listed that contain a a particular key in correct order
 | |
| 	// and that for one key different node pagination options are correct
 | |
| 	t.Run("key nodes", func(t *testing.T) {
 | |
| 		var limitCheckKey string
 | |
| 
 | |
| 		for key, addrs := range keyNodes {
 | |
| 			testNodes(handler, addrs, 0, key)(t)
 | |
| 			if limitCheckKey == "" {
 | |
| 				limitCheckKey = key
 | |
| 			}
 | |
| 		}
 | |
| 		testNodes(handler, nil, 0, invalidKey)(t)
 | |
| 
 | |
| 		limitCheckKeys := keyNodes[limitCheckKey]
 | |
| 		t.Run("limit 0", testNodes(handler, limitCheckKeys, 0, limitCheckKey))
 | |
| 		t.Run("limit default", testNodes(handler, limitCheckKeys, mock.DefaultLimit, limitCheckKey))
 | |
| 		t.Run("limit 2x default", testNodes(handler, limitCheckKeys, 2*mock.DefaultLimit, limitCheckKey))
 | |
| 		t.Run("limit 0.5x default", testNodes(handler, limitCheckKeys, mock.DefaultLimit/2, limitCheckKey))
 | |
| 		t.Run("limit max", testNodes(handler, limitCheckKeys, mock.MaxLimit, limitCheckKey))
 | |
| 		t.Run("limit 2x max", testNodes(handler, limitCheckKeys, 2*mock.MaxLimit, limitCheckKey))
 | |
| 		t.Run("limit negative", testNodes(handler, limitCheckKeys, -10, limitCheckKey))
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // testsKeys returns a test function that validates wantKeys against a series of /api/keys
 | |
| // HTTP responses with provided limit and node options.
 | |
| func testKeys(handler http.Handler, wantKeys []string, limit int, node string) func(t *testing.T) {
 | |
| 	return func(t *testing.T) {
 | |
| 		t.Helper()
 | |
| 
 | |
| 		wantLimit := limit
 | |
| 		if wantLimit <= 0 {
 | |
| 			wantLimit = mock.DefaultLimit
 | |
| 		}
 | |
| 		if wantLimit > mock.MaxLimit {
 | |
| 			wantLimit = mock.MaxLimit
 | |
| 		}
 | |
| 		wantKeysLen := len(wantKeys)
 | |
| 		var i int
 | |
| 		var startKey string
 | |
| 		for {
 | |
| 			var wantNext string
 | |
| 			start := i * wantLimit
 | |
| 			end := (i + 1) * wantLimit
 | |
| 			if end < wantKeysLen {
 | |
| 				wantNext = wantKeys[end]
 | |
| 			} else {
 | |
| 				end = wantKeysLen
 | |
| 			}
 | |
| 			testKeysResponse(t, handler, node, startKey, limit, KeysResponse{
 | |
| 				Keys: wantKeys[start:end],
 | |
| 				Next: wantNext,
 | |
| 			})
 | |
| 			if wantNext == "" {
 | |
| 				break
 | |
| 			}
 | |
| 			startKey = wantNext
 | |
| 			i++
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // testNodes returns a test function that validates wantAddrs against a series of /api/nodes
 | |
| // HTTP responses with provided limit and key options.
 | |
| func testNodes(handler http.Handler, wantAddrs []string, limit int, key string) func(t *testing.T) {
 | |
| 	return func(t *testing.T) {
 | |
| 		t.Helper()
 | |
| 
 | |
| 		wantLimit := limit
 | |
| 		if wantLimit <= 0 {
 | |
| 			wantLimit = mock.DefaultLimit
 | |
| 		}
 | |
| 		if wantLimit > mock.MaxLimit {
 | |
| 			wantLimit = mock.MaxLimit
 | |
| 		}
 | |
| 		wantAddrsLen := len(wantAddrs)
 | |
| 		var i int
 | |
| 		var startKey string
 | |
| 		for {
 | |
| 			var wantNext string
 | |
| 			start := i * wantLimit
 | |
| 			end := (i + 1) * wantLimit
 | |
| 			if end < wantAddrsLen {
 | |
| 				wantNext = wantAddrs[end]
 | |
| 			} else {
 | |
| 				end = wantAddrsLen
 | |
| 			}
 | |
| 			testNodesResponse(t, handler, key, startKey, limit, NodesResponse{
 | |
| 				Nodes: wantAddrs[start:end],
 | |
| 				Next:  wantNext,
 | |
| 			})
 | |
| 			if wantNext == "" {
 | |
| 				break
 | |
| 			}
 | |
| 			startKey = wantNext
 | |
| 			i++
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // testStatusResponse validates a response made on url if it matches
 | |
| // the expected StatusResponse.
 | |
| func testStatusResponse(t *testing.T, handler http.Handler, url string, code int) {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	resp := httpGet(t, handler, url)
 | |
| 
 | |
| 	if resp.StatusCode != code {
 | |
| 		t.Errorf("got status code %v, want %v", resp.StatusCode, code)
 | |
| 	}
 | |
| 	if got := resp.Header.Get("Content-Type"); got != jsonContentType {
 | |
| 		t.Errorf("got Content-Type header %q, want %q", got, jsonContentType)
 | |
| 	}
 | |
| 	var r StatusResponse
 | |
| 	if err := json.NewDecoder(resp.Body).Decode(&r); err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 	if r.Code != code {
 | |
| 		t.Errorf("got response code %v, want %v", r.Code, code)
 | |
| 	}
 | |
| 	if r.Message != http.StatusText(code) {
 | |
| 		t.Errorf("got response message %q, want %q", r.Message, http.StatusText(code))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // testKeysResponse validates response returned from handler on /api/keys
 | |
| // with node, start and limit options against KeysResponse.
 | |
| func testKeysResponse(t *testing.T, handler http.Handler, node, start string, limit int, want KeysResponse) {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	u, err := url.Parse("/api/keys")
 | |
| 	if err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 	q := u.Query()
 | |
| 	if node != "" {
 | |
| 		q.Set("node", node)
 | |
| 	}
 | |
| 	if start != "" {
 | |
| 		q.Set("start", start)
 | |
| 	}
 | |
| 	if limit != 0 {
 | |
| 		q.Set("limit", strconv.Itoa(limit))
 | |
| 	}
 | |
| 	u.RawQuery = q.Encode()
 | |
| 
 | |
| 	resp := httpGet(t, handler, u.String())
 | |
| 
 | |
| 	if resp.StatusCode != http.StatusOK {
 | |
| 		t.Errorf("got status code %v, want %v", resp.StatusCode, http.StatusOK)
 | |
| 	}
 | |
| 	if got := resp.Header.Get("Content-Type"); got != jsonContentType {
 | |
| 		t.Errorf("got Content-Type header %q, want %q", got, jsonContentType)
 | |
| 	}
 | |
| 	var r KeysResponse
 | |
| 	if err := json.NewDecoder(resp.Body).Decode(&r); err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 	if fmt.Sprint(r.Keys) != fmt.Sprint(want.Keys) {
 | |
| 		t.Errorf("got keys %v, want %v", r.Keys, want.Keys)
 | |
| 	}
 | |
| 	if r.Next != want.Next {
 | |
| 		t.Errorf("got next %s, want %s", r.Next, want.Next)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // testNodesResponse validates response returned from handler on /api/nodes
 | |
| // with key, start and limit options against NodesResponse.
 | |
| func testNodesResponse(t *testing.T, handler http.Handler, key, start string, limit int, want NodesResponse) {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	u, err := url.Parse("/api/nodes")
 | |
| 	if err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 	q := u.Query()
 | |
| 	if key != "" {
 | |
| 		q.Set("key", key)
 | |
| 	}
 | |
| 	if start != "" {
 | |
| 		q.Set("start", start)
 | |
| 	}
 | |
| 	if limit != 0 {
 | |
| 		q.Set("limit", strconv.Itoa(limit))
 | |
| 	}
 | |
| 	u.RawQuery = q.Encode()
 | |
| 
 | |
| 	resp := httpGet(t, handler, u.String())
 | |
| 
 | |
| 	if resp.StatusCode != http.StatusOK {
 | |
| 		t.Errorf("got status code %v, want %v", resp.StatusCode, http.StatusOK)
 | |
| 	}
 | |
| 	if got := resp.Header.Get("Content-Type"); got != jsonContentType {
 | |
| 		t.Errorf("got Content-Type header %q, want %q", got, jsonContentType)
 | |
| 	}
 | |
| 	var r NodesResponse
 | |
| 	if err := json.NewDecoder(resp.Body).Decode(&r); err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 	if fmt.Sprint(r.Nodes) != fmt.Sprint(want.Nodes) {
 | |
| 		t.Errorf("got nodes %v, want %v", r.Nodes, want.Nodes)
 | |
| 	}
 | |
| 	if r.Next != want.Next {
 | |
| 		t.Errorf("got next %s, want %s", r.Next, want.Next)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // httpGet uses httptest recorder to provide a response on handler's url.
 | |
| func httpGet(t *testing.T, handler http.Handler, url string) (r *http.Response) {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	req, err := http.NewRequest(http.MethodGet, url, nil)
 | |
| 	if err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 	w := httptest.NewRecorder()
 | |
| 	handler.ServeHTTP(w, req)
 | |
| 	return w.Result()
 | |
| }
 | |
| 
 | |
| // sortCaseInsensitive performs a case insensitive sort on a string slice.
 | |
| func sortCaseInsensitive(s []string) {
 | |
| 	sort.Slice(s, func(i, j int) bool {
 | |
| 		return strings.ToLower(s[i]) < strings.ToLower(s[j])
 | |
| 	})
 | |
| }
 |