blob: 0a1b880807e68a792e7fc5de0011fb9277362057 [file] [log] [blame]
// 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 "config.h"
#include "modules/credentialmanager/CredentialsContainer.h"
#include "bindings/core/v8/Dictionary.h"
#include "bindings/core/v8/ScriptPromise.h"
#include "bindings/core/v8/ScriptPromiseResolver.h"
#include "core/dom/DOMException.h"
#include "core/dom/ExceptionCode.h"
#include "core/dom/ExecutionContext.h"
#include "modules/credentialmanager/Credential.h"
#include "modules/credentialmanager/CredentialManagerClient.h"
#include "modules/credentialmanager/CredentialRequestOptions.h"
#include "modules/credentialmanager/FederatedCredential.h"
#include "modules/credentialmanager/FederatedCredentialRequestOptions.h"
#include "modules/credentialmanager/PasswordCredential.h"
#include "platform/weborigin/SecurityOrigin.h"
#include "public/platform/Platform.h"
#include "public/platform/WebCredential.h"
#include "public/platform/WebCredentialManagerClient.h"
#include "public/platform/WebCredentialManagerError.h"
#include "public/platform/WebFederatedCredential.h"
#include "public/platform/WebPasswordCredential.h"
namespace blink {
static void rejectDueToCredentialManagerError(ScriptPromiseResolver* resolver, WebCredentialManagerError* reason)
{
switch (reason->errorType) {
case WebCredentialManagerError::ErrorTypeDisabled:
resolver->reject(DOMException::create(InvalidStateError, "The credential manager is disabled."));
break;
case WebCredentialManagerError::ErrorTypeUnknown:
default:
resolver->reject(DOMException::create(NotReadableError, "An unknown error occured while talking to the credential manager."));
break;
}
}
class NotificationCallbacks : public WebCredentialManagerClient::NotificationCallbacks {
WTF_MAKE_NONCOPYABLE(NotificationCallbacks);
public:
explicit NotificationCallbacks(ScriptPromiseResolver* resolver) : m_resolver(resolver) { }
~NotificationCallbacks() override { }
void onSuccess() override
{
m_resolver->resolve();
}
void onError(WebCredentialManagerError* reason) override
{
rejectDueToCredentialManagerError(m_resolver, reason);
}
private:
const Persistent<ScriptPromiseResolver> m_resolver;
};
class RequestCallbacks : public WebCredentialManagerClient::RequestCallbacks {
WTF_MAKE_NONCOPYABLE(RequestCallbacks);
public:
explicit RequestCallbacks(ScriptPromiseResolver* resolver) : m_resolver(resolver) { }
~RequestCallbacks() override { }
void onSuccess(WebCredential* credential) override
{
if (!credential) {
m_resolver->resolve();
return;
}
ASSERT(credential->isPasswordCredential() || credential->isFederatedCredential());
if (credential->isPasswordCredential())
m_resolver->resolve(PasswordCredential::create(static_cast<WebPasswordCredential*>(credential)));
else
m_resolver->resolve(FederatedCredential::create(static_cast<WebFederatedCredential*>(credential)));
}
void onError(WebCredentialManagerError* reason) override
{
rejectDueToCredentialManagerError(m_resolver, reason);
}
private:
const Persistent<ScriptPromiseResolver> m_resolver;
};
CredentialsContainer* CredentialsContainer::create()
{
return new CredentialsContainer();
}
CredentialsContainer::CredentialsContainer()
{
}
static bool checkBoilerplate(ScriptPromiseResolver* resolver)
{
CredentialManagerClient* client = CredentialManagerClient::from(resolver->scriptState()->executionContext());
if (!client) {
resolver->reject(DOMException::create(InvalidStateError, "Could not establish connection to the credential manager."));
return false;
}
String errorMessage;
if (!resolver->scriptState()->executionContext()->isSecureContext(errorMessage)) {
resolver->reject(DOMException::create(SecurityError, errorMessage));
return false;
}
return true;
}
ScriptPromise CredentialsContainer::get(ScriptState* scriptState, const CredentialRequestOptions& options)
{
ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState);
ScriptPromise promise = resolver->promise();
if (!checkBoilerplate(resolver))
return promise;
Vector<KURL> providers;
if (options.hasFederated() && options.federated().hasProviders()) {
for (const auto& string : options.federated().providers()) {
KURL url = KURL(KURL(), string);
if (url.isValid())
providers.append(url);
}
}
CredentialManagerClient::from(scriptState->executionContext())->dispatchGet(options.suppressUI(), providers, new RequestCallbacks(resolver));
return promise;
}
ScriptPromise CredentialsContainer::store(ScriptState* scriptState, Credential* credential)
{
ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState);
ScriptPromise promise = resolver->promise();
if (!checkBoilerplate(resolver))
return promise;
CredentialManagerClient::from(scriptState->executionContext())->dispatchStore(WebCredential::create(credential->platformCredential()), new NotificationCallbacks(resolver));
return promise;
}
ScriptPromise CredentialsContainer::requireUserMediation(ScriptState* scriptState)
{
ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState);
ScriptPromise promise = resolver->promise();
if (!checkBoilerplate(resolver))
return promise;
CredentialManagerClient::from(scriptState->executionContext())->dispatchRequireUserMediation(new NotificationCallbacks(resolver));
return promise;
}
} // namespace blink