OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
xmpp_proto.cc
Go to the documentation of this file.
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 
27 }
28 
30 }
31 
33 }
34 
35 int XmppProto::EncodeStream(const XmppStreamMessage &str, string &to,
36  string &from, const string &xmlns, uint8_t *buf,
37  size_t size) {
38  int len = 0;
39 
40  switch (str.strmtype) {
42  len = EncodeOpen(buf, to, from, xmlns, size);
43  break;
45  len = EncodeOpenResp(buf, to, from, size);
46  break;
48  switch (str.strmtlstype) {
50  len = EncodeFeatureTlsRequest(buf);
51  break;
53  len = EncodeFeatureTlsStart(buf);
54  break;
56  len = EncodeFeatureTlsProceed(buf);
57  break;
58  }
59  break;
60  default:
61  break;
62  }
63 
64  return len;
65 }
66 
67 int XmppProto::EncodeStream(const XmppStanza::XmppMessage &str, uint8_t *buf,
68  size_t size) {
69  int ret = 0;
70 
72  return EncodeWhitespace(buf);
73  }
74 
75  return ret;
76 }
77 
78 int XmppProto::EncodeMessage(XmlBase *dom, uint8_t *buf, size_t size) {
79  int len = dom->WriteDoc(buf);
80 
81  return len;
82 }
83 
84 int XmppProto::EncodePresence(uint8_t *buf, size_t size) {
85  return 0;
86 }
87 
89  XmlBase *doc, uint8_t *buf, size_t size) {
90  unique_ptr<XmlBase> send_doc_(AllocXmppXmlImpl());
91 
92  // create
93  send_doc_->LoadDoc("");
94  send_doc_->AddNode("iq", "");
95 
96  switch(iq->stype) {
98  send_doc_->AddAttribute("type", "get");
99  break;
101  send_doc_->AddAttribute("type", "set");
102  break;
104  send_doc_->AddAttribute("type", "result");
105  break;
107  send_doc_->AddAttribute("type", "error");
108  break;
109  default:
110  break;
111  }
112  send_doc_->AddAttribute("from", iq->from);
113  send_doc_->AddAttribute("to", iq->to);
114  send_doc_->AddAttribute("id", "id1");
115 
116  send_doc_->AddChildNode("pubsub", "");
117  send_doc_->AddAttribute("xmlns", "http://jabber.org/protocol/pubsub");
118 
119  send_doc_->AppendDoc("pubsub", doc);
120 
121  //Returns byte encoded in the doc
122  int len = send_doc_->WriteDoc(buf);
123 
124  return len;
125 }
126 
127 int XmppProto::EncodeWhitespace(uint8_t *buf) {
128  string str(sXMPP_WHITESPACE);
129 
130  int len = str.size();
131  if (len > 0) {
132  memcpy(buf, str.data(), len);
133  }
134 
135  return len;
136 }
137 
138 int XmppProto::EncodeOpenResp(uint8_t *buf, string &to, string &from,
139  size_t max_size) {
140 
141  unique_ptr<XmlBase> resp_doc(XmppStanza::AllocXmppXmlImpl(sXMPP_STREAM_RESP));
142 
143  if (resp_doc.get() == NULL) {
144  return 0;
145  }
146 
147  SetTo(to, resp_doc.get());
148  SetFrom(from, resp_doc.get());
149 
150  std::stringstream ss;
151  resp_doc->PrintDoc(ss);
152  std::string msg;
153  msg = ss.str();
154  size_t len = msg.size();
155  if (len > max_size) {
156  LOG(ERROR, "\n (Open Confirm) size greater than max buffer size \n");
157  return 0;
158  } else {
159  boost::algorithm::ireplace_last(msg, "/", " ");
160  memcpy(buf, msg.c_str(), len);
161  return len;
162  }
163 }
164 
165 int XmppProto::EncodeOpen(uint8_t *buf, string &to, string &from,
166  const string &xmlns, size_t max_size) {
167 
168  if (open_doc_.get() == NULL) {
169  return 0;
170  }
171 
172  SetTo(to, open_doc_.get());
173  SetFrom(from, open_doc_.get());
174  SetXmlns(xmlns, open_doc_.get());
175 
176  //Returns byte encoded in the doc
177  std::stringstream ss;
178  open_doc_->PrintDoc(ss);
179  std::string msg;
180  msg = ss.str();
181  size_t len = msg.size();
182  if (len > max_size) {
183  LOG(ERROR, "\n (Open Message) size greater than max buffer size \n");
184  return 0;
185  } else {
186  boost::algorithm::ireplace_last(msg, "/", " ");
187  memcpy(buf, msg.c_str(), len);
188  return len;
189  }
190 }
191 
193  unique_ptr<XmlBase> resp_doc(XmppStanza::AllocXmppXmlImpl(sXMPP_STREAM_FEATURE_TLS));
194  //Returns byte encoded in the doc
195  int len = resp_doc->WriteDoc(buf);
196  return len;
197 }
198 
200  unique_ptr<XmlBase> resp_doc(XmppStanza::AllocXmppXmlImpl(sXMPP_STREAM_START_TLS));
201  //Returns byte encoded in the doc
202  int len = resp_doc->WriteDoc(buf);
203  return len;
204 }
205 
207  unique_ptr<XmlBase> resp_doc(XmppStanza::AllocXmppXmlImpl(sXMPP_STREAM_PROCEED_TLS));
208  //Returns byte encoded in the doc
209  int len = resp_doc->WriteDoc(buf);
210  return len;
211 }
212 
214  const string &ts) {
216  if (impl == nullptr) {
217  return nullptr;
218  }
219 
220  XmppStanza::XmppMessage *msg = DecodeInternal(connection, ts, impl);
221  if (!msg) {
222  return nullptr;
223  }
224 
225  // transfer ownership of the dom implementation
226  msg->dom.reset(impl);
227 
228  return msg;
229 }
230 
232  const XmppConnection *connection, const string &ts, XmlBase *impl) {
233  XmppStanza::XmppMessage *ret = nullptr;
234 
235  string ns(sXMPP_STREAM_O);
236  string ws(sXMPP_WHITESPACE);
237  string iq(sXMPP_IQ_KEY);
238 
239  if (ts.find(sXMPP_IQ) != string::npos) {
240  string ts_tmp = ts;
241 
242  if (impl->LoadDoc(ts) == -1) {
243  XMPP_WARNING(XmppIqMessageParseFail, connection->ToUVEKey(),
245  assert(false);
246  goto done;
247  }
248 
250  impl->ReadNode(iq);
251  msg->to = XmppProto::GetTo(impl);
252  msg->from = XmppProto::GetFrom(impl);
253  msg->id = XmppProto::GetId(impl);
254  msg->iq_type = XmppProto::GetType(impl);
255  // action is subscribe,publish,collection
256  const char *action = XmppProto::GetAction(impl, msg->iq_type);
257  if (action) {
258  msg->action = action;
259  }
260  if (XmppProto::GetNode(impl, msg->action)) {
261  msg->node = XmppProto::GetNode(impl, msg->action);
262  }
263  //associate or dissociate collection node
264  if (msg->action.compare("collection") == 0) {
265  if (XmppProto::GetAsNode(impl)) {
266  msg->as_node = XmppProto::GetAsNode(impl);
267  msg->is_as_node = true;
268  } else if (XmppProto::GetDsNode(impl)) {
269  msg->as_node = XmppProto::GetDsNode(impl);
270  msg->is_as_node = false;
271  }
272  }
273 
274  //msg->dom.reset(impl);
275 
276  ret = msg;
277 
278  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  goto done;
282 
283  } else if (ts.find(sXMPP_MESSAGE) != string::npos) {
284 
285  if (impl->LoadDoc(ts) == -1) {
286  XMPP_WARNING(XmppChatMessageParseFail, connection->ToUVEKey(),
288  goto done;
289  }
291  STATE_NONE);
293 
294  msg->to = XmppProto::GetTo(impl);
295  msg->from = XmppProto::GetFrom(impl);
296  ret = msg;
297 
298  XMPP_UTDEBUG(XmppChatMessageProcess, connection->ToUVEKey(),
299  XMPP_PEER_DIR_IN, msg->type, msg->from, msg->to);
300  goto done;
301 
302  } else if (ts.find(sXMPP_STREAM_O) != string::npos) {
303 
304  // ensusre stream open is at the beginning of the message
305  string ts_tmp = ts;
306  ts_tmp.erase(std::remove(ts_tmp.begin(), ts_tmp.end(), '\n'), ts_tmp.end());
307 
308  if ((ts_tmp.compare(0, strlen(sXMPP_STREAM_START),
309  sXMPP_STREAM_START) != 0) &&
310  (ts_tmp.compare(0, strlen(sXMPP_STREAM_START_S),
311  sXMPP_STREAM_START_S) != 0)) {
312  XMPP_WARNING(XmppBadMessage, connection->ToUVEKey(),
314  "Open message not at the beginning.", ts);
315  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  boost::algorithm::replace_last(ts_tmp, ">", "/>");
322  if (impl->LoadDoc(ts_tmp) == -1) {
323  XMPP_WARNING(XmppBadMessage, connection->ToUVEKey(),
324  XMPP_PEER_DIR_IN, "Open message parse failed.", ts);
325  goto done;
326  }
327 
331  impl->ReadNode(ns);
332  strm->to = XmppProto::GetTo(impl);
333  strm->from = XmppProto::GetFrom(impl);
334  strm->xmlns = XmppProto::GetXmlns(impl);
335 
336  ret = strm;
337 
338  XMPP_UTDEBUG(XmppRxOpenMessage, connection->ToUVEKey(),
339  XMPP_PEER_DIR_IN, strm->from, strm->to);
340 
341  } else if (ts.find(sXMPP_STREAM_NS_TLS) != string::npos) {
342 
343  if (impl->LoadDoc(ts) == -1) {
344  XMPP_WARNING(XmppBadMessage, connection->ToUVEKey(),
345  XMPP_PEER_DIR_IN, "Stream TLS parse failed.", ts);
346  goto done;
347  }
348 
349  // find stream:features tls required
350  if ((ts.find(sXMPP_STREAM_FEATURES_O) != string::npos) &&
351  (ts.find(sXMPP_STREAM_STARTTLS_O) != string::npos) &&
352  (ts.find(sXMPP_REQUIRED_O) != string::npos)) {
353 
358 
359  ret = strm;
360 
361  XMPP_UTDEBUG(XmppRxStreamTlsRequired, connection->ToUVEKey(),
363 
364  } else if (ts.find(sXMPP_STREAM_STARTTLS_O) != string::npos) {
369  ret = strm;
370 
371  XMPP_UTDEBUG(XmppRxStreamStartTls, connection->ToUVEKey(),
373 
374  } else if (ts.find(sXMPP_STREAM_PROCEED_O) != string::npos) {
379 
380  ret = strm;
381 
382  XMPP_UTDEBUG(XmppRxStreamProceed, connection->ToUVEKey(),
384  }
385  goto done;
386 
387  } else if (ts.find_first_of(sXMPP_VALIDWS) != string::npos) {
388 
390  new XmppStanza::XmppMessage(WHITESPACE_MESSAGE_STANZA);
391  return msg;
392  } else {
393  XMPP_WARNING(XmppBadMessage, connection->ToUVEKey(),
394  XMPP_PEER_DIR_IN, "Message not supported", ts);
395  }
396 
397 done:
398 
399  return ret;
400 }
401 
402 int XmppProto::SetTo(string &to, XmlBase *doc) {
403  if (!doc) return -1;
404 
405  string ns(sXMPP_STREAM_O);
406  doc->ReadNode(ns);
407  doc->ModifyAttribute("to", to);
408 
409  return 0;
410 }
411 
412 int XmppProto::SetFrom(string &from, XmlBase *doc) {
413  if (!doc) return -1;
414 
415  string ns(sXMPP_STREAM_O);
416  doc->ReadNode(ns);
417  return doc->ModifyAttribute("from", from);
418 }
419 
420 int XmppProto::SetXmlns(const string &xmlns, XmlBase *doc) {
421  if (!doc)
422  return -1;
423 
424  string ns(sXMPP_STREAM_O);
425  doc->ReadNode(ns);
426  return doc->ModifyAttribute("xmlns", xmlns);
427 }
428 
429 const char *XmppProto::GetTo(XmlBase *doc) {
430  if (!doc) return NULL;
431 
432  string tmp("to");
433  return doc->ReadAttrib(tmp);
434 }
435 
436 const char *XmppProto::GetFrom(XmlBase *doc) {
437  if (!doc) return NULL;
438 
439  string tmp("from");
440  return doc->ReadAttrib(tmp);
441 }
442 
443 const char *XmppProto::GetXmlns(XmlBase *doc) {
444  if (!doc)
445  return NULL;
446 
447  string tmp("xmlns");
448  return doc->ReadAttrib(tmp);
449 }
450 
451 const char *XmppProto::GetId(XmlBase *doc) {
452  if (!doc) return NULL;
453 
454  string tmp("id");
455  return doc->ReadAttrib(tmp);
456 }
457 
458 const char *XmppProto::GetType(XmlBase *doc) {
459  if (!doc) return NULL;
460 
461  string tmp("type");
462  return doc->ReadAttrib(tmp);
463 }
464 
465 const char *XmppProto::GetAction(XmlBase *doc, const string &str) {
466  if (!doc) return NULL;
467 
468  if (str.compare("set") == 0) {
469  doc->ReadNode("pubsub");
470  return(doc->ReadChildNodeName());
471  } else if (str.compare("get") == 0) {
472  }
473 
474  return(NULL);
475 }
476 
477 const char *XmppProto::GetNode(XmlBase *doc, const string &str) {
478  if (!doc) return NULL;
479 
480  if (!str.empty()) {
481  return(doc->ReadAttrib("node"));
482  }
483 
484  return(NULL);
485 }
486 
487 const char *XmppProto::GetAsNode(XmlBase *doc) {
488  if (!doc) return NULL;
489 
490  const char *node = doc->ReadNode("associate");
491  if (node != NULL) {
492  return(doc->ReadAttrib("node"));
493  }
494 
495  return(NULL);
496 }
497 
498 const char *XmppProto::GetDsNode(XmlBase *doc) {
499  if (!doc) return NULL;
500 
501  const char *node = doc->ReadNode("dissociate");
502  if (node != NULL) {
503  return(doc->ReadAttrib("node"));
504  }
505 
506  return(NULL);
507 }
#define sXMPP_IQ
Definition: xmpp_str.h:19
static int EncodeOpen(uint8_t *data, std::string &to, std::string &from, const std::string &xmlns, size_t size)
Definition: xmpp_proto.cc:165
static const char * GetXmlns(XmlBase *doc)
Definition: xmpp_proto.cc:443
#define XMPP_UTDEBUG(obj,...)
Definition: xmpp_log.h:67
static const char * GetId(XmlBase *doc)
Definition: xmpp_proto.cc:451
XmppMessageType type
Definition: xmpp_proto.h:57
#define sXMPP_IQ_KEY
Definition: xmpp_str.h:18
virtual int LoadDoc(const std::string &doc)=0
XmppStreamTlsType strmtlstype
Definition: xmpp_proto.h:99
static XmlBase * AllocXmppXmlImpl(const char *doc=NULL)
Definition: xmpp_proto.h:167
#define XMPP_PEER_DIR_IN
Definition: xmpp_log.h:15
virtual const char * ReadChildNodeName()=0
virtual const char * ReadNode(const std::string &name)=0
#define sXMPP_MESSAGE_KEY
Definition: xmpp_str.h:20
#define sXMPP_MESSAGE
Definition: xmpp_str.h:21
static const char * GetType(XmlBase *doc)
Definition: xmpp_proto.cc:458
static int EncodeMessage(const XmppChatMessage *, uint8_t *data, size_t size)
static const char * GetDsNode(XmlBase *doc)
Definition: xmpp_proto.cc:498
#define sXMPP_VALIDWS
Definition: xmpp_str.h:48
XmppStreamMsgType strmtype
Definition: xmpp_proto.h:98
#define sXMPP_STREAM_RESP
Definition: xmpp_str.h:41
#define sXMPP_STREAM_START_TLS
Definition: xmpp_str.h:43
static int EncodePresence(uint8_t *data, size_t size)
Definition: xmpp_proto.cc:84
virtual int ModifyAttribute(const std::string &key, const std::string &value)=0
static const char * GetFrom(XmlBase *doc)
Definition: xmpp_proto.cc:436
static int SetFrom(std::string &from, XmlBase *doc)
Definition: xmpp_proto.cc:412
#define sXMPP_STREAM_START
Definition: xmpp_str.h:51
std::unique_ptr< XmlBase > dom
Definition: xmpp_proto.h:62
static XmppStanza::XmppMessage * DecodeInternal(const XmppConnection *connection, const std::string &ts, XmlBase *impl)
Definition: xmpp_proto.cc:231
static std::unique_ptr< XmlBase > open_doc_
Definition: xmpp_proto.h:220
#define sXMPP_STREAM_PROCEED_TLS
Definition: xmpp_str.h:44
#define sXMPP_STREAM_O
Definition: xmpp_str.h:13
#define sXMPP_STREAM_NS_TLS
Definition: xmpp_str.h:34
static int SetXmlns(const std::string &from, XmlBase *doc)
Definition: xmpp_proto.cc:420
virtual const char * ReadAttrib(const std::string &str)=0
static int EncodeIq(const XmppMessageIq *iq, XmlBase *doc, uint8_t *data, size_t size)
Definition: xmpp_proto.cc:88
static int EncodeFeatureTlsProceed(uint8_t *data)
Definition: xmpp_proto.cc:206
#define sXMPP_STREAM_START_S
Definition: xmpp_str.h:52
static const char * GetAction(XmlBase *doc, const std::string &str)
Definition: xmpp_proto.cc:465
#define sXMPP_REQUIRED_O
Definition: xmpp_str.h:25
static XmppStanza::XmppMessage * Decode(const XmppConnection *connection, const std::string &ts)
Definition: xmpp_proto.cc:213
#define sXMPP_STREAM_PROCEED_O
Definition: xmpp_str.h:24
static const char * GetAsNode(XmlBase *doc)
Definition: xmpp_proto.cc:487
#define sXMPP_STREAM_OPEN
Definition: xmpp_str.h:40
static int SetTo(std::string &to, XmlBase *doc)
Definition: xmpp_proto.cc:402
static int EncodeWhitespace(uint8_t *data)
Definition: xmpp_proto.cc:127
virtual int WriteDoc(uint8_t *buf)=0
#define LOG(_Level, _Msg)
Definition: logging.h:33
#define sXMPP_STREAM_FEATURES_O
Definition: xmpp_str.h:14
static int EncodeStream(const XmppStreamMessage &str, std::string &to, std::string &from, const std::string &xmlns, uint8_t *data, size_t size)
#define sXMPP_STREAM_FEATURE_TLS
Definition: xmpp_str.h:42
static int EncodeFeatureTlsStart(uint8_t *data)
Definition: xmpp_proto.cc:199
static int EncodeFeatureTlsRequest(uint8_t *data)
Definition: xmpp_proto.cc:192
#define sXMPP_WHITESPACE
Definition: xmpp_str.h:46
static int EncodeOpenResp(uint8_t *data, std::string &to, std::string &from, size_t size)
Definition: xmpp_proto.cc:138
const std::string & ToUVEKey() const
static const char * GetNode(XmlBase *doc, const std::string &str)
Definition: xmpp_proto.cc:477
static const char * GetTo(XmlBase *doc)
Definition: xmpp_proto.cc:429
#define XMPP_WARNING(obj,...)
Definition: xmpp_log.h:35
#define sXMPP_STREAM_STARTTLS_O
Definition: xmpp_str.h:22