// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef REMOTING_JINGLE_GLUE_IQ_SENDER_H_ #define REMOTING_JINGLE_GLUE_IQ_SENDER_H_ #include <map> #include <string> #include "base/callback.h" #include "base/compiler_specific.h" #include "base/gtest_prod_util.h" #include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "remoting/jingle_glue/signal_strategy.h" namespace base { class TimeDelta; } // namespace base namespace buzz { class XmlElement; } // namespace buzz namespace remoting { class IqRequest; class SignalStrategy; // IqSender handles sending iq requests and routing of responses to // those requests. class IqSender : public SignalStrategy::Listener { public: // Callback that is called when an Iq response is received. Called // with the |response| set to NULL in case of a timeout. typedef base::Callback<void(IqRequest* request, const buzz::XmlElement* response)> ReplyCallback; explicit IqSender(SignalStrategy* signal_strategy); virtual ~IqSender(); // Send an iq stanza. Returns an IqRequest object that represends // the request. |callback| is called when response to |stanza| is // received. Destroy the returned IqRequest to cancel the callback. // Caller must take ownership of the result. Result must be // destroyed before sender is destroyed. scoped_ptr<IqRequest> SendIq(scoped_ptr<buzz::XmlElement> stanza, const ReplyCallback& callback); // Same as above, but also formats the message. scoped_ptr<IqRequest> SendIq(const std::string& type, const std::string& addressee, scoped_ptr<buzz::XmlElement> iq_body, const ReplyCallback& callback); // SignalStrategy::Listener implementation. virtual void OnSignalStrategyStateChange( SignalStrategy::State state) OVERRIDE; virtual bool OnSignalStrategyIncomingStanza( const buzz::XmlElement* stanza) OVERRIDE; private: typedef std::map<std::string, IqRequest*> IqRequestMap; friend class IqRequest; // Helper function used to create iq stanzas. static scoped_ptr<buzz::XmlElement> MakeIqStanza( const std::string& type, const std::string& addressee, scoped_ptr<buzz::XmlElement> iq_body); // Removes |request| from the list of pending requests. Called by IqRequest. void RemoveRequest(IqRequest* request); SignalStrategy* signal_strategy_; IqRequestMap requests_; DISALLOW_COPY_AND_ASSIGN(IqSender); }; // This call must only be used on the thread it was created on. class IqRequest : public base::SupportsWeakPtr<IqRequest> { public: IqRequest(IqSender* sender, const IqSender::ReplyCallback& callback, const std::string& addressee); ~IqRequest(); // Sets timeout for the request. When the timeout expires the // callback is called with the |response| set to NULL. void SetTimeout(base::TimeDelta timeout); private: friend class IqSender; void CallCallback(const buzz::XmlElement* stanza); void OnTimeout(); // Called by IqSender when a response is received. void OnResponse(const buzz::XmlElement* stanza); void DeliverResponse(scoped_ptr<buzz::XmlElement> stanza); IqSender* sender_; IqSender::ReplyCallback callback_; std::string addressee_; DISALLOW_COPY_AND_ASSIGN(IqRequest); }; } // namespace remoting #endif // REMOTING_JINGLE_GLUE_IQ_SENDER_H_