blob: 04c4f9466da575a6dd45593e2e83fb792d9c8c20 [file] [log] [blame]
/*
* Copyright (C) 2009 Apple Inc. All Rights Reserved.
* Copyright (C) 2009, 2011 Google Inc. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``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 APPLE INC. OR
* 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 "WorkerScriptLoader.h"
#include "ContentSecurityPolicy.h"
#include "ResourceResponse.h"
#include "ScriptExecutionContext.h"
#include "SecurityOrigin.h"
#include "TextResourceDecoder.h"
#include "WorkerGlobalScope.h"
#include "WorkerScriptLoaderClient.h"
#include "WorkerThreadableLoader.h"
#include <wtf/Ref.h>
#include <wtf/RefPtr.h>
namespace WebCore {
WorkerScriptLoader::WorkerScriptLoader()
: m_client(0)
, m_failed(false)
, m_identifier(0)
, m_finishing(false)
{
}
WorkerScriptLoader::~WorkerScriptLoader()
{
}
void WorkerScriptLoader::loadSynchronously(ScriptExecutionContext* scriptExecutionContext, const URL& url, FetchOptions::Mode mode, ContentSecurityPolicyEnforcement contentSecurityPolicyEnforcement)
{
ASSERT(scriptExecutionContext);
m_url = url;
std::unique_ptr<ResourceRequest> request(createResourceRequest());
if (!request)
return;
ASSERT_WITH_SECURITY_IMPLICATION(is<WorkerGlobalScope>(scriptExecutionContext));
// Only used for importScripts that prescribes NoCors mode.
ASSERT(mode == FetchOptions::Mode::NoCors);
ThreadableLoaderOptions options;
options.credentials = FetchOptions::Credentials::Include;
options.mode = mode;
options.sendLoadCallbacks = SendCallbacks;
options.contentSecurityPolicyEnforcement = contentSecurityPolicyEnforcement;
WorkerThreadableLoader::loadResourceSynchronously(downcast<WorkerGlobalScope>(*scriptExecutionContext), WTFMove(*request), *this, options);
}
void WorkerScriptLoader::loadAsynchronously(ScriptExecutionContext* scriptExecutionContext, const URL& url, FetchOptions::Mode mode, ContentSecurityPolicyEnforcement contentSecurityPolicyEnforcement, WorkerScriptLoaderClient* client)
{
ASSERT(client);
ASSERT(scriptExecutionContext);
m_client = client;
m_url = url;
std::unique_ptr<ResourceRequest> request(createResourceRequest());
if (!request)
return;
// Only used for loading worker scripts in classic mode.
// FIXME: We should add an option to set credential mode.
ASSERT(mode == FetchOptions::Mode::SameOrigin);
ThreadableLoaderOptions options;
options.credentials = FetchOptions::Credentials::SameOrigin;
options.mode = mode;
options.sendLoadCallbacks = SendCallbacks;
options.contentSecurityPolicyEnforcement = contentSecurityPolicyEnforcement;
// During create, callbacks may happen which remove the last reference to this object.
Ref<WorkerScriptLoader> protectedThis(*this);
m_threadableLoader = ThreadableLoader::create(*scriptExecutionContext, *this, WTFMove(*request), options);
}
const URL& WorkerScriptLoader::responseURL() const
{
ASSERT(!failed());
return m_responseURL;
}
std::unique_ptr<ResourceRequest> WorkerScriptLoader::createResourceRequest()
{
auto request = std::make_unique<ResourceRequest>(m_url);
request->setHTTPMethod("GET");
return request;
}
void WorkerScriptLoader::didReceiveResponse(unsigned long identifier, const ResourceResponse& response)
{
if (response.httpStatusCode() / 100 != 2 && response.httpStatusCode()) {
m_failed = true;
return;
}
m_responseURL = response.url();
m_responseEncoding = response.textEncodingName();
if (m_client)
m_client->didReceiveResponse(identifier, response);
}
void WorkerScriptLoader::didReceiveData(const char* data, int len)
{
if (m_failed)
return;
if (!m_decoder) {
if (!m_responseEncoding.isEmpty())
m_decoder = TextResourceDecoder::create("text/javascript", m_responseEncoding);
else
m_decoder = TextResourceDecoder::create("text/javascript", "UTF-8");
}
if (!len)
return;
if (len == -1)
len = strlen(data);
m_script.append(m_decoder->decode(data, len));
}
void WorkerScriptLoader::didFinishLoading(unsigned long identifier, double)
{
if (m_failed) {
notifyError();
return;
}
if (m_decoder)
m_script.append(m_decoder->flush());
m_identifier = identifier;
notifyFinished();
}
void WorkerScriptLoader::didFail(const ResourceError&)
{
notifyError();
}
void WorkerScriptLoader::notifyError()
{
m_failed = true;
notifyFinished();
}
String WorkerScriptLoader::script()
{
return m_script.toString();
}
void WorkerScriptLoader::notifyFinished()
{
if (!m_client || m_finishing)
return;
m_finishing = true;
m_client->notifyFinished();
}
} // namespace WebCore