blob: 79ca442d0b7095b66b8d774fb26109917d957b2d [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/public/test/service_worker_test_helpers.h"
#include <memory>
#include <utility>
#include "base/bind.h"
#include "base/bind_helpers.h"
#include "base/run_loop.h"
#include "base/task/post_task.h"
#include "content/browser/service_worker/service_worker_context_core_observer.h"
#include "content/browser/service_worker/service_worker_context_wrapper.h"
#include "content/public/browser/browser_task_traits.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/service_worker_context.h"
#include "third_party/blink/public/common/notifications/platform_notification_data.h"
#include "url/gurl.h"
namespace content {
namespace {
class StoppedObserver : public base::RefCountedThreadSafe<StoppedObserver> {
public:
static void StartObserving(ServiceWorkerContextWrapper* context,
int64_t service_worker_version_id,
base::OnceClosure completion_callback_ui) {
auto observer = base::WrapRefCounted(
new StoppedObserver(std::move(completion_callback_ui)));
// Adds a ref to StoppedObserver to keep |this| around until the worker is
// stopped.
observer->inner_observer_ = std::make_unique<Observer>(
context, service_worker_version_id,
base::BindOnce(&StoppedObserver::OnStopped, observer));
}
private:
friend class base::RefCountedThreadSafe<StoppedObserver>;
explicit StoppedObserver(base::OnceClosure completion_callback_ui)
: completion_callback_ui_(std::move(completion_callback_ui)) {}
~StoppedObserver() {}
class Observer : public ServiceWorkerContextCoreObserver {
public:
Observer(ServiceWorkerContextWrapper* context,
int64_t service_worker_version_id,
base::OnceClosure stopped_callback)
: context_(context),
version_id_(service_worker_version_id),
stopped_callback_(std::move(stopped_callback)) {
context_->AddObserver(this);
}
// ServiceWorkerContextCoreObserver:
void OnStopped(int64_t version_id) override {
DCHECK_CURRENTLY_ON(ServiceWorkerContext::GetCoreThreadId());
if (version_id != version_id_)
return;
std::move(stopped_callback_).Run();
}
~Observer() override { context_->RemoveObserver(this); }
private:
ServiceWorkerContextWrapper* const context_;
int64_t version_id_;
base::OnceClosure stopped_callback_;
};
void OnStopped() {
if (!BrowserThread::CurrentlyOn(BrowserThread::UI)) {
base::PostTask(FROM_HERE, {BrowserThread::UI},
base::BindOnce(&StoppedObserver::OnStopped, this));
return;
}
std::move(completion_callback_ui_).Run();
}
std::unique_ptr<Observer> inner_observer_;
base::OnceClosure completion_callback_ui_;
DISALLOW_COPY_AND_ASSIGN(StoppedObserver);
};
void StopServiceWorkerForRegistration(
scoped_refptr<ServiceWorkerContextWrapper> context_wrapper,
base::OnceClosure completion_callback_ui,
blink::ServiceWorkerStatusCode service_worker_status,
scoped_refptr<ServiceWorkerRegistration> service_worker_registration) {
DCHECK(BrowserThread::CurrentlyOn(ServiceWorkerContext::GetCoreThreadId()));
DCHECK_EQ(blink::ServiceWorkerStatusCode::kOk, service_worker_status);
int64_t version_id =
service_worker_registration->active_version()->version_id();
StoppedObserver::StartObserving(context_wrapper.get(), version_id,
std::move(completion_callback_ui));
service_worker_registration->active_version()->embedded_worker()->Stop();
}
void DispatchNotificationClickForRegistration(
const blink::PlatformNotificationData& notification_data,
blink::ServiceWorkerStatusCode service_worker_status,
scoped_refptr<ServiceWorkerRegistration> service_worker_registration) {
DCHECK(BrowserThread::CurrentlyOn(ServiceWorkerContext::GetCoreThreadId()));
DCHECK_EQ(blink::ServiceWorkerStatusCode::kOk, service_worker_status);
scoped_refptr<ServiceWorkerVersion> version =
service_worker_registration->active_version();
version->StartRequest(ServiceWorkerMetrics::EventType::NOTIFICATION_CLICK,
base::DoNothing());
version->endpoint()->DispatchNotificationClickEvent(
"notification_id", notification_data, -1 /* action_index */,
base::nullopt /* reply */,
base::BindOnce([](blink::mojom::ServiceWorkerEventStatus event_status) {
DCHECK_EQ(blink::mojom::ServiceWorkerEventStatus::COMPLETED,
event_status);
}));
}
void FindReadyRegistrationForScope(
scoped_refptr<ServiceWorkerContextWrapper> context_wrapper,
const GURL& scope,
base::OnceCallback<void(blink::ServiceWorkerStatusCode,
scoped_refptr<ServiceWorkerRegistration>)>
callback) {
if (!BrowserThread::CurrentlyOn(ServiceWorkerContext::GetCoreThreadId())) {
base::PostTask(
FROM_HERE, ServiceWorkerContext::GetCoreThreadId(),
base::BindOnce(&FindReadyRegistrationForScope,
std::move(context_wrapper), scope, std::move(callback)));
return;
}
context_wrapper->FindReadyRegistrationForScope(scope, std::move(callback));
}
} // namespace
void StopServiceWorkerForScope(ServiceWorkerContext* context,
const GURL& scope,
base::OnceClosure completion_callback_ui) {
DCHECK(context);
scoped_refptr<ServiceWorkerContextWrapper> context_wrapper(
static_cast<ServiceWorkerContextWrapper*>(context));
FindReadyRegistrationForScope(
context_wrapper, scope,
base::BindOnce(&StopServiceWorkerForRegistration, context_wrapper,
std::move(completion_callback_ui)));
}
void DispatchServiceWorkerNotificationClick(
ServiceWorkerContext* context,
const GURL& scope,
const blink::PlatformNotificationData& notification_data) {
DCHECK(context);
scoped_refptr<ServiceWorkerContextWrapper> context_wrapper(
static_cast<ServiceWorkerContextWrapper*>(context));
FindReadyRegistrationForScope(
std::move(context_wrapper), scope,
base::BindOnce(&DispatchNotificationClickForRegistration,
notification_data));
}
} // namespace content