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

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: Addressed review comments 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 _syncHeaderValuesCache(name);
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);
38 } 40 }
39 41
40 void remove(String name, Object value) { 42 void remove(String name, Object value) {
41 _checkMutable(); 43 _checkMutable();
42 name = name.toLowerCase(); 44 name = name.toLowerCase();
43 List<String> values = _headers[name]; 45 List<String> values = _headers[name];
44 if (values != null) { 46 if (values != null) {
45 int index = values.indexOf(value); 47 int index = values.indexOf(value);
46 if (index != -1) { 48 if (index != -1) {
47 values.removeRange(index, 1); 49 values.removeRange(index, 1);
48 } 50 }
49 } 51 }
52 _clearHeaderValueCache(name);
50 } 53 }
51 54
52 void removeAll(String name) { 55 void removeAll(String name) {
53 _checkMutable(); 56 _checkMutable();
54 name = name.toLowerCase(); 57 name = name.toLowerCase();
55 _headers.remove(name); 58 _headers.remove(name);
59 _clearHeaderValueCache(name);
56 } 60 }
57 61
58 void forEach(void f(String name, List<String> values)) { 62 void forEach(void f(String name, List<String> values)) {
63 _syncHeaderValuesCache();
59 _headers.forEach(f); 64 _headers.forEach(f);
60 } 65 }
61 66
62 String get host() => _host; 67 String get host() => _host;
63 68
64 void set host(String host) { 69 void set host(String host) {
65 _checkMutable(); 70 _checkMutable();
66 _host = host; 71 _host = host;
67 _updateHostHeader(); 72 _updateHostHeader();
68 } 73 }
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after
108 } 113 }
109 114
110 void set expires(Date expires) { 115 void set expires(Date expires) {
111 _checkMutable(); 116 _checkMutable();
112 // Format "Expires" header with date in Greenwich Mean Time (GMT). 117 // Format "Expires" header with date in Greenwich Mean Time (GMT).
113 String formatted = 118 String formatted =
114 _HttpUtils.formatDate(expires.changeTimeZone(new TimeZone.utc())); 119 _HttpUtils.formatDate(expires.changeTimeZone(new TimeZone.utc()));
115 _set("expires", formatted); 120 _set("expires", formatted);
116 } 121 }
117 122
123 void get contentType() {
124 if (_contentType == null) {
125 var values = _headers["content-type"];
126 if (values != null) {
127 _contentType = new ContentType.fromString(values[0]);
128 } else {
129 _contentType = new ContentType();
130 }
131 }
132 return _contentType;
133 }
134
118 void _add(String name, Object value) { 135 void _add(String name, Object value) {
119 // TODO(sgjesse): Add immutable state throw HttpException is immutable. 136 // TODO(sgjesse): Add immutable state throw HttpException is immutable.
120 if (name.toLowerCase() == "date") { 137 if (name.toLowerCase() == "date") {
121 if (value is Date) { 138 if (value is Date) {
122 date = value; 139 date = value;
123 } else if (value is String) { 140 } else if (value is String) {
124 _set("date", value); 141 _set("date", value);
125 } else { 142 } else {
126 throw new HttpException("Unexpected type for header named $name"); 143 throw new HttpException("Unexpected type for header named $name");
127 } 144 }
(...skipping 20 matching lines...) Expand all
148 _port = HttpClient.DEFAULT_HTTP_PORT; 165 _port = HttpClient.DEFAULT_HTTP_PORT;
149 } else { 166 } else {
150 try { 167 try {
151 _port = Math.parseInt(value.substring(pos + 1)); 168 _port = Math.parseInt(value.substring(pos + 1));
152 } catch (BadNumberFormatException e) { 169 } catch (BadNumberFormatException e) {
153 _port = null; 170 _port = null;
154 } 171 }
155 } 172 }
156 _set("host", value); 173 _set("host", value);
157 } 174 }
175 } else if (name.toLowerCase() == "content-type") {
176 _set("content-type", value);
177 _contentType = null;
158 } else { 178 } else {
159 name = name.toLowerCase(); 179 name = name.toLowerCase();
160 List<String> values = _headers[name]; 180 List<String> values = _headers[name];
161 if (values == null) { 181 if (values == null) {
162 values = new List<String>(); 182 values = new List<String>();
163 _headers[name] = values; 183 _headers[name] = values;
164 } 184 }
165 values.add(value.toString()); 185 values.add(value.toString());
166 } 186 }
167 } 187 }
168 188
169 void _set(String name, String value) { 189 void _set(String name, String value) {
170 name = name.toLowerCase(); 190 name = name.toLowerCase();
171 List<String> values = new List<String>(); 191 List<String> values = new List<String>();
172 _headers[name] = values; 192 _headers[name] = values;
173 values.add(value); 193 values.add(value);
174 } 194 }
175 195
176 _checkMutable() { 196 _checkMutable() {
177 if (!_mutable) throw new HttpException("HTTP headers are not mutable"); 197 if (!_mutable) throw new HttpException("HTTP headers are not mutable");
178 } 198 }
179 199
180 _updateHostHeader() { 200 _updateHostHeader() {
181 bool defaultPort = _port == null || _port == HttpClient.DEFAULT_HTTP_PORT; 201 bool defaultPort = _port == null || _port == HttpClient.DEFAULT_HTTP_PORT;
182 String portPart = defaultPort ? "" : ":$_port"; 202 String portPart = defaultPort ? "" : ":$_port";
183 _set("host", "$host$portPart"); 203 _set("host", "$host$portPart");
184 } 204 }
185 205
206 void _syncHeaderValuesCache([String name]) {
207 if (name == null) {
208 if (_contentType != null) {
209 _set("content-type", _contentType.toString());
210 }
211 } else if (name == "content-type" && _contentType != null) {
212 _set("content-type", _contentType.toString());
213 }
214 }
215
216 void _clearHeaderValueCache([String name]) {
217 if (name == null) {
218 _contentType = null;
219 } else if (name == "content-type") {
220 _contentType = null;
221 }
222 }
223
186 _write(_HttpConnectionBase connection) { 224 _write(_HttpConnectionBase connection) {
187 final COLONSP = const [_CharCode.COLON, _CharCode.SP]; 225 final COLONSP = const [_CharCode.COLON, _CharCode.SP];
188 final COMMASP = const [_CharCode.COMMA, _CharCode.SP]; 226 final COMMASP = const [_CharCode.COMMA, _CharCode.SP];
189 final CRLF = const [_CharCode.CR, _CharCode.LF]; 227 final CRLF = const [_CharCode.CR, _CharCode.LF];
190 228
229 _syncHeaderValuesCache();
230
191 // Format headers. 231 // Format headers.
192 _headers.forEach((String name, List<String> values) { 232 _headers.forEach((String name, List<String> values) {
193 List<int> data; 233 List<int> data;
194 data = name.charCodes(); 234 data = name.charCodes();
195 connection._write(data); 235 connection._write(data);
196 connection._write(COLONSP); 236 connection._write(COLONSP);
197 for (int i = 0; i < values.length; i++) { 237 for (int i = 0; i < values.length; i++) {
198 if (i > 0) { 238 if (i > 0) {
199 connection._write(COMMASP); 239 connection._write(COMMASP);
200 } 240 }
201 data = values[i].charCodes(); 241 data = values[i].charCodes();
202 connection._write(data); 242 connection._write(data);
203 } 243 }
204 connection._write(CRLF); 244 connection._write(CRLF);
205 }); 245 });
206 } 246 }
207 247
208 String toString() { 248 String toString() {
249 _syncHeaderValuesCache();
209 StringBuffer sb = new StringBuffer(); 250 StringBuffer sb = new StringBuffer();
210 _headers.forEach((String name, List<String> values) { 251 _headers.forEach((String name, List<String> values) {
211 sb.add(name); 252 sb.add(name);
212 sb.add(": "); 253 sb.add(": ");
213 for (int i = 0; i < values.length; i++) { 254 for (int i = 0; i < values.length; i++) {
214 if (i > 0) { 255 if (i > 0) {
215 sb.add(", "); 256 sb.add(", ");
216 } 257 }
217 sb.add(values[i]); 258 sb.add(values[i]);
218 } 259 }
219 sb.add("\n"); 260 sb.add("\n");
220 }); 261 });
221 return sb.toString(); 262 return sb.toString();
222 } 263 }
223 264
224 bool _mutable = true; // Are the headers currently mutable? 265 bool _mutable = true; // Are the headers currently mutable?
225 Map<String, List<String>> _headers; 266 Map<String, List<String>> _headers;
226 267
227 String _host; 268 String _host;
228 int _port; 269 int _port;
270 ContentType _contentType;
229 } 271 }
230 272
231 273
274 class _HeaderValue implements HeaderValue {
275 _HeaderValue([String this.value = ""]);
276
277 _HeaderValue.fromString(String value) {
278 // Parse the string.
279 _parse(value);
280 }
281
282 Map<String, String> get parameters() {
283 if (_parameters == null) _parameters = new Map<String, String>();
284 return _parameters;
285 }
286
287 String toString() {
288 StringBuffer sb = new StringBuffer();
289 sb.add(value);
290 if (parameters != null && parameters.length > 0) {
291 _parameters.forEach((String name, String value) {
292 sb.add("; ");
293 sb.add(name);
294 sb.add("=");
295 sb.add(value);
296 });
297 }
298 return sb.toString();
299 }
300
301 void _parse(String s) {
302 int index = 0;
303
304 bool done() => index == s.length;
305
306 void skipWS() {
307 while (!done()) {
308 if (s[index] != " " && s[index] != "\t") return;
309 index++;
310 }
311 }
312
313 String parseValue() {
314 int start = index;
315 while (!done()) {
316 if (s[index] == " " || s[index] == "\t" || s[index] == ";") break;
317 index++;
318 }
319 return s.substring(start, index).toLowerCase();
320 }
321
322 void expect(String expected) {
323 if (done()) throw new HttpException("Failed to parse header value [$s]");
324 if (s[index] != expected) {
325 throw new HttpException("Failed to parse header value [$s]");
326 }
327 index++;
328 }
329
330 void parseParameters() {
331 _parameters = new Map<String, String>();
332
333 String parseParameterName() {
334 int start = index;
335 while (!done()) {
336 if (s[index] == " " || s[index] == "\t" || s[index] == "=") break;
337 index++;
338 }
339 return s.substring(start, index).toLowerCase();
340 }
341
342 String parseParameterValue() {
343 if (s[index] == "\"") {
344 // Parse quoted value.
345 StringBuffer sb = new StringBuffer();
346 index++;
347 while (!done()) {
348 if (s[index] == "\\") {
349 if (index + 1 == s.length) {
350 throw new HttpException("Failed to parse header value [$s]");
351 }
352 index++;
353 } else if (s[index] == "\"") {
354 index++;
355 break;
356 }
357 sb.add(s[index]);
358 index++;
359 }
360 return sb.toString();
361 } else {
362 // Parse non-quoted value.
363 return parseValue();
364 }
365 }
366
367 while (!done()) {
368 skipWS();
369 if (done()) return;
370 String name = parseParameterName();
371 skipWS();
372 expect("=");
373 skipWS();
374 String value = parseParameterValue();
375 _parameters[name] = value;
376 skipWS();
377 if (done()) return;
378 expect(";");
379 }
380 }
381
382 skipWS();
383 value = parseValue();
384 skipWS();
385 if (done()) return;
386 expect(";");
387 parseParameters();
388 }
389
390 String value;
391 Map<String, String> _parameters;
392 }
393
394
395 class _ContentType extends _HeaderValue implements ContentType {
396 _ContentType([String this._primaryType = "", String this._subType = ""]);
397
398 _ContentType.fromString(String value) : super.fromString(value);
399
400 String get value() => "$_primaryType/$_subType";
401
402 void set value(String s) {
403 int index = s.indexOf("/");
404 if (index == -1 || s.length == index - 1) {
Mads Ager (google) 2012/05/21 11:49:33 I agree with Anders about this being: index == (s.
Søren Gjesse 2012/05/22 12:47:16 Of cause you are both right.
405 primaryType = s.trim().toLowerCase();
406 } else {
407 primaryType = s.substring(0, index).trim().toLowerCase();
408 subType = s.substring(index + 1).trim().toLowerCase();
409 }
410 }
411
412 String get primaryType() => _primaryType;
413
414 void set primaryType(String s) {
415 _primaryType = s;
416 }
417
418 String get subType() => _subType;
419
420 void set subType(String s) {
421 _subType = s;
422 }
423
424 String get charset() => parameters["charset"];
425
426 void set charset(String s) {
427 parameters["charset"] = s;
428 }
429
430 String _primaryType = "";
431 String _subType = "";
432 }
433
434
232 class _HttpRequestResponseBase { 435 class _HttpRequestResponseBase {
233 final int START = 0; 436 final int START = 0;
234 final int HEADER_SENT = 1; 437 final int HEADER_SENT = 1;
235 final int DONE = 2; 438 final int DONE = 2;
236 final int UPGRADED = 3; 439 final int UPGRADED = 3;
237 440
238 _HttpRequestResponseBase(_HttpConnectionBase this._httpConnection) 441 _HttpRequestResponseBase(_HttpConnectionBase this._httpConnection)
239 : _headers = new _HttpHeaders() { 442 : _headers = new _HttpHeaders() {
240 _state = START; 443 _state = START;
241 } 444 }
(...skipping 1405 matching lines...) Expand 10 before | Expand all | Expand 10 after
1647 1850
1648 1851
1649 class _RedirectInfo implements RedirectInfo { 1852 class _RedirectInfo implements RedirectInfo {
1650 const _RedirectInfo(int this.statusCode, 1853 const _RedirectInfo(int this.statusCode,
1651 String this.method, 1854 String this.method,
1652 Uri this.location); 1855 Uri this.location);
1653 final int statusCode; 1856 final int statusCode;
1654 final String method; 1857 final String method;
1655 final Uri location; 1858 final Uri location;
1656 } 1859 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698