Skip to content

Commit 654231b

Browse files
aschmahmannhacdias
andauthored
fix(gw): IPIP-402 CARs return useful blocks on not found errors (#440)
Co-authored-by: Henrique Dias <hacdias@gmail.com>
1 parent 7ec68c5 commit 654231b

File tree

4 files changed

+41
-6
lines changed

4 files changed

+41
-6
lines changed

CHANGELOG.md

+2
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,8 @@ The following emojis are used to highlight certain changes:
3232
### Fixed
3333

3434
- Address a Bitswap findpeers / connect race condition that can prevent peer communication ([#435](https://github.com/ipfs/boxo/issues/435))
35+
- HTTP Gateway API: Not having a block will result in a 5xx error rather than 404
36+
- HTTP Gateway API: CAR requests will return 200s and a CAR file proving a requested path does not exist rather than returning an error
3537

3638
### Security
3739

gateway/blocks_backend.go

+36
Original file line numberDiff line numberDiff line change
@@ -232,9 +232,45 @@ func (bb *BlocksBackend) Head(ctx context.Context, path ImmutablePath) (ContentP
232232
return md, fileNode, nil
233233
}
234234

235+
// emptyRoot is a CAR root with the empty identity CID. CAR files are recommended
236+
// to always include a CID in their root, even if it's just the empty CID.
237+
// https://ipld.io/specs/transport/car/carv1/#number-of-roots
238+
var emptyRoot = []cid.Cid{cid.MustParse("bafkqaaa")}
239+
235240
func (bb *BlocksBackend) GetCAR(ctx context.Context, p ImmutablePath, params CarParams) (ContentPathMetadata, io.ReadCloser, error) {
236241
pathMetadata, err := bb.ResolvePath(ctx, p)
237242
if err != nil {
243+
rootCid, err := cid.Decode(strings.Split(p.String(), "/")[2])
244+
if err != nil {
245+
return ContentPathMetadata{}, nil, err
246+
}
247+
248+
var buf bytes.Buffer
249+
cw, err := storage.NewWritable(&buf, emptyRoot, car.WriteAsCarV1(true))
250+
if err != nil {
251+
return ContentPathMetadata{}, nil, err
252+
}
253+
254+
blockGetter := merkledag.NewDAGService(bb.blockService).Session(ctx)
255+
256+
blockGetter = &nodeGetterToCarExporer{
257+
ng: blockGetter,
258+
cw: cw,
259+
}
260+
261+
// Setup the UnixFS resolver.
262+
f := newNodeGetterFetcherSingleUseFactory(ctx, blockGetter)
263+
pathResolver := resolver.NewBasicResolver(f)
264+
ip := ipfspath.FromString(p.String())
265+
_, _, err = pathResolver.ResolveToLastNode(ctx, ip)
266+
267+
if isErrNotFound(err) {
268+
return ContentPathMetadata{
269+
PathSegmentRoots: nil,
270+
LastSegment: ifacepath.NewResolvedPath(ip, rootCid, rootCid, ""),
271+
ContentType: "",
272+
}, io.NopCloser(&buf), nil
273+
}
238274
return ContentPathMetadata{}, nil, err
239275
}
240276

gateway/errors.go

+2-5
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,6 @@ import (
1212
"github.com/ipfs/boxo/gateway/assets"
1313
"github.com/ipfs/boxo/path/resolver"
1414
"github.com/ipfs/go-cid"
15-
ipld "github.com/ipfs/go-ipld-format"
1615
"github.com/ipld/go-ipld-prime/datamodel"
1716
)
1817

@@ -177,11 +176,9 @@ func webError(w http.ResponseWriter, r *http.Request, c *Config, err error, defa
177176
}
178177
}
179178

179+
// isErrNotFound returns true for IPLD errors that should return 4xx errors (e.g. the path doesn't exist, the data is
180+
// the wrong type, etc.), rather than issues with just finding and retrieving the data.
180181
func isErrNotFound(err error) bool {
181-
if ipld.IsNotFound(err) {
182-
return true
183-
}
184-
185182
// Checks if err is of a type that does not implement the .Is interface and
186183
// cannot be directly compared to. Therefore, errors.Is cannot be used.
187184
for {

gateway/gateway_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -763,7 +763,7 @@ func TestErrorBubblingFromBackend(t *testing.T) {
763763
})
764764
}
765765

766-
testError("404 Not Found from IPLD", &ipld.ErrNotFound{}, http.StatusNotFound)
766+
testError("500 Not Found from IPLD", &ipld.ErrNotFound{}, http.StatusInternalServerError)
767767
testError("404 Not Found from path resolver", resolver.ErrNoLink{}, http.StatusNotFound)
768768
testError("502 Bad Gateway", ErrBadGateway, http.StatusBadGateway)
769769
testError("504 Gateway Timeout", ErrGatewayTimeout, http.StatusGatewayTimeout)

0 commit comments

Comments
 (0)