mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-24 05:11:16 +00:00
42a2a6727f
Differential Revision: https://phabricator.services.mozilla.com/D114080
220 lines
6.3 KiB
C++
220 lines
6.3 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 "PermissionRequestBase.h"
|
|
|
|
#include "IndexedDBCommon.h"
|
|
|
|
#include "MainThreadUtils.h"
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/Services.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "nsIObserverService.h"
|
|
#include "nsIPrincipal.h"
|
|
#include "nsXULAppAPI.h"
|
|
|
|
namespace mozilla::dom::indexedDB {
|
|
|
|
using namespace mozilla::services;
|
|
|
|
namespace {
|
|
|
|
#define IDB_PREFIX "indexedDB"
|
|
#define TOPIC_PREFIX IDB_PREFIX "-permissions-"
|
|
|
|
const nsLiteralCString kPermissionString = nsLiteralCString(IDB_PREFIX);
|
|
|
|
const char kPermissionPromptTopic[] = TOPIC_PREFIX "prompt";
|
|
|
|
#ifdef DEBUG
|
|
const char kPermissionResponseTopic[] = TOPIC_PREFIX "response";
|
|
#endif
|
|
|
|
#undef TOPIC_PREFIX
|
|
#undef IDB_PREFIX
|
|
|
|
const uint32_t kPermissionDefault = nsIPermissionManager::UNKNOWN_ACTION;
|
|
|
|
void AssertSanity() {
|
|
MOZ_ASSERT(XRE_IsParentProcess());
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
}
|
|
|
|
} // namespace
|
|
|
|
PermissionRequestBase::PermissionRequestBase(Element* aOwnerElement,
|
|
nsIPrincipal* aPrincipal)
|
|
: mOwnerElement(aOwnerElement), mPrincipal(aPrincipal) {
|
|
AssertSanity();
|
|
MOZ_ASSERT(aOwnerElement);
|
|
MOZ_ASSERT(aPrincipal);
|
|
}
|
|
|
|
PermissionRequestBase::~PermissionRequestBase() { AssertSanity(); }
|
|
|
|
// static
|
|
Result<PermissionRequestBase::PermissionValue, nsresult>
|
|
PermissionRequestBase::GetCurrentPermission(nsIPrincipal& aPrincipal) {
|
|
AssertSanity();
|
|
|
|
const nsCOMPtr<nsIPermissionManager> permMan = GetPermissionManager();
|
|
QM_TRY(OkIf(permMan), Err(NS_ERROR_FAILURE));
|
|
|
|
QM_TRY_INSPECT(const uint32_t& intPermission,
|
|
MOZ_TO_RESULT_INVOKE(permMan, TestExactPermissionFromPrincipal,
|
|
&aPrincipal, kPermissionString));
|
|
|
|
const PermissionValue permission =
|
|
PermissionValueForIntPermission(intPermission);
|
|
|
|
MOZ_ASSERT(permission == kPermissionAllowed ||
|
|
permission == kPermissionDenied ||
|
|
permission == kPermissionPrompt);
|
|
|
|
return permission;
|
|
}
|
|
|
|
// static
|
|
auto PermissionRequestBase::PermissionValueForIntPermission(
|
|
uint32_t aIntPermission) -> PermissionValue {
|
|
AssertSanity();
|
|
|
|
switch (aIntPermission) {
|
|
case kPermissionDefault:
|
|
return kPermissionPrompt;
|
|
case kPermissionAllowed:
|
|
return kPermissionAllowed;
|
|
case kPermissionDenied:
|
|
return kPermissionDenied;
|
|
default:
|
|
MOZ_CRASH("Bad permission!");
|
|
}
|
|
|
|
MOZ_CRASH("Should never get here!");
|
|
}
|
|
|
|
Result<PermissionRequestBase::PermissionValue, nsresult>
|
|
PermissionRequestBase::PromptIfNeeded() {
|
|
AssertSanity();
|
|
MOZ_ASSERT(mPrincipal);
|
|
|
|
// Tricky, we want to release the window and principal in all cases except
|
|
// when we successfully prompt.
|
|
nsCOMPtr<Element> element = std::move(mOwnerElement);
|
|
nsCOMPtr<nsIPrincipal> principal = std::move(mPrincipal);
|
|
|
|
QM_TRY_INSPECT(const PermissionValue& currentValue,
|
|
GetCurrentPermission(*principal));
|
|
MOZ_ASSERT(currentValue != kPermissionDefault);
|
|
|
|
if (currentValue == kPermissionPrompt) {
|
|
nsCOMPtr<nsIObserverService> obsSvc = GetObserverService();
|
|
QM_TRY(OkIf(obsSvc), Err(NS_ERROR_FAILURE));
|
|
|
|
// We're about to prompt so move the members back.
|
|
mOwnerElement = std::move(element);
|
|
mPrincipal = std::move(principal);
|
|
|
|
QM_TRY(obsSvc->NotifyObservers(static_cast<nsIObserver*>(this),
|
|
kPermissionPromptTopic, nullptr),
|
|
QM_PROPAGATE, [this](const auto&) {
|
|
// Finally release if we failed the prompt.
|
|
mOwnerElement = nullptr;
|
|
mPrincipal = nullptr;
|
|
});
|
|
}
|
|
|
|
return currentValue;
|
|
}
|
|
|
|
void PermissionRequestBase::SetExplicitPermission(nsIPrincipal* aPrincipal,
|
|
uint32_t aIntPermission) {
|
|
AssertSanity();
|
|
MOZ_ASSERT(aPrincipal);
|
|
MOZ_ASSERT(aIntPermission == kPermissionAllowed ||
|
|
aIntPermission == kPermissionDenied);
|
|
|
|
nsCOMPtr<nsIPermissionManager> permMan = GetPermissionManager();
|
|
if (NS_WARN_IF(!permMan)) {
|
|
return;
|
|
}
|
|
|
|
nsresult rv =
|
|
permMan->AddFromPrincipal(aPrincipal, kPermissionString, aIntPermission,
|
|
nsIPermissionManager::EXPIRE_NEVER,
|
|
/* aExpireTime */ 0);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
NS_IMPL_ISUPPORTS(PermissionRequestBase, nsIObserver, nsIIDBPermissionsRequest)
|
|
|
|
NS_IMETHODIMP
|
|
PermissionRequestBase::GetBrowserElement(Element** aElement) {
|
|
AssertSanity();
|
|
*aElement = do_AddRef(mOwnerElement).take();
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
PermissionRequestBase::GetResponseObserver(nsIObserver** aObserver) {
|
|
AssertSanity();
|
|
*aObserver = do_AddRef(this).take();
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
PermissionRequestBase::Observe(nsISupports* aSubject, const char* aTopic,
|
|
const char16_t* aData) {
|
|
AssertSanity();
|
|
MOZ_ASSERT(!strcmp(aTopic, kPermissionResponseTopic));
|
|
MOZ_ASSERT(mOwnerElement);
|
|
MOZ_ASSERT(mPrincipal);
|
|
|
|
const nsCOMPtr<Element> element = std::move(mOwnerElement);
|
|
Unused << element;
|
|
const nsCOMPtr<nsIPrincipal> principal = std::move(mPrincipal);
|
|
|
|
nsresult rv;
|
|
uint32_t promptResult = nsDependentString(aData).ToInteger(&rv);
|
|
MOZ_ALWAYS_SUCCEEDS(rv);
|
|
|
|
// The UI prompt code will only return one of these three values. We have to
|
|
// transform it to our values.
|
|
MOZ_ASSERT(promptResult == kPermissionDefault ||
|
|
promptResult == kPermissionAllowed ||
|
|
promptResult == kPermissionDenied);
|
|
|
|
if (promptResult != kPermissionDefault) {
|
|
// Save explicitly allowed or denied permissions now.
|
|
SetExplicitPermission(principal, promptResult);
|
|
}
|
|
|
|
PermissionValue permission;
|
|
switch (promptResult) {
|
|
case kPermissionDefault:
|
|
permission = kPermissionPrompt;
|
|
break;
|
|
|
|
case kPermissionAllowed:
|
|
permission = kPermissionAllowed;
|
|
break;
|
|
|
|
case kPermissionDenied:
|
|
permission = kPermissionDenied;
|
|
break;
|
|
|
|
default:
|
|
MOZ_CRASH("Bad prompt result!");
|
|
}
|
|
|
|
OnPromptComplete(permission);
|
|
return NS_OK;
|
|
}
|
|
|
|
} // namespace mozilla::dom::indexedDB
|