3 * Copyright 2011 Google Inc.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * 3. The name of the author may not be used to endorse or promote products
14 * derived from this software without specific prior written permission.
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19 * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 #include "talk/base/fakesslidentity.h"
29 #include "talk/base/gunit.h"
30 #include "talk/base/thread.h"
31 #include "talk/p2p/base/constants.h"
32 #include "talk/p2p/base/fakesession.h"
33 #include "talk/p2p/base/parsing.h"
34 #include "talk/p2p/base/p2ptransport.h"
35 #include "talk/p2p/base/rawtransport.h"
36 #include "talk/p2p/base/sessionmessages.h"
37 #include "talk/xmllite/xmlelement.h"
38 #include "talk/xmpp/constants.h"
40 using cricket::Candidate;
41 using cricket::Candidates;
42 using cricket::Transport;
43 using cricket::FakeTransport;
44 using cricket::TransportChannel;
45 using cricket::FakeTransportChannel;
46 using cricket::IceRole;
47 using cricket::TransportDescription;
48 using cricket::WriteError;
49 using cricket::ParseError;
50 using talk_base::SocketAddress;
52 static const char kIceUfrag1[] = "TESTICEUFRAG0001";
53 static const char kIcePwd1[] = "TESTICEPWD00000000000001";
55 class TransportTest : public testing::Test,
56 public sigslot::has_slots<> {
59 : thread_(talk_base::Thread::Current()),
60 transport_(new FakeTransport(
61 thread_, thread_, "test content name", NULL)),
63 connecting_signalled_(false),
66 transport_->SignalConnecting.connect(this, &TransportTest::OnConnecting);
67 transport_->SignalCompleted.connect(this, &TransportTest::OnCompleted);
68 transport_->SignalFailed.connect(this, &TransportTest::OnFailed);
71 transport_->DestroyAllChannels();
74 channel_ = CreateChannel(1);
75 return (channel_ != NULL);
77 FakeTransportChannel* CreateChannel(int component) {
78 return static_cast<FakeTransportChannel*>(
79 transport_->CreateChannel(component));
81 void DestroyChannel() {
82 transport_->DestroyChannel(1);
87 void OnConnecting(Transport* transport) {
88 connecting_signalled_ = true;
90 void OnCompleted(Transport* transport) {
93 void OnFailed(Transport* transport) {
97 talk_base::Thread* thread_;
98 talk_base::scoped_ptr<FakeTransport> transport_;
99 FakeTransportChannel* channel_;
100 bool connecting_signalled_;
105 class FakeCandidateTranslator : public cricket::CandidateTranslator {
107 void AddMapping(int component, const std::string& channel_name) {
108 name_to_component[channel_name] = component;
109 component_to_name[component] = channel_name;
112 bool GetChannelNameFromComponent(
113 int component, std::string* channel_name) const {
114 if (component_to_name.find(component) == component_to_name.end()) {
117 *channel_name = component_to_name.find(component)->second;
120 bool GetComponentFromChannelName(
121 const std::string& channel_name, int* component) const {
122 if (name_to_component.find(channel_name) == name_to_component.end()) {
125 *component = name_to_component.find(channel_name)->second;
129 std::map<std::string, int> name_to_component;
130 std::map<int, std::string> component_to_name;
133 // Test that calling ConnectChannels triggers an OnConnecting signal.
134 TEST_F(TransportTest, TestConnectChannelsDoesSignal) {
135 EXPECT_TRUE(SetupChannel());
136 transport_->ConnectChannels();
137 EXPECT_FALSE(connecting_signalled_);
139 EXPECT_TRUE_WAIT(connecting_signalled_, 100);
142 // Test that DestroyAllChannels kills any pending OnConnecting signals.
143 TEST_F(TransportTest, TestDestroyAllClearsPosts) {
144 EXPECT_TRUE(transport_->CreateChannel(1) != NULL);
146 transport_->ConnectChannels();
147 transport_->DestroyAllChannels();
149 thread_->ProcessMessages(0);
150 EXPECT_FALSE(connecting_signalled_);
153 // This test verifies channels are created with proper ICE
154 // role, tiebreaker and remote ice mode and credentials after offer and
155 // answer negotiations.
156 TEST_F(TransportTest, TestChannelIceParameters) {
157 transport_->SetIceRole(cricket::ICEROLE_CONTROLLING);
158 transport_->SetIceTiebreaker(99U);
159 cricket::TransportDescription local_desc(
160 cricket::NS_JINGLE_ICE_UDP, kIceUfrag1, kIcePwd1);
161 ASSERT_TRUE(transport_->SetLocalTransportDescription(local_desc,
164 EXPECT_EQ(cricket::ICEROLE_CONTROLLING, transport_->ice_role());
165 EXPECT_TRUE(SetupChannel());
166 EXPECT_EQ(cricket::ICEROLE_CONTROLLING, channel_->GetIceRole());
167 EXPECT_EQ(cricket::ICEMODE_FULL, channel_->remote_ice_mode());
168 EXPECT_EQ(kIceUfrag1, channel_->ice_ufrag());
169 EXPECT_EQ(kIcePwd1, channel_->ice_pwd());
171 cricket::TransportDescription remote_desc(
172 cricket::NS_JINGLE_ICE_UDP, kIceUfrag1, kIcePwd1);
173 ASSERT_TRUE(transport_->SetRemoteTransportDescription(remote_desc,
176 EXPECT_EQ(cricket::ICEROLE_CONTROLLING, channel_->GetIceRole());
177 EXPECT_EQ(99U, channel_->IceTiebreaker());
178 EXPECT_EQ(cricket::ICEMODE_FULL, channel_->remote_ice_mode());
179 // Changing the transport role from CONTROLLING to CONTROLLED.
180 transport_->SetIceRole(cricket::ICEROLE_CONTROLLED);
181 EXPECT_EQ(cricket::ICEROLE_CONTROLLED, channel_->GetIceRole());
182 EXPECT_EQ(cricket::ICEMODE_FULL, channel_->remote_ice_mode());
183 EXPECT_EQ(kIceUfrag1, channel_->remote_ice_ufrag());
184 EXPECT_EQ(kIcePwd1, channel_->remote_ice_pwd());
187 // This test verifies that the Completed and Failed states can be reached.
188 TEST_F(TransportTest, TestChannelCompletedAndFailed) {
189 transport_->SetIceRole(cricket::ICEROLE_CONTROLLING);
190 cricket::TransportDescription local_desc(
191 cricket::NS_JINGLE_ICE_UDP, kIceUfrag1, kIcePwd1);
192 ASSERT_TRUE(transport_->SetLocalTransportDescription(local_desc,
195 EXPECT_TRUE(SetupChannel());
197 cricket::TransportDescription remote_desc(
198 cricket::NS_JINGLE_ICE_UDP, kIceUfrag1, kIcePwd1);
199 ASSERT_TRUE(transport_->SetRemoteTransportDescription(remote_desc,
203 channel_->SetConnectionCount(2);
204 channel_->SignalCandidatesAllocationDone(channel_);
205 channel_->SetWritable(true);
206 EXPECT_TRUE_WAIT(transport_->all_channels_writable(), 100);
207 // ICE is not yet completed because there is still more than one connection.
208 EXPECT_FALSE(completed_);
209 EXPECT_FALSE(transport_->completed());
210 EXPECT_FALSE(failed_);
212 // When the connection count drops to 1, SignalCompleted should be emitted,
213 // and completed() should be true.
214 channel_->SetConnectionCount(1);
215 EXPECT_TRUE_WAIT(completed_, 100);
216 EXPECT_TRUE(transport_->completed());
219 // When the connection count drops to 0, SignalFailed should be emitted, and
220 // completed() should be false.
221 channel_->SetConnectionCount(0);
222 EXPECT_TRUE_WAIT(failed_, 100);
223 EXPECT_FALSE(transport_->completed());
224 EXPECT_FALSE(completed_);
227 // Tests channel role is reversed after receiving ice-lite from remote.
228 TEST_F(TransportTest, TestSetRemoteIceLiteInOffer) {
229 transport_->SetIceRole(cricket::ICEROLE_CONTROLLED);
230 cricket::TransportDescription remote_desc(
231 cricket::NS_JINGLE_ICE_UDP, std::vector<std::string>(),
232 kIceUfrag1, kIcePwd1, cricket::ICEMODE_LITE,
233 cricket::CONNECTIONROLE_ACTPASS, NULL, cricket::Candidates());
234 ASSERT_TRUE(transport_->SetRemoteTransportDescription(remote_desc,
237 cricket::TransportDescription local_desc(
238 cricket::NS_JINGLE_ICE_UDP, kIceUfrag1, kIcePwd1);
239 ASSERT_TRUE(transport_->SetLocalTransportDescription(local_desc,
242 EXPECT_EQ(cricket::ICEROLE_CONTROLLING, transport_->ice_role());
243 EXPECT_TRUE(SetupChannel());
244 EXPECT_EQ(cricket::ICEROLE_CONTROLLING, channel_->GetIceRole());
245 EXPECT_EQ(cricket::ICEMODE_LITE, channel_->remote_ice_mode());
248 // Tests ice-lite in remote answer.
249 TEST_F(TransportTest, TestSetRemoteIceLiteInAnswer) {
250 transport_->SetIceRole(cricket::ICEROLE_CONTROLLING);
251 cricket::TransportDescription local_desc(
252 cricket::NS_JINGLE_ICE_UDP, kIceUfrag1, kIcePwd1);
253 ASSERT_TRUE(transport_->SetLocalTransportDescription(local_desc,
256 EXPECT_EQ(cricket::ICEROLE_CONTROLLING, transport_->ice_role());
257 EXPECT_TRUE(SetupChannel());
258 EXPECT_EQ(cricket::ICEROLE_CONTROLLING, channel_->GetIceRole());
259 // Channels will be created in ICEFULL_MODE.
260 EXPECT_EQ(cricket::ICEMODE_FULL, channel_->remote_ice_mode());
261 cricket::TransportDescription remote_desc(
262 cricket::NS_JINGLE_ICE_UDP, std::vector<std::string>(),
263 kIceUfrag1, kIcePwd1, cricket::ICEMODE_LITE,
264 cricket::CONNECTIONROLE_NONE, NULL, cricket::Candidates());
265 ASSERT_TRUE(transport_->SetRemoteTransportDescription(remote_desc,
268 EXPECT_EQ(cricket::ICEROLE_CONTROLLING, channel_->GetIceRole());
269 // After receiving remote description with ICEMODE_LITE, channel should
270 // have mode set to ICEMODE_LITE.
271 EXPECT_EQ(cricket::ICEMODE_LITE, channel_->remote_ice_mode());
274 // Tests that we can properly serialize/deserialize candidates.
275 TEST_F(TransportTest, TestP2PTransportWriteAndParseCandidate) {
276 Candidate test_candidate(
278 talk_base::SocketAddress("2001:db8:fefe::1", 9999),
279 738197504, "abcdef", "ghijkl", "foo", "testnet", 50, "");
280 Candidate test_candidate2(
282 talk_base::SocketAddress("192.168.7.1", 9999),
283 1107296256, "mnopqr", "stuvwx", "bar", "testnet2", 100, "");
284 talk_base::SocketAddress host_address("www.google.com", 24601);
285 host_address.SetResolvedIP(talk_base::IPAddress(0x0A000001));
286 Candidate test_candidate3(
287 "", 3, "spdy", host_address, 1476395008, "yzabcd",
288 "efghij", "baz", "testnet3", 150, "");
289 WriteError write_error;
290 ParseError parse_error;
291 talk_base::scoped_ptr<buzz::XmlElement> elem;
292 cricket::Candidate parsed_candidate;
293 cricket::P2PTransportParser parser;
295 FakeCandidateTranslator translator;
296 translator.AddMapping(1, "test");
297 translator.AddMapping(2, "test2");
298 translator.AddMapping(3, "test3");
300 EXPECT_TRUE(parser.WriteGingleCandidate(test_candidate, &translator,
301 elem.accept(), &write_error));
302 EXPECT_EQ("", write_error.text);
303 EXPECT_EQ("test", elem->Attr(buzz::QN_NAME));
304 EXPECT_EQ("udp", elem->Attr(cricket::QN_PROTOCOL));
305 EXPECT_EQ("2001:db8:fefe::1", elem->Attr(cricket::QN_ADDRESS));
306 EXPECT_EQ("9999", elem->Attr(cricket::QN_PORT));
307 EXPECT_EQ("0.34", elem->Attr(cricket::QN_PREFERENCE));
308 EXPECT_EQ("abcdef", elem->Attr(cricket::QN_USERNAME));
309 EXPECT_EQ("ghijkl", elem->Attr(cricket::QN_PASSWORD));
310 EXPECT_EQ("foo", elem->Attr(cricket::QN_TYPE));
311 EXPECT_EQ("testnet", elem->Attr(cricket::QN_NETWORK));
312 EXPECT_EQ("50", elem->Attr(cricket::QN_GENERATION));
314 EXPECT_TRUE(parser.ParseGingleCandidate(elem.get(), &translator,
315 &parsed_candidate, &parse_error));
316 EXPECT_TRUE(test_candidate.IsEquivalent(parsed_candidate));
318 EXPECT_TRUE(parser.WriteGingleCandidate(test_candidate2, &translator,
319 elem.accept(), &write_error));
320 EXPECT_EQ("test2", elem->Attr(buzz::QN_NAME));
321 EXPECT_EQ("tcp", elem->Attr(cricket::QN_PROTOCOL));
322 EXPECT_EQ("192.168.7.1", elem->Attr(cricket::QN_ADDRESS));
323 EXPECT_EQ("9999", elem->Attr(cricket::QN_PORT));
324 EXPECT_EQ("0.51", elem->Attr(cricket::QN_PREFERENCE));
325 EXPECT_EQ("mnopqr", elem->Attr(cricket::QN_USERNAME));
326 EXPECT_EQ("stuvwx", elem->Attr(cricket::QN_PASSWORD));
327 EXPECT_EQ("bar", elem->Attr(cricket::QN_TYPE));
328 EXPECT_EQ("testnet2", elem->Attr(cricket::QN_NETWORK));
329 EXPECT_EQ("100", elem->Attr(cricket::QN_GENERATION));
331 EXPECT_TRUE(parser.ParseGingleCandidate(elem.get(), &translator,
332 &parsed_candidate, &parse_error));
333 EXPECT_TRUE(test_candidate2.IsEquivalent(parsed_candidate));
335 // Check that an ip is preferred over hostname.
336 EXPECT_TRUE(parser.WriteGingleCandidate(test_candidate3, &translator,
337 elem.accept(), &write_error));
338 EXPECT_EQ("test3", elem->Attr(cricket::QN_NAME));
339 EXPECT_EQ("spdy", elem->Attr(cricket::QN_PROTOCOL));
340 EXPECT_EQ("10.0.0.1", elem->Attr(cricket::QN_ADDRESS));
341 EXPECT_EQ("24601", elem->Attr(cricket::QN_PORT));
342 EXPECT_EQ("0.69", elem->Attr(cricket::QN_PREFERENCE));
343 EXPECT_EQ("yzabcd", elem->Attr(cricket::QN_USERNAME));
344 EXPECT_EQ("efghij", elem->Attr(cricket::QN_PASSWORD));
345 EXPECT_EQ("baz", elem->Attr(cricket::QN_TYPE));
346 EXPECT_EQ("testnet3", elem->Attr(cricket::QN_NETWORK));
347 EXPECT_EQ("150", elem->Attr(cricket::QN_GENERATION));
349 EXPECT_TRUE(parser.ParseGingleCandidate(elem.get(), &translator,
350 &parsed_candidate, &parse_error));
351 EXPECT_TRUE(test_candidate3.IsEquivalent(parsed_candidate));
354 TEST_F(TransportTest, TestGetStats) {
355 EXPECT_TRUE(SetupChannel());
356 cricket::TransportStats stats;
357 EXPECT_TRUE(transport_->GetStats(&stats));
358 // Note that this tests the behavior of a FakeTransportChannel.
359 ASSERT_EQ(1U, stats.channel_stats.size());
360 EXPECT_EQ(1, stats.channel_stats[0].component);
361 transport_->ConnectChannels();
362 EXPECT_TRUE(transport_->GetStats(&stats));
363 ASSERT_EQ(1U, stats.channel_stats.size());
364 EXPECT_EQ(1, stats.channel_stats[0].component);