mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 05:37:11 +00:00
1367963d3d
- Bug 1198124 - Enable -Wshadow in directories that have no -Wshadow warnings. r=glandium (d03cfc1bb6) - Bug 1204403 - Fix -Wshadow warnings in xpcom. r=mccr8 (3247b5b146) - Bug 1140120 - Remove a couple of unused test functions on ServiceWorkerContainer; r=baku (efe1119749) - Bug 1180148 - Clear service workers registered for a site when clearing the cookies and stored data is not working in b2g. r=baku (7fb1d522ff) - Bug 1162333 - Add stronger assertions to PromiseWorkerProxy in debug builds. r=catalinb (0005a76252) - Bug 1190672 - Fix use of AutoReleasePromiseWorkerProxy in PushManager. r=catalinb (18f7358163) - Bug 1191495 - Relax PushManager callback assertions. r=kitcambridge (8a3822969f) - Bug 1189543 - pushManager.getSubscription() should return null for nonexistent push subscriptions in workers. r=nsm (b990083262) - spacing (44b87017a7) - Bug 1180295 - Rip out the Fennec code to set the screen render offset. r=rbarker (67183ae8bf) - Bug 1187804 - Un-adjust fixed layers by the async transform of the scroll frame that they're fixed with respect to, not of the nearest ancestor scroll frame. r=kats (7c5e7ac29c) - Bug 1131359 - Fix an include-what-you-use error in AsyncCompositionManager.cpp. r=kats (fe04815939) - Bug 1187804 - When a layer is scrolled by multiple scroll frames, do an AlignFixedAndStickyLayers pass on its subtree for each of the scroll frames. r=kats (031a5686f1) - Bug 1180295 - Stop clipping the content while the toolbar is in the process of sliding off. r=rbarker (9d806870e4) - Bug 1186968 - Fix Compositor::SetScreenRotation() call r=nical (e554c4b4a2) - Bug 1180295 - Implement seamless snapping to the stable state. r=rbarker (d1e7978eb7) - Bug 1131470 - Part 1: Rename existing use of ScreenOrientation to ScreenOrientationInternal. r=baku (63f0cb994b) - Bug 1131470 - Part 2: Update screen configuration HAL to report orientation angle. r=snorp,mwu (39b537f008) - Bug 1167597 - Mark PromiseReportRejectFeature::mPromise as MOZ_NON_OWNING_REF. r=ehsan (3ad520790a) - Bug 1086627 - Rename Promise constructs to more closely match the specification. r=nsm,jst (cea1d71cb1) - Bug 1086627 - Rename ThenableResolverTask to PromiseResolveThenableJob to more closely match Promise spec. r=nsm (e7102fe506) - Bug 1164725 - Convert flags in nsIDocument into bit fields. r=smaug (9f5d5f352d) - Bug 1192467 - Formalize precondition that SetDisplayDocument must not be called with null. - r=peterv (1aa8e91d14) - Bug 1153841 - Remove the 'it must be 2D' restriction for transforms on fixed-pos layers. r=BenWa,mattwoodrow (5bab86ec6a) - Bug 1173521 - Properly handle unapplying 3D projective transforms to rectangles in APZ code. r=botond (45c9850d5c) - Bug 1166301 - When applying an async adjustment to a fixed layer, only adjust its mask layer under the same circumstances as its clip rect. r=mattwoodrow (2801c48fe9) - Bug 1166301 - Update an old comment in AsyncCompositionManager::AlignFixedAndStickyLayers. r=kats (7e1d5c97d0) - Bug 1166301 - Allow async scrolling to move the clip rects of fixed background layers. r=mattwoodrow (edd889ec2b) - Bug 1200399 - Only send the RCD's metrics to Java-land for fennec-apz. r=rbarker (3b36a52f3a) - Bug 1201625 - Special-case B2GDroid since it doesn't have a root content document. r=rbarker (4fcdda1b0a) - Bug 1201529 - Fall back to rootmost metrics if there is no root content document metrics. r=botond (615c2ee727) - Bug 1201581 - Extract a helper method. r=rbarker (f2cccb8666) - Bug 1201581 - Hook up the syncFrameMetrics call to sync metrics info from the compositor to Java-land on each composite. r=rbarker (48e15bbde8) - Bug 1203760 P1 Allow pipe to wake up multiple streams at the same time. r=froydnj (5baed00637) - Bug 1203760 P2 Add gtests for nsPipeInputStream AsyncRead(). r=froydnj (52e5129dc8) - Bug 1201889 - When adjusting fixed and sticky layers in AsyncCompsitionManager, unapply all async transforms on the path from the fixed layer to the layer it's fixed with respect to. r=kats (dbb203178e) - Bug 1165536. Don't include resolution compensation when adjusting the cliprect of ancestors of scrollbars. r=botond (3bef5f2c18) - Fix async transforms on scrollbars not including any existing shadow transform. (bug 1128740 part 2, r=botond) (af12d76775) - missing of Bug 943728 - Replace double quotes with single quotes in Makefiles (e11470540f) - Bug 1188766 - Avoid exporting UPLOAD_EXTRA_FILES from mozconfig. r=mshal (4241b9e80b) - Fixup for bug 1188766 for valgrind bustage on a CLOSED TREE. r=me (4401dedf06) - Bug 1077622 - Fixes cp missing operand message if no .mozconfig found r=glandium (6eca1e5214) - Bug 1200523 - Remove leftover debugging code from js/src/Makefile.in; r=ted (414b72c74e) - spacing (99cda3f8a7)
793 lines
20 KiB
C++
793 lines
20 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "mozilla/dom/PushManager.h"
|
|
|
|
#include "mozilla/Preferences.h"
|
|
#include "mozilla/Services.h"
|
|
#include "mozilla/dom/PushManagerBinding.h"
|
|
#include "mozilla/dom/PushSubscriptionBinding.h"
|
|
#include "mozilla/dom/ServiceWorkerGlobalScopeBinding.h"
|
|
|
|
#include "mozilla/dom/Promise.h"
|
|
#include "mozilla/dom/PromiseWorkerProxy.h"
|
|
|
|
#include "nsIGlobalObject.h"
|
|
#include "nsIPermissionManager.h"
|
|
#include "nsIPrincipal.h"
|
|
#include "nsIPushClient.h"
|
|
|
|
#include "nsFrameMessageManager.h"
|
|
#include "nsContentCID.h"
|
|
|
|
#include "WorkerRunnable.h"
|
|
#include "WorkerPrivate.h"
|
|
#include "WorkerScope.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
using namespace workers;
|
|
|
|
class UnsubscribeResultCallback final : public nsIUnsubscribeResultCallback
|
|
{
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
|
|
explicit UnsubscribeResultCallback(Promise* aPromise)
|
|
: mPromise(aPromise)
|
|
{
|
|
AssertIsOnMainThread();
|
|
}
|
|
|
|
NS_IMETHOD
|
|
OnUnsubscribe(nsresult aStatus, bool aSuccess) override
|
|
{
|
|
if (NS_SUCCEEDED(aStatus)) {
|
|
mPromise->MaybeResolve(aSuccess);
|
|
} else {
|
|
mPromise->MaybeReject(NS_ERROR_DOM_NETWORK_ERR);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
~UnsubscribeResultCallback()
|
|
{}
|
|
|
|
nsRefPtr<Promise> mPromise;
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS(UnsubscribeResultCallback, nsIUnsubscribeResultCallback)
|
|
|
|
already_AddRefed<Promise>
|
|
PushSubscription::Unsubscribe(ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(mPrincipal);
|
|
|
|
nsCOMPtr<nsIPushClient> client =
|
|
do_CreateInstance("@mozilla.org/push/PushClient;1");
|
|
if (NS_WARN_IF(!client)) {
|
|
aRv = NS_ERROR_FAILURE;
|
|
return nullptr;
|
|
}
|
|
|
|
nsRefPtr<Promise> p = Promise::Create(mGlobal, aRv);
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsRefPtr<UnsubscribeResultCallback> callback =
|
|
new UnsubscribeResultCallback(p);
|
|
client->Unsubscribe(mScope, mPrincipal, callback);
|
|
return p.forget();
|
|
}
|
|
|
|
PushSubscription::PushSubscription(nsIGlobalObject* aGlobal,
|
|
const nsAString& aEndpoint,
|
|
const nsAString& aScope)
|
|
: mGlobal(aGlobal), mEndpoint(aEndpoint), mScope(aScope)
|
|
{
|
|
}
|
|
|
|
PushSubscription::~PushSubscription()
|
|
{}
|
|
|
|
JSObject*
|
|
PushSubscription::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return PushSubscriptionBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
void
|
|
PushSubscription::SetPrincipal(nsIPrincipal* aPrincipal)
|
|
{
|
|
MOZ_ASSERT(!mPrincipal);
|
|
mPrincipal = aPrincipal;
|
|
}
|
|
|
|
// static
|
|
already_AddRefed<PushSubscription>
|
|
PushSubscription::Constructor(GlobalObject& aGlobal, const nsAString& aEndpoint, const nsAString& aScope, ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(!aEndpoint.IsEmpty());
|
|
MOZ_ASSERT(!aScope.IsEmpty());
|
|
nsCOMPtr<nsIGlobalObject> global = do_QueryInterface(aGlobal.GetAsSupports());
|
|
nsRefPtr<PushSubscription> sub = new PushSubscription(global, aEndpoint, aScope);
|
|
return sub.forget();
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE(PushSubscription, mGlobal, mPrincipal)
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(PushSubscription)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(PushSubscription)
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(PushSubscription)
|
|
NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
|
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
PushManager::PushManager(nsIGlobalObject* aGlobal, const nsAString& aScope)
|
|
: mGlobal(aGlobal), mScope(aScope)
|
|
{
|
|
AssertIsOnMainThread();
|
|
}
|
|
|
|
PushManager::~PushManager()
|
|
{}
|
|
|
|
JSObject*
|
|
PushManager::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
// XXXnsm I don't know if this is the right way to do it, but I want to assert
|
|
// that an implementation has been set before this object gets exposed to JS.
|
|
MOZ_ASSERT(mImpl);
|
|
return PushManagerBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
void
|
|
PushManager::SetPushManagerImpl(PushManagerImpl& foo, ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
MOZ_ASSERT(!mImpl);
|
|
mImpl = &foo;
|
|
}
|
|
|
|
already_AddRefed<Promise>
|
|
PushManager::Subscribe(ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(mImpl);
|
|
return mImpl->Subscribe(aRv);
|
|
}
|
|
|
|
already_AddRefed<Promise>
|
|
PushManager::GetSubscription(ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(mImpl);
|
|
return mImpl->GetSubscription(aRv);
|
|
}
|
|
|
|
already_AddRefed<Promise>
|
|
PushManager::PermissionState(ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(mImpl);
|
|
return mImpl->PermissionState(aRv);
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE(PushManager, mGlobal, mImpl)
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(PushManager)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(PushManager)
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(PushManager)
|
|
NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
|
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
// WorkerPushSubscription
|
|
|
|
WorkerPushSubscription::WorkerPushSubscription(const nsAString& aEndpoint,
|
|
const nsAString& aScope)
|
|
: mEndpoint(aEndpoint), mScope(aScope)
|
|
{
|
|
MOZ_ASSERT(!aScope.IsEmpty());
|
|
MOZ_ASSERT(!aEndpoint.IsEmpty());
|
|
}
|
|
|
|
WorkerPushSubscription::~WorkerPushSubscription()
|
|
{}
|
|
|
|
JSObject*
|
|
WorkerPushSubscription::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return PushSubscriptionBinding_workers::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
// static
|
|
already_AddRefed<WorkerPushSubscription>
|
|
WorkerPushSubscription::Constructor(GlobalObject& aGlobal, const nsAString& aEndpoint, const nsAString& aScope, ErrorResult& aRv)
|
|
{
|
|
WorkerPrivate* worker = GetCurrentThreadWorkerPrivate();
|
|
MOZ_ASSERT(worker);
|
|
worker->AssertIsOnWorkerThread();
|
|
|
|
nsRefPtr<WorkerPushSubscription> sub = new WorkerPushSubscription(aEndpoint, aScope);
|
|
return sub.forget();
|
|
}
|
|
|
|
namespace {
|
|
// The caller MUST take ownership of the proxy's lock before it calls this.
|
|
void
|
|
ReleasePromiseWorkerProxy(already_AddRefed<PromiseWorkerProxy> aProxy)
|
|
{
|
|
AssertIsOnMainThread();
|
|
nsRefPtr<PromiseWorkerProxy> proxy = aProxy;
|
|
MOZ_ASSERT(proxy);
|
|
proxy->GetCleanUpLock().AssertCurrentThreadOwns();
|
|
if (proxy->IsClean()) {
|
|
return;
|
|
}
|
|
|
|
AutoJSAPI jsapi;
|
|
jsapi.Init();
|
|
|
|
nsRefPtr<PromiseWorkerProxyControlRunnable> cr =
|
|
new PromiseWorkerProxyControlRunnable(proxy->GetWorkerPrivate(),
|
|
proxy);
|
|
|
|
MOZ_ALWAYS_TRUE(cr->Dispatch(jsapi.cx()));
|
|
}
|
|
} // anonymous namespace
|
|
|
|
class UnsubscribeResultRunnable final : public WorkerRunnable
|
|
{
|
|
public:
|
|
UnsubscribeResultRunnable(PromiseWorkerProxy* aProxy,
|
|
nsresult aStatus,
|
|
bool aSuccess)
|
|
: WorkerRunnable(aProxy->GetWorkerPrivate(), WorkerThreadModifyBusyCount)
|
|
, mProxy(aProxy)
|
|
, mStatus(aStatus)
|
|
, mSuccess(aSuccess)
|
|
{
|
|
AssertIsOnMainThread();
|
|
}
|
|
|
|
bool
|
|
WorkerRun(JSContext* aCx, WorkerPrivate* aWorkerPrivate) override
|
|
{
|
|
MOZ_ASSERT(aWorkerPrivate);
|
|
aWorkerPrivate->AssertIsOnWorkerThread();
|
|
|
|
nsRefPtr<PromiseWorkerProxy> proxy = mProxy.forget();
|
|
nsRefPtr<Promise> promise = proxy->GetWorkerPromise();
|
|
if (NS_SUCCEEDED(mStatus)) {
|
|
promise->MaybeResolve(mSuccess);
|
|
} else {
|
|
promise->MaybeReject(NS_ERROR_DOM_NETWORK_ERR);
|
|
}
|
|
|
|
proxy->CleanUp(aCx);
|
|
return true;
|
|
}
|
|
private:
|
|
~UnsubscribeResultRunnable()
|
|
{}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> mProxy;
|
|
nsresult mStatus;
|
|
bool mSuccess;
|
|
};
|
|
|
|
class WorkerUnsubscribeResultCallback final : public nsIUnsubscribeResultCallback
|
|
{
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
|
|
explicit WorkerUnsubscribeResultCallback(PromiseWorkerProxy* aProxy)
|
|
: mProxy(aProxy)
|
|
{
|
|
AssertIsOnMainThread();
|
|
}
|
|
|
|
NS_IMETHOD
|
|
OnUnsubscribe(nsresult aStatus, bool aSuccess) override
|
|
{
|
|
AssertIsOnMainThread();
|
|
if (!mProxy) {
|
|
return NS_OK;
|
|
}
|
|
|
|
MutexAutoLock lock(mProxy->GetCleanUpLock());
|
|
if (mProxy->IsClean()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
AutoJSAPI jsapi;
|
|
jsapi.Init();
|
|
|
|
nsRefPtr<UnsubscribeResultRunnable> r =
|
|
new UnsubscribeResultRunnable(mProxy, aStatus, aSuccess);
|
|
if (!r->Dispatch(jsapi.cx())) {
|
|
ReleasePromiseWorkerProxy(mProxy.forget());
|
|
}
|
|
|
|
mProxy = nullptr;
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
~WorkerUnsubscribeResultCallback()
|
|
{
|
|
AssertIsOnMainThread();
|
|
if (mProxy) {
|
|
MutexAutoLock lock(mProxy->GetCleanUpLock());
|
|
if (!mProxy->IsClean()) {
|
|
AutoJSAPI jsapi;
|
|
jsapi.Init();
|
|
nsRefPtr<PromiseWorkerProxyControlRunnable> cr =
|
|
new PromiseWorkerProxyControlRunnable(mProxy->GetWorkerPrivate(), mProxy);
|
|
cr->Dispatch(jsapi.cx());
|
|
}
|
|
}
|
|
}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> mProxy;
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS(WorkerUnsubscribeResultCallback, nsIUnsubscribeResultCallback)
|
|
|
|
class UnsubscribeRunnable final : public nsRunnable
|
|
{
|
|
public:
|
|
UnsubscribeRunnable(PromiseWorkerProxy* aProxy,
|
|
const nsAString& aScope)
|
|
: mProxy(aProxy)
|
|
, mScope(aScope)
|
|
{
|
|
MOZ_ASSERT(aProxy);
|
|
MOZ_ASSERT(!aScope.IsEmpty());
|
|
}
|
|
|
|
NS_IMETHOD
|
|
Run() override
|
|
{
|
|
AssertIsOnMainThread();
|
|
MutexAutoLock lock(mProxy->GetCleanUpLock());
|
|
if (mProxy->IsClean()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsRefPtr<WorkerUnsubscribeResultCallback> callback =
|
|
new WorkerUnsubscribeResultCallback(mProxy);
|
|
|
|
nsCOMPtr<nsIPushClient> client =
|
|
do_CreateInstance("@mozilla.org/push/PushClient;1");
|
|
if (!client) {
|
|
callback->OnUnsubscribe(NS_ERROR_FAILURE, false);
|
|
}
|
|
|
|
nsCOMPtr<nsIPrincipal> principal = mProxy->GetWorkerPrivate()->GetPrincipal();
|
|
if (NS_WARN_IF(NS_FAILED(client->Unsubscribe(mScope, principal, callback)))) {
|
|
callback->OnUnsubscribe(NS_ERROR_FAILURE, false);
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
~UnsubscribeRunnable()
|
|
{}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> mProxy;
|
|
nsString mScope;
|
|
};
|
|
|
|
already_AddRefed<Promise>
|
|
WorkerPushSubscription::Unsubscribe(ErrorResult &aRv)
|
|
{
|
|
WorkerPrivate* worker = GetCurrentThreadWorkerPrivate();
|
|
MOZ_ASSERT(worker);
|
|
worker->AssertIsOnWorkerThread();
|
|
|
|
nsCOMPtr<nsIGlobalObject> global = worker->GlobalScope();
|
|
nsRefPtr<Promise> p = Promise::Create(global, aRv);
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> proxy = PromiseWorkerProxy::Create(worker, p);
|
|
if (!proxy) {
|
|
p->MaybeReject(NS_ERROR_DOM_NETWORK_ERR);
|
|
return p.forget();
|
|
}
|
|
|
|
nsRefPtr<UnsubscribeRunnable> r =
|
|
new UnsubscribeRunnable(proxy, mScope);
|
|
MOZ_ALWAYS_TRUE(NS_SUCCEEDED(NS_DispatchToMainThread(r)));
|
|
|
|
return p.forget();
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE_0(WorkerPushSubscription)
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(WorkerPushSubscription)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(WorkerPushSubscription)
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(WorkerPushSubscription)
|
|
NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
|
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
// WorkerPushManager
|
|
|
|
WorkerPushManager::WorkerPushManager(const nsAString& aScope)
|
|
: mScope(aScope)
|
|
{
|
|
}
|
|
|
|
JSObject*
|
|
WorkerPushManager::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return PushManagerBinding_workers::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
class GetSubscriptionResultRunnable final : public WorkerRunnable
|
|
{
|
|
public:
|
|
GetSubscriptionResultRunnable(PromiseWorkerProxy* aProxy,
|
|
nsresult aStatus,
|
|
const nsAString& aEndpoint,
|
|
const nsAString& aScope)
|
|
: WorkerRunnable(aProxy->GetWorkerPrivate(), WorkerThreadModifyBusyCount)
|
|
, mProxy(aProxy)
|
|
, mStatus(aStatus)
|
|
, mEndpoint(aEndpoint)
|
|
, mScope(aScope)
|
|
{ }
|
|
|
|
bool
|
|
WorkerRun(JSContext* aCx, WorkerPrivate* aWorkerPrivate) override
|
|
{
|
|
nsRefPtr<PromiseWorkerProxy> proxy = mProxy.forget();
|
|
nsRefPtr<Promise> promise = proxy->GetWorkerPromise();
|
|
if (NS_SUCCEEDED(mStatus)) {
|
|
if (mEndpoint.IsEmpty()) {
|
|
promise->MaybeResolve(JS::NullHandleValue);
|
|
} else {
|
|
nsRefPtr<WorkerPushSubscription> sub =
|
|
new WorkerPushSubscription(mEndpoint, mScope);
|
|
promise->MaybeResolve(sub);
|
|
}
|
|
} else {
|
|
promise->MaybeReject(NS_ERROR_DOM_ABORT_ERR);
|
|
}
|
|
|
|
proxy->CleanUp(aCx);
|
|
return true;
|
|
}
|
|
private:
|
|
~GetSubscriptionResultRunnable()
|
|
{}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> mProxy;
|
|
nsresult mStatus;
|
|
nsString mEndpoint;
|
|
nsString mScope;
|
|
};
|
|
|
|
class GetSubscriptionCallback final : public nsIPushEndpointCallback
|
|
{
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
|
|
explicit GetSubscriptionCallback(PromiseWorkerProxy* aProxy,
|
|
const nsAString& aScope)
|
|
: mProxy(aProxy)
|
|
, mScope(aScope)
|
|
{}
|
|
|
|
NS_IMETHOD
|
|
OnPushEndpoint(nsresult aStatus, const nsAString& aEndpoint) override
|
|
{
|
|
AssertIsOnMainThread();
|
|
|
|
if (!mProxy) {
|
|
return NS_OK;
|
|
}
|
|
|
|
MutexAutoLock lock(mProxy->GetCleanUpLock());
|
|
if (mProxy->IsClean()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
AutoJSAPI jsapi;
|
|
jsapi.Init();
|
|
|
|
nsRefPtr<GetSubscriptionResultRunnable> r =
|
|
new GetSubscriptionResultRunnable(mProxy, aStatus, aEndpoint, mScope);
|
|
if (!r->Dispatch(jsapi.cx())) {
|
|
ReleasePromiseWorkerProxy(mProxy.forget());
|
|
}
|
|
|
|
mProxy = nullptr;
|
|
return NS_OK;
|
|
}
|
|
|
|
protected:
|
|
~GetSubscriptionCallback()
|
|
{
|
|
AssertIsOnMainThread();
|
|
if (mProxy) {
|
|
MutexAutoLock lock(mProxy->GetCleanUpLock());
|
|
if (!mProxy->IsClean()) {
|
|
AutoJSAPI jsapi;
|
|
jsapi.Init();
|
|
nsRefPtr<PromiseWorkerProxyControlRunnable> cr =
|
|
new PromiseWorkerProxyControlRunnable(mProxy->GetWorkerPrivate(), mProxy);
|
|
cr->Dispatch(jsapi.cx());
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
nsRefPtr<PromiseWorkerProxy> mProxy;
|
|
nsString mScope;
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS(GetSubscriptionCallback, nsIPushEndpointCallback)
|
|
|
|
class GetSubscriptionRunnable final : public nsRunnable
|
|
{
|
|
public:
|
|
GetSubscriptionRunnable(PromiseWorkerProxy* aProxy,
|
|
const nsAString& aScope,
|
|
WorkerPushManager::SubscriptionAction aAction)
|
|
: mProxy(aProxy)
|
|
, mScope(aScope), mAction(aAction)
|
|
{}
|
|
|
|
NS_IMETHOD
|
|
Run() override
|
|
{
|
|
AssertIsOnMainThread();
|
|
MutexAutoLock lock(mProxy->GetCleanUpLock());
|
|
if (mProxy->IsClean()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsRefPtr<GetSubscriptionCallback> callback = new GetSubscriptionCallback(mProxy, mScope);
|
|
|
|
nsCOMPtr<nsIPermissionManager> permManager =
|
|
mozilla::services::GetPermissionManager();
|
|
if (!permManager) {
|
|
callback->OnPushEndpoint(NS_ERROR_FAILURE, EmptyString());
|
|
return NS_OK;
|
|
}
|
|
|
|
uint32_t permission = nsIPermissionManager::DENY_ACTION;
|
|
nsresult rv = permManager->TestExactPermissionFromPrincipal(
|
|
mProxy->GetWorkerPrivate()->GetPrincipal(),
|
|
"push",
|
|
&permission);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv)) || permission != nsIPermissionManager::ALLOW_ACTION) {
|
|
callback->OnPushEndpoint(NS_ERROR_FAILURE, EmptyString());
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIPushClient> client =
|
|
do_CreateInstance("@mozilla.org/push/PushClient;1");
|
|
if (!client) {
|
|
callback->OnPushEndpoint(NS_ERROR_FAILURE, EmptyString());
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIPrincipal> principal = mProxy->GetWorkerPrivate()->GetPrincipal();
|
|
mProxy = nullptr;
|
|
|
|
if (mAction == WorkerPushManager::SubscribeAction) {
|
|
rv = client->Subscribe(mScope, principal, callback);
|
|
} else {
|
|
MOZ_ASSERT(mAction == WorkerPushManager::GetSubscriptionAction);
|
|
rv = client->GetSubscription(mScope, principal, callback);
|
|
}
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
callback->OnPushEndpoint(NS_ERROR_FAILURE, EmptyString());
|
|
return rv;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
~GetSubscriptionRunnable()
|
|
{}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> mProxy;
|
|
nsString mScope;
|
|
WorkerPushManager::SubscriptionAction mAction;
|
|
};
|
|
|
|
already_AddRefed<Promise>
|
|
WorkerPushManager::PerformSubscriptionAction(SubscriptionAction aAction, ErrorResult& aRv)
|
|
{
|
|
WorkerPrivate* worker = GetCurrentThreadWorkerPrivate();
|
|
MOZ_ASSERT(worker);
|
|
worker->AssertIsOnWorkerThread();
|
|
|
|
nsCOMPtr<nsIGlobalObject> global = worker->GlobalScope();
|
|
nsRefPtr<Promise> p = Promise::Create(global, aRv);
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> proxy = PromiseWorkerProxy::Create(worker, p);
|
|
if (!proxy) {
|
|
p->MaybeReject(NS_ERROR_DOM_ABORT_ERR);
|
|
return p.forget();
|
|
}
|
|
|
|
nsRefPtr<GetSubscriptionRunnable> r =
|
|
new GetSubscriptionRunnable(proxy, mScope, aAction);
|
|
MOZ_ALWAYS_TRUE(NS_SUCCEEDED(NS_DispatchToMainThread(r)));
|
|
|
|
return p.forget();
|
|
}
|
|
|
|
already_AddRefed<Promise>
|
|
WorkerPushManager::Subscribe(ErrorResult& aRv)
|
|
{
|
|
return PerformSubscriptionAction(SubscribeAction, aRv);
|
|
}
|
|
|
|
already_AddRefed<Promise>
|
|
WorkerPushManager::GetSubscription(ErrorResult& aRv)
|
|
{
|
|
return PerformSubscriptionAction(GetSubscriptionAction, aRv);
|
|
}
|
|
|
|
class PermissionResultRunnable final : public WorkerRunnable
|
|
{
|
|
public:
|
|
PermissionResultRunnable(PromiseWorkerProxy *aProxy,
|
|
nsresult aStatus,
|
|
PushPermissionState aState)
|
|
: WorkerRunnable(aProxy->GetWorkerPrivate(), WorkerThreadModifyBusyCount)
|
|
, mProxy(aProxy)
|
|
, mStatus(aStatus)
|
|
, mState(aState)
|
|
{
|
|
AssertIsOnMainThread();
|
|
}
|
|
|
|
bool
|
|
WorkerRun(JSContext* aCx, WorkerPrivate* aWorkerPrivate) override
|
|
{
|
|
MOZ_ASSERT(aWorkerPrivate);
|
|
aWorkerPrivate->AssertIsOnWorkerThread();
|
|
|
|
nsRefPtr<PromiseWorkerProxy> proxy = mProxy.forget();
|
|
nsRefPtr<Promise> promise = proxy->GetWorkerPromise();
|
|
if (NS_SUCCEEDED(mStatus)) {
|
|
MOZ_ASSERT(uint32_t(mState) < ArrayLength(PushPermissionStateValues::strings));
|
|
nsAutoCString stringState(PushPermissionStateValues::strings[uint32_t(mState)].value, PushPermissionStateValues::strings[uint32_t(mState)].length);
|
|
promise->MaybeResolve(NS_ConvertUTF8toUTF16(stringState));
|
|
} else {
|
|
promise->MaybeReject(aCx, JS::UndefinedHandleValue);
|
|
}
|
|
|
|
proxy->CleanUp(aCx);
|
|
return true;
|
|
}
|
|
|
|
private:
|
|
~PermissionResultRunnable()
|
|
{}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> mProxy;
|
|
nsresult mStatus;
|
|
PushPermissionState mState;
|
|
};
|
|
|
|
class PermissionStateRunnable final : public nsRunnable
|
|
{
|
|
public:
|
|
explicit PermissionStateRunnable(PromiseWorkerProxy* aProxy)
|
|
: mProxy(aProxy)
|
|
{}
|
|
|
|
NS_IMETHOD
|
|
Run() override
|
|
{
|
|
AssertIsOnMainThread();
|
|
MutexAutoLock lock(mProxy->GetCleanUpLock());
|
|
if (mProxy->IsClean()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIPermissionManager> permManager =
|
|
mozilla::services::GetPermissionManager();
|
|
|
|
nsresult rv = NS_ERROR_FAILURE;
|
|
PushPermissionState state = PushPermissionState::Denied;
|
|
|
|
if (permManager) {
|
|
uint32_t permission = nsIPermissionManager::DENY_ACTION;
|
|
rv = permManager->TestExactPermissionFromPrincipal(
|
|
mProxy->GetWorkerPrivate()->GetPrincipal(),
|
|
"push",
|
|
&permission);
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
switch (permission) {
|
|
case nsIPermissionManager::ALLOW_ACTION:
|
|
state = PushPermissionState::Granted;
|
|
break;
|
|
case nsIPermissionManager::DENY_ACTION:
|
|
state = PushPermissionState::Denied;
|
|
break;
|
|
case nsIPermissionManager::PROMPT_ACTION:
|
|
state = PushPermissionState::Prompt;
|
|
break;
|
|
default:
|
|
MOZ_CRASH("Unexpected case!");
|
|
}
|
|
}
|
|
}
|
|
|
|
AutoJSAPI jsapi;
|
|
jsapi.Init();
|
|
nsRefPtr<PermissionResultRunnable> r =
|
|
new PermissionResultRunnable(mProxy, rv, state);
|
|
if (!r->Dispatch(jsapi.cx())) {
|
|
ReleasePromiseWorkerProxy(mProxy.forget());
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
~PermissionStateRunnable()
|
|
{}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> mProxy;
|
|
};
|
|
|
|
already_AddRefed<Promise>
|
|
WorkerPushManager::PermissionState(ErrorResult& aRv)
|
|
{
|
|
WorkerPrivate* worker = GetCurrentThreadWorkerPrivate();
|
|
MOZ_ASSERT(worker);
|
|
worker->AssertIsOnWorkerThread();
|
|
|
|
nsCOMPtr<nsIGlobalObject> global = worker->GlobalScope();
|
|
nsRefPtr<Promise> p = Promise::Create(global, aRv);
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsRefPtr<PromiseWorkerProxy> proxy = PromiseWorkerProxy::Create(worker, p);
|
|
if (!proxy) {
|
|
p->MaybeReject(worker->GetJSContext(), JS::UndefinedHandleValue);
|
|
return p.forget();
|
|
}
|
|
|
|
nsRefPtr<PermissionStateRunnable> r =
|
|
new PermissionStateRunnable(proxy);
|
|
NS_DispatchToMainThread(r);
|
|
|
|
return p.forget();
|
|
}
|
|
|
|
WorkerPushManager::~WorkerPushManager()
|
|
{}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE_0(WorkerPushManager)
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(WorkerPushManager)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(WorkerPushManager)
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(WorkerPushManager)
|
|
NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
|
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
|
NS_INTERFACE_MAP_END
|
|
} // namespace dom
|
|
} // namespace mozilla
|