blob: 5fde09d7f5fb31a28ff51e65a9fb71e6c41422fc [file] [log] [blame]
/*
* Copyright (C) 2009 Google Inc. All rights reserved.
* Copyright (C) 2010 Apple 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:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * 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.
* * Neither the name of Google 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 THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT
* OWNER 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 "third_party/blink/renderer/core/workers/shared_worker.h"
#include "base/optional.h"
#include "mojo/public/cpp/bindings/pending_remote.h"
#include "third_party/blink/public/common/blob/blob_utils.h"
#include "third_party/blink/public/mojom/fetch/fetch_api_request.mojom-blink.h"
#include "third_party/blink/public/mojom/worker/shared_worker_info.mojom-blink.h"
#include "third_party/blink/renderer/bindings/core/v8/v8_worker_options.h"
#include "third_party/blink/renderer/core/execution_context/execution_context.h"
#include "third_party/blink/renderer/core/fetch/request.h"
#include "third_party/blink/renderer/core/fileapi/public_url_manager.h"
#include "third_party/blink/renderer/core/frame/local_dom_window.h"
#include "third_party/blink/renderer/core/messaging/message_channel.h"
#include "third_party/blink/renderer/core/messaging/message_port.h"
#include "third_party/blink/renderer/core/probe/core_probes.h"
#include "third_party/blink/renderer/core/script/script.h"
#include "third_party/blink/renderer/core/workers/shared_worker_client_holder.h"
#include "third_party/blink/renderer/platform/bindings/exception_state.h"
#include "third_party/blink/renderer/platform/instrumentation/use_counter.h"
#include "third_party/blink/renderer/platform/weborigin/kurl.h"
#include "third_party/blink/renderer/platform/weborigin/security_origin.h"
namespace blink {
namespace {
void RecordSharedWorkerUsage(LocalDOMWindow* window) {
UseCounter::Count(window, WebFeature::kSharedWorkerStart);
if (window->IsCrossSiteSubframe())
UseCounter::Count(window, WebFeature::kThirdPartySharedWorker);
}
} // namespace
SharedWorker::SharedWorker(ExecutionContext* context)
: AbstractWorker(context),
is_being_connected_(false),
feature_handle_for_scheduler_(context->GetScheduler()->RegisterFeature(
SchedulingPolicy::Feature::kSharedWorker,
{SchedulingPolicy::DisableBackForwardCache()})) {}
SharedWorker* SharedWorker::Create(ExecutionContext* context,
const String& url,
const StringOrWorkerOptions& name_or_options,
ExceptionState& exception_state) {
DCHECK(IsMainThread());
// We don't currently support nested workers, so workers can only be created
// from windows.
LocalDOMWindow* window = To<LocalDOMWindow>(context);
RecordSharedWorkerUsage(window);
SharedWorker* worker = MakeGarbageCollected<SharedWorker>(context);
worker->UpdateStateIfNeeded();
auto* channel = MakeGarbageCollected<MessageChannel>(context);
worker->port_ = channel->port1();
MessagePortChannel remote_port = channel->port2()->Disentangle();
if (!window->GetSecurityOrigin()->CanAccessSharedWorkers()) {
exception_state.ThrowSecurityError(
"Access to shared workers is denied to origin '" +
window->GetSecurityOrigin()->ToString() + "'.");
return nullptr;
} else if (window->GetSecurityOrigin()->IsLocal()) {
UseCounter::Count(window, WebFeature::kFileAccessedSharedWorker);
}
KURL script_url = ResolveURL(context, url, exception_state);
if (script_url.IsEmpty())
return nullptr;
mojo::PendingRemote<mojom::blink::BlobURLToken> blob_url_token;
if (script_url.ProtocolIs("blob")) {
window->GetPublicURLManager().Resolve(
script_url, blob_url_token.InitWithNewPipeAndPassReceiver());
}
auto options = mojom::blink::WorkerOptions::New();
if (name_or_options.IsString()) {
options->name = name_or_options.GetAsString();
} else if (name_or_options.IsWorkerOptions()) {
WorkerOptions* worker_options = name_or_options.GetAsWorkerOptions();
if (worker_options->type() == "module" &&
!RuntimeEnabledFeatures::ModuleSharedWorkerEnabled()) {
exception_state.ThrowTypeError(
"Module scripts are not supported on SharedWorker yet. "
"(see https://crbug.com/824646)");
return nullptr;
}
options->name = worker_options->name();
base::Optional<mojom::blink::ScriptType> type_result =
Script::ParseScriptType(worker_options->type());
DCHECK(type_result);
options->type = type_result.value();
base::Optional<network::mojom::CredentialsMode> credentials_result =
Request::ParseCredentialsMode(worker_options->credentials());
DCHECK(credentials_result);
options->credentials = credentials_result.value();
} else {
NOTREACHED();
}
DCHECK(!options->name.IsNull());
if (options->type == mojom::blink::ScriptType::kClassic)
UseCounter::Count(window, WebFeature::kClassicSharedWorker);
else if (options->type == mojom::blink::ScriptType::kModule)
UseCounter::Count(window, WebFeature::kModuleSharedWorker);
SharedWorkerClientHolder::From(*window)->Connect(
worker, std::move(remote_port), script_url, std::move(blob_url_token),
std::move(options), context->UkmSourceID());
return worker;
}
SharedWorker::~SharedWorker() = default;
const AtomicString& SharedWorker::InterfaceName() const {
return event_target_names::kSharedWorker;
}
bool SharedWorker::HasPendingActivity() const {
return is_being_connected_;
}
void SharedWorker::ContextLifecycleStateChanged(
mojom::FrameLifecycleState state) {}
void SharedWorker::Trace(Visitor* visitor) const {
visitor->Trace(port_);
AbstractWorker::Trace(visitor);
Supplementable<SharedWorker>::Trace(visitor);
}
} // namespace blink