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