@@ -30,16 +30,7 @@ func (i *handler) serveCAR(ctx context.Context, w http.ResponseWriter, r *http.R
30
30
ctx , cancel := context .WithCancel (ctx )
31
31
defer cancel ()
32
32
33
- switch rq .responseParams ["version" ] {
34
- case "" : // noop, client does not care about version
35
- case "1" : // noop, we support this
36
- default :
37
- err := fmt .Errorf ("unsupported CAR version: only version=1 is supported" )
38
- i .webError (w , r , err , http .StatusBadRequest )
39
- return false
40
- }
41
-
42
- params , err := getCarParams (r )
33
+ params , err := buildCarParams (r , rq .responseParams )
43
34
if err != nil {
44
35
i .webError (w , r , err , http .StatusBadRequest )
45
36
return false
@@ -90,7 +81,7 @@ func (i *handler) serveCAR(ctx context.Context, w http.ResponseWriter, r *http.R
90
81
// sub-DAGs and IPLD selectors: https://github.com/ipfs/go-ipfs/issues/8769
91
82
w .Header ().Set ("Accept-Ranges" , "none" )
92
83
93
- w .Header ().Set ("Content-Type" , carResponseFormat + "; version=1" )
84
+ w .Header ().Set ("Content-Type" , buildContentTypeFromCarParams ( params ) )
94
85
w .Header ().Set ("X-Content-Type-Options" , "nosniff" ) // no funny business in the browsers :^)
95
86
96
87
_ , copyErr := io .Copy (w , carFile )
@@ -113,7 +104,15 @@ func (i *handler) serveCAR(ctx context.Context, w http.ResponseWriter, r *http.R
113
104
return true
114
105
}
115
106
116
- func getCarParams (r * http.Request ) (CarParams , error ) {
107
+ // buildCarParams returns CarParams based on the request, any optional parameters
108
+ // passed in URL, Accept header and the implicit defaults specific to boxo
109
+ // implementation, such as block order and duplicates status.
110
+ //
111
+ // If any of the optional content type parameters (e.g., CAR order or
112
+ // duplicates) are unspecified or empty, the function will automatically infer
113
+ // default values.
114
+ func buildCarParams (r * http.Request , contentTypeParams map [string ]string ) (CarParams , error ) {
115
+ // URL query parameters
117
116
queryParams := r .URL .Query ()
118
117
rangeStr , hasRange := queryParams .Get (carRangeBytesKey ), queryParams .Has (carRangeBytesKey )
119
118
scopeStr , hasScope := queryParams .Get (carTerminalElementTypeKey ), queryParams .Has (carTerminalElementTypeKey )
@@ -122,7 +121,7 @@ func getCarParams(r *http.Request) (CarParams, error) {
122
121
if hasRange {
123
122
rng , err := NewDagByteRange (rangeStr )
124
123
if err != nil {
125
- err = fmt .Errorf ("invalid entity-bytes: %w" , err )
124
+ err = fmt .Errorf ("invalid application/vnd.ipld.car entity-bytes URL parameter : %w" , err )
126
125
return CarParams {}, err
127
126
}
128
127
params .Range = & rng
@@ -133,16 +132,78 @@ func getCarParams(r *http.Request) (CarParams, error) {
133
132
case DagScopeEntity , DagScopeAll , DagScopeBlock :
134
133
params .Scope = s
135
134
default :
136
- err := fmt .Errorf ("unsupported dag-scope %s " , scopeStr )
135
+ err := fmt .Errorf ("unsupported application/vnd.ipld.car dag-scope URL parameter: %q " , scopeStr )
137
136
return CarParams {}, err
138
137
}
139
138
} else {
140
139
params .Scope = DagScopeAll
141
140
}
142
141
142
+ // application/vnd.ipld.car content type parameters from Accept header
143
+
144
+ // version of CAR format
145
+ switch contentTypeParams ["version" ] {
146
+ case "" : // noop, client does not care about version
147
+ case "1" : // noop, we support this
148
+ default :
149
+ return CarParams {}, fmt .Errorf ("unsupported application/vnd.ipld.car version: only version=1 is supported" )
150
+ }
151
+
152
+ // optional order from IPIP-412
153
+ if order := DagOrder (contentTypeParams ["order" ]); order != DagOrderUnspecified {
154
+ switch order {
155
+ case DagOrderUnknown , DagOrderDFS :
156
+ params .Order = order
157
+ default :
158
+ return CarParams {}, fmt .Errorf ("unsupported application/vnd.ipld.car content type order parameter: %q" , order )
159
+ }
160
+ } else {
161
+ // when order is not specified, we use DFS as the implicit default
162
+ // as this has always been the default behavior and we should not break
163
+ // legacy clients
164
+ params .Order = DagOrderDFS
165
+ }
166
+
167
+ // optional dups from IPIP-412
168
+ if dups := NewDuplicateBlocksPolicy (contentTypeParams ["dups" ]); dups != DuplicateBlocksUnspecified {
169
+ switch dups {
170
+ case DuplicateBlocksExcluded , DuplicateBlocksIncluded :
171
+ params .Duplicates = dups
172
+ default :
173
+ return CarParams {}, fmt .Errorf ("unsupported application/vnd.ipld.car content type dups parameter: %q" , dups )
174
+ }
175
+ } else {
176
+ // when duplicate block preference is not specified, we set it to
177
+ // false, as this has always been the default behavior, we should
178
+ // not break legacy clients, and responses to requests made via ?format=car
179
+ // should benefit from block deduplication
180
+ params .Duplicates = DuplicateBlocksExcluded
181
+
182
+ }
183
+
143
184
return params , nil
144
185
}
145
186
187
+ // buildContentTypeFromCarParams returns a string for Content-Type header.
188
+ // It does not change any values, CarParams are respected as-is.
189
+ func buildContentTypeFromCarParams (params CarParams ) string {
190
+ h := strings.Builder {}
191
+ h .WriteString (carResponseFormat )
192
+ h .WriteString ("; version=1" )
193
+
194
+ if params .Order != DagOrderUnspecified {
195
+ h .WriteString ("; order=" )
196
+ h .WriteString (string (params .Order ))
197
+ }
198
+
199
+ if params .Duplicates != DuplicateBlocksUnspecified {
200
+ h .WriteString ("; dups=" )
201
+ h .WriteString (params .Duplicates .String ())
202
+ }
203
+
204
+ return h .String ()
205
+ }
206
+
146
207
func getCarRootCidAndLastSegment (imPath ImmutablePath ) (cid.Cid , string , error ) {
147
208
imPathStr := imPath .String ()
148
209
if ! strings .HasPrefix (imPathStr , "/ipfs/" ) {
@@ -167,14 +228,25 @@ func getCarRootCidAndLastSegment(imPath ImmutablePath) (cid.Cid, string, error)
167
228
func getCarEtag (imPath ImmutablePath , params CarParams , rootCid cid.Cid ) string {
168
229
data := imPath .String ()
169
230
if params .Scope != DagScopeAll {
170
- data += "." + string (params .Scope )
231
+ data += string (params .Scope )
232
+ }
233
+
234
+ // 'order' from IPIP-412 impact Etag only if set to something else
235
+ // than DFS (which is the implicit default)
236
+ if params .Order != DagOrderDFS {
237
+ data += string (params .Order )
238
+ }
239
+
240
+ // 'dups' from IPIP-412 impact Etag only if 'y'
241
+ if dups := params .Duplicates .String (); dups == "y" {
242
+ data += dups
171
243
}
172
244
173
245
if params .Range != nil {
174
246
if params .Range .From != 0 || params .Range .To != nil {
175
- data += "." + strconv .FormatInt (params .Range .From , 10 )
247
+ data += strconv .FormatInt (params .Range .From , 10 )
176
248
if params .Range .To != nil {
177
- data += "." + strconv .FormatInt (* params .Range .To , 10 )
249
+ data += strconv .FormatInt (* params .Range .To , 10 )
178
250
}
179
251
}
180
252
}
0 commit comments