2018-06-26 14:09:39 +02:00
|
|
|
#include "providers/twitch/TwitchAccountManager.hpp"
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-06-26 15:33:51 +02:00
|
|
|
#include "common/Common.hpp"
|
2020-11-21 16:20:10 +01:00
|
|
|
#include "common/QLogging.hpp"
|
Sort and force grouping of includes (#4172)
This change enforces strict include grouping using IncludeCategories
In addition to adding this to the .clang-format file and applying it in the tests/src and src directories, I also did the following small changes:
In ChatterSet.hpp, I changed lrucache to a <>include
In Irc2.hpp, I change common/SignalVector.hpp to a "project-include"
In AttachedWindow.cpp, NativeMessaging.cpp, WindowsHelper.hpp, BaseWindow.cpp, and StreamerMode.cpp, I disabled clang-format for the windows-includes
In WindowDescriptors.hpp, I added the missing vector include. It was previously not needed because the include was handled by another file that was previously included first.
clang-format minimum version has been bumped, so Ubuntu version used in the check-formatting job has been bumped to 22.04 (which is the latest LTS)
2022-11-27 19:32:53 +01:00
|
|
|
#include "providers/twitch/api/Helix.hpp"
|
2018-08-11 22:23:06 +02:00
|
|
|
#include "providers/twitch/TwitchAccount.hpp"
|
2018-06-26 17:25:24 +02:00
|
|
|
#include "providers/twitch/TwitchCommon.hpp"
|
2022-12-31 15:41:01 +01:00
|
|
|
#include "providers/twitch/TwitchUser.hpp"
|
|
|
|
#include "util/SharedPtrElementLess.hpp"
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
namespace chatterino {
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
TwitchAccountManager::TwitchAccountManager()
|
2020-02-23 19:49:45 +01:00
|
|
|
: accounts(SharedPtrElementLess<TwitchAccount>{})
|
|
|
|
, anonymousUser_(new TwitchAccount(ANONYMOUS_USERNAME, "", "", ""))
|
2017-12-31 00:50:07 +01:00
|
|
|
{
|
2018-05-12 20:34:13 +02:00
|
|
|
this->currentUserChanged.connect([this] {
|
|
|
|
auto currentUser = this->getCurrent();
|
2021-02-14 14:01:13 +01:00
|
|
|
currentUser->loadBlocks();
|
2023-07-29 11:49:44 +02:00
|
|
|
currentUser->loadSeventvUserID();
|
2018-05-12 20:34:13 +02:00
|
|
|
});
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2023-09-16 13:52:51 +02:00
|
|
|
// We can safely ignore this signal connection since accounts will always be removed
|
|
|
|
// before TwitchAccountManager
|
|
|
|
std::ignore = this->accounts.itemRemoved.connect([this](const auto &acc) {
|
2019-08-23 16:52:04 +02:00
|
|
|
this->removeUser(acc.item.get());
|
|
|
|
});
|
2017-12-31 00:50:07 +01:00
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-02-05 15:11:50 +01:00
|
|
|
std::shared_ptr<TwitchAccount> TwitchAccountManager::getCurrent()
|
2017-12-31 00:50:07 +01:00
|
|
|
{
|
2018-07-06 19:23:47 +02:00
|
|
|
if (!this->currentUser_)
|
|
|
|
{
|
|
|
|
return this->anonymousUser_;
|
2017-12-31 00:50:07 +01:00
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-07-06 19:23:47 +02:00
|
|
|
return this->currentUser_;
|
2017-12-31 00:50:07 +01:00
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
std::vector<QString> TwitchAccountManager::getUsernames() const
|
|
|
|
{
|
|
|
|
std::vector<QString> userNames;
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-07-06 19:23:47 +02:00
|
|
|
std::lock_guard<std::mutex> lock(this->mutex_);
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2019-07-31 22:29:07 +02:00
|
|
|
for (const auto &user : this->accounts)
|
2018-05-28 08:34:54 +02:00
|
|
|
{
|
2017-12-31 00:50:07 +01:00
|
|
|
userNames.push_back(user->getUserName());
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
return userNames;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-02-05 15:11:50 +01:00
|
|
|
std::shared_ptr<TwitchAccount> TwitchAccountManager::findUserByUsername(
|
2017-12-31 00:50:07 +01:00
|
|
|
const QString &username) const
|
|
|
|
{
|
2018-07-06 19:23:47 +02:00
|
|
|
std::lock_guard<std::mutex> lock(this->mutex_);
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2019-07-31 22:29:07 +02:00
|
|
|
for (const auto &user : this->accounts)
|
2018-05-28 08:34:54 +02:00
|
|
|
{
|
2017-12-31 00:50:07 +01:00
|
|
|
if (username.compare(user->getUserName(), Qt::CaseInsensitive) == 0)
|
|
|
|
{
|
|
|
|
return user;
|
|
|
|
}
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
return nullptr;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
bool TwitchAccountManager::userExists(const QString &username) const
|
|
|
|
{
|
|
|
|
return this->findUserByUsername(username) != nullptr;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
void TwitchAccountManager::reloadUsers()
|
|
|
|
{
|
|
|
|
auto keys = pajlada::Settings::SettingManager::getObjectKeys("/accounts");
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
UserData userData;
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
bool listUpdated = false;
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
for (const auto &uid : keys)
|
|
|
|
{
|
|
|
|
if (uid == "current")
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-11-03 13:00:07 +01:00
|
|
|
auto username = pajlada::Settings::Setting<QString>::get(
|
2017-12-31 00:50:07 +01:00
|
|
|
"/accounts/" + uid + "/username");
|
2018-11-03 13:00:07 +01:00
|
|
|
auto userID = pajlada::Settings::Setting<QString>::get("/accounts/" +
|
|
|
|
uid + "/userID");
|
|
|
|
auto clientID = pajlada::Settings::Setting<QString>::get(
|
2017-12-31 00:50:07 +01:00
|
|
|
"/accounts/" + uid + "/clientID");
|
2018-11-03 13:00:07 +01:00
|
|
|
auto oauthToken = pajlada::Settings::Setting<QString>::get(
|
2017-12-31 00:50:07 +01:00
|
|
|
"/accounts/" + uid + "/oauthToken");
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-11-03 13:00:07 +01:00
|
|
|
if (username.isEmpty() || userID.isEmpty() || clientID.isEmpty() ||
|
|
|
|
oauthToken.isEmpty())
|
2017-12-31 00:50:07 +01:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-11-03 13:00:07 +01:00
|
|
|
userData.username = username.trimmed();
|
|
|
|
userData.userID = userID.trimmed();
|
|
|
|
userData.clientID = clientID.trimmed();
|
|
|
|
userData.oauthToken = oauthToken.trimmed();
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
switch (this->addUser(userData))
|
|
|
|
{
|
2019-09-26 00:51:05 +02:00
|
|
|
case AddUserResponse::UserAlreadyExists: {
|
2020-11-21 16:20:10 +01:00
|
|
|
qCDebug(chatterinoTwitch)
|
|
|
|
<< "User" << userData.username << "already exists";
|
2017-12-31 00:50:07 +01:00
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
break;
|
2019-09-26 00:51:05 +02:00
|
|
|
case AddUserResponse::UserValuesUpdated: {
|
2020-11-21 16:20:10 +01:00
|
|
|
qCDebug(chatterinoTwitch)
|
|
|
|
<< "User" << userData.username
|
|
|
|
<< "already exists, and values updated!";
|
2018-02-05 15:11:50 +01:00
|
|
|
if (userData.username == this->getCurrent()->getUserName())
|
|
|
|
{
|
2020-11-21 16:20:10 +01:00
|
|
|
qCDebug(chatterinoTwitch)
|
|
|
|
<< "It was the current user, so we need to "
|
|
|
|
"reconnect stuff!";
|
2022-05-07 17:22:39 +02:00
|
|
|
this->currentUserChanged();
|
2017-12-31 00:50:07 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2019-09-26 00:51:05 +02:00
|
|
|
case AddUserResponse::UserAdded: {
|
2020-11-21 16:20:10 +01:00
|
|
|
qCDebug(chatterinoTwitch) << "Added user" << userData.username;
|
2017-12-31 00:50:07 +01:00
|
|
|
listUpdated = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
if (listUpdated)
|
|
|
|
{
|
|
|
|
this->userListUpdated.invoke();
|
|
|
|
}
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-04-22 14:38:10 +02:00
|
|
|
void TwitchAccountManager::load()
|
|
|
|
{
|
|
|
|
this->reloadUsers();
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-11-03 13:00:07 +01:00
|
|
|
this->currentUsername.connect([this](const QString &newUsername) {
|
2018-04-22 14:38:10 +02:00
|
|
|
auto user = this->findUserByUsername(newUsername);
|
|
|
|
if (user)
|
|
|
|
{
|
2020-11-21 16:20:10 +01:00
|
|
|
qCDebug(chatterinoTwitch)
|
|
|
|
<< "Twitch user updated to" << newUsername;
|
2020-03-14 12:13:57 +01:00
|
|
|
getHelix()->update(user->getOAuthClient(), user->getOAuthToken());
|
2018-07-06 19:23:47 +02:00
|
|
|
this->currentUser_ = user;
|
2018-04-22 14:38:10 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-11-21 16:20:10 +01:00
|
|
|
qCDebug(chatterinoTwitch) << "Twitch user updated to anonymous";
|
2018-07-06 19:23:47 +02:00
|
|
|
this->currentUser_ = this->anonymousUser_;
|
2018-04-22 14:38:10 +02:00
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2022-05-07 17:22:39 +02:00
|
|
|
this->currentUserChanged();
|
2018-04-22 14:38:10 +02:00
|
|
|
});
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-06-24 15:09:56 +02:00
|
|
|
bool TwitchAccountManager::isLoggedIn() const
|
|
|
|
{
|
2018-07-06 19:23:47 +02:00
|
|
|
if (!this->currentUser_)
|
|
|
|
{
|
2018-06-24 15:09:56 +02:00
|
|
|
return false;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-06-24 15:09:56 +02:00
|
|
|
// Once `TwitchAccount` class has a way to check, we should also return
|
|
|
|
// false if the credentials are incorrect
|
2018-07-06 19:23:47 +02:00
|
|
|
return !this->currentUser_->isAnon();
|
2018-06-24 15:09:56 +02:00
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-05-28 08:51:39 +02:00
|
|
|
bool TwitchAccountManager::removeUser(TwitchAccount *account)
|
2017-12-31 00:50:07 +01:00
|
|
|
{
|
2020-01-03 20:51:37 +01:00
|
|
|
static const QString accountFormat("/accounts/uid%1");
|
|
|
|
|
2018-11-03 13:00:07 +01:00
|
|
|
auto userID(account->getUserId());
|
|
|
|
if (!userID.isEmpty())
|
2018-05-28 08:51:39 +02:00
|
|
|
{
|
2018-11-03 13:00:07 +01:00
|
|
|
pajlada::Settings::SettingManager::removeSetting(
|
2020-01-03 20:51:37 +01:00
|
|
|
accountFormat.arg(userID).toStdString());
|
2018-05-28 08:34:54 +02:00
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-11-03 13:00:07 +01:00
|
|
|
if (account->getUserName() == this->currentUsername)
|
2018-05-28 08:51:39 +02:00
|
|
|
{
|
2017-12-31 00:50:07 +01:00
|
|
|
// The user that was removed is the current user, log into the anonymous
|
|
|
|
// user
|
|
|
|
this->currentUsername = "";
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
this->userListUpdated.invoke();
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
return true;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
TwitchAccountManager::AddUserResponse TwitchAccountManager::addUser(
|
|
|
|
const TwitchAccountManager::UserData &userData)
|
|
|
|
{
|
|
|
|
auto previousUser = this->findUserByUsername(userData.username);
|
|
|
|
if (previousUser)
|
|
|
|
{
|
|
|
|
bool userUpdated = false;
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
if (previousUser->setOAuthClient(userData.clientID))
|
|
|
|
{
|
|
|
|
userUpdated = true;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
if (previousUser->setOAuthToken(userData.oauthToken))
|
|
|
|
{
|
|
|
|
userUpdated = true;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
if (userUpdated)
|
|
|
|
{
|
|
|
|
return AddUserResponse::UserValuesUpdated;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return AddUserResponse::UserAlreadyExists;
|
|
|
|
}
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-04-15 15:09:31 +02:00
|
|
|
auto newUser =
|
|
|
|
std::make_shared<TwitchAccount>(userData.username, userData.oauthToken,
|
|
|
|
userData.clientID, userData.userID);
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-05-28 08:51:39 +02:00
|
|
|
// std::lock_guard<std::mutex> lock(this->mutex);
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2020-02-23 19:44:13 +01:00
|
|
|
this->accounts.insert(newUser);
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2017-12-31 00:50:07 +01:00
|
|
|
return AddUserResponse::UserAdded;
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2018-01-18 18:20:40 +01:00
|
|
|
} // namespace chatterino
|