Line data Source code
1 : /*
2 : * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3 : */
4 :
5 : #include "xmpp/xmpp_proto.h"
6 : #include <iostream>
7 : #include <string>
8 : #include <boost/algorithm/string/replace.hpp>
9 : #include <boost/algorithm/string/predicate.hpp>
10 :
11 : #include "xmpp/xmpp_connection.h"
12 : #include "xmpp/xmpp_log.h"
13 : #include "xmpp/xmpp_session.h"
14 : #include "xmpp/xmpp_str.h"
15 :
16 : #include "sandesh/sandesh_trace.h"
17 : #include "sandesh/common/vns_types.h"
18 : #include "sandesh/common/vns_constants.h"
19 : #include "sandesh/xmpp_message_sandesh_types.h"
20 : #include "sandesh/xmpp_trace_sandesh_types.h"
21 :
22 : using namespace std;
23 :
24 : unique_ptr<XmlBase> XmppProto::open_doc_(AllocXmppXmlImpl(sXMPP_STREAM_OPEN));
25 :
26 0 : XmppStanza::XmppStanza() {
27 0 : }
28 :
29 0 : XmppProto::XmppProto() {
30 0 : }
31 :
32 0 : XmppProto::~XmppProto() {
33 0 : }
34 :
35 20318 : int XmppProto::EncodeStream(const XmppStreamMessage &str, string &to,
36 : string &from, const string &xmlns, uint8_t *buf,
37 : size_t size) {
38 20318 : int len = 0;
39 :
40 20318 : switch (str.strmtype) {
41 8045 : case (XmppStanza::XmppStreamMessage::INIT_STREAM_HEADER):
42 8045 : len = EncodeOpen(buf, to, from, xmlns, size);
43 8045 : break;
44 7828 : case (XmppStanza::XmppStreamMessage::INIT_STREAM_HEADER_RESP):
45 7828 : len = EncodeOpenResp(buf, to, from, size);
46 7829 : break;
47 4445 : case (XmppStanza::XmppStreamMessage::FEATURE_TLS):
48 4445 : switch (str.strmtlstype) {
49 1512 : case (XmppStanza::XmppStreamMessage::TLS_FEATURE_REQUEST):
50 1512 : len = EncodeFeatureTlsRequest(buf);
51 1512 : break;
52 1469 : case (XmppStanza::XmppStreamMessage::TLS_START):
53 1469 : len = EncodeFeatureTlsStart(buf);
54 1469 : break;
55 1464 : case (XmppStanza::XmppStreamMessage::TLS_PROCEED):
56 1464 : len = EncodeFeatureTlsProceed(buf);
57 1464 : break;
58 : }
59 4445 : break;
60 0 : default:
61 0 : break;
62 : }
63 :
64 20319 : return len;
65 : }
66 :
67 6448 : int XmppProto::EncodeStream(const XmppStanza::XmppMessage &str, uint8_t *buf,
68 : size_t size) {
69 6448 : int ret = 0;
70 :
71 6448 : if (str.type == XmppStanza::WHITESPACE_MESSAGE_STANZA) {
72 6448 : return EncodeWhitespace(buf);
73 : }
74 :
75 0 : return ret;
76 : }
77 :
78 0 : int XmppProto::EncodeMessage(XmlBase *dom, uint8_t *buf, size_t size) {
79 0 : int len = dom->WriteDoc(buf);
80 :
81 0 : return len;
82 : }
83 :
84 0 : int XmppProto::EncodePresence(uint8_t *buf, size_t size) {
85 0 : return 0;
86 : }
87 :
88 0 : int XmppProto::EncodeIq(const XmppStanza::XmppMessageIq *iq,
89 : XmlBase *doc, uint8_t *buf, size_t size) {
90 0 : unique_ptr<XmlBase> send_doc_(AllocXmppXmlImpl());
91 :
92 : // create
93 0 : send_doc_->LoadDoc("");
94 0 : send_doc_->AddNode("iq", "");
95 :
96 0 : switch(iq->stype) {
97 0 : case XmppStanza::XmppMessageIq::GET:
98 0 : send_doc_->AddAttribute("type", "get");
99 0 : break;
100 0 : case XmppStanza::XmppMessageIq::SET:
101 0 : send_doc_->AddAttribute("type", "set");
102 0 : break;
103 0 : case XmppStanza::XmppMessageIq::RESULT:
104 0 : send_doc_->AddAttribute("type", "result");
105 0 : break;
106 0 : case XmppStanza::XmppMessageIq::ERROR:
107 0 : send_doc_->AddAttribute("type", "error");
108 0 : break;
109 0 : default:
110 0 : break;
111 : }
112 0 : send_doc_->AddAttribute("from", iq->from);
113 0 : send_doc_->AddAttribute("to", iq->to);
114 0 : send_doc_->AddAttribute("id", "id1");
115 :
116 0 : send_doc_->AddChildNode("pubsub", "");
117 0 : send_doc_->AddAttribute("xmlns", "http://jabber.org/protocol/pubsub");
118 :
119 0 : send_doc_->AppendDoc("pubsub", doc);
120 :
121 : //Returns byte encoded in the doc
122 0 : int len = send_doc_->WriteDoc(buf);
123 :
124 0 : return len;
125 0 : }
126 :
127 6448 : int XmppProto::EncodeWhitespace(uint8_t *buf) {
128 6448 : string str(sXMPP_WHITESPACE);
129 :
130 6448 : int len = str.size();
131 6448 : if (len > 0) {
132 6448 : memcpy(buf, str.data(), len);
133 : }
134 :
135 6448 : return len;
136 6448 : }
137 :
138 7828 : int XmppProto::EncodeOpenResp(uint8_t *buf, string &to, string &from,
139 : size_t max_size) {
140 :
141 7828 : unique_ptr<XmlBase> resp_doc(XmppStanza::AllocXmppXmlImpl(sXMPP_STREAM_RESP));
142 :
143 7829 : if (resp_doc.get() == NULL) {
144 0 : return 0;
145 : }
146 :
147 7829 : SetTo(to, resp_doc.get());
148 7826 : SetFrom(from, resp_doc.get());
149 :
150 7828 : std::stringstream ss;
151 7829 : resp_doc->PrintDoc(ss);
152 7829 : std::string msg;
153 7829 : msg = ss.str();
154 7829 : size_t len = msg.size();
155 7829 : if (len > max_size) {
156 0 : LOG(ERROR, "\n (Open Confirm) size greater than max buffer size \n");
157 0 : return 0;
158 : } else {
159 7829 : boost::algorithm::ireplace_last(msg, "/", " ");
160 7829 : memcpy(buf, msg.c_str(), len);
161 7829 : return len;
162 : }
163 7829 : }
164 :
165 8045 : int XmppProto::EncodeOpen(uint8_t *buf, string &to, string &from,
166 : const string &xmlns, size_t max_size) {
167 :
168 8045 : if (open_doc_.get() == NULL) {
169 0 : return 0;
170 : }
171 :
172 8045 : SetTo(to, open_doc_.get());
173 8045 : SetFrom(from, open_doc_.get());
174 8045 : SetXmlns(xmlns, open_doc_.get());
175 :
176 : //Returns byte encoded in the doc
177 8045 : std::stringstream ss;
178 8045 : open_doc_->PrintDoc(ss);
179 8045 : std::string msg;
180 8045 : msg = ss.str();
181 8045 : size_t len = msg.size();
182 8045 : if (len > max_size) {
183 0 : LOG(ERROR, "\n (Open Message) size greater than max buffer size \n");
184 0 : return 0;
185 : } else {
186 8045 : boost::algorithm::ireplace_last(msg, "/", " ");
187 8045 : memcpy(buf, msg.c_str(), len);
188 8045 : return len;
189 : }
190 8045 : }
191 :
192 1512 : int XmppProto::EncodeFeatureTlsRequest(uint8_t *buf) {
193 1512 : unique_ptr<XmlBase> resp_doc(XmppStanza::AllocXmppXmlImpl(sXMPP_STREAM_FEATURE_TLS));
194 : //Returns byte encoded in the doc
195 1512 : int len = resp_doc->WriteDoc(buf);
196 1512 : return len;
197 1512 : }
198 :
199 1469 : int XmppProto::EncodeFeatureTlsStart(uint8_t *buf) {
200 1469 : unique_ptr<XmlBase> resp_doc(XmppStanza::AllocXmppXmlImpl(sXMPP_STREAM_START_TLS));
201 : //Returns byte encoded in the doc
202 1469 : int len = resp_doc->WriteDoc(buf);
203 1469 : return len;
204 1469 : }
205 :
206 1464 : int XmppProto::EncodeFeatureTlsProceed(uint8_t *buf) {
207 1464 : unique_ptr<XmlBase> resp_doc(XmppStanza::AllocXmppXmlImpl(sXMPP_STREAM_PROCEED_TLS));
208 : //Returns byte encoded in the doc
209 1464 : int len = resp_doc->WriteDoc(buf);
210 1464 : return len;
211 1464 : }
212 :
213 4243150 : XmppStanza::XmppMessage *XmppProto::Decode(const XmppConnection *connection,
214 : const string &ts) {
215 4243150 : XmlBase *impl = XmppStanza::AllocXmppXmlImpl();
216 4243113 : if (impl == nullptr) {
217 0 : return nullptr;
218 : }
219 :
220 4243113 : XmppStanza::XmppMessage *msg = DecodeInternal(connection, ts, impl);
221 4243170 : if (!msg) {
222 0 : return nullptr;
223 : }
224 :
225 : // transfer ownership of the dom implementation
226 4243170 : msg->dom.reset(impl);
227 :
228 4243133 : return msg;
229 : }
230 :
231 4243111 : XmppStanza::XmppMessage *XmppProto::DecodeInternal(
232 : const XmppConnection *connection, const string &ts, XmlBase *impl) {
233 4243111 : XmppStanza::XmppMessage *ret = nullptr;
234 :
235 4243111 : string ns(sXMPP_STREAM_O);
236 4243108 : string ws(sXMPP_WHITESPACE);
237 4243095 : string iq(sXMPP_IQ_KEY);
238 :
239 4243085 : if (ts.find(sXMPP_IQ) != string::npos) {
240 98576 : string ts_tmp = ts;
241 :
242 98571 : if (impl->LoadDoc(ts) == -1) {
243 0 : XMPP_WARNING(XmppIqMessageParseFail, connection->ToUVEKey(),
244 : XMPP_PEER_DIR_IN);
245 0 : assert(false);
246 : goto done;
247 : }
248 :
249 98580 : XmppStanza::XmppMessageIq *msg = new XmppStanza::XmppMessageIq;
250 98574 : impl->ReadNode(iq);
251 98573 : msg->to = XmppProto::GetTo(impl);
252 98576 : msg->from = XmppProto::GetFrom(impl);
253 98577 : msg->id = XmppProto::GetId(impl);
254 98577 : msg->iq_type = XmppProto::GetType(impl);
255 : // action is subscribe,publish,collection
256 98577 : const char *action = XmppProto::GetAction(impl, msg->iq_type);
257 98577 : if (action) {
258 98562 : msg->action = action;
259 : }
260 98577 : if (XmppProto::GetNode(impl, msg->action)) {
261 98563 : msg->node = XmppProto::GetNode(impl, msg->action);
262 : }
263 : //associate or dissociate collection node
264 98576 : if (msg->action.compare("collection") == 0) {
265 41673 : if (XmppProto::GetAsNode(impl)) {
266 27306 : msg->as_node = XmppProto::GetAsNode(impl);
267 27306 : msg->is_as_node = true;
268 14367 : } else if (XmppProto::GetDsNode(impl)) {
269 14339 : msg->as_node = XmppProto::GetDsNode(impl);
270 14339 : msg->is_as_node = false;
271 : }
272 : }
273 :
274 : //msg->dom.reset(impl);
275 :
276 98577 : ret = msg;
277 :
278 98577 : XMPP_UTDEBUG(XmppIqMessageProcess, connection->ToUVEKey(),
279 : XMPP_PEER_DIR_IN, msg->node, msg->action, msg->from,
280 : msg->to, msg->id, msg->iq_type);
281 98579 : goto done;
282 :
283 4243182 : } else if (ts.find(sXMPP_MESSAGE) != string::npos) {
284 :
285 1585922 : if (impl->LoadDoc(ts) == -1) {
286 0 : XMPP_WARNING(XmppChatMessageParseFail, connection->ToUVEKey(),
287 : XMPP_PEER_DIR_IN);
288 0 : goto done;
289 : }
290 : XmppStanza::XmppMessage *msg = new XmppStanza::XmppChatMessage(
291 1585922 : STATE_NONE);
292 1585922 : impl->ReadNode(sXMPP_MESSAGE_KEY);
293 :
294 1585922 : msg->to = XmppProto::GetTo(impl);
295 1585922 : msg->from = XmppProto::GetFrom(impl);
296 1585922 : ret = msg;
297 :
298 1585922 : XMPP_UTDEBUG(XmppChatMessageProcess, connection->ToUVEKey(),
299 : XMPP_PEER_DIR_IN, msg->type, msg->from, msg->to);
300 1585922 : goto done;
301 :
302 2558682 : } else if (ts.find(sXMPP_STREAM_O) != string::npos) {
303 :
304 : // ensusre stream open is at the beginning of the message
305 15716 : string ts_tmp = ts;
306 15716 : ts_tmp.erase(std::remove(ts_tmp.begin(), ts_tmp.end(), '\n'), ts_tmp.end());
307 :
308 15716 : if ((ts_tmp.compare(0, strlen(sXMPP_STREAM_START),
309 31432 : sXMPP_STREAM_START) != 0) &&
310 15716 : (ts_tmp.compare(0, strlen(sXMPP_STREAM_START_S),
311 : sXMPP_STREAM_START_S) != 0)) {
312 0 : XMPP_WARNING(XmppBadMessage, connection->ToUVEKey(),
313 : XMPP_PEER_DIR_IN,
314 : "Open message not at the beginning.", ts);
315 0 : goto done;
316 : }
317 :
318 : // check if the buf is xmpp open or response message
319 : // As end tag will be missing we need to modify the
320 : // string for stream open, else dom decoder will fail
321 15716 : boost::algorithm::replace_last(ts_tmp, ">", "/>");
322 15715 : if (impl->LoadDoc(ts_tmp) == -1) {
323 0 : XMPP_WARNING(XmppBadMessage, connection->ToUVEKey(),
324 : XMPP_PEER_DIR_IN, "Open message parse failed.", ts);
325 0 : goto done;
326 : }
327 :
328 : XmppStanza::XmppStreamMessage *strm =
329 15716 : new XmppStanza::XmppStreamMessage();
330 15713 : strm->strmtype = XmppStanza::XmppStreamMessage::INIT_STREAM_HEADER;
331 15713 : impl->ReadNode(ns);
332 15712 : strm->to = XmppProto::GetTo(impl);
333 15715 : strm->from = XmppProto::GetFrom(impl);
334 15713 : strm->xmlns = XmppProto::GetXmlns(impl);
335 :
336 15714 : ret = strm;
337 :
338 15714 : XMPP_UTDEBUG(XmppRxOpenMessage, connection->ToUVEKey(),
339 : XMPP_PEER_DIR_IN, strm->from, strm->to);
340 :
341 2558680 : } else if (ts.find(sXMPP_STREAM_NS_TLS) != string::npos) {
342 :
343 4365 : if (impl->LoadDoc(ts) == -1) {
344 0 : XMPP_WARNING(XmppBadMessage, connection->ToUVEKey(),
345 : XMPP_PEER_DIR_IN, "Stream TLS parse failed.", ts);
346 0 : goto done;
347 : }
348 :
349 : // find stream:features tls required
350 5820 : if ((ts.find(sXMPP_STREAM_FEATURES_O) != string::npos) &&
351 5820 : (ts.find(sXMPP_STREAM_STARTTLS_O) != string::npos) &&
352 1455 : (ts.find(sXMPP_REQUIRED_O) != string::npos)) {
353 :
354 : XmppStanza::XmppStreamMessage *strm =
355 1455 : new XmppStanza::XmppStreamMessage();
356 1455 : strm->strmtype = XmppStanza::XmppStreamMessage::FEATURE_TLS;
357 1455 : strm->strmtlstype = XmppStanza::XmppStreamMessage::TLS_FEATURE_REQUEST;
358 :
359 1455 : ret = strm;
360 :
361 1455 : XMPP_UTDEBUG(XmppRxStreamTlsRequired, connection->ToUVEKey(),
362 : XMPP_PEER_DIR_IN);
363 :
364 2910 : } else if (ts.find(sXMPP_STREAM_STARTTLS_O) != string::npos) {
365 : XmppStanza::XmppStreamMessage *strm =
366 1455 : new XmppStanza::XmppStreamMessage();
367 1455 : strm->strmtype = XmppStanza::XmppStreamMessage::FEATURE_TLS;
368 1455 : strm->strmtlstype = XmppStanza::XmppStreamMessage::TLS_START;
369 1455 : ret = strm;
370 :
371 1455 : XMPP_UTDEBUG(XmppRxStreamStartTls, connection->ToUVEKey(),
372 : XMPP_PEER_DIR_IN);
373 :
374 1455 : } else if (ts.find(sXMPP_STREAM_PROCEED_O) != string::npos) {
375 : XmppStanza::XmppStreamMessage *strm =
376 1455 : new XmppStanza::XmppStreamMessage();
377 1455 : strm->strmtype = XmppStanza::XmppStreamMessage::FEATURE_TLS;
378 1455 : strm->strmtlstype = XmppStanza::XmppStreamMessage::TLS_PROCEED;
379 :
380 1455 : ret = strm;
381 :
382 1455 : XMPP_UTDEBUG(XmppRxStreamProceed, connection->ToUVEKey(),
383 : XMPP_PEER_DIR_IN);
384 : }
385 4365 : goto done;
386 :
387 2538604 : } else if (ts.find_first_of(sXMPP_VALIDWS) != string::npos) {
388 :
389 : XmppStanza::XmppMessage *msg =
390 2538604 : new XmppStanza::XmppMessage(WHITESPACE_MESSAGE_STANZA);
391 2538604 : return msg;
392 : } else {
393 0 : XMPP_WARNING(XmppBadMessage, connection->ToUVEKey(),
394 : XMPP_PEER_DIR_IN, "Message not supported", ts);
395 : }
396 :
397 1704574 : done:
398 :
399 1704574 : return ret;
400 4243178 : }
401 :
402 15873 : int XmppProto::SetTo(string &to, XmlBase *doc) {
403 15873 : if (!doc) return -1;
404 :
405 15873 : string ns(sXMPP_STREAM_O);
406 15873 : doc->ReadNode(ns);
407 15871 : doc->ModifyAttribute("to", to);
408 :
409 15871 : return 0;
410 15871 : }
411 :
412 15871 : int XmppProto::SetFrom(string &from, XmlBase *doc) {
413 15871 : if (!doc) return -1;
414 :
415 15871 : string ns(sXMPP_STREAM_O);
416 15871 : doc->ReadNode(ns);
417 15871 : return doc->ModifyAttribute("from", from);
418 15873 : }
419 :
420 8045 : int XmppProto::SetXmlns(const string &xmlns, XmlBase *doc) {
421 8045 : if (!doc)
422 0 : return -1;
423 :
424 8045 : string ns(sXMPP_STREAM_O);
425 8045 : doc->ReadNode(ns);
426 8045 : return doc->ModifyAttribute("xmlns", xmlns);
427 8045 : }
428 :
429 1700177 : const char *XmppProto::GetTo(XmlBase *doc) {
430 1700177 : if (!doc) return NULL;
431 :
432 1700177 : string tmp("to");
433 1700180 : return doc->ReadAttrib(tmp);
434 1700181 : }
435 :
436 1700191 : const char *XmppProto::GetFrom(XmlBase *doc) {
437 1700191 : if (!doc) return NULL;
438 :
439 1700191 : string tmp("from");
440 1700192 : return doc->ReadAttrib(tmp);
441 1700193 : }
442 :
443 15712 : const char *XmppProto::GetXmlns(XmlBase *doc) {
444 15712 : if (!doc)
445 0 : return NULL;
446 :
447 15712 : string tmp("xmlns");
448 15713 : return doc->ReadAttrib(tmp);
449 15715 : }
450 :
451 98578 : const char *XmppProto::GetId(XmlBase *doc) {
452 98578 : if (!doc) return NULL;
453 :
454 98578 : string tmp("id");
455 98577 : return doc->ReadAttrib(tmp);
456 98577 : }
457 :
458 98576 : const char *XmppProto::GetType(XmlBase *doc) {
459 98576 : if (!doc) return NULL;
460 :
461 98576 : string tmp("type");
462 98576 : return doc->ReadAttrib(tmp);
463 98577 : }
464 :
465 98577 : const char *XmppProto::GetAction(XmlBase *doc, const string &str) {
466 98577 : if (!doc) return NULL;
467 :
468 98577 : if (str.compare("set") == 0) {
469 98565 : doc->ReadNode("pubsub");
470 98562 : return(doc->ReadChildNodeName());
471 15 : } else if (str.compare("get") == 0) {
472 : }
473 :
474 15 : return(NULL);
475 : }
476 :
477 197138 : const char *XmppProto::GetNode(XmlBase *doc, const string &str) {
478 197138 : if (!doc) return NULL;
479 :
480 197138 : if (!str.empty()) {
481 197123 : return(doc->ReadAttrib("node"));
482 : }
483 :
484 15 : return(NULL);
485 : }
486 :
487 68978 : const char *XmppProto::GetAsNode(XmlBase *doc) {
488 68978 : if (!doc) return NULL;
489 :
490 68978 : const char *node = doc->ReadNode("associate");
491 68977 : if (node != NULL) {
492 54610 : return(doc->ReadAttrib("node"));
493 : }
494 :
495 14367 : return(NULL);
496 : }
497 :
498 28705 : const char *XmppProto::GetDsNode(XmlBase *doc) {
499 28705 : if (!doc) return NULL;
500 :
501 28705 : const char *node = doc->ReadNode("dissociate");
502 28704 : if (node != NULL) {
503 28676 : return(doc->ReadAttrib("node"));
504 : }
505 :
506 28 : return(NULL);
507 : }
|