diff options
author | Lorry Tar Creator <lorry-tar-importer@lorry> | 2017-06-27 06:07:23 +0000 |
---|---|---|
committer | Lorry Tar Creator <lorry-tar-importer@lorry> | 2017-06-27 06:07:23 +0000 |
commit | 1bf1084f2b10c3b47fd1a588d85d21ed0eb41d0c (patch) | |
tree | 46dcd36c86e7fbc6e5df36deb463b33e9967a6f7 /Source/WebCore/Modules/fetch/FetchBodyOwner.cpp | |
parent | 32761a6cee1d0dee366b885b7b9c777e67885688 (diff) | |
download | WebKitGtk-tarball-master.tar.gz |
webkitgtk-2.16.5HEADwebkitgtk-2.16.5master
Diffstat (limited to 'Source/WebCore/Modules/fetch/FetchBodyOwner.cpp')
-rw-r--r-- | Source/WebCore/Modules/fetch/FetchBodyOwner.cpp | 273 |
1 files changed, 273 insertions, 0 deletions
diff --git a/Source/WebCore/Modules/fetch/FetchBodyOwner.cpp b/Source/WebCore/Modules/fetch/FetchBodyOwner.cpp new file mode 100644 index 000000000..380c570e9 --- /dev/null +++ b/Source/WebCore/Modules/fetch/FetchBodyOwner.cpp @@ -0,0 +1,273 @@ +/* + * Copyright (C) 2016 Canon Inc. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted, provided that the following conditions + * are required to be met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. Neither the name of Canon Inc. nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY CANON INC. AND ITS CONTRIBUTORS "AS IS" AND ANY + * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL CANON INC. AND ITS CONTRIBUTORS BE LIABLE FOR + * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER + * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, + * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#include "config.h" +#include "FetchBodyOwner.h" + +#if ENABLE(FETCH_API) + +#include "FetchLoader.h" +#include "HTTPParsers.h" +#include "JSBlob.h" +#include "ResourceResponse.h" + +namespace WebCore { + +FetchBodyOwner::FetchBodyOwner(ScriptExecutionContext& context, std::optional<FetchBody>&& body, Ref<FetchHeaders>&& headers) + : ActiveDOMObject(&context) + , m_body(WTFMove(body)) + , m_headers(WTFMove(headers)) +{ + suspendIfNeeded(); +} + +void FetchBodyOwner::stop() +{ + if (m_body) + m_body->cleanConsumePromise(); + + if (m_blobLoader) { + bool isUniqueReference = hasOneRef(); + if (m_blobLoader->loader) + m_blobLoader->loader->stop(); + // After that point, 'this' may be destroyed, since unsetPendingActivity should have been called. + ASSERT_UNUSED(isUniqueReference, isUniqueReference || !m_blobLoader); + } +} + +bool FetchBodyOwner::isDisturbedOrLocked() const +{ + if (m_isDisturbed) + return true; + +#if ENABLE(READABLE_STREAM_API) + if (m_readableStreamSource && m_readableStreamSource->isReadableStreamLocked()) + return true; +#endif + + return false; +} + +void FetchBodyOwner::arrayBuffer(Ref<DeferredPromise>&& promise) +{ + if (isBodyNull()) { + fulfillPromiseWithArrayBuffer(WTFMove(promise), nullptr, 0); + return; + } + if (isDisturbedOrLocked()) { + promise->reject(TypeError); + return; + } + m_isDisturbed = true; + m_body->arrayBuffer(*this, WTFMove(promise)); +} + +void FetchBodyOwner::blob(Ref<DeferredPromise>&& promise) +{ + if (isBodyNull()) { + promise->resolve<IDLInterface<Blob>>(Blob::create({ }, Blob::normalizedContentType(extractMIMETypeFromMediaType(m_contentType))).get()); + return; + } + if (isDisturbedOrLocked()) { + promise->reject(TypeError); + return; + } + m_isDisturbed = true; + m_body->blob(*this, WTFMove(promise), m_contentType); +} + +void FetchBodyOwner::cloneBody(const FetchBodyOwner& owner) +{ + m_contentType = owner.m_contentType; + if (owner.isBodyNull()) + return; + m_body = owner.m_body->clone(); +} + +void FetchBodyOwner::extractBody(ScriptExecutionContext& context, JSC::ExecState& state, JSC::JSValue value) +{ + m_body = FetchBody::extract(context, state, value, m_contentType); +} + +void FetchBodyOwner::updateContentType() +{ + String contentType = m_headers->fastGet(HTTPHeaderName::ContentType); + if (!contentType.isNull()) { + m_contentType = WTFMove(contentType); + return; + } + if (!m_contentType.isNull()) + m_headers->fastSet(HTTPHeaderName::ContentType, m_contentType); +} + +void FetchBodyOwner::consumeOnceLoadingFinished(FetchBodyConsumer::Type type, Ref<DeferredPromise>&& promise) +{ + if (isDisturbedOrLocked()) { + promise->reject(TypeError); + return; + } + m_isDisturbed = true; + m_body->consumeOnceLoadingFinished(type, WTFMove(promise)); +} + +void FetchBodyOwner::formData(Ref<DeferredPromise>&& promise) +{ + if (isBodyNull()) { + promise->reject(); + return; + } + if (isDisturbedOrLocked()) { + promise->reject(TypeError); + return; + } + m_isDisturbed = true; + m_body->formData(*this, WTFMove(promise)); +} + +void FetchBodyOwner::json(Ref<DeferredPromise>&& promise) +{ + if (isBodyNull()) { + promise->reject(SYNTAX_ERR); + return; + } + if (isDisturbedOrLocked()) { + promise->reject(TypeError); + return; + } + m_isDisturbed = true; + m_body->json(*this, WTFMove(promise)); +} + +void FetchBodyOwner::text(Ref<DeferredPromise>&& promise) +{ + if (isBodyNull()) { + promise->resolve<IDLDOMString>({ }); + return; + } + if (isDisturbedOrLocked()) { + promise->reject(TypeError); + return; + } + m_isDisturbed = true; + m_body->text(*this, WTFMove(promise)); +} + +void FetchBodyOwner::loadBlob(const Blob& blob, FetchBodyConsumer* consumer) +{ + // Can only be called once for a body instance. + ASSERT(isDisturbed()); + ASSERT(!m_blobLoader); + ASSERT(!isBodyNull()); + + if (!scriptExecutionContext()) { + m_body->loadingFailed(); + return; + } + + m_blobLoader.emplace(*this); + m_blobLoader->loader = std::make_unique<FetchLoader>(*m_blobLoader, consumer); + + m_blobLoader->loader->start(*scriptExecutionContext(), blob); + if (!m_blobLoader->loader->isStarted()) { + m_body->loadingFailed(); + m_blobLoader = std::nullopt; + return; + } + setPendingActivity(this); +} + +void FetchBodyOwner::finishBlobLoading() +{ + ASSERT(m_blobLoader); + + m_blobLoader = std::nullopt; + unsetPendingActivity(this); +} + +void FetchBodyOwner::blobLoadingSucceeded() +{ + ASSERT(!isBodyNull()); +#if ENABLE(READABLE_STREAM_API) + if (m_readableStreamSource) { + m_readableStreamSource->close(); + m_readableStreamSource = nullptr; + } +#endif + m_body->loadingSucceeded(); + finishBlobLoading(); +} + +void FetchBodyOwner::blobLoadingFailed() +{ + ASSERT(!isBodyNull()); +#if ENABLE(READABLE_STREAM_API) + if (m_readableStreamSource) { + if (!m_readableStreamSource->isCancelling()) + m_readableStreamSource->error(ASCIILiteral("Blob loading failed")); + m_readableStreamSource = nullptr; + } else +#endif + m_body->loadingFailed(); + + finishBlobLoading(); +} + +void FetchBodyOwner::blobChunk(const char* data, size_t size) +{ + ASSERT(data); +#if ENABLE(READABLE_STREAM_API) + ASSERT(m_readableStreamSource); + if (!m_readableStreamSource->enqueue(ArrayBuffer::tryCreate(data, size))) + stop(); +#else + UNUSED_PARAM(data); + UNUSED_PARAM(size); +#endif +} + +FetchBodyOwner::BlobLoader::BlobLoader(FetchBodyOwner& owner) + : owner(owner) +{ +} + +void FetchBodyOwner::BlobLoader::didReceiveResponse(const ResourceResponse& response) +{ + if (response.httpStatusCode() != 200) + didFail(); +} + +void FetchBodyOwner::BlobLoader::didFail() +{ + // didFail might be called within FetchLoader::start call. + if (loader->isStarted()) + owner.blobLoadingFailed(); +} + +} // namespace WebCore + +#endif // ENABLE(FETCH_API) |