Home | History | Annotate | Download | only in jingle_glue
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "remoting/jingle_glue/jingle_info_request.h"
      6 
      7 #include "base/bind.h"
      8 #include "base/message_loop/message_loop.h"
      9 #include "base/stl_util.h"
     10 #include "base/strings/string_number_conversions.h"
     11 #include "base/time/time.h"
     12 #include "net/base/net_util.h"
     13 #include "remoting/jingle_glue/iq_sender.h"
     14 #include "third_party/libjingle/source/talk/base/socketaddress.h"
     15 #include "third_party/libjingle/source/talk/xmllite/xmlelement.h"
     16 #include "third_party/libjingle/source/talk/xmpp/constants.h"
     17 
     18 namespace remoting {
     19 
     20 const int kRequestTimeoutSeconds = 5;
     21 
     22 JingleInfoRequest::JingleInfoRequest(SignalStrategy* signal_strategy)
     23     : iq_sender_(signal_strategy) {
     24 }
     25 
     26 JingleInfoRequest::~JingleInfoRequest() {}
     27 
     28 void JingleInfoRequest::Send(const OnJingleInfoCallback& callback) {
     29   on_jingle_info_cb_ = callback;
     30   scoped_ptr<buzz::XmlElement> iq_body(
     31       new buzz::XmlElement(buzz::QN_JINGLE_INFO_QUERY, true));
     32   request_ = iq_sender_.SendIq(
     33       buzz::STR_GET, buzz::STR_EMPTY, iq_body.Pass(),
     34       base::Bind(&JingleInfoRequest::OnResponse, base::Unretained(this)));
     35   if (!request_) {
     36     // If we failed to send IqRequest it means that SignalStrategy is
     37     // disconnected. Notify the caller.
     38     std::vector<talk_base::SocketAddress> stun_hosts;
     39     std::vector<std::string> relay_hosts;
     40     std::string relay_token;
     41     on_jingle_info_cb_.Run(relay_token, relay_hosts, stun_hosts);
     42     return;
     43   }
     44   request_->SetTimeout(base::TimeDelta::FromSeconds(kRequestTimeoutSeconds));
     45 }
     46 
     47 void JingleInfoRequest::OnResponse(IqRequest* request,
     48                                    const buzz::XmlElement* stanza) {
     49   std::vector<talk_base::SocketAddress> stun_hosts;
     50   std::vector<std::string> relay_hosts;
     51   std::string relay_token;
     52 
     53   if (!stanza) {
     54     LOG(WARNING) << "Jingle info request has timed out.";
     55     on_jingle_info_cb_.Run(relay_token, relay_hosts, stun_hosts);
     56     return;
     57   }
     58 
     59   const buzz::XmlElement* query =
     60       stanza->FirstNamed(buzz::QN_JINGLE_INFO_QUERY);
     61   if (query == NULL) {
     62     LOG(WARNING) << "No Jingle info found in Jingle Info query response."
     63                  << stanza->Str();
     64     on_jingle_info_cb_.Run(relay_token, relay_hosts, stun_hosts);
     65     return;
     66   }
     67 
     68   const buzz::XmlElement* stun = query->FirstNamed(buzz::QN_JINGLE_INFO_STUN);
     69   if (stun) {
     70     for (const buzz::XmlElement* server =
     71          stun->FirstNamed(buzz::QN_JINGLE_INFO_SERVER);
     72          server != NULL;
     73          server = server->NextNamed(buzz::QN_JINGLE_INFO_SERVER)) {
     74       std::string host = server->Attr(buzz::QN_JINGLE_INFO_HOST);
     75       std::string port_str = server->Attr(buzz::QN_JINGLE_INFO_UDP);
     76       if (host != buzz::STR_EMPTY && port_str != buzz::STR_EMPTY) {
     77         int port;
     78         if (!base::StringToInt(port_str, &port)) {
     79           LOG(WARNING) << "Unable to parse port in stanza" << stanza->Str();
     80           continue;
     81         }
     82 
     83         stun_hosts.push_back(talk_base::SocketAddress(host, port));
     84       }
     85     }
     86   }
     87 
     88   const buzz::XmlElement* relay = query->FirstNamed(buzz::QN_JINGLE_INFO_RELAY);
     89   if (relay) {
     90     relay_token = relay->TextNamed(buzz::QN_JINGLE_INFO_TOKEN);
     91     for (const buzz::XmlElement* server =
     92          relay->FirstNamed(buzz::QN_JINGLE_INFO_SERVER);
     93          server != NULL;
     94          server = server->NextNamed(buzz::QN_JINGLE_INFO_SERVER)) {
     95       std::string host = server->Attr(buzz::QN_JINGLE_INFO_HOST);
     96       if (host != buzz::STR_EMPTY)
     97         relay_hosts.push_back(host);
     98     }
     99   }
    100 
    101   on_jingle_info_cb_.Run(relay_token, relay_hosts, stun_hosts);
    102 }
    103 
    104 }  // namespace remoting
    105