| // Copyright 2014 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/push_messaging/push_messaging_router.h" |
| |
| #include <string> |
| |
| #include "base/bind.h" |
| #include "base/metrics/histogram_macros.h" |
| #include "base/task/post_task.h" |
| #include "content/browser/devtools/devtools_background_services_context_impl.h" |
| #include "content/browser/service_worker/service_worker_context_wrapper.h" |
| #include "content/browser/service_worker/service_worker_registration.h" |
| #include "content/browser/service_worker/service_worker_storage.h" |
| #include "content/browser/storage_partition_impl.h" |
| #include "content/public/browser/browser_context.h" |
| #include "content/public/browser/browser_task_traits.h" |
| #include "content/public/browser/browser_thread.h" |
| #include "third_party/blink/public/common/service_worker/service_worker_status_code.h" |
| #include "third_party/blink/public/mojom/push_messaging/push_messaging_status.mojom.h" |
| |
| namespace content { |
| |
| namespace { |
| |
| void RunDeliverCallback( |
| const PushMessagingRouter::DeliverMessageCallback& deliver_message_callback, |
| blink::mojom::PushDeliveryStatus delivery_status) { |
| DCHECK_CURRENTLY_ON(BrowserThread::IO); |
| base::PostTaskWithTraits( |
| FROM_HERE, {BrowserThread::UI}, |
| base::BindOnce(deliver_message_callback, delivery_status)); |
| } |
| |
| } // namespace |
| |
| // static |
| void PushMessagingRouter::DeliverMessage( |
| BrowserContext* browser_context, |
| const GURL& origin, |
| int64_t service_worker_registration_id, |
| const std::string& message_id, |
| base::Optional<std::string> payload, |
| const DeliverMessageCallback& deliver_message_callback) { |
| DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| StoragePartition* partition = |
| BrowserContext::GetStoragePartitionForSite(browser_context, origin); |
| scoped_refptr<ServiceWorkerContextWrapper> service_worker_context = |
| static_cast<ServiceWorkerContextWrapper*>( |
| partition->GetServiceWorkerContext()); |
| auto devtools_context = |
| base::WrapRefCounted<DevToolsBackgroundServicesContextImpl>( |
| service_worker_context->storage_partition() |
| ->GetDevToolsBackgroundServicesContext()); |
| base::PostTaskWithTraits( |
| FROM_HERE, {BrowserThread::IO}, |
| base::BindOnce(&PushMessagingRouter::FindServiceWorkerRegistration, |
| std::move(service_worker_context), |
| std::move(devtools_context), origin, |
| service_worker_registration_id, message_id, |
| std::move(payload), deliver_message_callback)); |
| } |
| |
| // static |
| void PushMessagingRouter::FindServiceWorkerRegistration( |
| scoped_refptr<ServiceWorkerContextWrapper> service_worker_context, |
| scoped_refptr<DevToolsBackgroundServicesContextImpl> devtools_context, |
| const GURL& origin, |
| int64_t service_worker_registration_id, |
| const std::string& message_id, |
| base::Optional<std::string> payload, |
| const DeliverMessageCallback& deliver_message_callback) { |
| DCHECK_CURRENTLY_ON(BrowserThread::IO); |
| // Try to acquire the registration from storage. If it's already live we'll |
| // receive it right away. If not, it will be revived from storage. |
| service_worker_context->FindReadyRegistrationForId( |
| service_worker_registration_id, origin, |
| base::BindOnce( |
| &PushMessagingRouter::FindServiceWorkerRegistrationCallback, |
| std::move(devtools_context), message_id, std::move(payload), |
| deliver_message_callback)); |
| } |
| |
| // static |
| void PushMessagingRouter::FindServiceWorkerRegistrationCallback( |
| scoped_refptr<DevToolsBackgroundServicesContextImpl> devtools_context, |
| const std::string& message_id, |
| base::Optional<std::string> payload, |
| const DeliverMessageCallback& deliver_message_callback, |
| blink::ServiceWorkerStatusCode service_worker_status, |
| scoped_refptr<ServiceWorkerRegistration> service_worker_registration) { |
| DCHECK_CURRENTLY_ON(BrowserThread::IO); |
| UMA_HISTOGRAM_ENUMERATION("PushMessaging.DeliveryStatus.FindServiceWorker", |
| service_worker_status); |
| if (service_worker_status == blink::ServiceWorkerStatusCode::kErrorNotFound) { |
| RunDeliverCallback(deliver_message_callback, |
| blink::mojom::PushDeliveryStatus::NO_SERVICE_WORKER); |
| return; |
| } |
| if (service_worker_status != blink::ServiceWorkerStatusCode::kOk) { |
| RunDeliverCallback(deliver_message_callback, |
| blink::mojom::PushDeliveryStatus::SERVICE_WORKER_ERROR); |
| return; |
| } |
| |
| ServiceWorkerVersion* version = service_worker_registration->active_version(); |
| DCHECK(version); |
| |
| // Hold on to the service worker registration in the callback to keep it |
| // alive until the callback dies. Otherwise the registration could be |
| // released when this method returns - before the event is delivered to the |
| // service worker. |
| version->RunAfterStartWorker( |
| ServiceWorkerMetrics::EventType::PUSH, |
| base::BindOnce(&PushMessagingRouter::DeliverMessageToWorker, |
| base::WrapRefCounted(version), |
| std::move(service_worker_registration), |
| std::move(devtools_context), message_id, |
| std::move(payload), deliver_message_callback)); |
| } |
| |
| // static |
| void PushMessagingRouter::DeliverMessageToWorker( |
| scoped_refptr<ServiceWorkerVersion> service_worker, |
| scoped_refptr<ServiceWorkerRegistration> service_worker_registration, |
| scoped_refptr<DevToolsBackgroundServicesContextImpl> devtools_context, |
| const std::string& message_id, |
| base::Optional<std::string> payload, |
| const DeliverMessageCallback& deliver_message_callback, |
| blink::ServiceWorkerStatusCode start_worker_status) { |
| DCHECK_CURRENTLY_ON(BrowserThread::IO); |
| if (start_worker_status != blink::ServiceWorkerStatusCode::kOk) { |
| DeliverMessageEnd(std::move(service_worker), |
| std::move(service_worker_registration), |
| std::move(devtools_context), message_id, |
| deliver_message_callback, start_worker_status); |
| return; |
| } |
| |
| int request_id = service_worker->StartRequestWithCustomTimeout( |
| ServiceWorkerMetrics::EventType::PUSH, |
| base::BindOnce(&PushMessagingRouter::DeliverMessageEnd, service_worker, |
| std::move(service_worker_registration), devtools_context, |
| message_id, deliver_message_callback), |
| base::TimeDelta::FromSeconds(blink::mojom::kPushEventTimeoutSeconds), |
| ServiceWorkerVersion::KILL_ON_TIMEOUT); |
| |
| service_worker->endpoint()->DispatchPushEvent( |
| payload, service_worker->CreateSimpleEventCallback(request_id)); |
| |
| if (devtools_context->IsRecording( |
| DevToolsBackgroundService::kPushMessaging)) { |
| std::map<std::string, std::string> event_metadata; |
| if (payload) |
| event_metadata["Payload"] = *payload; |
| devtools_context->LogBackgroundServiceEventOnIO( |
| service_worker->registration_id(), service_worker->script_origin(), |
| DevToolsBackgroundService::kPushMessaging, "Push event dispatched", |
| message_id, event_metadata); |
| } |
| } |
| |
| // static |
| void PushMessagingRouter::DeliverMessageEnd( |
| scoped_refptr<ServiceWorkerVersion> service_worker, |
| scoped_refptr<ServiceWorkerRegistration> service_worker_registration, |
| scoped_refptr<DevToolsBackgroundServicesContextImpl> devtools_context, |
| const std::string& message_id, |
| const DeliverMessageCallback& deliver_message_callback, |
| blink::ServiceWorkerStatusCode service_worker_status) { |
| DCHECK_CURRENTLY_ON(BrowserThread::IO); |
| UMA_HISTOGRAM_ENUMERATION("PushMessaging.DeliveryStatus.ServiceWorkerEvent", |
| service_worker_status); |
| blink::mojom::PushDeliveryStatus delivery_status = |
| blink::mojom::PushDeliveryStatus::SERVICE_WORKER_ERROR; |
| std::string status_description; |
| switch (service_worker_status) { |
| case blink::ServiceWorkerStatusCode::kOk: |
| delivery_status = blink::mojom::PushDeliveryStatus::SUCCESS; |
| status_description = "Success"; |
| break; |
| case blink::ServiceWorkerStatusCode::kErrorEventWaitUntilRejected: |
| delivery_status = |
| blink::mojom::PushDeliveryStatus::EVENT_WAITUNTIL_REJECTED; |
| status_description = "waitUntil Rejected"; |
| break; |
| case blink::ServiceWorkerStatusCode::kErrorTimeout: |
| delivery_status = blink::mojom::PushDeliveryStatus::TIMEOUT; |
| status_description = "Timeout"; |
| break; |
| case blink::ServiceWorkerStatusCode::kErrorFailed: |
| case blink::ServiceWorkerStatusCode::kErrorAbort: |
| case blink::ServiceWorkerStatusCode::kErrorStartWorkerFailed: |
| case blink::ServiceWorkerStatusCode::kErrorProcessNotFound: |
| case blink::ServiceWorkerStatusCode::kErrorNotFound: |
| case blink::ServiceWorkerStatusCode::kErrorIpcFailed: |
| case blink::ServiceWorkerStatusCode::kErrorScriptEvaluateFailed: |
| case blink::ServiceWorkerStatusCode::kErrorDiskCache: |
| case blink::ServiceWorkerStatusCode::kErrorRedundant: |
| case blink::ServiceWorkerStatusCode::kErrorDisallowed: |
| delivery_status = blink::mojom::PushDeliveryStatus::SERVICE_WORKER_ERROR; |
| break; |
| case blink::ServiceWorkerStatusCode::kErrorExists: |
| case blink::ServiceWorkerStatusCode::kErrorInstallWorkerFailed: |
| case blink::ServiceWorkerStatusCode::kErrorActivateWorkerFailed: |
| case blink::ServiceWorkerStatusCode::kErrorNetwork: |
| case blink::ServiceWorkerStatusCode::kErrorSecurity: |
| case blink::ServiceWorkerStatusCode::kErrorState: |
| case blink::ServiceWorkerStatusCode::kErrorInvalidArguments: |
| NOTREACHED() << "Got unexpected error code: " |
| << static_cast<uint32_t>(service_worker_status) << " " |
| << blink::ServiceWorkerStatusToString(service_worker_status); |
| delivery_status = blink::mojom::PushDeliveryStatus::SERVICE_WORKER_ERROR; |
| break; |
| } |
| RunDeliverCallback(deliver_message_callback, delivery_status); |
| |
| if (devtools_context->IsRecording( |
| DevToolsBackgroundService::kPushMessaging) && |
| delivery_status != |
| blink::mojom::PushDeliveryStatus::SERVICE_WORKER_ERROR) { |
| devtools_context->LogBackgroundServiceEventOnIO( |
| service_worker->registration_id(), service_worker->script_origin(), |
| DevToolsBackgroundService::kPushMessaging, "Push event completed", |
| message_id, {{"Status", status_description}}); |
| } |
| } |
| |
| } // namespace content |