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: "name"", 569 "cannot resolve nonhash: immutable address not a content hash: "nonhash"", 570 "cannot resolve nonhash: no DNS to resolve name: "nonhash"", 571 "cannot resolve nonhash: no DNS to resolve name: "nonhash"", 572 "cannot resolve nonhash: no DNS to resolve name: "nonhash"", 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 }