summaryrefslogtreecommitdiffstats
blob: 6625e7770b47b77ea5185040b2df743499d13087 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
/*
 * Copyright (c) 2014 Kevin Smith and Remko Tronçon
 * Licensed under the GNU General Public License v3.
 * See Documentation/Licenses/GPLv3.txt for more information.
 */

#include <boost/lexical_cast.hpp>
#include <boost/optional.hpp>
#include <Swiften/Base/DateTime.h>
#include <Swiften/Parser/IQParser.h>
#include <Swiften/Parser/MessageParser.h>
#include <Swiften/Parser/PayloadParserFactory.h>
#include <Swiften/Parser/PayloadParserFactoryCollection.h>
#include <Swiften/Parser/PayloadParsers/DelayParser.h>
#include <Swiften/Parser/PayloadParsers/ForwardedParser.h>
#include <Swiften/Parser/PresenceParser.h>


using namespace Swift;

ForwardedParser::ForwardedParser(PayloadParserFactoryCollection* factories) : factories_(factories), level_(TopLevel) {
}

void ForwardedParser::handleStartElement(const std::string& element, const std::string& ns, const AttributeMap& attributes) {
	if (level_ == PayloadLevel) {
		if (element == "iq" && ns == "jabber:client") { /* begin parsing a nested stanza? */
			childParser_ = boost::dynamic_pointer_cast<StanzaParser>(boost::make_shared<IQParser>(factories_));
		} else if (element == "message" && ns == "jabber:client") {
			childParser_ = boost::dynamic_pointer_cast<StanzaParser>(boost::make_shared<MessageParser>(factories_));
		} else if (element == "presence" && ns == "jabber:client") {
			childParser_ = boost::dynamic_pointer_cast<StanzaParser>(boost::make_shared<PresenceParser>(factories_));
		} else if (element == "delay" && ns == "urn:xmpp:delay") { /* nested delay payload */
			delayParser_ = boost::make_shared<DelayParser>();
		}
	}
	if (childParser_) { /* parsing a nested stanza? */
		childParser_->handleStartElement(element, ns, attributes);
	}
	if (delayParser_) { /* parsing a nested delay payload? */
		delayParser_->handleStartElement(element, ns, attributes);
	}
	++level_;
}

void ForwardedParser::handleEndElement(const std::string& element, const std::string& ns) {
	--level_;
	if (childParser_ && level_ >= PayloadLevel) {
		childParser_->handleEndElement(element, ns);
	}
	if (childParser_ && level_ == PayloadLevel) {
		/* done parsing nested stanza */
		getPayloadInternal()->setStanza(childParser_->getStanza());
		childParser_.reset();
	}
	if (delayParser_ && level_ >= PayloadLevel) {
		delayParser_->handleEndElement(element, ns);
	}
	if (delayParser_ && level_ == PayloadLevel) {
		/* done parsing nested delay payload */
		getPayloadInternal()->setDelay(boost::dynamic_pointer_cast<Delay>(delayParser_->getPayload()));
		delayParser_.reset();
	}
}

void ForwardedParser::handleCharacterData(const std::string& data) {
	if (childParser_) {
		childParser_->handleCharacterData(data);
	}
	if (delayParser_) {
		delayParser_->handleCharacterData(data);
	}
}