Skip to content

Commit 582e297

Browse files
aschmahmannJorropo
authored andcommitted
fix: handle _redirects for If-None-Match headers (#412)
* fix: handle _redirects for If-None-Match headers * fix: handle _redirects for If-None-Match headers * fix(gateway): HEAD requests now respect _redirects * feat: add _redirects regression test * docs: add changelog entry (cherry picked from commit 1f5df74)
1 parent 61f2939 commit 582e297

File tree

5 files changed

+87
-5
lines changed

5 files changed

+87
-5
lines changed

CHANGELOG.md

+2
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,8 @@ The following emojis are used to highlight certain changes:
2222

2323
### Fixed
2424

25+
- Handle `_redirects` file when `If-None-Match` header is present ([#412](https://github.com/ipfs/boxo/pull/412))
26+
2527
### Security
2628

2729
## [0.10.2] - 2023-06-29

gateway/gateway_test.go

+55
Original file line numberDiff line numberDiff line change
@@ -523,6 +523,61 @@ func TestRedirects(t *testing.T) {
523523
res = mustDoWithoutRedirect(t, req)
524524
require.Equal(t, http.StatusNotFound, res.StatusCode)
525525
})
526+
527+
t.Run("_redirects file with If-None-Match header", func(t *testing.T) {
528+
t.Parallel()
529+
530+
backend, root := newMockBackend(t, "redirects-spa.car")
531+
backend.namesys["/ipns/example.com"] = path.FromCid(root)
532+
533+
ts := newTestServerWithConfig(t, backend, Config{
534+
Headers: map[string][]string{},
535+
NoDNSLink: false,
536+
PublicGateways: map[string]*PublicGateway{
537+
"example.com": {
538+
UseSubdomains: true,
539+
DeserializedResponses: true,
540+
},
541+
},
542+
DeserializedResponses: true,
543+
})
544+
545+
missingPageURL := ts.URL + "/missing-page"
546+
547+
do := func(method string) {
548+
// Make initial request to non-existing page that should return the contents
549+
// of index.html as per the _redirects file.
550+
req := mustNewRequest(t, method, missingPageURL, nil)
551+
req.Header.Add("Accept", "text/html")
552+
req.Host = "example.com"
553+
554+
res := mustDoWithoutRedirect(t, req)
555+
defer res.Body.Close()
556+
557+
// Check statuses and body.
558+
require.Equal(t, http.StatusOK, res.StatusCode)
559+
body, err := io.ReadAll(res.Body)
560+
require.NoError(t, err)
561+
require.Equal(t, "hello world\n", string(body))
562+
563+
// Check Etag.
564+
etag := res.Header.Get("Etag")
565+
require.NotEmpty(t, etag)
566+
567+
// Repeat request with Etag as If-None-Match value. Expect 304 Not Modified.
568+
req = mustNewRequest(t, method, missingPageURL, nil)
569+
req.Header.Add("Accept", "text/html")
570+
req.Host = "example.com"
571+
req.Header.Add("If-None-Match", etag)
572+
573+
res = mustDoWithoutRedirect(t, req)
574+
defer res.Body.Close()
575+
require.Equal(t, http.StatusNotModified, res.StatusCode)
576+
}
577+
578+
do(http.MethodGet)
579+
do(http.MethodHead)
580+
})
526581
}
527582

528583
func TestDeserializedResponses(t *testing.T) {

gateway/handler.go

+13-3
Original file line numberDiff line numberDiff line change
@@ -700,9 +700,19 @@ func (i *handler) handleIfNoneMatch(w http.ResponseWriter, r *http.Request, rq *
700700
if ifNoneMatch := r.Header.Get("If-None-Match"); ifNoneMatch != "" {
701701
pathMetadata, err := i.backend.ResolvePath(r.Context(), rq.immutablePath)
702702
if err != nil {
703-
err = fmt.Errorf("failed to resolve %s: %w", debugStr(rq.contentPath.String()), err)
704-
i.webError(w, r, err, http.StatusInternalServerError)
705-
return true
703+
var forwardedPath ImmutablePath
704+
var continueProcessing bool
705+
if isWebRequest(rq.responseFormat) {
706+
forwardedPath, continueProcessing = i.handleWebRequestErrors(w, r, rq.mostlyResolvedPath(), rq.immutablePath, rq.contentPath, err, rq.logger)
707+
if continueProcessing {
708+
pathMetadata, err = i.backend.ResolvePath(r.Context(), forwardedPath)
709+
}
710+
}
711+
if !continueProcessing || err != nil {
712+
err = fmt.Errorf("failed to resolve %s: %w", debugStr(rq.contentPath.String()), err)
713+
i.webError(w, r, err, http.StatusInternalServerError)
714+
return true
715+
}
706716
}
707717

708718
pathCid := pathMetadata.LastSegment.Cid()

gateway/handler_defaults.go

+17-2
Original file line numberDiff line numberDiff line change
@@ -33,8 +33,23 @@ func (i *handler) serveDefaults(ctx context.Context, w http.ResponseWriter, r *h
3333
case http.MethodHead:
3434
var data files.Node
3535
pathMetadata, data, err = i.backend.Head(ctx, rq.mostlyResolvedPath())
36-
if !i.handleRequestErrors(w, r, rq.contentPath, err) {
37-
return false
36+
if err != nil {
37+
if isWebRequest(rq.responseFormat) {
38+
forwardedPath, continueProcessing := i.handleWebRequestErrors(w, r, rq.mostlyResolvedPath(), rq.immutablePath, rq.contentPath, err, rq.logger)
39+
if !continueProcessing {
40+
return false
41+
}
42+
pathMetadata, data, err = i.backend.Head(ctx, forwardedPath)
43+
if err != nil {
44+
err = fmt.Errorf("failed to resolve %s: %w", debugStr(rq.contentPath.String()), err)
45+
i.webError(w, r, err, http.StatusInternalServerError)
46+
return false
47+
}
48+
} else {
49+
if !i.handleRequestErrors(w, r, rq.contentPath, err) {
50+
return false
51+
}
52+
}
3853
}
3954
defer data.Close()
4055
if _, ok := data.(files.Directory); ok {

gateway/testdata/redirects-spa.car

360 Bytes
Binary file not shown.

0 commit comments

Comments
 (0)