Use "= default" to denote default constructor or destructor
[WebKit-https.git] / Source / WebCore / workers / WorkerScriptLoader.cpp
1 /*
2  * Copyright (C) 2009-2017 Apple Inc. All Rights Reserved.
3  * Copyright (C) 2009, 2011 Google Inc. All Rights Reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
15  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
18  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
19  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
20  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
21  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
22  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
24  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
25  */
26
27 #include "config.h"
28 #include "WorkerScriptLoader.h"
29
30 #include "ContentSecurityPolicy.h"
31 #include "ResourceResponse.h"
32 #include "ScriptExecutionContext.h"
33 #include "TextResourceDecoder.h"
34 #include "WorkerGlobalScope.h"
35 #include "WorkerScriptLoaderClient.h"
36 #include "WorkerThreadableLoader.h"
37 #include <wtf/Ref.h>
38
39 namespace WebCore {
40
41 WorkerScriptLoader::WorkerScriptLoader() = default;
42
43 WorkerScriptLoader::~WorkerScriptLoader() = default;
44
45 void WorkerScriptLoader::loadSynchronously(ScriptExecutionContext* scriptExecutionContext, const URL& url, FetchOptions::Mode mode, ContentSecurityPolicyEnforcement contentSecurityPolicyEnforcement, const String& initiatorIdentifier)
46 {
47     ASSERT(scriptExecutionContext);
48
49     m_url = url;
50
51     std::unique_ptr<ResourceRequest> request(createResourceRequest(initiatorIdentifier));
52     if (!request)
53         return;
54
55     ASSERT_WITH_SECURITY_IMPLICATION(is<WorkerGlobalScope>(scriptExecutionContext));
56
57     // Only used for importScripts that prescribes NoCors mode.
58     ASSERT(mode == FetchOptions::Mode::NoCors);
59
60     ThreadableLoaderOptions options;
61     options.credentials = FetchOptions::Credentials::Include;
62     options.mode = mode;
63     options.sendLoadCallbacks = SendCallbacks;
64     options.contentSecurityPolicyEnforcement = contentSecurityPolicyEnforcement;
65
66     WorkerThreadableLoader::loadResourceSynchronously(downcast<WorkerGlobalScope>(*scriptExecutionContext), WTFMove(*request), *this, options);
67 }
68
69 void WorkerScriptLoader::loadAsynchronously(ScriptExecutionContext* scriptExecutionContext, const URL& url, FetchOptions::Mode mode, ContentSecurityPolicyEnforcement contentSecurityPolicyEnforcement, const String& initiatorIdentifier, WorkerScriptLoaderClient* client)
70 {
71     ASSERT(client);
72     ASSERT(scriptExecutionContext);
73
74     m_client = client;
75     m_url = url;
76
77     std::unique_ptr<ResourceRequest> request(createResourceRequest(initiatorIdentifier));
78     if (!request)
79         return;
80
81     // Only used for loading worker scripts in classic mode.
82     // FIXME: We should add an option to set credential mode.
83     ASSERT(mode == FetchOptions::Mode::SameOrigin);
84
85     ThreadableLoaderOptions options;
86     options.credentials = FetchOptions::Credentials::SameOrigin;
87     options.mode = mode;
88     options.sendLoadCallbacks = SendCallbacks;
89     options.contentSecurityPolicyEnforcement = contentSecurityPolicyEnforcement;
90
91     // During create, callbacks may happen which remove the last reference to this object.
92     Ref<WorkerScriptLoader> protectedThis(*this);
93     m_threadableLoader = ThreadableLoader::create(*scriptExecutionContext, *this, WTFMove(*request), options);
94 }
95
96 const URL& WorkerScriptLoader::responseURL() const
97 {
98     ASSERT(!failed());
99     return m_responseURL;
100 }
101
102 std::unique_ptr<ResourceRequest> WorkerScriptLoader::createResourceRequest(const String& initiatorIdentifier)
103 {
104     auto request = std::make_unique<ResourceRequest>(m_url);
105     request->setHTTPMethod(ASCIILiteral("GET"));
106     request->setInitiatorIdentifier(initiatorIdentifier);
107     return request;
108 }
109
110 void WorkerScriptLoader::didReceiveResponse(unsigned long identifier, const ResourceResponse& response)
111 {
112     if (response.httpStatusCode() / 100 != 2 && response.httpStatusCode()) {
113         m_failed = true;
114         return;
115     }
116
117     if (!isScriptAllowedByNosniff(response)) {
118         m_failed = true;
119         return;
120     }
121
122     m_responseURL = response.url();
123     m_responseEncoding = response.textEncodingName();
124     if (m_client)
125         m_client->didReceiveResponse(identifier, response);
126 }
127
128 void WorkerScriptLoader::didReceiveData(const char* data, int len)
129 {
130     if (m_failed)
131         return;
132
133     if (!m_decoder) {
134         if (!m_responseEncoding.isEmpty())
135             m_decoder = TextResourceDecoder::create(ASCIILiteral("text/javascript"), m_responseEncoding);
136         else
137             m_decoder = TextResourceDecoder::create(ASCIILiteral("text/javascript"), "UTF-8");
138     }
139
140     if (!len)
141         return;
142     
143     if (len == -1)
144         len = strlen(data);
145     
146     m_script.append(m_decoder->decode(data, len));
147 }
148
149 void WorkerScriptLoader::didFinishLoading(unsigned long identifier)
150 {
151     if (m_failed) {
152         notifyError();
153         return;
154     }
155
156     if (m_decoder)
157         m_script.append(m_decoder->flush());
158
159     m_identifier = identifier;
160     notifyFinished();
161 }
162
163 void WorkerScriptLoader::didFail(const ResourceError& error)
164 {
165     m_error = error;
166     notifyError();
167 }
168
169 void WorkerScriptLoader::notifyError()
170 {
171     m_failed = true;
172     notifyFinished();
173 }
174
175 String WorkerScriptLoader::script()
176 {
177     return m_script.toString();
178 }
179
180 void WorkerScriptLoader::notifyFinished()
181 {
182     if (!m_client || m_finishing)
183         return;
184
185     m_finishing = true;
186     m_client->notifyFinished();
187 }
188
189 } // namespace WebCore