Home | History | Annotate | Download | only in signaling
      1 // Copyright 2014 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/signaling/fake_signal_strategy.h"
      6 
      7 #include "base/bind.h"
      8 #include "base/location.h"
      9 #include "base/logging.h"
     10 #include "base/single_thread_task_runner.h"
     11 #include "base/stl_util.h"
     12 #include "base/strings/string_number_conversions.h"
     13 #include "base/thread_task_runner_handle.h"
     14 #include "third_party/libjingle/source/talk/xmpp/constants.h"
     15 #include "third_party/webrtc/libjingle/xmllite/xmlelement.h"
     16 
     17 namespace remoting {
     18 
     19 // static
     20 void FakeSignalStrategy::Connect(FakeSignalStrategy* peer1,
     21                                  FakeSignalStrategy* peer2) {
     22   DCHECK(peer1->main_thread_->BelongsToCurrentThread());
     23   DCHECK(peer2->main_thread_->BelongsToCurrentThread());
     24   peer1->ConnectTo(peer2);
     25   peer2->ConnectTo(peer1);
     26 }
     27 
     28 FakeSignalStrategy::FakeSignalStrategy(const std::string& jid)
     29     : main_thread_(base::ThreadTaskRunnerHandle::Get()),
     30       jid_(jid),
     31       last_id_(0),
     32       weak_factory_(this) {
     33 
     34 }
     35 
     36 FakeSignalStrategy::~FakeSignalStrategy() {
     37   while (!received_messages_.empty()) {
     38     delete received_messages_.front();
     39     received_messages_.pop_front();
     40   }
     41 }
     42 
     43 void FakeSignalStrategy::ConnectTo(FakeSignalStrategy* peer) {
     44   PeerCallback peer_callback =
     45       base::Bind(&FakeSignalStrategy::DeliverMessageOnThread,
     46                  main_thread_,
     47                  weak_factory_.GetWeakPtr());
     48   if (peer->main_thread_->BelongsToCurrentThread()) {
     49     peer->SetPeerCallback(peer_callback);
     50   } else {
     51     peer->main_thread_->PostTask(
     52         FROM_HERE,
     53         base::Bind(&FakeSignalStrategy::SetPeerCallback,
     54                    base::Unretained(peer),
     55                    peer_callback));
     56   }
     57 }
     58 
     59 void FakeSignalStrategy::Connect() {
     60   DCHECK(CalledOnValidThread());
     61   FOR_EACH_OBSERVER(Listener, listeners_,
     62                     OnSignalStrategyStateChange(CONNECTED));
     63 }
     64 
     65 void FakeSignalStrategy::Disconnect() {
     66   DCHECK(CalledOnValidThread());
     67   FOR_EACH_OBSERVER(Listener, listeners_,
     68                     OnSignalStrategyStateChange(DISCONNECTED));
     69 }
     70 
     71 SignalStrategy::State FakeSignalStrategy::GetState() const {
     72   return CONNECTED;
     73 }
     74 
     75 SignalStrategy::Error FakeSignalStrategy::GetError() const {
     76   return OK;
     77 }
     78 
     79 std::string FakeSignalStrategy::GetLocalJid() const {
     80   DCHECK(CalledOnValidThread());
     81   return jid_;
     82 }
     83 
     84 void FakeSignalStrategy::AddListener(Listener* listener) {
     85   DCHECK(CalledOnValidThread());
     86   listeners_.AddObserver(listener);
     87 }
     88 
     89 void FakeSignalStrategy::RemoveListener(Listener* listener) {
     90   DCHECK(CalledOnValidThread());
     91   listeners_.RemoveObserver(listener);
     92 }
     93 
     94 bool FakeSignalStrategy::SendStanza(scoped_ptr<buzz::XmlElement> stanza) {
     95   DCHECK(CalledOnValidThread());
     96 
     97   stanza->SetAttr(buzz::QN_FROM, jid_);
     98 
     99   if (!peer_callback_.is_null()) {
    100     peer_callback_.Run(stanza.Pass());
    101     return true;
    102   } else {
    103     return false;
    104   }
    105 }
    106 
    107 std::string FakeSignalStrategy::GetNextId() {
    108   ++last_id_;
    109   return base::IntToString(last_id_);
    110 }
    111 
    112 // static
    113 void FakeSignalStrategy::DeliverMessageOnThread(
    114     scoped_refptr<base::SingleThreadTaskRunner> thread,
    115     base::WeakPtr<FakeSignalStrategy> target,
    116     scoped_ptr<buzz::XmlElement> stanza) {
    117   thread->PostTask(FROM_HERE,
    118                    base::Bind(&FakeSignalStrategy::OnIncomingMessage,
    119                               target, base::Passed(&stanza)));
    120 }
    121 
    122 void FakeSignalStrategy::OnIncomingMessage(
    123     scoped_ptr<buzz::XmlElement> stanza) {
    124   DCHECK(CalledOnValidThread());
    125 
    126   buzz::XmlElement* stanza_ptr = stanza.get();
    127   received_messages_.push_back(stanza.release());
    128 
    129   const std::string& to_field = stanza_ptr->Attr(buzz::QN_TO);
    130   if (to_field != jid_) {
    131     LOG(WARNING) << "Dropping stanza that is addressed to " << to_field
    132                  << ". Local jid: " << jid_
    133                  << ". Message content: " << stanza_ptr->Str();
    134     return;
    135   }
    136 
    137   ObserverListBase<Listener>::Iterator it(listeners_);
    138   Listener* listener;
    139   while ((listener = it.GetNext()) != NULL) {
    140     if (listener->OnSignalStrategyIncomingStanza(stanza_ptr))
    141       break;
    142   }
    143 }
    144 
    145 void FakeSignalStrategy::SetPeerCallback(const PeerCallback& peer_callback) {
    146   peer_callback_ = peer_callback;
    147 }
    148 
    149 }  // namespace remoting
    150