/* * Squawk messenger. * Copyright (C) 2019 Yury Gubich * * This program 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. * * This program 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. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "squawk.h" #include #include #include #include Core::Squawk::Squawk(QObject* parent): QObject(parent), accounts(), amap(), network() { connect(&network, &NetworkAccess::fileLocalPathResponse, this, &Squawk::fileLocalPathResponse); connect(&network, &NetworkAccess::downloadFileProgress, this, &Squawk::downloadFileProgress); connect(&network, &NetworkAccess::downloadFileError, this, &Squawk::downloadFileError); connect(&network, &NetworkAccess::uploadFileProgress, this, &Squawk::uploadFileProgress); connect(&network, &NetworkAccess::uploadFileError, this, &Squawk::uploadFileError); } Core::Squawk::~Squawk() { Accounts::const_iterator itr = accounts.begin(); Accounts::const_iterator end = accounts.end(); for (; itr != end; ++itr) { delete (*itr); } } void Core::Squawk::stop() { qDebug("Stopping squawk core.."); network.stop(); QSettings settings; settings.beginGroup("core"); settings.beginWriteArray("accounts"); for (int i = 0; i < accounts.size(); ++i) { settings.setArrayIndex(i); Account* acc = accounts[i]; settings.setValue("name", acc->getName()); settings.setValue("server", acc->getServer()); settings.setValue("login", acc->getLogin()); settings.setValue("password", acc->getPassword()); settings.setValue("resource", acc->getResource()); } settings.endArray(); settings.endGroup(); settings.sync(); emit quit(); } void Core::Squawk::start() { qDebug("Starting squawk core.."); QSettings settings; settings.beginGroup("core"); int size = settings.beginReadArray("accounts"); for (int i = 0; i < size; ++i) { settings.setArrayIndex(i); addAccount( settings.value("login").toString(), settings.value("server").toString(), settings.value("password").toString(), settings.value("name").toString(), settings.value("resource").toString() ); } settings.endArray(); settings.endGroup(); network.start(); } void Core::Squawk::newAccountRequest(const QMap& map) { QString name = map.value("name").toString(); QString login = map.value("login").toString(); QString server = map.value("server").toString(); QString password = map.value("password").toString(); QString resource = map.value("resource").toString(); addAccount(login, server, password, name, resource); } void Core::Squawk::addAccount(const QString& login, const QString& server, const QString& password, const QString& name, const QString& resource) { QSettings settings; unsigned int reconnects = settings.value("reconnects", 2).toUInt(); Account* acc = new Account(login, server, password, name, &network); acc->setResource(resource); acc->setReconnectTimes(reconnects); accounts.push_back(acc); amap.insert(std::make_pair(name, acc)); connect(acc, &Account::connectionStateChanged, this, &Squawk::onAccountConnectionStateChanged); connect(acc, &Account::changed, this, &Squawk::onAccountChanged); connect(acc, &Account::error, this, &Squawk::onAccountError); connect(acc, &Account::availabilityChanged, this, &Squawk::onAccountAvailabilityChanged); connect(acc, &Account::addContact, this, &Squawk::onAccountAddContact); connect(acc, &Account::addGroup, this, &Squawk::onAccountAddGroup); connect(acc, &Account::removeGroup, this, &Squawk::onAccountRemoveGroup); connect(acc, qOverload(&Account::removeContact), this, qOverload(&Squawk::onAccountRemoveContact)); connect(acc, qOverload(&Account::removeContact), this, qOverload(&Squawk::onAccountRemoveContact)); connect(acc, &Account::changeContact, this, &Squawk::onAccountChangeContact); connect(acc, &Account::addPresence, this, &Squawk::onAccountAddPresence); connect(acc, &Account::removePresence, this, &Squawk::onAccountRemovePresence); connect(acc, &Account::message, this, &Squawk::onAccountMessage); connect(acc, &Account::responseArchive, this, &Squawk::onAccountResponseArchive); connect(acc, &Account::addRoom, this, &Squawk::onAccountAddRoom); connect(acc, &Account::changeRoom, this, &Squawk::onAccountChangeRoom); connect(acc, &Account::removeRoom, this, &Squawk::onAccountRemoveRoom); connect(acc, &Account::addRoomParticipant, this, &Squawk::onAccountAddRoomPresence); connect(acc, &Account::changeRoomParticipant, this, &Squawk::onAccountChangeRoomPresence); connect(acc, &Account::removeRoomParticipant, this, &Squawk::onAccountRemoveRoomPresence); connect(acc, &Account::receivedVCard, this, &Squawk::responseVCard); connect(acc, &Account::uploadFileError, this, &Squawk::uploadFileError); QMap map = { {"login", login}, {"server", server}, {"name", name}, {"password", password}, {"resource", resource}, {"state", Shared::disconnected}, {"offline", Shared::offline}, {"error", ""}, {"avatarPath", acc->getAvatarPath()} }; emit newAccount(map); } void Core::Squawk::changeState(int p_state) { Shared::Availability avail; if (p_state < Shared::availabilityLowest && p_state > Shared::availabilityHighest) { qDebug("An attempt to set invalid availability to Squawk core, skipping"); } avail = static_cast(p_state); state = avail; for (std::deque::iterator itr = accounts.begin(), end = accounts.end(); itr != end; ++itr) { (*itr)->setAvailability(state); } } void Core::Squawk::connectAccount(const QString& account) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug("An attempt to connect non existing account, skipping"); return; } itr->second->connect(); } void Core::Squawk::disconnectAccount(const QString& account) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug("An attempt to connect non existing account, skipping"); return; } itr->second->disconnect(); } void Core::Squawk::onAccountConnectionStateChanged(int state) { Account* acc = static_cast(sender()); emit changeAccount(acc->getName(), {{"state", state}}); if (state == Shared::disconnected) { bool equals = true; for (Accounts::const_iterator itr = accounts.begin(), end = accounts.end(); itr != end; itr++) { if ((*itr)->getState() != Shared::disconnected) { equals = false; } } if (equals) { state = Shared::offline; emit stateChanged(state); } } } void Core::Squawk::onAccountAddContact(const QString& jid, const QString& group, const QMap& data) { Account* acc = static_cast(sender()); emit addContact(acc->getName(), jid, group, data); } void Core::Squawk::onAccountAddGroup(const QString& name) { Account* acc = static_cast(sender()); emit addGroup(acc->getName(), name); } void Core::Squawk::onAccountRemoveGroup(const QString& name) { Account* acc = static_cast(sender()); emit removeGroup(acc->getName(), name); } void Core::Squawk::onAccountChangeContact(const QString& jid, const QMap& data) { Account* acc = static_cast(sender()); emit changeContact(acc->getName(), jid, data); } void Core::Squawk::onAccountRemoveContact(const QString& jid) { Account* acc = static_cast(sender()); emit removeContact(acc->getName(), jid); } void Core::Squawk::onAccountRemoveContact(const QString& jid, const QString& group) { Account* acc = static_cast(sender()); emit removeContact(acc->getName(), jid, group); } void Core::Squawk::onAccountAddPresence(const QString& jid, const QString& name, const QMap& data) { Account* acc = static_cast(sender()); emit addPresence(acc->getName(), jid, name, data); } void Core::Squawk::onAccountRemovePresence(const QString& jid, const QString& name) { Account* acc = static_cast(sender()); emit removePresence(acc->getName(), jid, name); } void Core::Squawk::onAccountAvailabilityChanged(int state) { Account* acc = static_cast(sender()); emit changeAccount(acc->getName(), {{"availability", state}}); } void Core::Squawk::onAccountChanged(const QMap& data) { Account* acc = static_cast(sender()); emit changeAccount(acc->getName(), data); } void Core::Squawk::onAccountMessage(const Shared::Message& data) { Account* acc = static_cast(sender()); emit accountMessage(acc->getName(), data); } void Core::Squawk::sendMessage(const QString& account, const Shared::Message& data) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug("An attempt to send a message with non existing account, skipping"); return; } itr->second->sendMessage(data); } void Core::Squawk::sendMessage(const QString& account, const Shared::Message& data, const QString& path) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug("An attempt to send a message with non existing account, skipping"); return; } itr->second->sendMessage(data, path); } void Core::Squawk::requestArchive(const QString& account, const QString& jid, int count, const QString& before) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug("An attempt to request an archive of non existing account, skipping"); return; } itr->second->requestArchive(jid, count, before); } void Core::Squawk::onAccountResponseArchive(const QString& jid, const std::list& list) { Account* acc = static_cast(sender()); emit responseArchive(acc->getName(), jid, list); } void Core::Squawk::modifyAccountRequest(const QString& name, const QMap& map) { AccountsMap::const_iterator itr = amap.find(name); if (itr == amap.end()) { qDebug("An attempt to modify non existing account, skipping"); return; } Core::Account* acc = itr->second; Shared::ConnectionState st = acc->getState(); if (st != Shared::disconnected) { acc->reconnect(); } QMap::const_iterator mItr; mItr = map.find("login"); if (mItr != map.end()) { acc->setLogin(mItr->toString()); } mItr = map.find("password"); if (mItr != map.end()) { acc->setPassword(mItr->toString()); } mItr = map.find("resource"); if (mItr != map.end()) { acc->setResource(mItr->toString()); } mItr = map.find("server"); if (mItr != map.end()) { acc->setServer(mItr->toString()); } emit changeAccount(name, map); } void Core::Squawk::onAccountError(const QString& text) { Account* acc = static_cast(sender()); emit changeAccount(acc->getName(), {{"error", text}}); } void Core::Squawk::removeAccountRequest(const QString& name) { AccountsMap::const_iterator itr = amap.find(name); if (itr == amap.end()) { qDebug() << "An attempt to remove non existing account " << name << " from core, skipping"; return; } Account* acc = itr->second; if (acc->getState() != Shared::disconnected) { acc->disconnect(); } for (Accounts::const_iterator aItr = accounts.begin(); aItr != accounts.end(); ++aItr) { if (*aItr == acc) { accounts.erase(aItr); break; } } amap.erase(itr); QString path(QStandardPaths::writableLocation(QStandardPaths::CacheLocation)); path += "/" + name; QDir dir(path); dir.removeRecursively(); emit removeAccount(name); acc->deleteLater(); } void Core::Squawk::subscribeContact(const QString& account, const QString& jid, const QString& reason) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug("An attempt to subscribe to the contact with non existing account, skipping"); return; } itr->second->subscribeToContact(jid, reason); } void Core::Squawk::unsubscribeContact(const QString& account, const QString& jid, const QString& reason) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug("An attempt to subscribe to the contact with non existing account, skipping"); return; } itr->second->unsubscribeFromContact(jid, reason); } void Core::Squawk::removeContactRequest(const QString& account, const QString& jid) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug("An attempt to remove contact from non existing account, skipping"); return; } itr->second->removeContactRequest(jid); } void Core::Squawk::addContactRequest(const QString& account, const QString& jid, const QString& name, const QSet& groups) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug("An attempt to add contact to a non existing account, skipping"); return; } itr->second->addContactRequest(jid, name, groups); } void Core::Squawk::onAccountAddRoom(const QString jid, const QMap& data) { Account* acc = static_cast(sender()); emit addRoom(acc->getName(), jid, data); } void Core::Squawk::onAccountChangeRoom(const QString jid, const QMap& data) { Account* acc = static_cast(sender()); emit changeRoom(acc->getName(), jid, data); } void Core::Squawk::onAccountRemoveRoom(const QString jid) { Account* acc = static_cast(sender()); emit removeRoom(acc->getName(), jid); } void Core::Squawk::setRoomJoined(const QString& account, const QString& jid, bool joined) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug() << "An attempt to set jouned to the room" << jid << "of non existing account" << account << ", skipping"; return; } itr->second->setRoomJoined(jid, joined); } void Core::Squawk::setRoomAutoJoin(const QString& account, const QString& jid, bool joined) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug() << "An attempt to set autoJoin to the room" << jid << "of non existing account" << account << ", skipping"; return; } itr->second->setRoomAutoJoin(jid, joined); } void Core::Squawk::onAccountAddRoomPresence(const QString& jid, const QString& nick, const QMap& data) { Account* acc = static_cast(sender()); emit addRoomParticipant(acc->getName(), jid, nick, data); } void Core::Squawk::onAccountChangeRoomPresence(const QString& jid, const QString& nick, const QMap& data) { Account* acc = static_cast(sender()); emit changeRoomParticipant(acc->getName(), jid, nick, data); } void Core::Squawk::onAccountRemoveRoomPresence(const QString& jid, const QString& nick) { Account* acc = static_cast(sender()); emit removeRoomParticipant(acc->getName(), jid, nick); } void Core::Squawk::removeRoomRequest(const QString& account, const QString& jid) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug() << "An attempt to remove the room" << jid << "of non existing account" << account << ", skipping"; return; } itr->second->removeRoomRequest(jid); } void Core::Squawk::addRoomRequest(const QString& account, const QString& jid, const QString& nick, const QString& password, bool autoJoin) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug() << "An attempt to add the room" << jid << "to non existing account" << account << ", skipping"; return; } itr->second->addRoomRequest(jid, nick, password, autoJoin); } void Core::Squawk::fileLocalPathRequest(const QString& messageId, const QString& url) { network.fileLocalPathRequest(messageId, url); } void Core::Squawk::downloadFileRequest(const QString& messageId, const QString& url) { network.downladFileRequest(messageId, url); } void Core::Squawk::addContactToGroupRequest(const QString& account, const QString& jid, const QString& groupName) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug() << "An attempt to add contact" << jid << "of non existing account" << account << "to the group" << groupName << ", skipping"; return; } itr->second->addContactToGroupRequest(jid, groupName); } void Core::Squawk::removeContactFromGroupRequest(const QString& account, const QString& jid, const QString& groupName) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug() << "An attempt to add contact" << jid << "of non existing account" << account << "to the group" << groupName << ", skipping"; return; } itr->second->removeContactFromGroupRequest(jid, groupName); } void Core::Squawk::renameContactRequest(const QString& account, const QString& jid, const QString& newName) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug() << "An attempt to rename contact" << jid << "of non existing account" << account << ", skipping"; return; } itr->second->renameContactRequest(jid, newName); } void Core::Squawk::requestVCard(const QString& account, const QString& jid) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug() << "An attempt to request" << jid << "vcard of non existing account" << account << ", skipping"; return; } itr->second->requestVCard(jid); } void Core::Squawk::uploadVCard(const QString& account, const Shared::VCard& card) { AccountsMap::const_iterator itr = amap.find(account); if (itr == amap.end()) { qDebug() << "An attempt to upload vcard to non existing account" << account << ", skipping"; return; } itr->second->uploadVCard(card); }