2 * Copyright (C) 2004, 2006, 2008, 2011 Apple Inc. All rights reserved.
3 * Copyright (C) 2009 Google Inc. All rights reserved.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Library General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Library General Public License for more details.
15 * You should have received a copy of the GNU Library General Public License
16 * along with this library; see the file COPYING.LIB. If not, write to
17 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
18 * Boston, MA 02110-1301, USA.
28 #include "ChromeClient.h"
31 #include "FileSystem.h"
32 #include "FormDataBuilder.h"
33 #include "FormDataList.h"
34 #include "MIMETypeRegistry.h"
36 #include "TextEncoding.h"
37 #include <wtf/Decoder.h>
38 #include <wtf/Encoder.h>
42 inline FormData::FormData()
44 , m_hasGeneratedFiles(false)
45 , m_alwaysStream(false)
46 , m_containsPasswordData(false)
50 inline FormData::FormData(const FormData& data)
51 : RefCounted<FormData>()
52 , m_elements(data.m_elements)
53 , m_identifier(data.m_identifier)
54 , m_hasGeneratedFiles(false)
55 , m_alwaysStream(false)
56 , m_containsPasswordData(data.m_containsPasswordData)
58 // We shouldn't be copying FormData that hasn't already removed its generated files
59 // but just in case, make sure the new FormData is ready to generate its own files.
60 if (data.m_hasGeneratedFiles) {
61 size_t n = m_elements.size();
62 for (size_t i = 0; i < n; ++i) {
63 FormDataElement& e = m_elements[i];
64 if (e.m_type == FormDataElement::encodedFile)
65 e.m_generatedFilename = String();
72 // This cleanup should've happened when the form submission finished.
73 // Just in case, let's assert, and do the cleanup anyway in release builds.
74 ASSERT(!m_hasGeneratedFiles);
75 removeGeneratedFilesIfNeeded();
78 PassRefPtr<FormData> FormData::create()
80 return adoptRef(new FormData);
83 PassRefPtr<FormData> FormData::create(const void* data, size_t size)
85 RefPtr<FormData> result = create();
86 result->appendData(data, size);
87 return result.release();
90 PassRefPtr<FormData> FormData::create(const CString& string)
92 RefPtr<FormData> result = create();
93 result->appendData(string.data(), string.length());
94 return result.release();
97 PassRefPtr<FormData> FormData::create(const Vector<char>& vector)
99 RefPtr<FormData> result = create();
100 result->appendData(vector.data(), vector.size());
101 return result.release();
104 PassRefPtr<FormData> FormData::create(const FormDataList& list, const TextEncoding& encoding, EncodingType encodingType)
106 RefPtr<FormData> result = create();
107 result->appendKeyValuePairItems(list, encoding, false, 0, encodingType);
108 return result.release();
111 PassRefPtr<FormData> FormData::createMultiPart(const FormDataList& list, const TextEncoding& encoding, Document* document)
113 RefPtr<FormData> result = create();
114 result->appendKeyValuePairItems(list, encoding, true, document);
115 return result.release();
118 PassRefPtr<FormData> FormData::copy() const
120 return adoptRef(new FormData(*this));
123 PassRefPtr<FormData> FormData::deepCopy() const
125 RefPtr<FormData> formData(create());
127 formData->m_alwaysStream = m_alwaysStream;
129 size_t n = m_elements.size();
130 formData->m_elements.reserveInitialCapacity(n);
131 for (size_t i = 0; i < n; ++i) {
132 const FormDataElement& e = m_elements[i];
134 case FormDataElement::data:
135 formData->m_elements.append(FormDataElement(e.m_data));
137 case FormDataElement::encodedFile:
139 formData->m_elements.append(FormDataElement(e.m_filename, e.m_fileStart, e.m_fileLength, e.m_expectedFileModificationTime, e.m_shouldGenerateFile));
141 formData->m_elements.append(FormDataElement(e.m_filename, e.m_shouldGenerateFile));
145 case FormDataElement::encodedBlob:
146 formData->m_elements.append(FormDataElement(e.m_blobURL));
151 return formData.release();
154 void FormData::appendData(const void* data, size_t size)
156 if (m_elements.isEmpty() || m_elements.last().m_type != FormDataElement::data)
157 m_elements.append(FormDataElement());
158 FormDataElement& e = m_elements.last();
159 size_t oldSize = e.m_data.size();
160 e.m_data.grow(oldSize + size);
161 memcpy(e.m_data.data() + oldSize, data, size);
164 void FormData::appendFile(const String& filename, bool shouldGenerateFile)
167 m_elements.append(FormDataElement(filename, 0, BlobDataItem::toEndOfFile, BlobDataItem::doNotCheckFileChange, shouldGenerateFile));
169 m_elements.append(FormDataElement(filename, shouldGenerateFile));
174 void FormData::appendFileRange(const String& filename, long long start, long long length, double expectedModificationTime, bool shouldGenerateFile)
176 m_elements.append(FormDataElement(filename, start, length, expectedModificationTime, shouldGenerateFile));
179 void FormData::appendBlob(const KURL& blobURL)
181 m_elements.append(FormDataElement(blobURL));
185 void FormData::appendKeyValuePairItems(const FormDataList& list, const TextEncoding& encoding, bool isMultiPartForm, Document* document, EncodingType encodingType)
188 m_boundary = FormDataBuilder::generateUniqueBoundaryString();
190 Vector<char> encodedData;
192 const Vector<FormDataList::Item>& items = list.items();
193 size_t formDataListSize = items.size();
194 ASSERT(!(formDataListSize % 2));
195 for (size_t i = 0; i < formDataListSize; i += 2) {
196 const FormDataList::Item& key = items[i];
197 const FormDataList::Item& value = items[i + 1];
198 if (isMultiPartForm) {
200 FormDataBuilder::beginMultiPartHeader(header, m_boundary.data(), key.data());
202 bool shouldGenerateFile = false;
204 // If the current type is blob, then we also need to include the filename
207 if (value.blob()->isFile()) {
208 // For file blob, use the filename (or relative path if it is present) as the name.
209 File* file = static_cast<File*>(value.blob());
210 #if ENABLE(DIRECTORY_UPLOAD)
211 name = file->webkitRelativePath().isEmpty() ? file->name() : file->webkitRelativePath();
215 // Let the application specify a filename if it's going to generate a replacement file for the upload.
216 const String& path = file->path();
217 if (!path.isEmpty()) {
218 if (Page* page = document->page()) {
219 String generatedFileName;
220 shouldGenerateFile = page->chrome()->client()->shouldReplaceWithGeneratedFileForUpload(path, generatedFileName);
221 if (shouldGenerateFile)
222 name = generatedFileName;
226 // For non-file blob, use the filename if it is passed in FormData.append().
227 if (!value.filename().isEmpty())
228 name = value.filename();
233 // We have to include the filename=".." part in the header, even if the filename is empty
234 FormDataBuilder::addFilenameToMultiPartHeader(header, encoding, name);
236 // Add the content type if available, or "application/octet-stream" otherwise (RFC 1867).
238 if (value.blob()->type().isEmpty())
239 contentType = "application/octet-stream";
241 contentType = value.blob()->type();
242 FormDataBuilder::addContentTypeToMultiPartHeader(header, contentType.latin1());
245 FormDataBuilder::finishMultiPartHeader(header);
248 appendData(header.data(), header.size());
250 if (value.blob()->isFile()) {
251 // Do not add the file if the path is empty.
252 if (!static_cast<File*>(value.blob())->path().isEmpty())
253 appendFile(static_cast<File*>(value.blob())->path(), shouldGenerateFile);
257 appendBlob(value.blob()->url());
260 appendData(value.data().data(), value.data().length());
261 appendData("\r\n", 2);
263 // Omit the name "isindex" if it's the first form data element.
264 // FIXME: Why is this a good rule? Is this obsolete now?
265 if (encodedData.isEmpty() && key.data() == "isindex")
266 FormDataBuilder::encodeStringAsFormData(encodedData, value.data());
268 FormDataBuilder::addKeyValuePairAsFormData(encodedData, key.data(), value.data(), encodingType);
273 FormDataBuilder::addBoundaryToMultiPartHeader(encodedData, m_boundary.data(), true);
275 appendData(encodedData.data(), encodedData.size());
278 void FormData::flatten(Vector<char>& data) const
280 // Concatenate all the byte arrays, but omit any files.
282 size_t n = m_elements.size();
283 for (size_t i = 0; i < n; ++i) {
284 const FormDataElement& e = m_elements[i];
285 if (e.m_type == FormDataElement::data)
286 data.append(e.m_data.data(), static_cast<size_t>(e.m_data.size()));
290 String FormData::flattenToString() const
294 return Latin1Encoding().decode(reinterpret_cast<const char*>(bytes.data()), bytes.size());
297 void FormData::generateFiles(Document* document)
299 ASSERT(!m_hasGeneratedFiles);
301 if (m_hasGeneratedFiles)
304 Page* page = document->page();
307 ChromeClient* client = page->chrome()->client();
309 size_t n = m_elements.size();
310 for (size_t i = 0; i < n; ++i) {
311 FormDataElement& e = m_elements[i];
312 if (e.m_type == FormDataElement::encodedFile && e.m_shouldGenerateFile) {
313 e.m_generatedFilename = client->generateReplacementFile(e.m_filename);
314 m_hasGeneratedFiles = true;
319 void FormData::removeGeneratedFilesIfNeeded()
321 if (!m_hasGeneratedFiles)
324 size_t n = m_elements.size();
325 for (size_t i = 0; i < n; ++i) {
326 FormDataElement& e = m_elements[i];
327 if (e.m_type == FormDataElement::encodedFile && !e.m_generatedFilename.isEmpty()) {
328 ASSERT(e.m_shouldGenerateFile);
329 String directory = directoryName(e.m_generatedFilename);
330 deleteFile(e.m_generatedFilename);
331 deleteEmptyDirectory(directory);
332 e.m_generatedFilename = String();
335 m_hasGeneratedFiles = false;
338 static void encode(Encoder& encoder, const FormDataElement& element)
340 encoder.encodeUInt32(element.m_type);
342 switch (element.m_type) {
343 case FormDataElement::data:
344 encoder.encodeBytes(reinterpret_cast<const uint8_t*>(element.m_data.data()), element.m_data.size());
347 case FormDataElement::encodedFile:
348 encoder.encodeString(element.m_filename);
349 encoder.encodeBool(element.m_shouldGenerateFile);
351 encoder.encodeInt64(element.m_fileStart);
352 encoder.encodeInt64(element.m_fileLength);
353 encoder.encodeDouble(element.m_expectedFileModificationTime);
355 encoder.encodeInt64(0);
356 encoder.encodeInt64(0);
357 encoder.encodeDouble(0);
362 case FormDataElement::encodedBlob:
363 encoder.encodeString(element.m_blobURL.string());
368 ASSERT_NOT_REACHED();
371 static bool decode(Decoder& decoder, FormDataElement& element)
374 if (!decoder.decodeUInt32(type))
378 case FormDataElement::data: {
379 element.m_type = FormDataElement::data;
380 Vector<uint8_t> data;
381 if (!decoder.decodeBytes(data))
383 size_t size = data.size();
384 element.m_data.resize(size);
385 memcpy(element.m_data.data(), data.data(), size);
389 case FormDataElement::encodedFile: {
390 element.m_type = FormDataElement::encodedFile;
391 if (!decoder.decodeString(element.m_filename))
393 if (!decoder.decodeBool(element.m_shouldGenerateFile))
396 if (!decoder.decodeInt64(fileStart))
401 if (!decoder.decodeInt64(fileLength))
403 if (fileLength < fileStart)
405 double expectedFileModificationTime;
406 if (!decoder.decodeDouble(expectedFileModificationTime))
409 element.m_fileStart = fileStart;
410 element.m_fileLength = fileLength;
411 element.m_expectedFileModificationTime = expectedFileModificationTime;
417 case FormDataElement::encodedBlob:
418 element.m_type = FormDataElement::encodedBlob;
419 String blobURLString;
420 if (!decoder.decodeString(blobURLString))
422 element.m_blobURL = KURL(KURL(), blobURLString);
430 void FormData::encodeForBackForward(Encoder& encoder) const
432 encoder.encodeBool(m_alwaysStream);
434 encoder.encodeBytes(reinterpret_cast<const uint8_t*>(m_boundary.data()), m_boundary.size());
436 size_t size = m_elements.size();
437 encoder.encodeUInt64(size);
438 for (size_t i = 0; i < size; ++i)
439 encode(encoder, m_elements[i]);
441 encoder.encodeBool(m_hasGeneratedFiles);
443 encoder.encodeInt64(m_identifier);
446 PassRefPtr<FormData> FormData::decodeForBackForward(Decoder& decoder)
448 RefPtr<FormData> data = FormData::create();
450 if (!decoder.decodeBool(data->m_alwaysStream))
453 Vector<uint8_t> boundary;
454 if (!decoder.decodeBytes(boundary))
456 size_t size = boundary.size();
457 data->m_boundary.resize(size);
458 memcpy(data->m_boundary.data(), boundary.data(), size);
460 uint64_t elementsSize;
461 if (!decoder.decodeUInt64(elementsSize))
463 for (size_t i = 0; i < elementsSize; ++i) {
464 FormDataElement element;
465 if (!decode(decoder, element))
467 data->m_elements.append(element);
470 if (!decoder.decodeBool(data->m_hasGeneratedFiles))
473 if (!decoder.decodeInt64(data->m_identifier))
476 return data.release();
479 } // namespace WebCore