2 * Copyright (C) 1999 Lars Knoll (knoll@kde.org)
3 * (C) 1999 Antti Koivisto (koivisto@kde.org)
4 * (C) 2001 Dirk Mueller (mueller@kde.org)
5 * Copyright (C) 2004, 2005, 2006, 2007, 2008 Apple Inc. All rights reserved.
6 * (C) 2006 Alexey Proskuryakov (ap@nypop.com)
7 * Copyright (C) 2008 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.com/)
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Library General Public
11 * License as published by the Free Software Foundation; either
12 * version 2 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Library General Public License for more details.
19 * You should have received a copy of the GNU Library General Public License
20 * along with this library; see the file COPYING.LIB. If not, write to
21 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
22 * Boston, MA 02110-1301, USA.
26 #include "platform/network/FormDataEncoder.h"
28 #include "wtf/CryptographicallyRandomNumber.h"
29 #include "wtf/HexNumber.h"
30 #include "wtf/text/CString.h"
31 #include "wtf/text/TextEncoding.h"
37 static inline void append(Vector
<char>& buffer
, char string
)
39 buffer
.append(string
);
42 static inline void append(Vector
<char>& buffer
, const char* string
)
44 buffer
.append(string
, strlen(string
));
47 static inline void append(Vector
<char>& buffer
, const CString
& string
)
49 buffer
.append(string
.data(), string
.length());
52 static void appendQuotedString(Vector
<char>& buffer
, const CString
& string
)
54 // Append a string as a quoted value, escaping quotes and line breaks.
55 // FIXME: Is it correct to use percent escaping here? Other browsers do not encode these characters yet,
56 // so we should test popular servers to find out if there is an encoding form they can handle.
57 size_t length
= string
.length();
58 for (size_t i
= 0; i
< length
; ++i
) {
59 char c
= string
.data()[i
];
63 append(buffer
, "%0A");
66 append(buffer
, "%0D");
69 append(buffer
, "%22");
77 WTF::TextEncoding
FormDataEncoder::encodingFromAcceptCharset(const String
& acceptCharset
, const String
& charset
, const String
& defaultCharset
)
79 String normalizedAcceptCharset
= acceptCharset
;
80 normalizedAcceptCharset
.replace(',', ' ');
82 Vector
<String
> charsets
;
83 normalizedAcceptCharset
.split(' ', charsets
);
85 WTF::TextEncoding encoding
;
87 Vector
<String
>::const_iterator end
= charsets
.end();
88 for (Vector
<String
>::const_iterator it
= charsets
.begin(); it
!= end
; ++it
) {
89 if ((encoding
= WTF::TextEncoding(*it
)).isValid())
93 if (charset
.isEmpty()) {
94 if (defaultCharset
.isEmpty())
95 return WTF::UTF8Encoding();
97 return defaultCharset
;
103 Vector
<char> FormDataEncoder::generateUniqueBoundaryString()
105 Vector
<char> boundary
;
107 // The RFC 2046 spec says the alphanumeric characters plus the
108 // following characters are legal for boundaries: '()+_,-./:=?
109 // However the following characters, though legal, cause some sites
111 // Note that our algorithm makes it twice as much likely for 'A' or 'B'
112 // to appear in the boundary string, because 0x41 and 0x42 are present in
113 // the below array twice.
114 static const char alphaNumericEncodingMap
[64] = {
115 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48,
116 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4E, 0x4F, 0x50,
117 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58,
118 0x59, 0x5A, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66,
119 0x67, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E,
120 0x6F, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76,
121 0x77, 0x78, 0x79, 0x7A, 0x30, 0x31, 0x32, 0x33,
122 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x41, 0x42
125 // Start with an informative prefix.
126 append(boundary
, "----WebKitFormBoundary");
128 // Append 16 random 7bit ascii AlphaNumeric characters.
129 Vector
<char> randomBytes
;
131 for (unsigned i
= 0; i
< 4; ++i
) {
132 uint32_t randomness
= cryptographicallyRandomNumber();
133 randomBytes
.append(alphaNumericEncodingMap
[(randomness
>> 24) & 0x3F]);
134 randomBytes
.append(alphaNumericEncodingMap
[(randomness
>> 16) & 0x3F]);
135 randomBytes
.append(alphaNumericEncodingMap
[(randomness
>> 8) & 0x3F]);
136 randomBytes
.append(alphaNumericEncodingMap
[randomness
& 0x3F]);
139 boundary
.appendVector(randomBytes
);
140 boundary
.append(0); // Add a 0 at the end so we can use this as a C-style string.
144 void FormDataEncoder::beginMultiPartHeader(Vector
<char>& buffer
, const CString
& boundary
, const CString
& name
)
146 addBoundaryToMultiPartHeader(buffer
, boundary
);
148 // FIXME: This loses data irreversibly if the input name includes characters you can't encode
149 // in the website's character set.
150 append(buffer
, "Content-Disposition: form-data; name=\"");
151 appendQuotedString(buffer
, name
);
155 void FormDataEncoder::addBoundaryToMultiPartHeader(Vector
<char>& buffer
, const CString
& boundary
, bool isLastBoundary
)
157 append(buffer
, "--");
158 append(buffer
, boundary
);
161 append(buffer
, "--");
163 append(buffer
, "\r\n");
166 void FormDataEncoder::addFilenameToMultiPartHeader(Vector
<char>& buffer
, const WTF::TextEncoding
& encoding
, const String
& filename
)
168 // FIXME: This loses data irreversibly if the filename includes characters you can't encode
169 // in the website's character set.
170 append(buffer
, "; filename=\"");
171 appendQuotedString(buffer
, encoding
.encode(filename
, WTF::QuestionMarksForUnencodables
));
175 void FormDataEncoder::addContentTypeToMultiPartHeader(Vector
<char>& buffer
, const CString
& mimeType
)
177 append(buffer
, "\r\nContent-Type: ");
178 append(buffer
, mimeType
);
181 void FormDataEncoder::finishMultiPartHeader(Vector
<char>& buffer
)
183 append(buffer
, "\r\n\r\n");
186 void FormDataEncoder::addKeyValuePairAsFormData(Vector
<char>& buffer
, const CString
& key
, const CString
& value
, EncodedFormData::EncodingType encodingType
)
188 if (encodingType
== EncodedFormData::TextPlain
) {
189 if (!buffer
.isEmpty())
190 append(buffer
, "\r\n");
193 append(buffer
, value
);
195 if (!buffer
.isEmpty())
197 encodeStringAsFormData(buffer
, key
);
199 encodeStringAsFormData(buffer
, value
);
203 void FormDataEncoder::encodeStringAsFormData(Vector
<char>& buffer
, const CString
& string
)
205 // Same safe characters as Netscape for compatibility.
206 static const char safeCharacters
[] = "-._*";
208 // http://www.w3.org/TR/html4/interact/forms.html#h-17.13.4.1
209 unsigned length
= string
.length();
210 for (unsigned i
= 0; i
< length
; ++i
) {
211 unsigned char c
= string
.data()[i
];
213 if ((c
>= 'A' && c
<= 'Z') || (c
>= 'a' && c
<= 'z') || (c
>= '0' && c
<= '9') || strchr(safeCharacters
, c
)) {
215 } else if (c
== ' ') {
217 } else if (c
== '\n' || (c
== '\r' && (i
+ 1 >= length
|| string
.data()[i
+ 1] != '\n'))) {
218 append(buffer
, "%0D%0A");
219 } else if (c
!= '\r') {
221 appendByteAsHex(c
, buffer
);