/** * XMPP - libpurple transport * * Copyright (C) 2009, Jan Kaluza * * 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 2 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, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA */ #include "transport/rostermanager.h" #include "transport/rosterstorage.h" #include "transport/storagebackend.h" #include "transport/buddy.h" #include "transport/usermanager.h" #include "transport/buddy.h" #include "transport/user.h" #include "Swiften/Roster/SetRosterRequest.h" #include "Swiften/Elements/RosterPayload.h" #include "Swiften/Elements/RosterItemPayload.h" #include "Swiften/Elements/RosterItemExchangePayload.h" namespace Transport { RosterManager::RosterManager(User *user, Component *component){ m_rosterStorage = NULL; m_user = user; m_component = component; m_setBuddyTimer = m_component->getFactories()->getTimerFactory()->createTimer(1000); m_RIETimer = m_component->getFactories()->getTimerFactory()->createTimer(5000); m_RIETimer->onTick.connect(boost::bind(&RosterManager::sendRIE, this)); } RosterManager::~RosterManager() { m_setBuddyTimer->stop(); m_RIETimer->stop(); if (m_rosterStorage) { // for (std::map::const_iterator it = m_buddies.begin(); it != m_buddies.end(); it++) { // Buddy *buddy = (*it).second; // m_rosterStorage->storeBuddy(buddy); // } m_rosterStorage->storeBuddies(); } for (std::map::const_iterator it = m_buddies.begin(); it != m_buddies.end(); it++) { Buddy *buddy = (*it).second; delete buddy; } if (m_rosterStorage) delete m_rosterStorage; } void RosterManager::setBuddy(Buddy *buddy) { // m_setBuddyTimer->onTick.connect(boost::bind(&RosterManager::setBuddyCallback, this, buddy)); // m_setBuddyTimer->start(); setBuddyCallback(buddy); } void RosterManager::sendBuddyRosterPush(Buddy *buddy) { Swift::RosterPayload::ref payload = Swift::RosterPayload::ref(new Swift::RosterPayload()); Swift::RosterItemPayload item; item.setJID(buddy->getJID().toBare()); item.setName(buddy->getAlias()); item.setGroups(buddy->getGroups()); payload->addItem(item); Swift::SetRosterRequest::ref request = Swift::SetRosterRequest::create(payload, m_user->getJID().toBare(), m_component->getIQRouter()); request->onResponse.connect(boost::bind(&RosterManager::handleBuddyRosterPushResponse, this, _1, buddy->getName())); request->send(); } void RosterManager::setBuddyCallback(Buddy *buddy) { m_setBuddyTimer->onTick.disconnect(boost::bind(&RosterManager::setBuddyCallback, this, buddy)); if (m_rosterStorage) { buddy->onBuddyChanged.connect(boost::bind(&RosterStorage::storeBuddy, m_rosterStorage, buddy)); } std::cout << "ADDING " << buddy->getName() << "\n"; m_buddies[buddy->getName()] = buddy; onBuddySet(buddy); // In server mode the only way is to send jabber:iq:roster push. // In component mode we send RIE or Subscribe presences, based on features. if (m_component->inServerMode()) { sendBuddyRosterPush(buddy); } else { if (m_setBuddyTimer->onTick.empty()) { m_setBuddyTimer->stop(); if (true /*&& rie_is_supported*/) { m_RIETimer->start(); } } } if (m_rosterStorage) m_rosterStorage->storeBuddy(buddy); } void RosterManager::unsetBuddy(Buddy *buddy) { m_buddies.erase(buddy->getName()); if (m_rosterStorage) m_rosterStorage->removeBuddyFromQueue(buddy); onBuddyUnset(buddy); } void RosterManager::storeBuddy(Buddy *buddy) { if (m_rosterStorage) { m_rosterStorage->storeBuddy(buddy); } } void RosterManager::handleBuddyRosterPushResponse(Swift::ErrorPayload::ref error, const std::string &key) { if (m_buddies[key] != NULL) { m_buddies[key]->buddyChanged(); } } Buddy *RosterManager::getBuddy(const std::string &name) { return m_buddies[name]; } void RosterManager::sendRIE() { m_RIETimer->stop(); Swift::RosterItemExchangePayload::ref payload = Swift::RosterItemExchangePayload::ref(new Swift::RosterItemExchangePayload()); for (std::map::const_iterator it = m_buddies.begin(); it != m_buddies.end(); it++) { Buddy *buddy = (*it).second; Swift::RosterItemExchangePayload::Item item; item.setJID(buddy->getJID().toBare()); item.setName(buddy->getAlias()); item.setAction(Swift::RosterItemExchangePayload::Item::Add); item.setGroups(buddy->getGroups()); payload->addItem(item); } boost::shared_ptr > request(new Swift::GenericRequest(Swift::IQ::Set, m_user->getJID(), payload, m_component->getIQRouter())); request->send(); } void RosterManager::handleSubscription(Swift::Presence::ref presence) { std::string legacyName = Buddy::JIDToLegacyName(presence->getTo()); } void RosterManager::setStorageBackend(StorageBackend *storageBackend) { if (m_rosterStorage) { return; } m_rosterStorage = new RosterStorage(m_user, storageBackend); std::list roster; storageBackend->getBuddies(m_user->getUserInfo().id, roster); for (std::list::const_iterator it = roster.begin(); it != roster.end(); it++) { Buddy *buddy = m_component->getFactory()->createBuddy(this, *it); std::cout << "CREATING BUDDY FROM DATABASE CACHE " << buddy->getName() << "\n"; m_buddies[buddy->getName()] = buddy; buddy->onBuddyChanged.connect(boost::bind(&RosterStorage::storeBuddy, m_rosterStorage, buddy)); onBuddySet(buddy); } } Swift::RosterPayload::ref RosterManager::generateRosterPayload() { Swift::RosterPayload::ref payload = Swift::RosterPayload::ref(new Swift::RosterPayload()); for (std::map::const_iterator it = m_buddies.begin(); it != m_buddies.end(); it++) { Buddy *buddy = (*it).second; Swift::RosterItemPayload item; item.setJID(buddy->getJID().toBare()); item.setName(buddy->getAlias()); item.setGroups(buddy->getGroups()); payload->addItem(item); } return payload; } }