// 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 "modules/push_messaging/PushManager.h" #include "bindings/core/v8/ScriptPromise.h" #include "bindings/core/v8/ScriptPromiseResolver.h" #include "bindings/core/v8/ScriptState.h" #include "core/dom/DOMException.h" #include "core/dom/Document.h" #include "core/dom/ExceptionCode.h" #include "core/dom/ExecutionContext.h" #include "modules/push_messaging/PushController.h" #include "modules/push_messaging/PushError.h" #include "modules/push_messaging/PushPermissionStatusCallbacks.h" #include "modules/push_messaging/PushSubscription.h" #include "modules/push_messaging/PushSubscriptionCallbacks.h" #include "modules/push_messaging/PushSubscriptionOptions.h" #include "modules/serviceworkers/ServiceWorkerRegistration.h" #include "public/platform/Platform.h" #include "public/platform/modules/push_messaging/WebPushClient.h" #include "public/platform/modules/push_messaging/WebPushProvider.h" #include "public/platform/modules/push_messaging/WebPushSubscriptionOptions.h" #include "wtf/RefPtr.h" namespace blink { namespace { WebPushProvider* pushProvider() { WebPushProvider* webPushProvider = Platform::current()->pushProvider(); ASSERT(webPushProvider); return webPushProvider; } String bufferSourceToString(const ArrayBufferOrArrayBufferView& applicationServerKey, ExceptionState& exceptionState) { // Check the validity of the sender info. It must be a 65 byte unencrypted key, // which has the byte 0x04 as the first byte as a marker. char* input; int length; if (applicationServerKey.isArrayBuffer()) { input = static_cast( applicationServerKey.getAsArrayBuffer()->data()); length = applicationServerKey.getAsArrayBuffer()->byteLength(); } else if (applicationServerKey.isArrayBufferView()) { input = static_cast( applicationServerKey.getAsArrayBufferView()->buffer()->data()); length = applicationServerKey.getAsArrayBufferView()->buffer()->byteLength(); } else { ASSERT_NOT_REACHED(); return String(); } if (length == 65 && input[0] == 0x04) return WebString::fromUTF8(input, length); exceptionState.throwDOMException(InvalidAccessError, "The provided applicationServerKey is not valid."); return String(); } } // namespace PushManager::PushManager(ServiceWorkerRegistration* registration) : m_registration(registration) { ASSERT(registration); } WebPushSubscriptionOptions PushManager::toWebPushSubscriptionOptions(const PushSubscriptionOptions& options, ExceptionState& exceptionState) { WebPushSubscriptionOptions webOptions; webOptions.userVisibleOnly = options.userVisibleOnly(); if (options.hasApplicationServerKey()) { webOptions.applicationServerKey = bufferSourceToString(options.applicationServerKey(), exceptionState); } return webOptions; } ScriptPromise PushManager::subscribe(ScriptState* scriptState, const PushSubscriptionOptions& options, ExceptionState& exceptionState) { if (!m_registration->active()) return ScriptPromise::rejectWithDOMException(scriptState, DOMException::create(AbortError, "Subscription failed - no active Service Worker")); const WebPushSubscriptionOptions& webOptions = toWebPushSubscriptionOptions(options, exceptionState); if (exceptionState.hadException()) return ScriptPromise(); ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); ScriptPromise promise = resolver->promise(); // The document context is the only reasonable context from which to ask the user for permission // to use the Push API. The embedder should persist the permission so that later calls in // different contexts can succeed. if (scriptState->getExecutionContext()->isDocument()) { Document* document = toDocument(scriptState->getExecutionContext()); if (!document->domWindow() || !document->frame()) return ScriptPromise::rejectWithDOMException(scriptState, DOMException::create(InvalidStateError, "Document is detached from window.")); PushController::clientFrom(document->frame()).subscribe(m_registration->webRegistration(), webOptions, new PushSubscriptionCallbacks(resolver, m_registration)); } else { pushProvider()->subscribe(m_registration->webRegistration(), webOptions, new PushSubscriptionCallbacks(resolver, m_registration)); } return promise; } ScriptPromise PushManager::getSubscription(ScriptState* scriptState) { ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); ScriptPromise promise = resolver->promise(); pushProvider()->getSubscription(m_registration->webRegistration(), new PushSubscriptionCallbacks(resolver, m_registration)); return promise; } ScriptPromise PushManager::permissionState(ScriptState* scriptState, const PushSubscriptionOptions& options, ExceptionState& exceptionState) { if (scriptState->getExecutionContext()->isDocument()) { Document* document = toDocument(scriptState->getExecutionContext()); if (!document->domWindow() || !document->frame()) return ScriptPromise::rejectWithDOMException(scriptState, DOMException::create(InvalidStateError, "Document is detached from window.")); } ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); ScriptPromise promise = resolver->promise(); pushProvider()->getPermissionStatus(m_registration->webRegistration(), toWebPushSubscriptionOptions(options, exceptionState), new PushPermissionStatusCallbacks(resolver)); return promise; } DEFINE_TRACE(PushManager) { visitor->trace(m_registration); } } // namespace blink