Files @ 78e71f9345c7
Branch filter:

Location: libtransport.git/src/Conversation.cpp - annotation

Jan Kaluza
Cleanup the includes, rename source code files to match the class name exactly
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
78e71f9345c7
78e71f9345c7
78e71f9345c7
78e71f9345c7
78e71f9345c7
78e71f9345c7
78e71f9345c7
78e71f9345c7
76c678dd6f1d
cc64a76c8be5
cc64a76c8be5
cc64a76c8be5
cc64a76c8be5
65ad5a210697
cc64a76c8be5
76c678dd6f1d
76c678dd6f1d
943dc1925b7c
76c678dd6f1d
943dc1925b7c
5b89dae8ca9a
8b6973539f23
70cf46773b85
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
76c678dd6f1d
76c678dd6f1d
b5c026470f07
76c678dd6f1d
76c678dd6f1d
4717bd007185
4717bd007185
4717bd007185
4717bd007185
4717bd007185
4717bd007185
4717bd007185
4717bd007185
4717bd007185
4717bd007185
4717bd007185
4717bd007185
4717bd007185
97a2cd5b2af5
97a2cd5b2af5
4717bd007185
4717bd007185
4717bd007185
3ff7f9249074
3ff7f9249074
3ff7f9249074
97a2cd5b2af5
97a2cd5b2af5
97a2cd5b2af5
3ff7f9249074
4717bd007185
97a2cd5b2af5
97a2cd5b2af5
fe47e0979be9
97a2cd5b2af5
4717bd007185
4717bd007185
4717bd007185
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
c0687aba7478
c0687aba7478
c0687aba7478
c0687aba7478
c0687aba7478
c0687aba7478
c0687aba7478
c0687aba7478
c0687aba7478
c0687aba7478
c0687aba7478
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
c0687aba7478
4f9457bdd83a
4f9457bdd83a
fe47e0979be9
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
c0687aba7478
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
fe47e0979be9
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
4f9457bdd83a
2add374fbd61
943dc1925b7c
2add374fbd61
2add374fbd61
2add374fbd61
761c746d1008
761c746d1008
0b0e020cdf91
0b0e020cdf91
0b0e020cdf91
761c746d1008
761c746d1008
761c746d1008
2add374fbd61
5b89dae8ca9a
ff3adaa3f33f
ff3adaa3f33f
ff3adaa3f33f
ff3adaa3f33f
ff3adaa3f33f
0a33a3100eb9
5b89dae8ca9a
2add374fbd61
ff3adaa3f33f
2add374fbd61
2add374fbd61
2add374fbd61
2add374fbd61
2add374fbd61
df3276119498
df3276119498
df3276119498
df3276119498
df3276119498
df3276119498
df3276119498
df3276119498
df3276119498
df3276119498
df3276119498
2add374fbd61
0a33a3100eb9
2add374fbd61
0a33a3100eb9
943dc1925b7c
ff3adaa3f33f
943dc1925b7c
943dc1925b7c
0185f9487299
0185f9487299
0185f9487299
0185f9487299
0185f9487299
943dc1925b7c
0a33a3100eb9
76c678dd6f1d
2add374fbd61
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
e14d3c30e92e
e14d3c30e92e
e14d3c30e92e
e14d3c30e92e
e14d3c30e92e
7e9ea5150d9c
7e9ea5150d9c
2add374fbd61
4f9457bdd83a
4f9457bdd83a
0a33a3100eb9
0a33a3100eb9
1055c577d2d3
e5aab1415d9e
e5aab1415d9e
fe47e0979be9
1055c577d2d3
1055c577d2d3
1055c577d2d3
7e9ea5150d9c
7e9ea5150d9c
8761e976aeca
8761e976aeca
8761e976aeca
8761e976aeca
8761e976aeca
8761e976aeca
fe47e0979be9
7e9ea5150d9c
7805d876f88d
7805d876f88d
7805d876f88d
7805d876f88d
7805d876f88d
7805d876f88d
7805d876f88d
7805d876f88d
fe47e0979be9
7805d876f88d
7805d876f88d
7e9ea5150d9c
7e9ea5150d9c
7e9ea5150d9c
1055c577d2d3
6699a898fbec
0a33a3100eb9
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
8c529dbabbbc
0a33a3100eb9
0a33a3100eb9
8e03fbec4701
8e03fbec4701
8e03fbec4701
8e03fbec4701
8e03fbec4701
99d1c435b90f
8e03fbec4701
99d1c435b90f
99d1c435b90f
8e03fbec4701
8e03fbec4701
0a33a3100eb9
0a33a3100eb9
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
58fbe0d388c6
58fbe0d388c6
58fbe0d388c6
58fbe0d388c6
58fbe0d388c6
58fbe0d388c6
58fbe0d388c6
58fbe0d388c6
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
65ad5a210697
70cf46773b85
70cf46773b85
70cf46773b85
70cf46773b85
70cf46773b85
65ad5a210697
65ad5a210697
76c678dd6f1d
457d19d1ec41
65ad5a210697
19786b3bb708
19786b3bb708
19786b3bb708
19786b3bb708
8e03fbec4701
1af263f488e8
19786b3bb708
19786b3bb708
8e03fbec4701
8e03fbec4701
457d19d1ec41
457d19d1ec41
457d19d1ec41
457d19d1ec41
457d19d1ec41
8e03fbec4701
457d19d1ec41
65ad5a210697
457d19d1ec41
0a33a3100eb9
1055c577d2d3
1055c577d2d3
1055c577d2d3
6b83d9f942fa
70cf46773b85
70cf46773b85
70cf46773b85
70cf46773b85
70cf46773b85
70cf46773b85
70cf46773b85
e5aab1415d9e
e5aab1415d9e
fe47e0979be9
e5aab1415d9e
e5aab1415d9e
e5aab1415d9e
1af263f488e8
1055c577d2d3
1055c577d2d3
1055c577d2d3
1055c577d2d3
1055c577d2d3
1055c577d2d3
e5aab1415d9e
1055c577d2d3
1055c577d2d3
1055c577d2d3
88607aff3c0b
88607aff3c0b
fe47e0979be9
88607aff3c0b
8e03fbec4701
8e03fbec4701
8e03fbec4701
8b6973539f23
8b6973539f23
fe47e0979be9
8b6973539f23
8b6973539f23
76c678dd6f1d
76c678dd6f1d
76c678dd6f1d
/**
 * XMPP - libpurple transport
 *
 * Copyright (C) 2009, Jan Kaluza <hanzz@soc.pidgin.im>
 *
 * 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 <iostream>
#include "transport/Conversation.h"
#include "transport/ConversationManager.h"
#include "transport/User.h"
#include "transport/Transport.h"
#include "transport/Buddy.h"
#include "transport/RosterManager.h"
#include "transport/Frontend.h"
#include "transport/Config.h"

#include "Swiften/Elements/MUCItem.h"
#include "Swiften/Elements/MUCOccupant.h"
#include "Swiften/Elements/MUCUserPayload.h"
#include "Swiften/Elements/Delay.h"
#include "Swiften/Elements/MUCPayload.h"

namespace Transport {

Conversation::Conversation(ConversationManager *conversationManager, const std::string &legacyName, bool isMUC) : m_conversationManager(conversationManager) {
	m_legacyName = legacyName;
	m_muc = isMUC;
	m_jid = m_conversationManager->getUser()->getJID().toBare();
	m_sentInitialPresence = false;
	m_nicknameChanged = false;

	if (CONFIG_BOOL_DEFAULTED(conversationManager->getComponent()->getConfig(), "features.rawxml", false)) {
		m_sentInitialPresence = true;
	}
}

Conversation::~Conversation() {
}

void Conversation::destroyRoom() {
	if (m_muc) {
		Swift::Presence::ref presence = Swift::Presence::create();
		std::string legacyName = m_legacyName;
		if (legacyName.find_last_of("@") != std::string::npos) {
			legacyName.replace(legacyName.find_last_of("@"), 1, "%"); // OK
		}
		presence->setFrom(Swift::JID(legacyName, m_conversationManager->getComponent()->getJID().toBare(), m_nickname));
		presence->setType(Swift::Presence::Unavailable);

		Swift::MUCItem item;
		item.affiliation = Swift::MUCOccupant::NoAffiliation;
		item.role = Swift::MUCOccupant::NoRole;
		item.actor = "Transport";
		item.reason = "Spectrum 2 transport is being shut down.";
		Swift::MUCUserPayload *p = new Swift::MUCUserPayload ();
		p->addItem(item);

		Swift::MUCUserPayload::StatusCode c;
		c.code = 332;
		p->addStatusCode(c);
		Swift::MUCUserPayload::StatusCode c2;
		c2.code = 307;
		p->addStatusCode(c2);

		presence->addPayload(boost::shared_ptr<Swift::Payload>(p));
		BOOST_FOREACH(const Swift::JID &jid, m_jids) {
			presence->setTo(jid);
			m_conversationManager->getComponent()->getFrontend()->sendPresence(presence);
		}
	}
}

void Conversation::setRoom(const std::string &room) {
	m_room = room;
	m_legacyName = m_room + "/" + m_legacyName;
}

void Conversation::cacheMessage(boost::shared_ptr<Swift::Message> &message) {
	boost::posix_time::ptime timestamp = boost::posix_time::second_clock::universal_time();
	boost::shared_ptr<Swift::Delay> delay(boost::make_shared<Swift::Delay>());
	delay->setStamp(timestamp);
	message->addPayload(delay);
	m_cachedMessages.push_back(message);
	if (m_cachedMessages.size() > 100) {
		m_cachedMessages.pop_front();
	}
}

void Conversation::handleRawMessage(boost::shared_ptr<Swift::Message> &message) {
	if (message->getType() != Swift::Message::Groupchat) {
		if (m_conversationManager->getComponent()->inServerMode() && m_conversationManager->getUser()->shouldCacheMessages()) {
			cacheMessage(message);
		}
		else {
			m_conversationManager->getComponent()->getFrontend()->sendMessage(message);
		}
	}
	else {
		if (m_jids.empty()) {
			cacheMessage(message);
		}
		else {
			BOOST_FOREACH(const Swift::JID &jid, m_jids) {
				message->setTo(jid);
				// Subject has to be sent after our own presence (the one with code 110)
				if (!message->getSubject().empty() && m_sentInitialPresence == false) {
					m_subject = message;
					return;
				}
				m_conversationManager->getComponent()->getFrontend()->sendMessage(message);
			}
		}
	}
}

void Conversation::handleMessage(boost::shared_ptr<Swift::Message> &message, const std::string &nickname) {
	if (m_muc) {
		message->setType(Swift::Message::Groupchat);
	}
	else {
		if (message->getType() == Swift::Message::Headline) {
			if (m_conversationManager->getUser()->getUserSetting("send_headlines") != "1") {
				message->setType(Swift::Message::Chat);
			}
		}
		else {
			message->setType(Swift::Message::Chat);
		}
	}

	std::string n = nickname;
	if (n.empty() && !m_room.empty() && !m_muc) {
		n = m_nickname;
	}

	if (message->getType() != Swift::Message::Groupchat) {
		message->setTo(m_jid);
		// normal message
		if (n.empty()) {
			Buddy *buddy = m_conversationManager->getUser()->getRosterManager()->getBuddy(m_legacyName);
			if (buddy) {
				message->setFrom(buddy->getJID());
			}
			else {
				std::string name = m_legacyName;
				if (CONFIG_BOOL_DEFAULTED(m_conversationManager->getComponent()->getConfig(), "service.jid_escaping", true)) {
					name = Swift::JID::getEscapedNode(m_legacyName);
				}
				else {
					if (name.find_last_of("@") != std::string::npos) {
						name.replace(name.find_last_of("@"), 1, "%");
					}
				}

				message->setFrom(Swift::JID(name, m_conversationManager->getComponent()->getJID().toBare(), "bot"));
			}
		}
		// PM message
		else {
			if (m_room.empty()) {
				message->setFrom(Swift::JID(n, m_conversationManager->getComponent()->getJID().toBare(), "user"));
			}
			else {
				std::string legacyName = m_room;
				if (legacyName.find_last_of("@") != std::string::npos) {
					legacyName.replace(legacyName.find_last_of("@"), 1, "%"); // OK
				}
				message->setFrom(Swift::JID(legacyName, m_conversationManager->getComponent()->getJID().toBare(), n));
			}
		}
	}
	else {
		std::string legacyName = m_legacyName;
		if (legacyName.find_last_of("@") != std::string::npos) {
			legacyName.replace(legacyName.find_last_of("@"), 1, "%"); // OK
		}

		std::string n = nickname;
		if (n.empty()) {
			n = " ";
		}

		message->setFrom(Swift::JID(legacyName, m_conversationManager->getComponent()->getJID().toBare(), n));
	}

	handleRawMessage(message);
}

void Conversation::sendParticipants(const Swift::JID &to) {
	for (std::map<std::string, Swift::Presence::ref>::iterator it = m_participants.begin(); it != m_participants.end(); it++) {
		(*it).second->setTo(to);
		m_conversationManager->getComponent()->getFrontend()->sendPresence((*it).second);
	}
}

void Conversation::sendCachedMessages(const Swift::JID &to) {
	for (std::list<boost::shared_ptr<Swift::Message> >::const_iterator it = m_cachedMessages.begin(); it != m_cachedMessages.end(); it++) {
		if (to.isValid()) {
			(*it)->setTo(to);
		}
		else {
			(*it)->setTo(m_jid.toBare());
		}
		m_conversationManager->getComponent()->getFrontend()->sendMessage(*it);
	}

	if (m_subject) {
		if (to.isValid()) {
			m_subject->setTo(to);
		}
		else {
			m_subject->setTo(m_jid.toBare());
		}
		m_conversationManager->getComponent()->getFrontend()->sendMessage(m_subject);
	}

	m_cachedMessages.clear();
}

Swift::Presence::ref Conversation::generatePresence(const std::string &nick, int flag, int status, const std::string &statusMessage, const std::string &newname) {
	std::string nickname = nick;
	Swift::Presence::ref presence = Swift::Presence::create();
	std::string legacyName = m_legacyName;
	if (m_muc) {
		if (legacyName.find_last_of("@") != std::string::npos) {
			legacyName.replace(legacyName.find_last_of("@"), 1, "%"); // OK
		}
	}
	presence->setFrom(Swift::JID(legacyName, m_conversationManager->getComponent()->getJID().toBare(), nickname));
	presence->setType(Swift::Presence::Available);

	if (!statusMessage.empty())
		presence->setStatus(statusMessage);

	Swift::StatusShow s((Swift::StatusShow::Type) status);

	if (s.getType() == Swift::StatusShow::None) {
		presence->setType(Swift::Presence::Unavailable);
	}

	presence->setShow(s.getType());

	Swift::MUCUserPayload *p = new Swift::MUCUserPayload ();
	if (m_nickname == nickname) {
		if (flag & PARTICIPANT_FLAG_CONFLICT) {
			delete p;
			presence->setType(Swift::Presence::Error);
			presence->addPayload(boost::shared_ptr<Swift::Payload>(new Swift::MUCPayload()));
			presence->addPayload(boost::shared_ptr<Swift::Payload>(new Swift::ErrorPayload(Swift::ErrorPayload::Conflict)));
			return presence;
		}
		else if (flag & PARTICIPANT_FLAG_NOT_AUTHORIZED) {
			delete p;
			presence->setType(Swift::Presence::Error);
			presence->addPayload(boost::shared_ptr<Swift::Payload>(new Swift::MUCPayload()));
			presence->addPayload(boost::shared_ptr<Swift::Payload>(new Swift::ErrorPayload(Swift::ErrorPayload::NotAuthorized, Swift::ErrorPayload::Auth, statusMessage)));
			return presence;
		}
		else if (flag & PARTICIPANT_FLAG_ROOM_NOT_FOUD) {
			delete p;
			presence->setType(Swift::Presence::Error);
			presence->addPayload(boost::shared_ptr<Swift::Payload>(new Swift::MUCPayload()));
			presence->addPayload(boost::shared_ptr<Swift::Payload>(new Swift::ErrorPayload(Swift::ErrorPayload::ItemNotFound, Swift::ErrorPayload::Cancel, statusMessage)));
			return presence;
		}
		else {
			Swift::MUCUserPayload::StatusCode c;
			c.code = 110;
			p->addStatusCode(c);
			if (m_nicknameChanged) {
				Swift::MUCUserPayload::StatusCode c;
				c.code = 210;
				p->addStatusCode(c);
			}
			m_sentInitialPresence = true;
		}
	}


	Swift::MUCItem item;
	
	item.affiliation = Swift::MUCOccupant::Member;
	item.role = Swift::MUCOccupant::Participant;

	if (flag & PARTICIPANT_FLAG_MODERATOR) {
		item.affiliation = Swift::MUCOccupant::Admin;
		item.role = Swift::MUCOccupant::Moderator;
	}

	if (!newname.empty()) {
		item.nick = newname;
		Swift::MUCUserPayload::StatusCode c;
		c.code = 303;
		p->addStatusCode(c);
		presence->setType(Swift::Presence::Unavailable);
	}
	
	p->addItem(item);
	presence->addPayload(boost::shared_ptr<Swift::Payload>(p));
	return presence;
}


void Conversation::setNickname(const std::string &nickname) {
	if (!nickname.empty() && m_nickname != nickname) {
		m_nicknameChanged = true;
	}
	m_nickname = nickname;
}

void Conversation::handleRawPresence(Swift::Presence::ref presence) {
	// TODO: Detect nickname change.
	m_conversationManager->getComponent()->getFrontend()->sendPresence(presence);
	m_participants[presence->getFrom().getResource()] = presence;
}

void Conversation::handleParticipantChanged(const std::string &nick, Conversation::ParticipantFlag flag, int status, const std::string &statusMessage, const std::string &newname) {
	Swift::Presence::ref presence = generatePresence(nick, flag, status, statusMessage, newname);

	if (presence->getType() == Swift::Presence::Unavailable) {
		m_participants.erase(nick);
	}
	else {
		m_participants[nick] = presence;
	}


	BOOST_FOREACH(const Swift::JID &jid, m_jids) {
		presence->setTo(jid);
		m_conversationManager->getComponent()->getFrontend()->sendPresence(presence);
	}
	if (!newname.empty()) {
		handleParticipantChanged(newname, flag, status, statusMessage);
	}

	if (m_sentInitialPresence && m_subject) {
		m_conversationManager->getComponent()->getFrontend()->sendMessage(m_subject);
		m_subject.reset();
	}
}

}