blob: b24330f9f2e90e43361acd86cb1d80a69596857f [file] [log] [blame]
// Copyright 2017 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "content/browser/service_worker/service_worker_script_loader_factory.h"
#include <memory>
#include <string>
#include <utility>
#include "base/debug/crash_logging.h"
#include "content/browser/service_worker/service_worker_cache_writer.h"
#include "content/browser/service_worker/service_worker_context_core.h"
#include "content/browser/service_worker/service_worker_installed_script_loader.h"
#include "content/browser/service_worker/service_worker_new_script_loader.h"
#include "content/browser/service_worker/service_worker_provider_host.h"
#include "content/browser/service_worker/service_worker_version.h"
#include "mojo/public/cpp/bindings/strong_binding.h"
#include "services/network/public/cpp/resource_response.h"
#include "services/network/public/cpp/shared_url_loader_factory.h"
#include "third_party/blink/public/common/service_worker/service_worker_utils.h"
namespace content {
ServiceWorkerScriptLoaderFactory::ServiceWorkerScriptLoaderFactory(
base::WeakPtr<ServiceWorkerContextCore> context,
base::WeakPtr<ServiceWorkerProviderHost> provider_host,
scoped_refptr<network::SharedURLLoaderFactory> loader_factory)
: context_(context),
provider_host_(provider_host),
loader_factory_(std::move(loader_factory)),
weak_factory_(this) {
DCHECK(provider_host_->IsProviderForServiceWorker());
DCHECK(loader_factory_);
}
ServiceWorkerScriptLoaderFactory::~ServiceWorkerScriptLoaderFactory() = default;
void ServiceWorkerScriptLoaderFactory::CreateLoaderAndStart(
network::mojom::URLLoaderRequest request,
int32_t routing_id,
int32_t request_id,
uint32_t options,
const network::ResourceRequest& resource_request,
network::mojom::URLLoaderClientPtr client,
const net::MutableNetworkTrafficAnnotationTag& traffic_annotation) {
DCHECK(blink::ServiceWorkerUtils::IsServicificationEnabled());
if (!CheckIfScriptRequestIsValid(resource_request)) {
// TODO(kinuko): Record the reason like what we do with netlog in
// ServiceWorkerContextRequestHandler.
client->OnComplete(network::URLLoaderCompletionStatus(net::ERR_ABORTED));
return;
}
// There are four cases of how to handle the request for the script.
// A) service worker is installed, script is installed: serve from storage
// (use ServceWorkerInstalledScriptLoader). Typically this case is handled
// by ServiceWorkerInstalledScriptsSender, but we can still get here when a
// new service worker starts up and becomes installed while it is running.
// B) service worker is installed, script is not installed: return a network
// error. This happens when the script is newly imported after
// installation, which is disallowed by the spec.
// C) service worker is not installed, script is installed: serve from
// storage (use ServiceWorkerInstalledScriptLoader)
// D) service worker is not installed, script is not installed:
// When ServiceWorkerImportedScriptsUpdateCheck is enabled, there are
// three sub cases:
// 1) If compared script info exists and specifies that the script is
// installed in an old service worker and content is not changed, then
// copy the old script into the new service worker and load it using
// ServiceWorkerInstalledScriptLoader.
// 2) If compared script info exists and specifies that the script is
// installed in an old service worker but content has changed, then
// resume the paused state in the compared script info to load the
// script.
// 3) For other cases or if ServiceWorkerImportedScriptsUpdateCheck is not
// enabled, serve from network with installing the script
// (use ServiceWorkerNewScriptLoader).
// This is the common case: load the script and install it.
// Case A and C:
scoped_refptr<ServiceWorkerVersion> version =
provider_host_->running_hosted_version();
int64_t resource_id =
version->script_cache_map()->LookupResourceId(resource_request.url);
if (resource_id != kInvalidServiceWorkerResourceId) {
std::unique_ptr<ServiceWorkerResponseReader> response_reader =
context_->storage()->CreateResponseReader(resource_id);
mojo::MakeStrongBinding(
std::make_unique<ServiceWorkerInstalledScriptLoader>(
options, std::move(client), std::move(response_reader), version,
resource_request.url),
std::move(request));
return;
}
// Case B:
if (ServiceWorkerVersion::IsInstalled(version->status())) {
client->OnComplete(network::URLLoaderCompletionStatus(net::ERR_FAILED));
return;
}
// Case D:
// Compared script info is used to decide which sub case should be used.
// If there is no compared script info, goto D.3 directly.
const auto& compared_script_info_map = version->compared_script_info_map();
if (!compared_script_info_map.empty()) {
// ServiceWorkerImportedScriptsUpdateCheck must be enabled if there is
// compared script info.
DCHECK(blink::ServiceWorkerUtils::IsImportedScriptUpdateCheckEnabled());
auto it = compared_script_info_map.find(resource_request.url);
if (it != compared_script_info_map.end()) {
switch (it->second.result) {
case ServiceWorkerSingleScriptUpdateChecker::Result::kFailed:
// Network failure is treated as D.1
case ServiceWorkerSingleScriptUpdateChecker::Result::kIdentical:
// Case D.1:
CopyScript(
it->first, it->second.old_resource_id,
base::BindOnce(
&ServiceWorkerScriptLoaderFactory::OnCopyScriptFinished,
weak_factory_.GetWeakPtr(), std::move(request), options,
resource_request, std::move(client)));
return;
case ServiceWorkerSingleScriptUpdateChecker::Result::kDifferent:
// Case D.2:
// TODO(https://crbug.com/648295): Currently, this case is treated
// the same as case D.3. In future, the paused state in compared
// script info should be resumed instead of a fresh download.
NOTIMPLEMENTED();
break;
case ServiceWorkerSingleScriptUpdateChecker::Result::kNotCompared:
// This is invalid, as scripts in compared script info must have been
// compared.
NOTREACHED();
return;
}
}
}
// Case D.3:
mojo::MakeStrongBinding(
std::make_unique<ServiceWorkerNewScriptLoader>(
routing_id, request_id, options, resource_request, std::move(client),
provider_host_->running_hosted_version(), loader_factory_,
traffic_annotation),
std::move(request));
}
void ServiceWorkerScriptLoaderFactory::Clone(
network::mojom::URLLoaderFactoryRequest request) {
bindings_.AddBinding(this, std::move(request));
}
bool ServiceWorkerScriptLoaderFactory::CheckIfScriptRequestIsValid(
const network::ResourceRequest& resource_request) {
if (!context_ || !provider_host_)
return false;
scoped_refptr<ServiceWorkerVersion> version =
provider_host_->running_hosted_version();
if (!version)
return false;
// Handle only the service worker main script (RESOURCE_TYPE_SERVICE_WORKER)
// or importScripts() (RESOURCE_TYPE_SCRIPT).
if (resource_request.resource_type != RESOURCE_TYPE_SERVICE_WORKER &&
resource_request.resource_type != RESOURCE_TYPE_SCRIPT) {
static auto* key = base::debug::AllocateCrashKeyString(
"swslf_bad_type", base::debug::CrashKeySize::Size32);
base::debug::SetCrashKeyString(
key, base::NumberToString(resource_request.resource_type));
mojo::ReportBadMessage("SWSLF_BAD_RESOURCE_TYPE");
return false;
}
if (version->status() == ServiceWorkerVersion::REDUNDANT) {
// This could happen if browser-side has set the status to redundant but
// the worker has not yet stopped. The worker is already doomed so just
// reject the request. Handle it specially here because otherwise it'd
// be unclear whether "REDUNDANT" should count as installed or not
// installed when making decisions about how to handle the request and
// logging UMA.
return false;
}
// TODO(falken): Make sure we don't handle a redirected request.
return true;
}
void ServiceWorkerScriptLoaderFactory::CopyScript(
const GURL& url,
int64_t resource_id,
base::OnceCallback<void(int64_t, net::Error)> callback) {
ServiceWorkerStorage* storage = context_->storage();
int64_t new_resource_id = storage->NewResourceId();
cache_writer_ = ServiceWorkerCacheWriter::CreateForCopy(
storage->CreateResponseReader(resource_id),
storage->CreateResponseWriter(new_resource_id));
scoped_refptr<ServiceWorkerVersion> version =
provider_host_->running_hosted_version();
version->script_cache_map()->NotifyStartedCaching(url, new_resource_id);
net::Error error = cache_writer_->StartCopy(
base::BindOnce(std::move(callback), new_resource_id));
// Run the callback directly if the operation completed or failed
// synchronously.
if (net::ERR_IO_PENDING != error) {
std::move(callback).Run(new_resource_id, error);
}
}
void ServiceWorkerScriptLoaderFactory::OnCopyScriptFinished(
network::mojom::URLLoaderRequest request,
uint32_t options,
const network::ResourceRequest& resource_request,
network::mojom::URLLoaderClientPtr client,
int64_t new_resource_id,
net::Error error) {
int64_t resource_size = cache_writer_->bytes_written();
cache_writer_.reset();
scoped_refptr<ServiceWorkerVersion> version =
provider_host_->running_hosted_version();
if (error != net::OK) {
version->script_cache_map()->NotifyFinishedCaching(
resource_request.url, resource_size, error,
kServiceWorkerCopyScriptError);
client->OnComplete(network::URLLoaderCompletionStatus(error));
return;
}
// The copy operation is successful, add the newly copied resource record to
// the script cache map to identify that the script is installed.
version->script_cache_map()->NotifyFinishedCaching(
resource_request.url, resource_size, net::OK, std::string());
// Use ServiceWorkerInstalledScriptLoader to load the new copy.
mojo::MakeStrongBinding(
std::make_unique<ServiceWorkerInstalledScriptLoader>(
options, std::move(client),
context_->storage()->CreateResponseReader(new_resource_id), version,
resource_request.url),
std::move(request));
}
} // namespace content