Skip to content

Commit 1f5df74

Browse files
aschmahmannhacdias
andauthored
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 --------- Co-authored-by: Henrique Dias <hacdias@gmail.com>
1 parent f6b448b commit 1f5df74

File tree

5 files changed

+86
-5
lines changed

5 files changed

+86
-5
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,7 @@ The following emojis are used to highlight certain changes:
4949
### Fixed
5050

5151
- Removed mentions of unused ARC algorithm ([#336](https://github.com/ipfs/boxo/issues/366#issuecomment-1597253540))
52+
- Handle `_redirects` file when `If-None-Match` header is present ([#412](https://github.com/ipfs/boxo/pull/412))
5253

5354
### Security
5455

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
@@ -703,9 +703,19 @@ func (i *handler) handleIfNoneMatch(w http.ResponseWriter, r *http.Request, rq *
703703
if ifNoneMatch := r.Header.Get("If-None-Match"); ifNoneMatch != "" {
704704
pathMetadata, err := i.backend.ResolvePath(r.Context(), rq.immutablePath)
705705
if err != nil {
706-
err = fmt.Errorf("failed to resolve %s: %w", debugStr(rq.contentPath.String()), err)
707-
i.webError(w, r, err, http.StatusInternalServerError)
708-
return true
706+
var forwardedPath ImmutablePath
707+
var continueProcessing bool
708+
if isWebRequest(rq.responseFormat) {
709+
forwardedPath, continueProcessing = i.handleWebRequestErrors(w, r, rq.mostlyResolvedPath(), rq.immutablePath, rq.contentPath, err, rq.logger)
710+
if continueProcessing {
711+
pathMetadata, err = i.backend.ResolvePath(r.Context(), forwardedPath)
712+
}
713+
}
714+
if !continueProcessing || err != nil {
715+
err = fmt.Errorf("failed to resolve %s: %w", debugStr(rq.contentPath.String()), err)
716+
i.webError(w, r, err, http.StatusInternalServerError)
717+
return true
718+
}
709719
}
710720

711721
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)