187 lines
5.0 KiB
C++
187 lines
5.0 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#include "data/components/passkeys.h"
|
|
|
|
#include "apiwrap.h"
|
|
#include "data/data_passkey_deserialize.h"
|
|
#include "main/main_app_config.h"
|
|
#include "main/main_session.h"
|
|
#include "platform/platform_webauthn.h"
|
|
|
|
namespace Data {
|
|
namespace {
|
|
|
|
constexpr auto kTimeoutMs = 5000;
|
|
|
|
[[nodiscard]] PasskeyEntry FromTL(const MTPDpasskey &data) {
|
|
return PasskeyEntry{
|
|
.id = qs(data.vid()),
|
|
.name = qs(data.vname()),
|
|
.date = data.vdate().v,
|
|
.softwareEmojiId = data.vsoftware_emoji_id().value_or(0),
|
|
.lastUsageDate = data.vlast_usage_date().value_or(0),
|
|
};
|
|
}
|
|
|
|
} // namespace
|
|
|
|
Passkeys::Passkeys(not_null<Main::Session*> session)
|
|
: _session(session) {
|
|
}
|
|
|
|
Passkeys::~Passkeys() = default;
|
|
|
|
void Passkeys::initRegistration(
|
|
Fn<void(const Data::Passkey::RegisterData&)> done) {
|
|
_session->api().request(MTPaccount_InitPasskeyRegistration(
|
|
)).done([=](const MTPaccount_PasskeyRegistrationOptions &result) {
|
|
const auto &data = result.data();
|
|
const auto jsonData = data.voptions().data().vdata().v;
|
|
if (const auto p = Data::Passkey::DeserializeRegisterData(jsonData)) {
|
|
done(*p);
|
|
}
|
|
}).send();
|
|
}
|
|
|
|
void Passkeys::registerPasskey(
|
|
const Platform::WebAuthn::RegisterResult &result,
|
|
Fn<void()> done) {
|
|
const auto credentialIdBase64 = QString::fromUtf8(
|
|
result.credentialId.toBase64(QByteArray::Base64UrlEncoding));
|
|
_session->api().request(MTPaccount_RegisterPasskey(
|
|
MTP_inputPasskeyCredentialPublicKey(
|
|
MTP_string(credentialIdBase64),
|
|
MTP_string(credentialIdBase64),
|
|
MTP_inputPasskeyResponseRegister(
|
|
MTP_dataJSON(MTP_bytes(result.clientDataJSON)),
|
|
MTP_bytes(result.attestationObject)))
|
|
)).done([=](const MTPPasskey &result) {
|
|
_passkeys.emplace_back(FromTL(result.data()));
|
|
_listUpdated.fire({});
|
|
done();
|
|
}).send();
|
|
}
|
|
|
|
void Passkeys::deletePasskey(
|
|
const QString &id,
|
|
Fn<void()> done,
|
|
Fn<void(QString)> fail) {
|
|
_session->api().request(MTPaccount_DeletePasskey(
|
|
MTP_string(id)
|
|
)).done([=] {
|
|
_lastRequestTime = 0;
|
|
_listKnown = false;
|
|
loadList();
|
|
done();
|
|
}).fail([=](const MTP::Error &error) {
|
|
fail(error.type());
|
|
}).send();
|
|
}
|
|
|
|
rpl::producer<> Passkeys::requestList() {
|
|
if (crl::now() - _lastRequestTime > kTimeoutMs) {
|
|
if (!_listRequestId) {
|
|
loadList();
|
|
}
|
|
return _listUpdated.events();
|
|
} else {
|
|
return _listUpdated.events_starting_with(rpl::empty_value());
|
|
}
|
|
}
|
|
|
|
const std::vector<PasskeyEntry> &Passkeys::list() const {
|
|
return _passkeys;
|
|
}
|
|
|
|
bool Passkeys::listKnown() const {
|
|
return _listKnown;
|
|
}
|
|
|
|
void Passkeys::loadList() {
|
|
_lastRequestTime = crl::now();
|
|
_listRequestId = _session->api().request(MTPaccount_GetPasskeys(
|
|
)).done([=](const MTPaccount_Passkeys &result) {
|
|
_listRequestId = 0;
|
|
_listKnown = true;
|
|
const auto &data = result.data();
|
|
_passkeys.clear();
|
|
_passkeys.reserve(data.vpasskeys().v.size());
|
|
for (const auto &passkey : data.vpasskeys().v) {
|
|
_passkeys.emplace_back(FromTL(passkey.data()));
|
|
}
|
|
_listUpdated.fire({});
|
|
}).fail([=] {
|
|
_listRequestId = 0;
|
|
}).send();
|
|
}
|
|
|
|
bool Passkeys::canRegister() const {
|
|
const auto max = _session->appConfig().passkeysAccountPasskeysMax();
|
|
return Platform::WebAuthn::IsSupported() && _passkeys.size() < max;
|
|
}
|
|
|
|
bool Passkeys::possible() const {
|
|
return _session->appConfig().settingsDisplayPasskeys();
|
|
}
|
|
|
|
void InitPasskeyLogin(
|
|
MTP::Sender &api,
|
|
Fn<void(const Data::Passkey::LoginData&)> done) {
|
|
api.request(MTPauth_InitPasskeyLogin(
|
|
MTP_int(ApiId),
|
|
MTP_string(ApiHash)
|
|
)).done([=](const MTPauth_PasskeyLoginOptions &result) {
|
|
const auto &data = result.data();
|
|
if (const auto p = Passkey::DeserializeLoginData(
|
|
data.voptions().data().vdata().v)) {
|
|
done(*p);
|
|
}
|
|
}).send();
|
|
}
|
|
|
|
void FinishPasskeyLogin(
|
|
MTP::Sender &api,
|
|
int initialDc,
|
|
const Platform::WebAuthn::LoginResult &result,
|
|
Fn<void(const MTPauth_Authorization&)> done,
|
|
Fn<void(QString)> fail) {
|
|
const auto userHandleStr = QString::fromUtf8(result.userHandle);
|
|
const auto parts = userHandleStr.split(':');
|
|
if (parts.size() != 2) {
|
|
return;
|
|
}
|
|
const auto userDc = parts[0].toInt();
|
|
const auto credentialIdBase64 = result.credentialId.toBase64(
|
|
QByteArray::Base64UrlEncoding | QByteArray::OmitTrailingEquals);
|
|
const auto credential = MTP_inputPasskeyCredentialPublicKey(
|
|
MTP_string(credentialIdBase64.toStdString()),
|
|
MTP_string(credentialIdBase64.toStdString()),
|
|
MTP_inputPasskeyResponseLogin(
|
|
MTP_dataJSON(MTP_bytes(result.clientDataJSON)),
|
|
MTP_bytes(result.authenticatorData),
|
|
MTP_bytes(result.signature),
|
|
MTP_string(userHandleStr.toStdString())
|
|
)
|
|
);
|
|
const auto flags = (userDc != initialDc)
|
|
? MTPauth_finishPasskeyLogin::Flag::f_from_dc_id
|
|
: MTPauth_finishPasskeyLogin::Flags(0);
|
|
api.request(MTPauth_FinishPasskeyLogin(
|
|
MTP_flags(flags),
|
|
credential,
|
|
MTP_int(initialDc),
|
|
MTP_long(0)
|
|
)).toDC(
|
|
userDc
|
|
).done(done).fail([=](const MTP::Error &error) {
|
|
fail(error.type());
|
|
}).send();
|
|
}
|
|
|
|
} // namespace Data
|