OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 } |
OLD | NEW |