Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(850)

Side by Side Diff: runtime/bin/http_impl.dart

Issue 10407002: Add special handling of the content type HTTP header (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Created 8 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
2 // for details. All rights reserved. Use of this source code is governed by a 2 // for details. All rights reserved. Use of this source code is governed by a
3 // BSD-style license that can be found in the LICENSE file. 3 // BSD-style license that can be found in the LICENSE file.
4 4
5 class _HttpHeaders implements HttpHeaders { 5 class _HttpHeaders implements HttpHeaders {
6 _HttpHeaders() : _headers = new Map<String, List<String>>(); 6 _HttpHeaders() : _headers = new Map<String, List<String>>();
7 7
8 List<String> operator[](String name) { 8 List<String> operator[](String name) {
9 name = name.toLowerCase(); 9 name = name.toLowerCase();
10 return _headers[name]; 10 return _headers[name];
11 } 11 }
12 12
13 String value(String name) { 13 String value(String name) {
14 name = name.toLowerCase(); 14 name = name.toLowerCase();
15 if (name == "content-type") _syncContentType();
15 List<String> values = _headers[name]; 16 List<String> values = _headers[name];
16 if (values == null) return null; 17 if (values == null) return null;
17 if (values.length > 1) { 18 if (values.length > 1) {
18 throw new HttpException("More than one value for header $name"); 19 throw new HttpException("More than one value for header $name");
19 } 20 }
20 return values[0]; 21 return values[0];
21 } 22 }
22 23
23 void add(String name, Object value) { 24 void add(String name, Object value) {
24 _checkMutable(); 25 _checkMutable();
25 if (value is List) { 26 if (value is List) {
26 for (int i = 0; i < value.length; i++) { 27 for (int i = 0; i < value.length; i++) {
27 _add(name, value[i]); 28 _add(name, value[i]);
28 } 29 }
29 } else { 30 } else {
30 _add(name, value); 31 _add(name, value);
31 } 32 }
32 } 33 }
33 34
34 void set(String name, Object value) { 35 void set(String name, Object value) {
36 name = name.toLowerCase();
35 _checkMutable(); 37 _checkMutable();
36 removeAll(name); 38 removeAll(name);
37 add(name, value); 39 add(name, value);
40 if (name == "content-type") contentType == null;
Mads Ager (google) 2012/05/21 07:40:39 This looks strange. If you are setting content-typ
Anders Johnsen 2012/05/21 07:43:21 =, not ==.
Søren Gjesse 2012/05/21 11:11:06 Done.
Søren Gjesse 2012/05/21 11:11:06 Added _clearHeaderValueCache and added test.
38 } 41 }
39 42
40 void remove(String name, Object value) { 43 void remove(String name, Object value) {
41 _checkMutable(); 44 _checkMutable();
42 name = name.toLowerCase(); 45 name = name.toLowerCase();
43 List<String> values = _headers[name]; 46 List<String> values = _headers[name];
44 if (values != null) { 47 if (values != null) {
45 int index = values.indexOf(value); 48 int index = values.indexOf(value);
46 if (index != -1) { 49 if (index != -1) {
47 values.removeRange(index, 1); 50 values.removeRange(index, 1);
48 } 51 }
49 } 52 }
53 if (name == "content-type") contentType == null;
Mads Ager (google) 2012/05/21 07:40:39 _contentType?
Anders Johnsen 2012/05/21 07:43:21 =, not ==.
Søren Gjesse 2012/05/21 11:11:06 Used _clearHeaderValueCache.
Søren Gjesse 2012/05/21 11:11:06 Done.
50 } 54 }
51 55
52 void removeAll(String name) { 56 void removeAll(String name) {
53 _checkMutable(); 57 _checkMutable();
54 name = name.toLowerCase(); 58 name = name.toLowerCase();
55 _headers.remove(name); 59 _headers.remove(name);
60 contentType == null;
Mads Ager (google) 2012/05/21 07:40:39 _contentType?
Anders Johnsen 2012/05/21 07:43:21 =, not ==.
Søren Gjesse 2012/05/21 11:11:06 Used _clearHeaderValueCache.
Søren Gjesse 2012/05/21 11:11:06 Done.
56 } 61 }
57 62
58 void forEach(void f(String name, List<String> values)) { 63 void forEach(void f(String name, List<String> values)) {
64 _syncContentType();
59 _headers.forEach(f); 65 _headers.forEach(f);
60 } 66 }
61 67
62 String get host() => _host; 68 String get host() => _host;
63 69
64 void set host(String host) { 70 void set host(String host) {
65 _checkMutable(); 71 _checkMutable();
66 _host = host; 72 _host = host;
67 _updateHostHeader(); 73 _updateHostHeader();
68 } 74 }
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after
108 } 114 }
109 115
110 void set expires(Date expires) { 116 void set expires(Date expires) {
111 _checkMutable(); 117 _checkMutable();
112 // Format "Expires" header with date in Greenwich Mean Time (GMT). 118 // Format "Expires" header with date in Greenwich Mean Time (GMT).
113 String formatted = 119 String formatted =
114 _HttpUtils.formatDate(expires.changeTimeZone(new TimeZone.utc())); 120 _HttpUtils.formatDate(expires.changeTimeZone(new TimeZone.utc()));
115 _set("expires", formatted); 121 _set("expires", formatted);
116 } 122 }
117 123
124 void get contentType() {
Anders Johnsen 2012/05/21 07:37:30 Would it be better/simpler to not have the field _
Søren Gjesse 2012/05/21 11:11:06 It would, however for code patterns like this ...
125 if (_contentType == null) {
126 var values = _headers["content-type"];
127 if (values != null) {
128 _contentType = new ContentType.fromString(values[0]);
129 } else {
130 _contentType = new ContentType();
131 }
132 }
133 return _contentType;
134 }
135
118 void _add(String name, Object value) { 136 void _add(String name, Object value) {
119 // TODO(sgjesse): Add immutable state throw HttpException is immutable. 137 // TODO(sgjesse): Add immutable state throw HttpException is immutable.
120 if (name.toLowerCase() == "date") { 138 if (name.toLowerCase() == "date") {
121 if (value is Date) { 139 if (value is Date) {
122 date = value; 140 date = value;
123 } else if (value is String) { 141 } else if (value is String) {
124 _set("date", value); 142 _set("date", value);
125 } else { 143 } else {
126 throw new HttpException("Unexpected type for header named $name"); 144 throw new HttpException("Unexpected type for header named $name");
127 } 145 }
(...skipping 20 matching lines...) Expand all
148 _port = HttpClient.DEFAULT_HTTP_PORT; 166 _port = HttpClient.DEFAULT_HTTP_PORT;
149 } else { 167 } else {
150 try { 168 try {
151 _port = Math.parseInt(value.substring(pos + 1)); 169 _port = Math.parseInt(value.substring(pos + 1));
152 } catch (BadNumberFormatException e) { 170 } catch (BadNumberFormatException e) {
153 _port = null; 171 _port = null;
154 } 172 }
155 } 173 }
156 _set("host", value); 174 _set("host", value);
157 } 175 }
176 } else if (name.toLowerCase() == "content-type") {
177 _set("content-type", value);
178 _contentType = null;
158 } else { 179 } else {
159 name = name.toLowerCase(); 180 name = name.toLowerCase();
160 List<String> values = _headers[name]; 181 List<String> values = _headers[name];
161 if (values == null) { 182 if (values == null) {
162 values = new List<String>(); 183 values = new List<String>();
163 _headers[name] = values; 184 _headers[name] = values;
164 } 185 }
165 values.add(value.toString()); 186 values.add(value.toString());
166 } 187 }
167 } 188 }
(...skipping 13 matching lines...) Expand all
181 bool defaultPort = _port == null || _port == HttpClient.DEFAULT_HTTP_PORT; 202 bool defaultPort = _port == null || _port == HttpClient.DEFAULT_HTTP_PORT;
182 String portPart = defaultPort ? "" : ":$_port"; 203 String portPart = defaultPort ? "" : ":$_port";
183 _set("host", "$host$portPart"); 204 _set("host", "$host$portPart");
184 } 205 }
185 206
186 _write(_HttpConnectionBase connection) { 207 _write(_HttpConnectionBase connection) {
187 final COLONSP = const [_CharCode.COLON, _CharCode.SP]; 208 final COLONSP = const [_CharCode.COLON, _CharCode.SP];
188 final COMMASP = const [_CharCode.COMMA, _CharCode.SP]; 209 final COMMASP = const [_CharCode.COMMA, _CharCode.SP];
189 final CRLF = const [_CharCode.CR, _CharCode.LF]; 210 final CRLF = const [_CharCode.CR, _CharCode.LF];
190 211
212 _syncContentType();
213
191 // Format headers. 214 // Format headers.
192 _headers.forEach((String name, List<String> values) { 215 _headers.forEach((String name, List<String> values) {
193 List<int> data; 216 List<int> data;
194 data = name.charCodes(); 217 data = name.charCodes();
195 connection._write(data); 218 connection._write(data);
196 connection._write(COLONSP); 219 connection._write(COLONSP);
197 for (int i = 0; i < values.length; i++) { 220 for (int i = 0; i < values.length; i++) {
198 if (i > 0) { 221 if (i > 0) {
199 connection._write(COMMASP); 222 connection._write(COMMASP);
200 } 223 }
201 data = values[i].charCodes(); 224 data = values[i].charCodes();
202 connection._write(data); 225 connection._write(data);
203 } 226 }
204 connection._write(CRLF); 227 connection._write(CRLF);
205 }); 228 });
206 } 229 }
207 230
231 _syncContentType() {
Mads Ager (google) 2012/05/21 07:40:39 Is there any reason to allow ContentType to be mut
Søren Gjesse 2012/05/21 11:11:06 As the HttpHeaders object is mutable it would be s
232 if (_contentType != null) {
233 _set("content-type", _contentType.toString());
234 }
235 }
236
208 String toString() { 237 String toString() {
238 _syncContentType();
209 StringBuffer sb = new StringBuffer(); 239 StringBuffer sb = new StringBuffer();
210 _headers.forEach((String name, List<String> values) { 240 _headers.forEach((String name, List<String> values) {
211 sb.add(name); 241 sb.add(name);
212 sb.add(": "); 242 sb.add(": ");
213 for (int i = 0; i < values.length; i++) { 243 for (int i = 0; i < values.length; i++) {
214 if (i > 0) { 244 if (i > 0) {
215 sb.add(", "); 245 sb.add(", ");
216 } 246 }
217 sb.add(values[i]); 247 sb.add(values[i]);
218 } 248 }
219 sb.add("\n"); 249 sb.add("\n");
220 }); 250 });
221 return sb.toString(); 251 return sb.toString();
222 } 252 }
223 253
224 bool _mutable = true; // Are the headers currently mutable? 254 bool _mutable = true; // Are the headers currently mutable?
225 Map<String, List<String>> _headers; 255 Map<String, List<String>> _headers;
226 256
227 String _host; 257 String _host;
228 int _port; 258 int _port;
259 ContentType _contentType;
229 } 260 }
230 261
231 262
263 class _HeaderValue implements HeaderValue {
264 _HeaderValue([String this.value = ""]);
265
266 _HeaderValue.fromString(String value) {
267 // Parse the string.
268 _parse(value);
269 }
270
271 Map<String, String> get parameters() {
272 if (_parameters == null) _parameters = new Map<String, String>();
273 return _parameters;
274 }
275
276 String toString() {
277 StringBuffer sb = new StringBuffer();
278 sb.add(value);
279 if (parameters != null && parameters.length > 0) {
280 _parameters.forEach((String name, String value) {
281 sb.add("; ");
282 sb.add(name);
283 sb.add("=");
284 sb.add(value);
285 });
286 }
287 return sb.toString();
288 }
289
290 void _parse(String s) {
291 int index = 0;
292
293 void skipWS() {
294 while (index < s.length) {
Anders Johnsen 2012/05/21 07:37:30 !done()
Søren Gjesse 2012/05/21 11:11:06 Done.
295 if (s[index] != " " && s[index] != "\t") return;
296 index++;
297 }
298 }
299
300 String parseValue() {
301 int start = index;
302 while (index < s.length) {
Anders Johnsen 2012/05/21 07:37:30 !done()
Søren Gjesse 2012/05/21 11:11:06 Done.
303 if (s[index] == " " || s[index] == "\t" || s[index] == ";") break;
304 index++;
305 }
306 return s.substring(start, index).toLowerCase();
307 }
308
309 void expect(String expected) {
310 if (index == s.length) throw new HttpException("YYY");
Anders Johnsen 2012/05/21 07:37:30 Some real string here :)
Anders Johnsen 2012/05/21 07:37:30 if(done()) ?
Mads Ager (google) 2012/05/21 07:40:39 More useful error message?
Søren Gjesse 2012/05/21 11:11:06 Done.
Søren Gjesse 2012/05/21 11:11:06 Done.
Søren Gjesse 2012/05/21 11:11:06 Done.
311 if (s[index] != expected) throw new HttpException("XXX $expected [${s[inde x]}]");
Anders Johnsen 2012/05/21 07:37:30 Ditto.
Mads Ager (google) 2012/05/21 07:40:39 Ditto and a long line.
Søren Gjesse 2012/05/21 11:11:06 Done.
Søren Gjesse 2012/05/21 11:11:06 Done and done.
312 index++;
313 }
314
315 bool done() => index == s.length;
316
317 void parseParameters() {
318 _parameters = new Map<String, String>();
319
320 String parseParameterName() {
321 int start = index;
322 while (index < s.length) {
Anders Johnsen 2012/05/21 07:37:30 !done()
Søren Gjesse 2012/05/21 11:11:06 Done.
323 if (s[index] == " " || s[index] == "\t" || s[index] == "=") break;
324 index++;
325 }
326 return s.substring(start, index).toLowerCase();
327 }
328
329 String parseParameterValue() {
330 if (s[index] == "\"") {
331 // Parse quoted value.
332 StringBuffer sb = new StringBuffer();
333 index++;
334 while (index < s.length) {
Anders Johnsen 2012/05/21 07:37:30 Maybe also !done() here, but that's 100% up to you
Søren Gjesse 2012/05/21 11:11:06 Done.
335 if (s[index] == "\\") {
Anders Johnsen 2012/05/21 07:37:30 Are we actually handling this correctly? Should we
Mads Ager (google) 2012/05/21 07:40:39 Does the backslash have no special meaning? It doe
Søren Gjesse 2012/05/21 11:11:06 According to the spec the backslash escapes the fo
Søren Gjesse 2012/05/21 11:11:06 See other comment.
336 if (index + 1 == s.length) throw new HttpException("ZZZ");
Anders Johnsen 2012/05/21 07:37:30 Real string.
Mads Ager (google) 2012/05/21 07:40:39 Update message.
Søren Gjesse 2012/05/21 11:11:06 Done.
Søren Gjesse 2012/05/21 11:11:06 Done.
337 index++;
338 } else if (s[index] == "\"") {
339 index++;
340 break;
341 }
342 sb.add(s[index]);
343 index++;
344 }
345 return sb.toString();
346 } else {
347 // Parse non-quoted value.
Anders Johnsen 2012/05/21 07:37:30 Change this to 'return parseValue()'?
Søren Gjesse 2012/05/21 11:11:06 Done.
348 int start = index;
349 while (index < s.length) {
350 if (s[index] == " " || s[index] == "\t" || s[index] == ";") break;
351 index++;
352 }
353 return s.substring(start, index).toLowerCase();
354 }
355 }
356
357 while (!done()) {
358 skipWS();
359 if (done()) return;
360 String name = parseParameterName();
361 skipWS();
362 expect("=");
363 skipWS();
364 String value = parseParameterValue();
365 _parameters[name] = value;
366 skipWS();
367 if (done()) return;
368 expect(";");
369 }
370 }
371
372 skipWS();
373 value = parseValue();
374 skipWS();
375 if (done()) return;
376 expect(";");
377 parseParameters();
378 }
379
380 String value;
381 Map<String, String> _parameters;
382 }
383
384
385 class _ContentType extends _HeaderValue implements ContentType {
386 _ContentType([String this._primaryType = "", String this._subType = ""]);
387
388 _ContentType.fromString(String value) {
Anders Johnsen 2012/05/21 07:37:30 : super.fromString(value); ?
Søren Gjesse 2012/05/21 11:11:06 Done.
389 // Parse the string.
390 _parse(value);
391 }
392
393 String get value() => "$_primaryType/$_subType";
394
395 void set value(String s) {
396 int index = s.indexOf("/");
397 if (index == -1 || s.length == index - 1) {
Anders Johnsen 2012/05/21 07:37:30 Did you mean 'index == s.length - 1'?
Søren Gjesse 2012/05/21 11:11:06 No, this is either not found or last char.
Anders Johnsen 2012/05/21 11:27:03 But the last char is the case where index is s.len
Søren Gjesse 2012/05/22 12:47:16 My brain meltdown - sorry for being stupid.
398 primaryType = s;
Mads Ager (google) 2012/05/21 07:40:39 s.trim().toLowerCase()?
Søren Gjesse 2012/05/21 11:11:06 Done.
399 } else {
400 primaryType = s.substring(0, index).trim().toLowerCase();
401 subType = s.substring(index + 1).trim().toLowerCase();
402 }
403 }
404
Mads Ager (google) 2012/05/21 07:40:39 Extra space
Søren Gjesse 2012/05/21 11:11:06 Done.
405
406 String get primaryType() => _primaryType;
407
408 void set primaryType(String s) {
409 _primaryType = s;
410 }
411
412 String get subType() => _subType;
413
414 void set subType(String s) {
415 _subType = s;
416 }
417
418 String get charset() => parameters["charset"];
419
420 void set charset(String s) {
421 parameters["charset"] = s;
422 }
423
424 String _primaryType = "";
425 String _subType = "";
426 }
427
428
232 class _HttpRequestResponseBase { 429 class _HttpRequestResponseBase {
233 final int START = 0; 430 final int START = 0;
234 final int HEADER_SENT = 1; 431 final int HEADER_SENT = 1;
235 final int DONE = 2; 432 final int DONE = 2;
236 final int UPGRADED = 3; 433 final int UPGRADED = 3;
237 434
238 _HttpRequestResponseBase(_HttpConnectionBase this._httpConnection) 435 _HttpRequestResponseBase(_HttpConnectionBase this._httpConnection)
239 : _headers = new _HttpHeaders() { 436 : _headers = new _HttpHeaders() {
240 _state = START; 437 _state = START;
241 } 438 }
(...skipping 1405 matching lines...) Expand 10 before | Expand all | Expand 10 after
1647 1844
1648 1845
1649 class _RedirectInfo implements RedirectInfo { 1846 class _RedirectInfo implements RedirectInfo {
1650 const _RedirectInfo(int this.statusCode, 1847 const _RedirectInfo(int this.statusCode,
1651 String this.method, 1848 String this.method,
1652 Uri this.location); 1849 Uri this.location);
1653 final int statusCode; 1850 final int statusCode;
1654 final String method; 1851 final String method;
1655 final Uri location; 1852 final Uri location;
1656 } 1853 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698