2017-02-23 09:32:28 +00:00
|
|
|
/*
|
|
|
|
This file is part of Telegram Desktop,
|
|
|
|
the official desktop version of Telegram messaging app, see https://telegram.org
|
|
|
|
|
|
|
|
Telegram Desktop is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
It is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
In addition, as a special exception, the copyright holders give permission
|
|
|
|
to link the code of portions of this program with the OpenSSL library.
|
|
|
|
|
|
|
|
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
|
|
|
|
Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
|
|
|
*/
|
|
|
|
#include "auth_session.h"
|
|
|
|
|
2017-04-06 19:02:40 +00:00
|
|
|
#include "apiwrap.h"
|
2017-02-23 10:59:19 +00:00
|
|
|
#include "messenger.h"
|
2017-03-04 19:36:59 +00:00
|
|
|
#include "storage/file_download.h"
|
2017-04-09 18:06:06 +00:00
|
|
|
#include "storage/localstorage.h"
|
2017-03-04 19:36:59 +00:00
|
|
|
#include "window/notifications_manager.h"
|
2017-04-15 18:48:54 +00:00
|
|
|
#include "platform/platform_specific.h"
|
2017-04-19 09:44:07 +00:00
|
|
|
#include "calls/calls_instance.h"
|
2017-04-15 18:48:54 +00:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
constexpr auto kAutoLockTimeoutLateMs = TimeMs(3000);
|
|
|
|
|
|
|
|
} // namespace
|
2017-02-23 09:32:28 +00:00
|
|
|
|
2017-03-28 12:30:38 +00:00
|
|
|
QByteArray AuthSessionData::serialize() const {
|
|
|
|
auto size = sizeof(qint32) * 2;
|
|
|
|
|
|
|
|
auto result = QByteArray();
|
|
|
|
result.reserve(size);
|
|
|
|
{
|
|
|
|
QBuffer buffer(&result);
|
|
|
|
if (!buffer.open(QIODevice::WriteOnly)) {
|
|
|
|
Unexpected("Can't open data for AuthSessionData::serialize()");
|
|
|
|
}
|
|
|
|
|
|
|
|
QDataStream stream(&buffer);
|
|
|
|
stream.setVersion(QDataStream::Qt_5_1);
|
2017-03-29 11:50:09 +00:00
|
|
|
stream << static_cast<qint32>(_variables.emojiPanelTab);
|
2017-03-28 12:30:38 +00:00
|
|
|
stream << qint32(_variables.lastSeenWarningSeen ? 1 : 0);
|
2017-04-09 18:06:06 +00:00
|
|
|
stream << qint32(_variables.tabbedSelectorSectionEnabled ? 1 : 0);
|
2017-03-28 12:30:38 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AuthSessionData::constructFromSerialized(const QByteArray &serialized) {
|
|
|
|
if (serialized.isEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto readonly = serialized;
|
|
|
|
QBuffer buffer(&readonly);
|
|
|
|
if (!buffer.open(QIODevice::ReadOnly)) {
|
|
|
|
Unexpected("Can't open data for DcOptions::constructFromSerialized()");
|
|
|
|
}
|
|
|
|
QDataStream stream(&buffer);
|
|
|
|
stream.setVersion(QDataStream::Qt_5_1);
|
2017-03-29 11:50:09 +00:00
|
|
|
qint32 emojiPanTab = static_cast<qint32>(EmojiPanelTab::Emoji);
|
2017-03-28 12:30:38 +00:00
|
|
|
qint32 lastSeenWarningSeen = 0;
|
2017-04-09 18:06:06 +00:00
|
|
|
qint32 tabbedSelectorSectionEnabled = 1;
|
2017-03-28 12:30:38 +00:00
|
|
|
stream >> emojiPanTab;
|
|
|
|
stream >> lastSeenWarningSeen;
|
2017-04-09 18:06:06 +00:00
|
|
|
if (!stream.atEnd()) {
|
|
|
|
stream >> tabbedSelectorSectionEnabled;
|
|
|
|
}
|
2017-03-28 12:30:38 +00:00
|
|
|
if (stream.status() != QDataStream::Ok) {
|
|
|
|
LOG(("App Error: Bad data for AuthSessionData::constructFromSerialized()"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-03-29 11:50:09 +00:00
|
|
|
auto uncheckedTab = static_cast<EmojiPanelTab>(emojiPanTab);
|
2017-03-28 12:30:38 +00:00
|
|
|
switch (uncheckedTab) {
|
2017-03-29 11:50:09 +00:00
|
|
|
case EmojiPanelTab::Emoji:
|
|
|
|
case EmojiPanelTab::Stickers:
|
|
|
|
case EmojiPanelTab::Gifs: _variables.emojiPanelTab = uncheckedTab; break;
|
2017-03-28 12:30:38 +00:00
|
|
|
}
|
|
|
|
_variables.lastSeenWarningSeen = (lastSeenWarningSeen == 1);
|
2017-04-09 18:06:06 +00:00
|
|
|
_variables.tabbedSelectorSectionEnabled = (tabbedSelectorSectionEnabled == 1);
|
2017-03-28 12:30:38 +00:00
|
|
|
}
|
|
|
|
|
2017-03-04 19:36:59 +00:00
|
|
|
AuthSession::AuthSession(UserId userId)
|
|
|
|
: _userId(userId)
|
2017-04-15 21:45:25 +00:00
|
|
|
, _autoLockTimer([this] { checkAutoLock(); })
|
2017-04-06 19:02:40 +00:00
|
|
|
, _api(std::make_unique<ApiWrap>())
|
2017-04-19 09:44:07 +00:00
|
|
|
, _calls(std::make_unique<Calls::Instance>())
|
2017-03-04 19:36:59 +00:00
|
|
|
, _downloader(std::make_unique<Storage::Downloader>())
|
2017-04-15 21:45:25 +00:00
|
|
|
, _notifications(std::make_unique<Window::Notifications::System>(this)) {
|
2017-03-28 12:30:38 +00:00
|
|
|
Expects(_userId != 0);
|
2017-04-09 18:06:06 +00:00
|
|
|
_saveDataTimer.setCallback([this] {
|
|
|
|
Local::writeUserSettings();
|
|
|
|
});
|
2017-04-15 18:48:54 +00:00
|
|
|
subscribe(Messenger::Instance().passcodedChanged(), [this] {
|
|
|
|
_shouldLockAt = 0;
|
|
|
|
notifications().updateAll();
|
|
|
|
});
|
2017-02-23 09:32:28 +00:00
|
|
|
}
|
|
|
|
|
2017-03-04 19:36:59 +00:00
|
|
|
bool AuthSession::Exists() {
|
2017-04-14 12:12:40 +00:00
|
|
|
if (auto messenger = Messenger::InstancePointer()) {
|
|
|
|
return (messenger->authSession() != nullptr);
|
|
|
|
}
|
|
|
|
return false;
|
2017-03-04 19:36:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AuthSession &AuthSession::Current() {
|
|
|
|
auto result = Messenger::Instance().authSession();
|
|
|
|
t_assert(result != nullptr);
|
|
|
|
return *result;
|
2017-02-23 09:32:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
UserData *AuthSession::CurrentUser() {
|
2017-03-04 19:36:59 +00:00
|
|
|
return App::user(CurrentUserId());
|
|
|
|
}
|
|
|
|
|
|
|
|
base::Observable<void> &AuthSession::CurrentDownloaderTaskFinished() {
|
2017-03-15 16:24:06 +00:00
|
|
|
return Current().downloader().taskFinished();
|
2017-03-04 19:36:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool AuthSession::validateSelf(const MTPUser &user) {
|
|
|
|
if (user.type() != mtpc_user || !user.c_user().is_self() || user.c_user().vid.v != userId()) {
|
|
|
|
LOG(("Auth Error: wrong self user received."));
|
|
|
|
App::logOutDelayed();
|
|
|
|
return false;
|
2017-02-23 09:32:28 +00:00
|
|
|
}
|
2017-03-04 19:36:59 +00:00
|
|
|
return true;
|
2017-02-23 09:32:28 +00:00
|
|
|
}
|
2017-03-04 19:36:59 +00:00
|
|
|
|
2017-04-09 18:06:06 +00:00
|
|
|
void AuthSession::saveDataDelayed(TimeMs delay) {
|
|
|
|
Expects(this == &AuthSession::Current());
|
|
|
|
_saveDataTimer.callOnce(delay);
|
|
|
|
}
|
|
|
|
|
2017-04-15 18:48:54 +00:00
|
|
|
void AuthSession::checkAutoLock() {
|
|
|
|
if (!Global::LocalPasscode() || App::passcoded()) return;
|
|
|
|
|
|
|
|
Messenger::Instance().checkLocalTime();
|
|
|
|
auto now = getms(true);
|
|
|
|
auto shouldLockInMs = Global::AutoLock() * 1000LL;
|
|
|
|
auto idleForMs = psIdleTime();
|
|
|
|
auto notPlayingVideoForMs = now - data().lastTimeVideoPlayedAt();
|
|
|
|
auto checkTimeMs = qMin(idleForMs, notPlayingVideoForMs);
|
|
|
|
if (checkTimeMs >= shouldLockInMs || (_shouldLockAt > 0 && now > _shouldLockAt + kAutoLockTimeoutLateMs)) {
|
|
|
|
Messenger::Instance().setupPasscode();
|
|
|
|
} else {
|
|
|
|
_shouldLockAt = now + (shouldLockInMs - checkTimeMs);
|
|
|
|
_autoLockTimer.callOnce(shouldLockInMs - checkTimeMs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AuthSession::checkAutoLockIn(TimeMs time) {
|
|
|
|
if (_autoLockTimer.isActive()) {
|
|
|
|
auto remain = _autoLockTimer.remainingTime();
|
|
|
|
if (remain > 0 && remain <= time) return;
|
|
|
|
}
|
|
|
|
_autoLockTimer.callOnce(time);
|
|
|
|
}
|
|
|
|
|
2017-03-04 19:36:59 +00:00
|
|
|
AuthSession::~AuthSession() = default;
|