github.com/muhammedhassanm/blockchain@v0.0.0-20200120143007-697261defd4d/go-ethereum-master/swarm/api/http/server_test.go (about)

     1  // Copyright 2017 The go-ethereum Authors
     2  // This file is part of the go-ethereum library.
     3  //
     4  // The go-ethereum library is free software: you can redistribute it and/or modify
     5  // it under the terms of the GNU Lesser General Public License as published by
     6  // the Free Software Foundation, either version 3 of the License, or
     7  // (at your option) any later version.
     8  //
     9  // The go-ethereum library is distributed in the hope that it will be useful,
    10  // but WITHOUT ANY WARRANTY; without even the implied warranty of
    11  // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
    12  // GNU Lesser General Public License for more details.
    13  //
    14  // You should have received a copy of the GNU Lesser General Public License
    15  // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
    16  
    17  package http
    18  
    19  import (
    20  	"bytes"
    21  	"context"
    22  	"crypto/rand"
    23  	"encoding/json"
    24  	"errors"
    25  	"flag"
    26  	"fmt"
    27  	"io/ioutil"
    28  	"net/http"
    29  	"os"
    30  	"strings"
    31  	"testing"
    32  
    33  	"github.com/ethereum/go-ethereum/common"
    34  	"github.com/ethereum/go-ethereum/common/hexutil"
    35  	"github.com/ethereum/go-ethereum/log"
    36  	"github.com/ethereum/go-ethereum/swarm/api"
    37  	swarm "github.com/ethereum/go-ethereum/swarm/api/client"
    38  	"github.com/ethereum/go-ethereum/swarm/multihash"
    39  	"github.com/ethereum/go-ethereum/swarm/storage"
    40  	"github.com/ethereum/go-ethereum/swarm/testutil"
    41  )
    42  
    43  func init() {
    44  	loglevel := flag.Int("loglevel", 2, "loglevel")
    45  	flag.Parse()
    46  	log.Root().SetHandler(log.CallerFileHandler(log.LvlFilterHandler(log.Lvl(*loglevel), log.StreamHandler(os.Stderr, log.TerminalFormat(true)))))
    47  }
    48  
    49  func TestResourcePostMode(t *testing.T) {
    50  	path := ""
    51  	errstr := "resourcePostMode for '%s' should be raw %v frequency %d, was raw %v, frequency %d"
    52  	r, f, err := resourcePostMode(path)
    53  	if err != nil {
    54  		t.Fatal(err)
    55  	} else if r || f != 0 {
    56  		t.Fatalf(errstr, path, false, 0, r, f)
    57  	}
    58  
    59  	path = "raw"
    60  	r, f, err = resourcePostMode(path)
    61  	if err != nil {
    62  		t.Fatal(err)
    63  	} else if !r || f != 0 {
    64  		t.Fatalf(errstr, path, true, 0, r, f)
    65  	}
    66  
    67  	path = "13"
    68  	r, f, err = resourcePostMode(path)
    69  	if err != nil {
    70  		t.Fatal(err)
    71  	} else if r || f == 0 {
    72  		t.Fatalf(errstr, path, false, 13, r, f)
    73  	}
    74  
    75  	path = "raw/13"
    76  	r, f, err = resourcePostMode(path)
    77  	if err != nil {
    78  		t.Fatal(err)
    79  	} else if !r || f == 0 {
    80  		t.Fatalf(errstr, path, true, 13, r, f)
    81  	}
    82  
    83  	path = "foo/13"
    84  	r, f, err = resourcePostMode(path)
    85  	if err == nil {
    86  		t.Fatal("resourcePostMode for 'foo/13' should fail, returned error nil")
    87  	}
    88  }
    89  
    90  func serverFunc(api *api.API) testutil.TestServer {
    91  	return NewServer(api)
    92  }
    93  
    94  // test the transparent resolving of multihash resource types with bzz:// scheme
    95  //
    96  // first upload data, and store the multihash to the resulting manifest in a resource update
    97  // retrieving the update with the multihash should return the manifest pointing directly to the data
    98  // and raw retrieve of that hash should return the data
    99  func TestBzzResourceMultihash(t *testing.T) {
   100  
   101  	srv := testutil.NewTestSwarmServer(t, serverFunc)
   102  	defer srv.Close()
   103  
   104  	// add the data our multihash aliased manifest will point to
   105  	databytes := "bar"
   106  	url := fmt.Sprintf("%s/bzz:/", srv.URL)
   107  	resp, err := http.Post(url, "text/plain", bytes.NewReader([]byte(databytes)))
   108  	if err != nil {
   109  		t.Fatal(err)
   110  	}
   111  	defer resp.Body.Close()
   112  	if resp.StatusCode != http.StatusOK {
   113  		t.Fatalf("err %s", resp.Status)
   114  	}
   115  	b, err := ioutil.ReadAll(resp.Body)
   116  
   117  	if err != nil {
   118  		t.Fatal(err)
   119  	}
   120  	s := common.FromHex(string(b))
   121  	mh := multihash.ToMultihash(s)
   122  
   123  	mhHex := hexutil.Encode(mh)
   124  	log.Info("added data", "manifest", string(b), "data", common.ToHex(mh))
   125  
   126  	// our mutable resource "name"
   127  	keybytes := "foo.eth"
   128  
   129  	// create the multihash update
   130  	url = fmt.Sprintf("%s/bzz-resource:/%s/13", srv.URL, keybytes)
   131  	resp, err = http.Post(url, "application/octet-stream", bytes.NewReader([]byte(mhHex)))
   132  	if err != nil {
   133  		t.Fatal(err)
   134  	}
   135  	defer resp.Body.Close()
   136  	if resp.StatusCode != http.StatusOK {
   137  		t.Fatalf("err %s", resp.Status)
   138  	}
   139  	b, err = ioutil.ReadAll(resp.Body)
   140  	if err != nil {
   141  		t.Fatal(err)
   142  	}
   143  	rsrcResp := &storage.Address{}
   144  	err = json.Unmarshal(b, rsrcResp)
   145  	if err != nil {
   146  		t.Fatalf("data %s could not be unmarshaled: %v", b, err)
   147  	}
   148  
   149  	correctManifestAddrHex := "d689648fb9e00ddc7ebcf474112d5881c5bf7dbc6e394681b1d224b11b59b5e0"
   150  	if rsrcResp.Hex() != correctManifestAddrHex {
   151  		t.Fatalf("Response resource key mismatch, expected '%s', got '%s'", correctManifestAddrHex, rsrcResp)
   152  	}
   153  
   154  	// get bzz manifest transparent resource resolve
   155  	url = fmt.Sprintf("%s/bzz:/%s", srv.URL, rsrcResp)
   156  	resp, err = http.Get(url)
   157  	if err != nil {
   158  		t.Fatal(err)
   159  	}
   160  	defer resp.Body.Close()
   161  	if resp.StatusCode != http.StatusOK {
   162  		t.Fatalf("err %s", resp.Status)
   163  	}
   164  	b, err = ioutil.ReadAll(resp.Body)
   165  	if err != nil {
   166  		t.Fatal(err)
   167  	}
   168  	if !bytes.Equal(b, []byte(databytes)) {
   169  		t.Fatalf("retrieved data mismatch, expected %x, got %x", databytes, b)
   170  	}
   171  }
   172  
   173  // Test resource updates using the raw update methods
   174  func TestBzzResource(t *testing.T) {
   175  	srv := testutil.NewTestSwarmServer(t, serverFunc)
   176  	defer srv.Close()
   177  
   178  	// our mutable resource "name"
   179  	keybytes := "foo.eth"
   180  
   181  	// data of update 1
   182  	databytes := make([]byte, 666)
   183  	_, err := rand.Read(databytes)
   184  	if err != nil {
   185  		t.Fatal(err)
   186  	}
   187  
   188  	// creates resource and sets update 1
   189  	url := fmt.Sprintf("%s/bzz-resource:/%s/raw/13", srv.URL, []byte(keybytes))
   190  	resp, err := http.Post(url, "application/octet-stream", bytes.NewReader(databytes))
   191  	if err != nil {
   192  		t.Fatal(err)
   193  	}
   194  	defer resp.Body.Close()
   195  	if resp.StatusCode != http.StatusOK {
   196  		t.Fatalf("err %s", resp.Status)
   197  	}
   198  	b, err := ioutil.ReadAll(resp.Body)
   199  	if err != nil {
   200  		t.Fatal(err)
   201  	}
   202  	rsrcResp := &storage.Address{}
   203  	err = json.Unmarshal(b, rsrcResp)
   204  	if err != nil {
   205  		t.Fatalf("data %s could not be unmarshaled: %v", b, err)
   206  	}
   207  
   208  	correctManifestAddrHex := "d689648fb9e00ddc7ebcf474112d5881c5bf7dbc6e394681b1d224b11b59b5e0"
   209  	if rsrcResp.Hex() != correctManifestAddrHex {
   210  		t.Fatalf("Response resource key mismatch, expected '%s', got '%s'", correctManifestAddrHex, rsrcResp.Hex())
   211  	}
   212  
   213  	// get the manifest
   214  	url = fmt.Sprintf("%s/bzz-raw:/%s", srv.URL, rsrcResp)
   215  	resp, err = http.Get(url)
   216  	if err != nil {
   217  		t.Fatal(err)
   218  	}
   219  	defer resp.Body.Close()
   220  	if resp.StatusCode != http.StatusOK {
   221  		t.Fatalf("err %s", resp.Status)
   222  	}
   223  	b, err = ioutil.ReadAll(resp.Body)
   224  	if err != nil {
   225  		t.Fatal(err)
   226  	}
   227  	manifest := &api.Manifest{}
   228  	err = json.Unmarshal(b, manifest)
   229  	if err != nil {
   230  		t.Fatal(err)
   231  	}
   232  	if len(manifest.Entries) != 1 {
   233  		t.Fatalf("Manifest has %d entries", len(manifest.Entries))
   234  	}
   235  
   236  	correctRootKeyHex := "f667277e004e8486c7a3631fd226802430e84e9a81b6085d31f512a591ae0065"
   237  	if manifest.Entries[0].Hash != correctRootKeyHex {
   238  		t.Fatalf("Expected manifest path '%s', got '%s'", correctRootKeyHex, manifest.Entries[0].Hash)
   239  	}
   240  
   241  	// get bzz manifest transparent resource resolve
   242  	url = fmt.Sprintf("%s/bzz:/%s", srv.URL, rsrcResp)
   243  	resp, err = http.Get(url)
   244  	if err != nil {
   245  		t.Fatal(err)
   246  	}
   247  	defer resp.Body.Close()
   248  	if resp.StatusCode != http.StatusOK {
   249  		t.Fatalf("err %s", resp.Status)
   250  	}
   251  	b, err = ioutil.ReadAll(resp.Body)
   252  	if err != nil {
   253  		t.Fatal(err)
   254  	}
   255  
   256  	// get non-existent name, should fail
   257  	url = fmt.Sprintf("%s/bzz-resource:/bar", srv.URL)
   258  	resp, err = http.Get(url)
   259  	if err != nil {
   260  		t.Fatal(err)
   261  	}
   262  	resp.Body.Close()
   263  
   264  	// get latest update (1.1) through resource directly
   265  	log.Info("get update latest = 1.1", "addr", correctManifestAddrHex)
   266  	url = fmt.Sprintf("%s/bzz-resource:/%s", srv.URL, correctManifestAddrHex)
   267  	resp, err = http.Get(url)
   268  	if err != nil {
   269  		t.Fatal(err)
   270  	}
   271  	defer resp.Body.Close()
   272  	if resp.StatusCode != http.StatusOK {
   273  		t.Fatalf("err %s", resp.Status)
   274  	}
   275  	b, err = ioutil.ReadAll(resp.Body)
   276  	if err != nil {
   277  		t.Fatal(err)
   278  	}
   279  	if !bytes.Equal(databytes, b) {
   280  		t.Fatalf("Expected body '%x', got '%x'", databytes, b)
   281  	}
   282  
   283  	// update 2
   284  	log.Info("update 2")
   285  	url = fmt.Sprintf("%s/bzz-resource:/%s/raw", srv.URL, correctManifestAddrHex)
   286  	data := []byte("foo")
   287  	resp, err = http.Post(url, "application/octet-stream", bytes.NewReader(data))
   288  	if err != nil {
   289  		t.Fatal(err)
   290  	}
   291  	defer resp.Body.Close()
   292  	if resp.StatusCode != http.StatusOK {
   293  		t.Fatalf("Update returned %s", resp.Status)
   294  	}
   295  
   296  	// get latest update (1.2) through resource directly
   297  	log.Info("get update 1.2")
   298  	url = fmt.Sprintf("%s/bzz-resource:/%s", srv.URL, correctManifestAddrHex)
   299  	resp, err = http.Get(url)
   300  	if err != nil {
   301  		t.Fatal(err)
   302  	}
   303  	defer resp.Body.Close()
   304  	if resp.StatusCode != http.StatusOK {
   305  		t.Fatalf("err %s", resp.Status)
   306  	}
   307  	b, err = ioutil.ReadAll(resp.Body)
   308  	if err != nil {
   309  		t.Fatal(err)
   310  	}
   311  	if !bytes.Equal(data, b) {
   312  		t.Fatalf("Expected body '%x', got '%x'", data, b)
   313  	}
   314  
   315  	// get latest update (1.2) with specified period
   316  	log.Info("get update latest = 1.2")
   317  	url = fmt.Sprintf("%s/bzz-resource:/%s/1", srv.URL, correctManifestAddrHex)
   318  	resp, err = http.Get(url)
   319  	if err != nil {
   320  		t.Fatal(err)
   321  	}
   322  	defer resp.Body.Close()
   323  	if resp.StatusCode != http.StatusOK {
   324  		t.Fatalf("err %s", resp.Status)
   325  	}
   326  	b, err = ioutil.ReadAll(resp.Body)
   327  	if err != nil {
   328  		t.Fatal(err)
   329  	}
   330  	if !bytes.Equal(data, b) {
   331  		t.Fatalf("Expected body '%x', got '%x'", data, b)
   332  	}
   333  
   334  	// get first update (1.1) with specified period and version
   335  	log.Info("get first update 1.1")
   336  	url = fmt.Sprintf("%s/bzz-resource:/%s/1/1", srv.URL, correctManifestAddrHex)
   337  	resp, err = http.Get(url)
   338  	if err != nil {
   339  		t.Fatal(err)
   340  	}
   341  	defer resp.Body.Close()
   342  	if resp.StatusCode != http.StatusOK {
   343  		t.Fatalf("err %s", resp.Status)
   344  	}
   345  	b, err = ioutil.ReadAll(resp.Body)
   346  	if err != nil {
   347  		t.Fatal(err)
   348  	}
   349  	if !bytes.Equal(databytes, b) {
   350  		t.Fatalf("Expected body '%x', got '%x'", databytes, b)
   351  	}
   352  }
   353  
   354  func TestBzzGetPath(t *testing.T) {
   355  	testBzzGetPath(false, t)
   356  	testBzzGetPath(true, t)
   357  }
   358  
   359  func testBzzGetPath(encrypted bool, t *testing.T) {
   360  	var err error
   361  
   362  	testmanifest := []string{
   363  		`{"entries":[{"path":"b","hash":"011b4d03dd8c01f1049143cf9c4c817e4b167f1d1b83e5c6f0f10d89ba1e7bce","contentType":"","status":0},{"path":"c","hash":"011b4d03dd8c01f1049143cf9c4c817e4b167f1d1b83e5c6f0f10d89ba1e7bce","contentType":"","status":0}]}`,
   364  		`{"entries":[{"path":"a","hash":"011b4d03dd8c01f1049143cf9c4c817e4b167f1d1b83e5c6f0f10d89ba1e7bce","contentType":"","status":0},{"path":"b/","hash":"<key0>","contentType":"application/bzz-manifest+json","status":0}]}`,
   365  		`{"entries":[{"path":"a/","hash":"<key1>","contentType":"application/bzz-manifest+json","status":0}]}`,
   366  	}
   367  
   368  	testrequests := make(map[string]int)
   369  	testrequests["/"] = 2
   370  	testrequests["/a/"] = 1
   371  	testrequests["/a/b/"] = 0
   372  	testrequests["/x"] = 0
   373  	testrequests[""] = 0
   374  
   375  	expectedfailrequests := []string{"", "/x"}
   376  
   377  	reader := [3]*bytes.Reader{}
   378  
   379  	addr := [3]storage.Address{}
   380  
   381  	srv := testutil.NewTestSwarmServer(t, serverFunc)
   382  	defer srv.Close()
   383  
   384  	for i, mf := range testmanifest {
   385  		reader[i] = bytes.NewReader([]byte(mf))
   386  		var wait func(context.Context) error
   387  		ctx := context.TODO()
   388  		addr[i], wait, err = srv.FileStore.Store(ctx, reader[i], int64(len(mf)), encrypted)
   389  		for j := i + 1; j < len(testmanifest); j++ {
   390  			testmanifest[j] = strings.Replace(testmanifest[j], fmt.Sprintf("<key%v>", i), addr[i].Hex(), -1)
   391  		}
   392  		if err != nil {
   393  			t.Fatal(err)
   394  		}
   395  		err = wait(ctx)
   396  		if err != nil {
   397  			t.Fatal(err)
   398  		}
   399  	}
   400  
   401  	rootRef := addr[2].Hex()
   402  
   403  	_, err = http.Get(srv.URL + "/bzz-raw:/" + rootRef + "/a")
   404  	if err != nil {
   405  		t.Fatalf("Failed to connect to proxy: %v", err)
   406  	}
   407  
   408  	for k, v := range testrequests {
   409  		var resp *http.Response
   410  		var respbody []byte
   411  
   412  		url := srv.URL + "/bzz-raw:/"
   413  		if k[:] != "" {
   414  			url += rootRef + "/" + k[1:] + "?content_type=text/plain"
   415  		}
   416  		resp, err = http.Get(url)
   417  		if err != nil {
   418  			t.Fatalf("Request failed: %v", err)
   419  		}
   420  		defer resp.Body.Close()
   421  		respbody, err = ioutil.ReadAll(resp.Body)
   422  
   423  		if string(respbody) != testmanifest[v] {
   424  			isexpectedfailrequest := false
   425  
   426  			for _, r := range expectedfailrequests {
   427  				if k[:] == r {
   428  					isexpectedfailrequest = true
   429  				}
   430  			}
   431  			if !isexpectedfailrequest {
   432  				t.Fatalf("Response body does not match, expected: %v, got %v", testmanifest[v], string(respbody))
   433  			}
   434  		}
   435  	}
   436  
   437  	for k, v := range testrequests {
   438  		var resp *http.Response
   439  		var respbody []byte
   440  
   441  		url := srv.URL + "/bzz-hash:/"
   442  		if k[:] != "" {
   443  			url += rootRef + "/" + k[1:]
   444  		}
   445  		resp, err = http.Get(url)
   446  		if err != nil {
   447  			t.Fatalf("Request failed: %v", err)
   448  		}
   449  		defer resp.Body.Close()
   450  		respbody, err = ioutil.ReadAll(resp.Body)
   451  		if err != nil {
   452  			t.Fatalf("Read request body: %v", err)
   453  		}
   454  
   455  		if string(respbody) != addr[v].Hex() {
   456  			isexpectedfailrequest := false
   457  
   458  			for _, r := range expectedfailrequests {
   459  				if k[:] == r {
   460  					isexpectedfailrequest = true
   461  				}
   462  			}
   463  			if !isexpectedfailrequest {
   464  				t.Fatalf("Response body does not match, expected: %v, got %v", addr[v], string(respbody))
   465  			}
   466  		}
   467  	}
   468  
   469  	ref := addr[2].Hex()
   470  
   471  	for _, c := range []struct {
   472  		path string
   473  		json string
   474  		html string
   475  	}{
   476  		{
   477  			path: "/",
   478  			json: `{"common_prefixes":["a/"]}`,
   479  			html: fmt.Sprintf("<!DOCTYPE html>\n<html>\n<head>\n  <meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">\n  <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">\n\t\t<link rel=\"shortcut icon\" type=\"image/x-icon\" href=\"\"/>\n\t<title>Swarm index of bzz:/%s/</title>\n</head>\n\n<body>\n  <h1>Swarm index of bzz:/%s/</h1>\n  <hr>\n  <table>\n    <thead>\n      <tr>\n\t<th>Path</th>\n\t<th>Type</th>\n\t<th>Size</th>\n      </tr>\n    </thead>\n\n    <tbody>\n      \n\t<tr>\n\t  <td><a href=\"a/\">a/</a></td>\n\t  <td>DIR</td>\n\t  <td>-</td>\n\t</tr>\n      \n\n      \n  </table>\n  <hr>\n</body>\n", ref, ref),
   480  		},
   481  		{
   482  			path: "/a/",
   483  			json: `{"common_prefixes":["a/b/"],"entries":[{"hash":"011b4d03dd8c01f1049143cf9c4c817e4b167f1d1b83e5c6f0f10d89ba1e7bce","path":"a/a","mod_time":"0001-01-01T00:00:00Z"}]}`,
   484  			html: fmt.Sprintf("<!DOCTYPE html>\n<html>\n<head>\n  <meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">\n  <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">\n\t\t<link rel=\"shortcut icon\" type=\"image/x-icon\" href=\"\"/>\n\t<title>Swarm index of bzz:/%s/a/</title>\n</head>\n\n<body>\n  <h1>Swarm index of bzz:/%s/a/</h1>\n  <hr>\n  <table>\n    <thead>\n      <tr>\n\t<th>Path</th>\n\t<th>Type</th>\n\t<th>Size</th>\n      </tr>\n    </thead>\n\n    <tbody>\n      \n\t<tr>\n\t  <td><a href=\"b/\">b/</a></td>\n\t  <td>DIR</td>\n\t  <td>-</td>\n\t</tr>\n      \n\n      \n\t<tr>\n\t  <td><a href=\"a\">a</a></td>\n\t  <td></td>\n\t  <td>0</td>\n\t</tr>\n      \n  </table>\n  <hr>\n</body>\n", ref, ref),
   485  		},
   486  		{
   487  			path: "/a/b/",
   488  			json: `{"entries":[{"hash":"011b4d03dd8c01f1049143cf9c4c817e4b167f1d1b83e5c6f0f10d89ba1e7bce","path":"a/b/b","mod_time":"0001-01-01T00:00:00Z"},{"hash":"011b4d03dd8c01f1049143cf9c4c817e4b167f1d1b83e5c6f0f10d89ba1e7bce","path":"a/b/c","mod_time":"0001-01-01T00:00:00Z"}]}`,
   489  			html: fmt.Sprintf("<!DOCTYPE html>\n<html>\n<head>\n  <meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">\n  <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">\n\t\t<link rel=\"shortcut icon\" type=\"image/x-icon\" href=\"\"/>\n\t<title>Swarm index of bzz:/%s/a/b/</title>\n</head>\n\n<body>\n  <h1>Swarm index of bzz:/%s/a/b/</h1>\n  <hr>\n  <table>\n    <thead>\n      <tr>\n\t<th>Path</th>\n\t<th>Type</th>\n\t<th>Size</th>\n      </tr>\n    </thead>\n\n    <tbody>\n      \n\n      \n\t<tr>\n\t  <td><a href=\"b\">b</a></td>\n\t  <td></td>\n\t  <td>0</td>\n\t</tr>\n      \n\t<tr>\n\t  <td><a href=\"c\">c</a></td>\n\t  <td></td>\n\t  <td>0</td>\n\t</tr>\n      \n  </table>\n  <hr>\n</body>\n", ref, ref),
   490  		},
   491  		{
   492  			path: "/x",
   493  		},
   494  		{
   495  			path: "",
   496  		},
   497  	} {
   498  		k := c.path
   499  		url := srv.URL + "/bzz-list:/"
   500  		if k[:] != "" {
   501  			url += rootRef + "/" + k[1:]
   502  		}
   503  		t.Run("json list "+c.path, func(t *testing.T) {
   504  			resp, err := http.Get(url)
   505  			if err != nil {
   506  				t.Fatalf("HTTP request: %v", err)
   507  			}
   508  			defer resp.Body.Close()
   509  			respbody, err := ioutil.ReadAll(resp.Body)
   510  			if err != nil {
   511  				t.Fatalf("Read response body: %v", err)
   512  			}
   513  
   514  			body := strings.TrimSpace(string(respbody))
   515  			if body != c.json {
   516  				isexpectedfailrequest := false
   517  
   518  				for _, r := range expectedfailrequests {
   519  					if k[:] == r {
   520  						isexpectedfailrequest = true
   521  					}
   522  				}
   523  				if !isexpectedfailrequest {
   524  					t.Errorf("Response list body %q does not match, expected: %v, got %v", k, c.json, body)
   525  				}
   526  			}
   527  		})
   528  		t.Run("html list "+c.path, func(t *testing.T) {
   529  			req, err := http.NewRequest(http.MethodGet, url, nil)
   530  			if err != nil {
   531  				t.Fatalf("New request: %v", err)
   532  			}
   533  			req.Header.Set("Accept", "text/html")
   534  			resp, err := http.DefaultClient.Do(req)
   535  			if err != nil {
   536  				t.Fatalf("HTTP request: %v", err)
   537  			}
   538  			defer resp.Body.Close()
   539  			respbody, err := ioutil.ReadAll(resp.Body)
   540  			if err != nil {
   541  				t.Fatalf("Read response body: %v", err)
   542  			}
   543  
   544  			if string(respbody) != c.html {
   545  				isexpectedfailrequest := false
   546  
   547  				for _, r := range expectedfailrequests {
   548  					if k[:] == r {
   549  						isexpectedfailrequest = true
   550  					}
   551  				}
   552  				if !isexpectedfailrequest {
   553  					t.Errorf("Response list body %q does not match, expected: %q, got %q", k, c.html, string(respbody))
   554  				}
   555  			}
   556  		})
   557  	}
   558  
   559  	nonhashtests := []string{
   560  		srv.URL + "/bzz:/name",
   561  		srv.URL + "/bzz-immutable:/nonhash",
   562  		srv.URL + "/bzz-raw:/nonhash",
   563  		srv.URL + "/bzz-list:/nonhash",
   564  		srv.URL + "/bzz-hash:/nonhash",
   565  	}
   566  
   567  	nonhashresponses := []string{
   568  		"cannot resolve name: no DNS to resolve name: &#34;name&#34;",
   569  		"cannot resolve nonhash: immutable address not a content hash: &#34;nonhash&#34;",
   570  		"cannot resolve nonhash: no DNS to resolve name: &#34;nonhash&#34;",
   571  		"cannot resolve nonhash: no DNS to resolve name: &#34;nonhash&#34;",
   572  		"cannot resolve nonhash: no DNS to resolve name: &#34;nonhash&#34;",
   573  	}
   574  
   575  	for i, url := range nonhashtests {
   576  		var resp *http.Response
   577  		var respbody []byte
   578  
   579  		resp, err = http.Get(url)
   580  
   581  		if err != nil {
   582  			t.Fatalf("Request failed: %v", err)
   583  		}
   584  		defer resp.Body.Close()
   585  		respbody, err = ioutil.ReadAll(resp.Body)
   586  		if err != nil {
   587  			t.Fatalf("ReadAll failed: %v", err)
   588  		}
   589  		if !strings.Contains(string(respbody), nonhashresponses[i]) {
   590  			t.Fatalf("Non-Hash response body does not match, expected: %v, got: %v", nonhashresponses[i], string(respbody))
   591  		}
   592  	}
   593  }
   594  
   595  // TestBzzRootRedirect tests that getting the root path of a manifest without
   596  // a trailing slash gets redirected to include the trailing slash so that
   597  // relative URLs work as expected.
   598  func TestBzzRootRedirect(t *testing.T) {
   599  	testBzzRootRedirect(false, t)
   600  }
   601  func TestBzzRootRedirectEncrypted(t *testing.T) {
   602  	testBzzRootRedirect(true, t)
   603  }
   604  
   605  func testBzzRootRedirect(toEncrypt bool, t *testing.T) {
   606  	srv := testutil.NewTestSwarmServer(t, serverFunc)
   607  	defer srv.Close()
   608  
   609  	// create a manifest with some data at the root path
   610  	client := swarm.NewClient(srv.URL)
   611  	data := []byte("data")
   612  	file := &swarm.File{
   613  		ReadCloser: ioutil.NopCloser(bytes.NewReader(data)),
   614  		ManifestEntry: api.ManifestEntry{
   615  			Path:        "",
   616  			ContentType: "text/plain",
   617  			Size:        int64(len(data)),
   618  		},
   619  	}
   620  	hash, err := client.Upload(file, "", toEncrypt)
   621  	if err != nil {
   622  		t.Fatal(err)
   623  	}
   624  
   625  	// define a CheckRedirect hook which ensures there is only a single
   626  	// redirect to the correct URL
   627  	redirected := false
   628  	httpClient := http.Client{
   629  		CheckRedirect: func(req *http.Request, via []*http.Request) error {
   630  			if redirected {
   631  				return errors.New("too many redirects")
   632  			}
   633  			redirected = true
   634  			expectedPath := "/bzz:/" + hash + "/"
   635  			if req.URL.Path != expectedPath {
   636  				return fmt.Errorf("expected redirect to %q, got %q", expectedPath, req.URL.Path)
   637  			}
   638  			return nil
   639  		},
   640  	}
   641  
   642  	// perform the GET request and assert the response
   643  	res, err := httpClient.Get(srv.URL + "/bzz:/" + hash)
   644  	if err != nil {
   645  		t.Fatal(err)
   646  	}
   647  	defer res.Body.Close()
   648  	if !redirected {
   649  		t.Fatal("expected GET /bzz:/<hash> to redirect to /bzz:/<hash>/ but it didn't")
   650  	}
   651  	gotData, err := ioutil.ReadAll(res.Body)
   652  	if err != nil {
   653  		t.Fatal(err)
   654  	}
   655  	if !bytes.Equal(gotData, data) {
   656  		t.Fatalf("expected response to equal %q, got %q", data, gotData)
   657  	}
   658  }
   659  
   660  func TestMethodsNotAllowed(t *testing.T) {
   661  	srv := testutil.NewTestSwarmServer(t, serverFunc)
   662  	defer srv.Close()
   663  	databytes := "bar"
   664  	for _, c := range []struct {
   665  		url  string
   666  		code int
   667  	}{
   668  		{
   669  			url:  fmt.Sprintf("%s/bzz-list:/", srv.URL),
   670  			code: 405,
   671  		}, {
   672  			url:  fmt.Sprintf("%s/bzz-hash:/", srv.URL),
   673  			code: 405,
   674  		},
   675  		{
   676  			url:  fmt.Sprintf("%s/bzz-immutable:/", srv.URL),
   677  			code: 405,
   678  		},
   679  	} {
   680  		res, _ := http.Post(c.url, "text/plain", bytes.NewReader([]byte(databytes)))
   681  		if res.StatusCode != c.code {
   682  			t.Fatal("should have failed")
   683  		}
   684  	}
   685  
   686  }