mirror-chatterino2/src/providers/twitch/twitchserver.cpp

237 lines
7.3 KiB
C++
Raw Normal View History

2018-02-05 15:11:50 +01:00
#include "twitchserver.hpp"
#include "application.hpp"
#include "controllers/accounts/accountcontroller.hpp"
2018-05-23 22:27:29 +02:00
#include "controllers/highlights/highlightcontroller.hpp"
2018-02-05 15:11:50 +01:00
#include "providers/twitch/ircmessagehandler.hpp"
#include "providers/twitch/twitchaccount.hpp"
#include "providers/twitch/twitchhelpers.hpp"
2018-02-05 15:11:50 +01:00
#include "providers/twitch/twitchmessagebuilder.hpp"
#include "util/posttothread.hpp"
2018-06-04 21:05:18 +02:00
#include <IrcCommand>
2018-02-05 15:11:50 +01:00
#include <cassert>
2018-06-04 21:05:18 +02:00
// using namespace Communi;
2018-02-05 15:11:50 +01:00
using namespace chatterino::singletons;
namespace chatterino {
namespace providers {
namespace twitch {
2018-02-05 15:11:50 +01:00
TwitchServer::TwitchServer()
2018-04-18 09:12:29 +02:00
: whispersChannel(new Channel("/whispers", Channel::TwitchWhispers))
, mentionsChannel(new Channel("/mentions", Channel::TwitchMentions))
2018-04-20 19:54:45 +02:00
, watchingChannel(Channel::getEmpty(), Channel::TwitchWatching)
2018-02-05 15:11:50 +01:00
{
qDebug() << "init TwitchServer";
2018-02-05 15:11:50 +01:00
}
void TwitchServer::initialize()
2018-02-05 15:11:50 +01:00
{
2018-05-26 20:26:25 +02:00
getApp()->accounts->twitch.currentUserChanged.connect(
[this]() { util::postToThread([this] { this->connect(); }); });
2018-02-05 15:11:50 +01:00
}
2018-06-04 21:05:18 +02:00
void TwitchServer::initializeConnection(providers::irc::IrcConnection *connection, bool isRead,
bool isWrite)
2018-02-05 15:11:50 +01:00
{
2018-05-26 20:26:25 +02:00
std::shared_ptr<TwitchAccount> account = getApp()->accounts->twitch.getCurrent();
2018-02-05 15:11:50 +01:00
qDebug() << "logging in as" << account->getUserName();
2018-02-05 15:11:50 +01:00
QString username = account->getUserName();
// QString oauthClient = account->getOAuthClient();
QString oauthToken = account->getOAuthToken();
if (!oauthToken.startsWith("oauth:")) {
oauthToken.prepend("oauth:");
}
connection->setUserName(username);
connection->setNickName(username);
connection->setRealName(username);
if (!account->isAnon()) {
connection->setPassword(oauthToken);
// fourtf: ignored users
// this->refreshIgnoredUsers(username, oauthClient, oauthToken);
}
2018-06-04 21:05:18 +02:00
connection->sendCommand(Communi::IrcCommand::createCapability("REQ", "twitch.tv/membership"));
connection->sendCommand(Communi::IrcCommand::createCapability("REQ", "twitch.tv/commands"));
connection->sendCommand(Communi::IrcCommand::createCapability("REQ", "twitch.tv/tags"));
2018-02-05 15:11:50 +01:00
connection->setHost("irc.chat.twitch.tv");
connection->setPort(6667);
}
std::shared_ptr<Channel> TwitchServer::createChannel(const QString &channelName)
{
TwitchChannel *channel = new TwitchChannel(channelName, this->getReadConnection());
2018-06-06 18:57:22 +02:00
channel->sendMessageSignal.connect([this, channel](auto chan, auto msg, bool &sent) {
{
std::lock_guard<std::mutex> guard(this->lastMessageMutex);
std::queue<QTime> &lastMessage =
channel->hasModRights() ? this->lastMessageMod : this->lastMessagePleb;
size_t maxMessageCount = channel->hasModRights() ? 99 : 19;
QTime now = QTime::currentTime();
if (lastMessage.size() > 0 &&
lastMessage.back().addMSecs(channel->hasModRights() ? 100 : 1100) > now) {
if (lastErrorTimeSpeed.addSecs(30) < now) {
auto errorMessage =
messages::Message::createSystemMessage("sending messages too fast");
channel->addMessage(errorMessage);
lastErrorTimeSpeed = now;
}
return;
}
while (lastMessage.size() > 0 && lastMessage.front().addSecs(32) < now) {
lastMessage.pop();
}
if (lastMessage.size() >= maxMessageCount) {
if (lastErrorTimeAmount.addSecs(30) < now) {
auto errorMessage =
messages::Message::createSystemMessage("sending too many messages");
channel->addMessage(errorMessage);
lastErrorTimeAmount = now;
}
return;
}
lastMessage.push(now);
}
this->sendMessage(chan, msg);
sent = true;
});
2018-02-05 15:11:50 +01:00
return std::shared_ptr<Channel>(channel);
}
2018-06-04 21:05:18 +02:00
void TwitchServer::privateMessageReceived(Communi::IrcPrivateMessage *message)
2018-02-05 15:11:50 +01:00
{
2018-06-04 12:23:23 +02:00
IrcMessageHandler::getInstance().handlePrivMessage(message, *this);
2018-02-05 15:11:50 +01:00
}
2018-06-04 21:05:18 +02:00
void TwitchServer::messageReceived(Communi::IrcMessage *message)
2018-02-05 15:11:50 +01:00
{
// this->readConnection
2018-06-04 21:05:18 +02:00
if (message->type() == Communi::IrcMessage::Type::Private) {
2018-02-05 15:11:50 +01:00
// We already have a handler for private messages
return;
}
const QString &command = message->command();
auto &handler = IrcMessageHandler::getInstance();
2018-02-05 15:11:50 +01:00
if (command == "ROOMSTATE") {
handler.handleRoomStateMessage(message);
2018-02-05 15:11:50 +01:00
} else if (command == "CLEARCHAT") {
handler.handleClearChatMessage(message);
2018-02-05 15:11:50 +01:00
} else if (command == "USERSTATE") {
handler.handleUserStateMessage(message);
2018-02-05 15:11:50 +01:00
} else if (command == "WHISPER") {
handler.handleWhisperMessage(message);
2018-02-05 15:11:50 +01:00
} else if (command == "USERNOTICE") {
2018-06-04 12:23:23 +02:00
handler.handleUserNoticeMessage(message, *this);
2018-02-05 15:11:50 +01:00
} else if (command == "MODE") {
handler.handleModeMessage(message);
2018-02-05 15:11:50 +01:00
} else if (command == "NOTICE") {
2018-06-04 21:05:18 +02:00
handler.handleNoticeMessage(static_cast<Communi::IrcNoticeMessage *>(message));
} else if (command == "JOIN") {
handler.handleJoinMessage(message);
} else if (command == "PART") {
handler.handlePartMessage(message);
2018-02-05 15:11:50 +01:00
}
}
2018-06-04 21:05:18 +02:00
void TwitchServer::writeConnectionMessageReceived(Communi::IrcMessage *message)
2018-02-05 15:11:50 +01:00
{
switch (message->type()) {
2018-06-04 21:05:18 +02:00
case Communi::IrcMessage::Type::Notice: {
2018-02-05 15:11:50 +01:00
IrcMessageHandler::getInstance().handleWriteConnectionNoticeMessage(
2018-06-04 21:05:18 +02:00
static_cast<Communi::IrcNoticeMessage *>(message));
2018-02-05 15:11:50 +01:00
} break;
2018-06-04 21:05:18 +02:00
default:;
2018-02-05 15:11:50 +01:00
}
}
std::shared_ptr<Channel> TwitchServer::getCustomChannel(const QString &channelName)
{
if (channelName == "/whispers") {
return whispersChannel;
}
if (channelName == "/mentions") {
return mentionsChannel;
}
return nullptr;
}
void TwitchServer::forEachChannelAndSpecialChannels(std::function<void(ChannelPtr)> func)
{
std::lock_guard<std::mutex> lock(this->channelMutex);
for (std::weak_ptr<Channel> &weak : this->channels) {
std::shared_ptr<Channel> chan = weak.lock();
if (!chan) {
continue;
}
func(chan);
}
func(this->whispersChannel);
func(this->mentionsChannel);
}
std::shared_ptr<Channel> TwitchServer::getChannelOrEmptyByID(const QString &channelID)
{
{
std::lock_guard<std::mutex> lock(this->channelMutex);
for (const auto &weakChannel : this->channels) {
auto channel = weakChannel.lock();
if (!channel) {
continue;
}
auto twitchChannel = std::dynamic_pointer_cast<TwitchChannel>(channel);
if (!twitchChannel) {
continue;
}
if (twitchChannel->roomID == channelID) {
return twitchChannel;
}
}
}
return Channel::getEmpty();
}
QString TwitchServer::cleanChannelName(const QString &dirtyChannelName)
{
return dirtyChannelName.toLower();
}
2018-02-05 15:11:50 +01:00
} // namespace twitch
} // namespace providers
} // namespace chatterino