/* * libjingle * Copyright 2011, Google Inc. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "webrtc/libjingle/xmpp/pubsubtasks.h" #include #include #include "webrtc/libjingle/xmpp/constants.h" #include "webrtc/libjingle/xmpp/receivetask.h" // An implementation of the tasks for XEP-0060 // (http://xmpp.org/extensions/xep-0060.html). namespace buzz { namespace { bool IsPubSubEventItemsElem(const XmlElement* stanza, const std::string& expected_node) { if (stanza->Name() != QN_MESSAGE) { return false; } const XmlElement* event_elem = stanza->FirstNamed(QN_PUBSUB_EVENT); if (event_elem == NULL) { return false; } const XmlElement* items_elem = event_elem->FirstNamed(QN_PUBSUB_EVENT_ITEMS); if (items_elem == NULL) { return false; } const std::string& actual_node = items_elem->Attr(QN_NODE); return (actual_node == expected_node); } // Creates XmlElement* CreatePubSubItemsElem(const std::string& node) { XmlElement* items_elem = new XmlElement(QN_PUBSUB_ITEMS, false); items_elem->AddAttr(QN_NODE, node); XmlElement* pubsub_elem = new XmlElement(QN_PUBSUB, false); pubsub_elem->AddElement(items_elem); return pubsub_elem; } // Creates payload... // Takes ownership of payload. XmlElement* CreatePubSubPublishItemElem( const std::string& node, const std::string& itemid, const std::vector& children) { XmlElement* pubsub_elem = new XmlElement(QN_PUBSUB, true); XmlElement* publish_elem = new XmlElement(QN_PUBSUB_PUBLISH, false); publish_elem->AddAttr(QN_NODE, node); XmlElement* item_elem = new XmlElement(QN_PUBSUB_ITEM, false); item_elem->AddAttr(QN_ID, itemid); for (std::vector::const_iterator child = children.begin(); child != children.end(); ++child) { item_elem->AddElement(*child); } publish_elem->AddElement(item_elem); pubsub_elem->AddElement(publish_elem); return pubsub_elem; } // Creates payload... // Takes ownership of payload. XmlElement* CreatePubSubRetractItemElem(const std::string& node, const std::string& itemid) { XmlElement* pubsub_elem = new XmlElement(QN_PUBSUB, true); XmlElement* retract_elem = new XmlElement(QN_PUBSUB_RETRACT, false); retract_elem->AddAttr(QN_NODE, node); retract_elem->AddAttr(QN_NOTIFY, "true"); XmlElement* item_elem = new XmlElement(QN_PUBSUB_ITEM, false); item_elem->AddAttr(QN_ID, itemid); retract_elem->AddElement(item_elem); pubsub_elem->AddElement(retract_elem); return pubsub_elem; } void ParseItem(const XmlElement* item_elem, std::vector* items) { PubSubItem item; item.itemid = item_elem->Attr(QN_ID); item.elem = item_elem; items->push_back(item); } // Right now, s are treated the same as items with empty // payloads. We may want to change it in the future, but right now // it's sufficient for our needs. void ParseRetract(const XmlElement* retract_elem, std::vector* items) { ParseItem(retract_elem, items); } void ParseEventItemsElem(const XmlElement* stanza, std::vector* items) { const XmlElement* event_elem = stanza->FirstNamed(QN_PUBSUB_EVENT); if (event_elem != NULL) { const XmlElement* items_elem = event_elem->FirstNamed(QN_PUBSUB_EVENT_ITEMS); if (items_elem != NULL) { for (const XmlElement* item_elem = items_elem->FirstNamed(QN_PUBSUB_EVENT_ITEM); item_elem != NULL; item_elem = item_elem->NextNamed(QN_PUBSUB_EVENT_ITEM)) { ParseItem(item_elem, items); } for (const XmlElement* retract_elem = items_elem->FirstNamed(QN_PUBSUB_EVENT_RETRACT); retract_elem != NULL; retract_elem = retract_elem->NextNamed(QN_PUBSUB_EVENT_RETRACT)) { ParseRetract(retract_elem, items); } } } } void ParsePubSubItemsElem(const XmlElement* stanza, std::vector* items) { const XmlElement* pubsub_elem = stanza->FirstNamed(QN_PUBSUB); if (pubsub_elem != NULL) { const XmlElement* items_elem = pubsub_elem->FirstNamed(QN_PUBSUB_ITEMS); if (items_elem != NULL) { for (const XmlElement* item_elem = items_elem->FirstNamed(QN_PUBSUB_ITEM); item_elem != NULL; item_elem = item_elem->NextNamed(QN_PUBSUB_ITEM)) { ParseItem(item_elem, items); } } } } } // namespace PubSubRequestTask::PubSubRequestTask(XmppTaskParentInterface* parent, const Jid& pubsubjid, const std::string& node) : IqTask(parent, STR_GET, pubsubjid, CreatePubSubItemsElem(node)) { } void PubSubRequestTask::HandleResult(const XmlElement* stanza) { std::vector items; ParsePubSubItemsElem(stanza, &items); SignalResult(this, items); } int PubSubReceiveTask::ProcessStart() { if (SignalUpdate.is_empty()) { return STATE_DONE; } return ReceiveTask::ProcessStart(); } bool PubSubReceiveTask::WantsStanza(const XmlElement* stanza) { return MatchStanzaFrom(stanza, pubsubjid_) && IsPubSubEventItemsElem(stanza, node_) && !SignalUpdate.is_empty(); } void PubSubReceiveTask::ReceiveStanza(const XmlElement* stanza) { std::vector items; ParseEventItemsElem(stanza, &items); SignalUpdate(this, items); } PubSubPublishTask::PubSubPublishTask(XmppTaskParentInterface* parent, const Jid& pubsubjid, const std::string& node, const std::string& itemid, const std::vector& children) : IqTask(parent, STR_SET, pubsubjid, CreatePubSubPublishItemElem(node, itemid, children)), itemid_(itemid) { } void PubSubPublishTask::HandleResult(const XmlElement* stanza) { SignalResult(this); } PubSubRetractTask::PubSubRetractTask(XmppTaskParentInterface* parent, const Jid& pubsubjid, const std::string& node, const std::string& itemid) : IqTask(parent, STR_SET, pubsubjid, CreatePubSubRetractItemElem(node, itemid)), itemid_(itemid) { } void PubSubRetractTask::HandleResult(const XmlElement* stanza) { SignalResult(this); } } // namespace buzz