Source code

Revision control

Copy as Markdown

Other Tools

/* -*- 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 "IdentityCredentialRequestManager.h"
#include "mozilla/ClearOnShutdown.h"
#include "nsContentUtils.h"
namespace mozilla {
NS_IMPL_ISUPPORTS0(IdentityCredentialRequestManager);
StaticRefPtr<IdentityCredentialRequestManager>
IdentityCredentialRequestManager::sSingleton;
// static
IdentityCredentialRequestManager*
IdentityCredentialRequestManager::GetInstance() {
if (!sSingleton) {
sSingleton = new IdentityCredentialRequestManager();
ClearOnShutdown(&sSingleton);
}
return sSingleton;
}
RefPtr<MozPromise<std::tuple<nsCString, Maybe<nsCString>>, nsresult, true>>
IdentityCredentialRequestManager::GetTokenFromPopup(
dom::WebIdentityParent* aRelyingPartyWindow, nsIURI* aURLToOpen) {
MOZ_ASSERT(aRelyingPartyWindow);
MOZ_ASSERT(aURLToOpen);
// Create the promise that will be resolved *after* the child process opens
// the window
RefPtr<MozPromise<std::tuple<nsCString, Maybe<nsCString>>, nsresult,
true>::Private>
result = new MozPromise<std::tuple<nsCString, Maybe<nsCString>>, nsresult,
true>::Private(__func__);
NotNull<nsIURI*> uri = WrapNotNull(aURLToOpen);
RefPtr<IdentityCredentialRequestManager> self = this;
// Tell the RP child to open an IDP popup.
// It will either resolve with a failing nsresult or a BC ID of the popup.
aRelyingPartyWindow->SendOpenContinuationWindow(
uri,
[result, self](const dom::OpenContinuationWindowResponse& response) {
// If it failed, reject now, rejecting the RP child's initial call.
if (response.type() == dom::OpenContinuationWindowResponse::Tnsresult) {
result->Reject(response.get_nsresult(), __func__);
return;
}
// If we have a BC ID, a popup opened.
if (response.type() == dom::OpenContinuationWindowResponse::Tuint64_t) {
RefPtr<dom::CanonicalBrowsingContext> bc =
dom::CanonicalBrowsingContext::Get(response.get_uint64_t());
if (!bc) {
result->Reject(NS_ERROR_DOM_NETWORK_ERR, __func__);
}
// Transform the BC ID into its top-chrome-window-bc, so we have
// something stable through navigation and can listen for the popup's
// close.
dom::CanonicalBrowsingContext* chromeBC =
bc->TopCrossChromeBoundary();
if (!chromeBC) {
result->Reject(NS_ERROR_DOM_NETWORK_ERR, __func__);
}
// There really shouldn't be more than one request per top window
MOZ_ASSERT(!self->mPendingTokenRequests.Contains(chromeBC->Id()));
// Insert a refptr to the promise so we can settle it later, and we
// can find it by the BC id!
self->mPendingTokenRequests.InsertOrUpdate(chromeBC->Id(), result);
// If the window closes before we have a chance to resolve it,
// remove the promise from our map and reject it.
chromeBC->AddFinalDiscardListener([self](uint64_t id) {
Maybe<RefPtr<MozPromise<std::tuple<nsCString, Maybe<nsCString>>,
nsresult, true>::Private>>
pending = self->mPendingTokenRequests.Extract(id);
// If it already settled before the window closed, just drop the
// promise ref.
if (pending.isNothing()) {
return;
}
pending.value()->Reject(NS_ERROR_DOM_NETWORK_ERR, __func__);
});
}
},
[result](const ipc::ResponseRejectReason& rejection) {
result->Reject(NS_ERROR_DOM_NETWORK_ERR, __func__);
});
return result.forget();
}
nsresult IdentityCredentialRequestManager::MaybeResolvePopup(
dom::WebIdentityParent* aPopupWindow, const nsCString& aToken,
const dom::IdentityResolveOptions& aOptions) {
// aPopupWindow is (theoretically) a popup window opened by
// SendOpenContinuationWindow. So try to get its top chrome bc so we can look
// into the map!
dom::WindowGlobalParent* manager =
static_cast<dom::WindowGlobalParent*>(aPopupWindow->Manager());
if (!manager) {
return NS_ERROR_DOM_NOT_ALLOWED_ERR;
}
dom::CanonicalBrowsingContext* bc = manager->BrowsingContext();
if (!bc) {
return NS_ERROR_DOM_NOT_ALLOWED_ERR;
}
dom::CanonicalBrowsingContext* chromeBC = bc->TopCrossChromeBoundary();
if (!chromeBC) {
return NS_ERROR_DOM_NOT_ALLOWED_ERR;
}
// Get its entry, removing it from the map.
Maybe<RefPtr<MozPromise<std::tuple<nsCString, Maybe<nsCString>>, nsresult,
true>::Private>>
pendingPromise = mPendingTokenRequests.Extract(chromeBC->Id());
// This will be Nothing if the function was called on a window not opened by
// SendOpenContinuationWindow. This error will be forwarded along to the JS
// caller.
if (!pendingPromise.isSome()) {
return NS_ERROR_DOM_NOT_ALLOWED_ERR;
}
// Convert the Optional to a Maybe and send a successful response to the RP
// window that opened this popup.
Maybe<nsCString> overrideAccountId = Nothing();
if (aOptions.mAccountId.WasPassed()) {
overrideAccountId = Some(aOptions.mAccountId.Value());
}
pendingPromise.value()->Resolve(std::make_tuple(aToken, overrideAccountId),
__func__);
return NS_OK;
}
bool IdentityCredentialRequestManager::IsActivePopup(
dom::WebIdentityParent* aPopupWindow) {
dom::WindowGlobalParent* manager =
static_cast<dom::WindowGlobalParent*>(aPopupWindow->Manager());
if (!manager) {
return false;
}
dom::CanonicalBrowsingContext* bc = manager->BrowsingContext();
if (!bc) {
return false;
}
// Transform the BC ID into its top-chrome-window-bc, so we have
// something stable through navigation and can listen for the popup's close.
dom::CanonicalBrowsingContext* chromeBC = bc->TopCrossChromeBoundary();
if (!chromeBC) {
return false;
}
return mPendingTokenRequests.Contains(chromeBC->Id());
}
} // namespace mozilla