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

Side by Side Diff: pkg/http_server/test/http_multipart_test.dart

Issue 730203008: Don't to do HTML entity decoding for multi-part form fields (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Created 6 years, 1 month 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) 2013, the Dart project authors. Please see the AUTHORS file 1 // Copyright (c) 2013, 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 import "package:http_server/http_server.dart"; 5 import "package:http_server/http_server.dart";
6 import "package:mime/mime.dart"; 6 import "package:mime/mime.dart";
7 import "package:unittest/unittest.dart"; 7 import "package:unittest/unittest.dart";
8 import 'dart:async'; 8 import 'dart:async';
9 import 'dart:io'; 9 import 'dart:io';
10 import 'dart:convert'; 10 import 'dart:convert';
11 11
12 class FormField { 12 class FormField {
13 final String name; 13 final String name;
14 final value; 14 final value;
kustermann 2014/11/18 16:34:20 Document that 'value' is either String or List<int
Søren Gjesse 2014/11/19 07:41:05 Done.
15 final String contentType; 15 final String contentType;
16 final String filename; 16 final String filename;
17 17
18 FormField(String this.name, 18 FormField(String this.name,
19 this.value, 19 this.value,
20 {String this.contentType, 20 {String this.contentType,
21 String this.filename}); 21 String this.filename});
22 22
23 bool operator==(other) { 23 bool operator==(other) {
24 if (value.length != other.value.length) return false; 24 if (value.length != other.value.length) return false;
25 for (int i = 0; i < value.length; i++) { 25 for (int i = 0; i < value.length; i++) {
26 if (value[i] != other.value[i]) { 26 if (value[i] != other.value[i]) {
27 return false; 27 return false;
28 } 28 }
29 } 29 }
30 return name == other.name && 30 return name == other.name &&
31 contentType == other.contentType && 31 contentType == other.contentType &&
32 filename == other.filename; 32 filename == other.filename;
33 } 33 }
kustermann 2014/11/18 16:34:20 If this one overrides operator== it should also ov
Søren Gjesse 2014/11/19 07:41:05 Oops. Just used name.hashCode.
34 34
35 String toString() { 35 String toString() {
36 return "FormField('$name', '$value', '$contentType', '$filename')"; 36 return "FormField('$name', '$value', '$contentType', '$filename')";
37 } 37 }
38 } 38 }
39 39
40 void postDataTest(List<int> message, 40 void postDataTest(List<int> message,
41 String contentType, 41 String contentType,
42 String boundary, 42 String boundary,
43 List<FormField> expectedFields) { 43 List<FormField> expectedFields,
44 {defaultEncoding: LATIN1}) {
44 HttpServer.bind("127.0.0.1", 0).then((server) { 45 HttpServer.bind("127.0.0.1", 0).then((server) {
45 server.listen((request) { 46 server.listen((request) {
46 String boundary = request.headers.contentType.parameters['boundary']; 47 String boundary = request.headers.contentType.parameters['boundary'];
47 request 48 request
48 .transform(new MimeMultipartTransformer(boundary)) 49 .transform(new MimeMultipartTransformer(boundary))
49 .map((part) => HttpMultipartFormData.parse( 50 .map((part) => HttpMultipartFormData.parse(
50 part, defaultEncoding: LATIN1)) 51 part, defaultEncoding: defaultEncoding))
51 .map((multipart) { 52 .map((multipart) {
52 var future; 53 var future;
53 if (multipart.isText) { 54 if (multipart.isText) {
54 future = multipart 55 future = multipart
55 .fold(new StringBuffer(), (b, s) => b..write(s)) 56 .fold(new StringBuffer(), (b, s) => b..write(s))
56 .then((b) => b.toString()); 57 .then((b) => b.toString());
kustermann 2014/11/18 16:34:20 Maybe just do a future = multipart.join('') here
Søren Gjesse 2014/11/19 07:41:05 Done.
57 } else { 58 } else {
58 future = multipart 59 future = multipart
59 .fold([], (b, s) => b..addAll(s)); 60 .fold([], (b, s) => b..addAll(s));
kustermann 2014/11/18 16:34:20 future = multipart.toList()
Søren Gjesse 2014/11/19 07:41:05 toList does not have the addAll semantics.
60 } 61 }
61 return future 62 return future
62 .then((data) { 63 .then((data) {
63 String contentType; 64 String contentType;
64 if (multipart.contentType != null) { 65 if (multipart.contentType != null) {
65 contentType = multipart.contentType.mimeType; 66 contentType = multipart.contentType.mimeType;
66 } 67 }
67 return new FormField( 68 return new FormField(
68 multipart.contentDisposition.parameters['name'], 69 multipart.contentDisposition.parameters['name'],
69 data, 70 data,
70 contentType: contentType, 71 contentType: contentType,
71 filename: 72 filename:
72 multipart.contentDisposition.parameters['filename']); 73 multipart.contentDisposition.parameters['filename']);
73 }); 74 });
74 }) 75 })
75 .fold([], (l, f) => l..add(f)) 76 .fold([], (l, f) => l..add(f))
kustermann 2014/11/18 16:34:20 Just do a '.toList()' instead ?
Søren Gjesse 2014/11/19 07:41:05 Done.
76 .then(Future.wait) 77 .then(Future.wait)
77 .then((fields) { 78 .then((fields) {
78 expect(fields, equals(expectedFields)); 79 expect(fields, equals(expectedFields));
79 request.response.close().then((_) => server.close()); 80 request.response.close().then((_) => server.close());
80 }); 81 });
81 }); 82 });
82 var client = new HttpClient(); 83 var client = new HttpClient();
83 client.post('127.0.0.1', server.port, '/') 84 client.post('127.0.0.1', server.port, '/')
84 .then((request) { 85 .then((request) {
85 request.headers.set('content-type', 86 request.headers.set('content-type',
(...skipping 93 matching lines...) Expand 10 before | Expand all | Expand 10 after
179 180
180 postDataTest(message, 181 postDataTest(message,
181 'multipart/form-data', 182 'multipart/form-data',
182 '----WebKitFormBoundaryQSql8kDALM7tAkC1', 183 '----WebKitFormBoundaryQSql8kDALM7tAkC1',
183 [new FormField('submit-name', 'Test'), 184 [new FormField('submit-name', 'Test'),
184 new FormField('files', 185 new FormField('files',
185 data, 186 data,
186 contentType: 'application/octet-stream', 187 contentType: 'application/octet-stream',
187 filename: 'VERSION')]); 188 filename: 'VERSION')]);
188 189
190 // In Chrome, Safari and Firefox HTML entity encoding might be used for
191 // values in form fields. The HTML entity encoding for ひらがな is
192 // &#12402;&#12425;&#12364;&#12394;
189 message = [ 193 message = [
190 45, 45, 45, 45, 45, 45, 87, 101, 98, 75, 105, 116, 70, 111, 114, 109, 66, 194 45, 45, 45, 45, 45, 45, 87, 101, 98, 75, 105, 116, 70, 111, 114, 109, 66,
191 111, 117, 110, 100, 97, 114, 121, 118, 65, 86, 122, 117, 103, 75, 77, 116, 195 111, 117, 110, 100, 97, 114, 121, 118, 65, 86, 122, 117, 103, 75, 77, 116,
192 90, 98, 121, 87, 111, 66, 71, 13, 10, 67, 111, 110, 116, 101, 110, 116, 196 90, 98, 121, 87, 111, 66, 71, 13, 10, 67, 111, 110, 116, 101, 110, 116,
193 45, 68, 105, 115, 112, 111, 115, 105, 116, 105, 111, 110, 58, 32, 102, 197 45, 68, 105, 115, 112, 111, 115, 105, 116, 105, 111, 110, 58, 32, 102,
194 111, 114, 109, 45, 100, 97, 116, 97, 59, 32, 110, 97, 109, 101, 61, 34, 198 111, 114, 109, 45, 100, 97, 116, 97, 59, 32, 110, 97, 109, 101, 61, 34,
195 110, 97, 109, 101, 34, 13, 10, 13, 10, 38, 35, 49, 50, 52, 48, 50, 59, 38, 199 110, 97, 109, 101, 34, 13, 10, 13, 10, 38, 35, 49, 50, 52, 48, 50, 59, 38,
196 35, 49, 50, 52, 50, 53, 59, 38, 35, 49, 50, 51, 54, 52, 59, 38, 35, 49, 200 35, 49, 50, 52, 50, 53, 59, 38, 35, 49, 50, 51, 54, 52, 59, 38, 35, 49,
197 50, 51, 57, 52, 59, 13, 10, 45, 45, 45, 45, 45, 45, 87, 101, 98, 75, 105, 201 50, 51, 57, 52, 59, 13, 10, 45, 45, 45, 45, 45, 45, 87, 101, 98, 75, 105,
198 116, 70, 111, 114, 109, 66, 111, 117, 110, 100, 97, 114, 121, 118, 65, 86, 202 116, 70, 111, 114, 109, 66, 111, 117, 110, 100, 97, 114, 121, 118, 65, 86,
199 122, 117, 103, 75, 77, 116, 90, 98, 121, 87, 111, 66, 71, 45, 45, 13, 10]; 203 122, 117, 103, 75, 77, 116, 90, 98, 121, 87, 111, 66, 71, 45, 45, 13, 10];
200 204
201 postDataTest(message, 205 postDataTest(message,
202 'multipart/form-data', 206 'multipart/form-data',
203 '----WebKitFormBoundaryvAVzugKMtZbyWoBG', 207 '----WebKitFormBoundaryvAVzugKMtZbyWoBG',
204 [new FormField('name', 'ひらがな')]); 208 [new FormField('name', '&#12402;&#12425;&#12364;&#12394;')],
209 defaultEncoding: UTF8);
210
211 // The UTF-8 encoding of ひらがな is
212 // [227, 129, 178, 227, 130, 137, 227, 129, 140, 227, 129, 170].
213 message = [
214 45, 45, 45, 45, 45, 45, 87, 101, 98, 75, 105, 116, 70, 111, 114, 109, 66,
215 111, 117, 110, 100, 97, 114, 121, 71, 88, 116, 66, 114, 99, 106, 120, 104,
216 101, 75, 101, 78, 54, 105, 48, 13, 10, 67, 111, 110, 116, 101, 110, 116,
217 45, 68, 105, 115, 112, 111, 115, 105, 116, 105, 111, 110, 58, 32, 102,
218 111, 114, 109, 45, 100, 97, 116, 97, 59, 32, 110, 97, 109, 101, 61, 34,
219 116, 101, 115, 116, 34, 13, 10, 13, 10, 227, 129, 178, 227, 130, 137, 227,
220 129, 140, 227, 129, 170, 13, 10, 45, 45, 45, 45, 45, 45, 87, 101, 98, 75,
221 105, 116, 70, 111, 114, 109, 66, 111, 117, 110, 100, 97, 114, 121, 71, 88,
222 116, 66, 114, 99, 106, 120, 104, 101, 75, 101, 78, 54, 105, 48, 45, 45, 13 ,
kustermann 2014/11/18 16:34:20 long line
Søren Gjesse 2014/11/19 07:41:05 Done.
223 10];
224
225 postDataTest(message,
226 'multipart/form-data',
227 '----WebKitFormBoundaryGXtBrcjxheKeN6i0',
228 [new FormField('test', 'ひらがな')],
229 defaultEncoding: UTF8);
205 230
206 message = [ 231 message = [
207 45, 45, 45, 45, 45, 45, 87, 101, 98, 75, 105, 116, 70, 111, 114, 109, 66, 232 45, 45, 45, 45, 45, 45, 87, 101, 98, 75, 105, 116, 70, 111, 114, 109, 66,
208 111, 117, 110, 100, 97, 114, 121, 102, 101, 48, 69, 122, 86, 49, 97, 78, 233 111, 117, 110, 100, 97, 114, 121, 102, 101, 48, 69, 122, 86, 49, 97, 78,
209 121, 115, 68, 49, 98, 80, 104, 13, 10, 67, 111, 110, 116, 101, 110, 116, 234 121, 115, 68, 49, 98, 80, 104, 13, 10, 67, 111, 110, 116, 101, 110, 116,
210 45, 68, 105, 115, 112, 111, 115, 105, 116, 105, 111, 110, 58, 32, 102, 235 45, 68, 105, 115, 112, 111, 115, 105, 116, 105, 111, 110, 58, 32, 102,
211 111, 114, 109, 45, 100, 97, 116, 97, 59, 32, 110, 97, 109, 101, 61, 34, 236 111, 114, 109, 45, 100, 97, 116, 97, 59, 32, 110, 97, 109, 101, 61, 34,
212 110, 97, 109, 101, 34, 13, 10, 13, 10, 248, 118, 13, 10, 45, 45, 45, 45, 237 110, 97, 109, 101, 34, 13, 10, 13, 10, 248, 118, 13, 10, 45, 45, 45, 45,
213 45, 45, 87, 101, 98, 75, 105, 116, 70, 111, 114, 109, 66, 111, 117, 110, 238 45, 45, 87, 101, 98, 75, 105, 116, 70, 111, 114, 109, 66, 111, 117, 110,
214 100, 97, 114, 121, 102, 101, 48, 69, 122, 86, 49, 97, 78, 121, 115, 68, 239 100, 97, 114, 121, 102, 101, 48, 69, 122, 86, 49, 97, 78, 121, 115, 68,
215 49, 98, 80, 104, 45, 45, 13, 10]; 240 49, 98, 80, 104, 45, 45, 13, 10];
216 241
217 postDataTest(message, 242 postDataTest(message,
218 'multipart/form-data', 243 'multipart/form-data',
219 '----WebKitFormBoundaryfe0EzV1aNysD1bPh', 244 '----WebKitFormBoundaryfe0EzV1aNysD1bPh',
220 [new FormField('name', 'øv')]); 245 [new FormField('name', 'øv')]);
221 } 246 }
222 247
223 248
224 void main() { 249 void main() {
225 testEmptyPostData(); 250 testEmptyPostData();
226 testPostData(); 251 testPostData();
227 } 252 }
OLDNEW
« pkg/http_server/lib/src/http_multipart_form_data_impl.dart ('K') | « pkg/http_server/pubspec.yaml ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698