blob: 0aeb080bcd83f235e160e9b5c39bd5c5208552cd [file] [log] [blame]
Harald Alvestrand19793842018-06-25 12:03:50 +02001/*
2 * Copyright 2017 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Yves Gerey3e707812018-11-28 16:47:49 +010011#include <memory>
12#include <set>
13#include <string>
14#include <utility>
15#include <vector>
Harald Alvestrand19793842018-06-25 12:03:50 +020016
Yves Gerey3e707812018-11-28 16:47:49 +010017#include "absl/types/optional.h"
Steve Anton10542f22019-01-11 09:11:00 -080018#include "api/call/call_factory_interface.h"
Harald Alvestrand42386282018-07-12 07:56:05 +020019#include "api/jsep.h"
Qingsi Wang1ba5dec2019-08-19 11:57:17 -070020#include "api/jsep_session_description.h"
Steve Anton10542f22019-01-11 09:11:00 -080021#include "api/peer_connection_interface.h"
22#include "api/peer_connection_proxy.h"
23#include "api/rtc_error.h"
Mirko Bonadeid9708072019-01-25 20:26:48 +010024#include "api/scoped_refptr.h"
Danil Chapovalov53d45ba2019-07-03 14:56:33 +020025#include "api/task_queue/default_task_queue_factory.h"
Steve Anton10542f22019-01-11 09:11:00 -080026#include "media/base/fake_media_engine.h"
Jeroen de Borstaf242c82019-04-24 13:13:48 -070027#include "p2p/base/mock_async_resolver.h"
Steve Anton10542f22019-01-11 09:11:00 -080028#include "p2p/base/port_allocator.h"
29#include "p2p/client/basic_port_allocator.h"
30#include "pc/peer_connection.h"
31#include "pc/peer_connection_factory.h"
32#include "pc/peer_connection_wrapper.h"
33#include "pc/sdp_utils.h"
34#include "pc/test/mock_peer_connection_observers.h"
Qingsi Wang1ba5dec2019-08-19 11:57:17 -070035#include "pc/webrtc_sdp.h"
Jeroen de Borstaf242c82019-04-24 13:13:48 -070036#include "rtc_base/arraysize.h"
Yves Gerey3e707812018-11-28 16:47:49 +010037#include "rtc_base/checks.h"
Qingsi Wangecd30542019-05-22 14:34:56 -070038#include "rtc_base/fake_mdns_responder.h"
Steve Anton10542f22019-01-11 09:11:00 -080039#include "rtc_base/fake_network.h"
Harald Alvestrand19793842018-06-25 12:03:50 +020040#include "rtc_base/gunit.h"
Steve Anton10542f22019-01-11 09:11:00 -080041#include "rtc_base/ref_counted_object.h"
42#include "rtc_base/rtc_certificate_generator.h"
Steve Anton10542f22019-01-11 09:11:00 -080043#include "rtc_base/socket_address.h"
Yves Gerey3e707812018-11-28 16:47:49 +010044#include "rtc_base/thread.h"
Steve Anton10542f22019-01-11 09:11:00 -080045#include "rtc_base/virtual_socket_server.h"
Mirko Bonadei17f48782018-09-28 08:51:10 +020046#include "system_wrappers/include/metrics.h"
Jeroen de Borstaf242c82019-04-24 13:13:48 -070047#include "test/gmock.h"
Harald Alvestrand19793842018-06-25 12:03:50 +020048
49namespace webrtc {
50
51using RTCConfiguration = PeerConnectionInterface::RTCConfiguration;
52using RTCOfferAnswerOptions = PeerConnectionInterface::RTCOfferAnswerOptions;
Jeroen de Borstaf242c82019-04-24 13:13:48 -070053using ::testing::NiceMock;
Harald Alvestrand19793842018-06-25 12:03:50 +020054using ::testing::Values;
55
Harald Alvestrandc0e97252018-07-26 10:39:55 +020056static const char kUsagePatternMetric[] = "WebRTC.PeerConnection.UsagePattern";
Harald Alvestrand19793842018-06-25 12:03:50 +020057static constexpr int kDefaultTimeout = 10000;
Jeroen de Borstaf242c82019-04-24 13:13:48 -070058static const rtc::SocketAddress kLocalAddrs[2] = {
59 rtc::SocketAddress("1.1.1.1", 0), rtc::SocketAddress("2.2.2.2", 0)};
Harald Alvestrand056d8112018-07-16 19:18:58 +020060static const rtc::SocketAddress kPrivateLocalAddress("10.1.1.1", 0);
Qingsi Wang1ba5dec2019-08-19 11:57:17 -070061static const rtc::SocketAddress kPrivateIpv6LocalAddress("fd12:3456:789a:1::1",
62 0);
Harald Alvestrand19793842018-06-25 12:03:50 +020063
64int MakeUsageFingerprint(std::set<PeerConnection::UsageEvent> events) {
65 int signature = 0;
66 for (const auto it : events) {
67 signature |= static_cast<int>(it);
68 }
69 return signature;
70}
71
72class PeerConnectionFactoryForUsageHistogramTest
73 : public rtc::RefCountedObject<PeerConnectionFactory> {
74 public:
75 PeerConnectionFactoryForUsageHistogramTest()
Danil Chapovalovf5258be2019-03-19 17:45:24 +010076 : rtc::RefCountedObject<PeerConnectionFactory>([] {
77 PeerConnectionFactoryDependencies dependencies;
78 dependencies.network_thread = rtc::Thread::Current();
79 dependencies.worker_thread = rtc::Thread::Current();
80 dependencies.signaling_thread = rtc::Thread::Current();
Danil Chapovalov53d45ba2019-07-03 14:56:33 +020081 dependencies.task_queue_factory = CreateDefaultTaskQueueFactory();
Danil Chapovalovf5258be2019-03-19 17:45:24 +010082 dependencies.media_engine =
Mirko Bonadei317a1f02019-09-17 17:06:18 +020083 std::make_unique<cricket::FakeMediaEngine>();
Danil Chapovalovf5258be2019-03-19 17:45:24 +010084 dependencies.call_factory = CreateCallFactory();
85 return dependencies;
86 }()) {}
Harald Alvestrand19793842018-06-25 12:03:50 +020087
88 void ActionsBeforeInitializeForTesting(PeerConnectionInterface* pc) override {
89 PeerConnection* internal_pc = static_cast<PeerConnection*>(pc);
90 if (return_histogram_very_quickly_) {
91 internal_pc->ReturnHistogramVeryQuicklyForTesting();
92 }
93 }
94
95 void ReturnHistogramVeryQuickly() { return_histogram_very_quickly_ = true; }
96
97 private:
Harald Alvestrand183e09d2018-06-28 12:04:41 +020098 bool return_histogram_very_quickly_ = false;
99};
100
101class PeerConnectionWrapperForUsageHistogramTest;
Yves Gerey3e707812018-11-28 16:47:49 +0100102
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200103typedef PeerConnectionWrapperForUsageHistogramTest* RawWrapperPtr;
104
105class ObserverForUsageHistogramTest : public MockPeerConnectionObserver {
106 public:
107 void OnIceCandidate(const webrtc::IceCandidateInterface* candidate) override;
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200108
109 void OnInterestingUsage(int usage_pattern) override {
110 interesting_usage_detected_ = usage_pattern;
111 }
112
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200113 void PrepareToExchangeCandidates(RawWrapperPtr other) {
114 candidate_target_ = other;
115 }
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200116
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200117 bool HaveDataChannel() { return last_datachannel_; }
118
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200119 absl::optional<int> interesting_usage_detected() {
120 return interesting_usage_detected_;
121 }
122
Harald Alvestrand7a1c7f72018-08-01 10:50:16 +0200123 void ClearInterestingUsageDetector() {
124 interesting_usage_detected_ = absl::optional<int>();
125 }
126
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700127 bool candidate_gathered() const { return candidate_gathered_; }
128
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200129 private:
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200130 absl::optional<int> interesting_usage_detected_;
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700131 bool candidate_gathered_ = false;
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200132 RawWrapperPtr candidate_target_; // Note: Not thread-safe against deletions.
Harald Alvestrand19793842018-06-25 12:03:50 +0200133};
134
135class PeerConnectionWrapperForUsageHistogramTest
136 : public PeerConnectionWrapper {
137 public:
138 using PeerConnectionWrapper::PeerConnectionWrapper;
139
140 PeerConnection* GetInternalPeerConnection() {
141 auto* pci =
142 static_cast<PeerConnectionProxyWithInternal<PeerConnectionInterface>*>(
143 pc());
144 return static_cast<PeerConnection*>(pci->internal());
145 }
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200146
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200147 // Override with different return type
148 ObserverForUsageHistogramTest* observer() {
149 return static_cast<ObserverForUsageHistogramTest*>(
150 PeerConnectionWrapper::observer());
151 }
152
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200153 void PrepareToExchangeCandidates(
154 PeerConnectionWrapperForUsageHistogramTest* other) {
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200155 observer()->PrepareToExchangeCandidates(other);
156 other->observer()->PrepareToExchangeCandidates(this);
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200157 }
158
159 bool IsConnected() {
160 return pc()->ice_connection_state() ==
161 PeerConnectionInterface::kIceConnectionConnected ||
162 pc()->ice_connection_state() ==
163 PeerConnectionInterface::kIceConnectionCompleted;
164 }
165
166 bool HaveDataChannel() {
167 return static_cast<ObserverForUsageHistogramTest*>(observer())
168 ->HaveDataChannel();
169 }
Harald Alvestrand42386282018-07-12 07:56:05 +0200170 void AddOrBufferIceCandidate(const webrtc::IceCandidateInterface* candidate) {
171 if (!pc()->AddIceCandidate(candidate)) {
172 std::string sdp;
173 EXPECT_TRUE(candidate->ToString(&sdp));
174 std::unique_ptr<webrtc::IceCandidateInterface> candidate_copy(
175 CreateIceCandidate(candidate->sdp_mid(), candidate->sdp_mline_index(),
176 sdp, nullptr));
177 buffered_candidates_.push_back(std::move(candidate_copy));
178 }
179 }
Harald Alvestrand056d8112018-07-16 19:18:58 +0200180
Harald Alvestrand42386282018-07-12 07:56:05 +0200181 void AddBufferedIceCandidates() {
182 for (const auto& candidate : buffered_candidates_) {
183 EXPECT_TRUE(pc()->AddIceCandidate(candidate.get()));
184 }
185 buffered_candidates_.clear();
186 }
Harald Alvestrand056d8112018-07-16 19:18:58 +0200187
Harald Alvestrand42386282018-07-12 07:56:05 +0200188 bool ConnectTo(PeerConnectionWrapperForUsageHistogramTest* callee) {
189 PrepareToExchangeCandidates(callee);
Harald Alvestrand056d8112018-07-16 19:18:58 +0200190 if (!ExchangeOfferAnswerWith(callee)) {
191 return false;
192 }
Harald Alvestrand42386282018-07-12 07:56:05 +0200193 AddBufferedIceCandidates();
194 callee->AddBufferedIceCandidates();
Harald Alvestrand056d8112018-07-16 19:18:58 +0200195 WAIT(IsConnected(), kDefaultTimeout);
196 WAIT(callee->IsConnected(), kDefaultTimeout);
Harald Alvestrand42386282018-07-12 07:56:05 +0200197 return IsConnected() && callee->IsConnected();
198 }
199
Harald Alvestrand056d8112018-07-16 19:18:58 +0200200 bool GenerateOfferAndCollectCandidates() {
201 auto offer = CreateOffer(RTCOfferAnswerOptions());
202 if (!offer) {
203 return false;
204 }
205 bool set_local_offer =
206 SetLocalDescription(CloneSessionDescription(offer.get()));
207 EXPECT_TRUE(set_local_offer);
208 if (!set_local_offer) {
209 return false;
210 }
211 EXPECT_TRUE_WAIT(observer()->ice_gathering_complete_, kDefaultTimeout);
212 return true;
213 }
214
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700215 webrtc::PeerConnectionInterface::IceGatheringState ice_gathering_state() {
216 return pc()->ice_gathering_state();
217 }
218
Harald Alvestrand42386282018-07-12 07:56:05 +0200219 private:
220 // Candidates that have been sent but not yet configured
221 std::vector<std::unique_ptr<webrtc::IceCandidateInterface>>
222 buffered_candidates_;
Harald Alvestrand19793842018-06-25 12:03:50 +0200223};
224
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200225void ObserverForUsageHistogramTest::OnIceCandidate(
226 const webrtc::IceCandidateInterface* candidate) {
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700227 // If target is not set, ignore. This happens in one-ended unit tests.
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200228 if (candidate_target_) {
Harald Alvestrand42386282018-07-12 07:56:05 +0200229 this->candidate_target_->AddOrBufferIceCandidate(candidate);
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200230 }
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700231 candidate_gathered_ = true;
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200232}
233
Harald Alvestrand19793842018-06-25 12:03:50 +0200234class PeerConnectionUsageHistogramTest : public ::testing::Test {
235 protected:
236 typedef std::unique_ptr<PeerConnectionWrapperForUsageHistogramTest>
237 WrapperPtr;
238
239 PeerConnectionUsageHistogramTest()
240 : vss_(new rtc::VirtualSocketServer()), main_(vss_.get()) {
Qingsi Wang7fc821d2018-07-12 12:54:53 -0700241 webrtc::metrics::Reset();
Harald Alvestrand19793842018-06-25 12:03:50 +0200242 }
243
244 WrapperPtr CreatePeerConnection() {
Harald Alvestrand056d8112018-07-16 19:18:58 +0200245 return CreatePeerConnection(RTCConfiguration(),
246 PeerConnectionFactoryInterface::Options(),
247 nullptr, false);
Harald Alvestrand19793842018-06-25 12:03:50 +0200248 }
249
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200250 WrapperPtr CreatePeerConnection(const RTCConfiguration& config) {
251 return CreatePeerConnection(
Harald Alvestrand056d8112018-07-16 19:18:58 +0200252 config, PeerConnectionFactoryInterface::Options(), nullptr, false);
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200253 }
254
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700255 WrapperPtr CreatePeerConnectionWithMdns(const RTCConfiguration& config) {
256 auto resolver_factory =
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200257 std::make_unique<NiceMock<webrtc::MockAsyncResolverFactory>>();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700258
259 webrtc::PeerConnectionDependencies deps(nullptr /* observer_in */);
260
261 auto fake_network = NewFakeNetwork();
Qingsi Wangecd30542019-05-22 14:34:56 -0700262 fake_network->set_mdns_responder(
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200263 std::make_unique<webrtc::FakeMdnsResponder>(rtc::Thread::Current()));
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700264 fake_network->AddInterface(NextLocalAddress());
265
266 std::unique_ptr<cricket::BasicPortAllocator> port_allocator(
267 new cricket::BasicPortAllocator(fake_network));
268
269 deps.async_resolver_factory = std::move(resolver_factory);
270 deps.allocator = std::move(port_allocator);
271
272 return CreatePeerConnection(config,
273 PeerConnectionFactoryInterface::Options(),
274 std::move(deps), false);
275 }
276
Harald Alvestrand19793842018-06-25 12:03:50 +0200277 WrapperPtr CreatePeerConnectionWithImmediateReport() {
Harald Alvestrand056d8112018-07-16 19:18:58 +0200278 return CreatePeerConnection(RTCConfiguration(),
279 PeerConnectionFactoryInterface::Options(),
280 nullptr, true);
281 }
282
283 WrapperPtr CreatePeerConnectionWithPrivateLocalAddresses() {
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700284 auto* fake_network = NewFakeNetwork();
285 fake_network->AddInterface(NextLocalAddress());
286 fake_network->AddInterface(kPrivateLocalAddress);
287
288 auto port_allocator =
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200289 std::make_unique<cricket::BasicPortAllocator>(fake_network);
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700290
Harald Alvestrand056d8112018-07-16 19:18:58 +0200291 return CreatePeerConnection(RTCConfiguration(),
292 PeerConnectionFactoryInterface::Options(),
293 std::move(port_allocator), false);
Harald Alvestrand19793842018-06-25 12:03:50 +0200294 }
295
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700296 WrapperPtr CreatePeerConnectionWithPrivateIpv6LocalAddresses() {
297 auto* fake_network = NewFakeNetwork();
298 fake_network->AddInterface(NextLocalAddress());
299 fake_network->AddInterface(kPrivateIpv6LocalAddress);
300
301 auto port_allocator =
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200302 std::make_unique<cricket::BasicPortAllocator>(fake_network);
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700303
304 return CreatePeerConnection(RTCConfiguration(),
305 PeerConnectionFactoryInterface::Options(),
306 std::move(port_allocator), false);
307 }
308
Harald Alvestrand19793842018-06-25 12:03:50 +0200309 WrapperPtr CreatePeerConnection(
310 const RTCConfiguration& config,
311 const PeerConnectionFactoryInterface::Options factory_options,
Harald Alvestrand056d8112018-07-16 19:18:58 +0200312 std::unique_ptr<cricket::PortAllocator> allocator,
Harald Alvestrand19793842018-06-25 12:03:50 +0200313 bool immediate_report) {
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700314 PeerConnectionDependencies deps(nullptr);
315 deps.allocator = std::move(allocator);
316
317 return CreatePeerConnection(config, factory_options, std::move(deps),
318 immediate_report);
319 }
320
321 WrapperPtr CreatePeerConnection(
322 const RTCConfiguration& config,
323 const PeerConnectionFactoryInterface::Options factory_options,
324 PeerConnectionDependencies deps,
325 bool immediate_report) {
Harald Alvestrand19793842018-06-25 12:03:50 +0200326 rtc::scoped_refptr<PeerConnectionFactoryForUsageHistogramTest> pc_factory(
327 new PeerConnectionFactoryForUsageHistogramTest());
328 pc_factory->SetOptions(factory_options);
329 RTC_CHECK(pc_factory->Initialize());
330 if (immediate_report) {
331 pc_factory->ReturnHistogramVeryQuickly();
332 }
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700333
334 // If no allocator is provided, one will be created using a network manager
335 // that uses the host network. This doesn't work on all trybots.
336 if (!deps.allocator) {
337 auto fake_network = NewFakeNetwork();
338 fake_network->AddInterface(NextLocalAddress());
339 deps.allocator =
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200340 std::make_unique<cricket::BasicPortAllocator>(fake_network);
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700341 }
342
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200343 auto observer = std::make_unique<ObserverForUsageHistogramTest>();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700344 deps.observer = observer.get();
345
346 auto pc = pc_factory->CreatePeerConnection(config, std::move(deps));
Harald Alvestrand19793842018-06-25 12:03:50 +0200347 if (!pc) {
348 return nullptr;
349 }
350
Yves Gerey4e933292018-10-31 15:36:05 +0100351 observer->SetPeerConnectionInterface(pc.get());
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200352 auto wrapper = std::make_unique<PeerConnectionWrapperForUsageHistogramTest>(
353 pc_factory, pc, std::move(observer));
Harald Alvestrand19793842018-06-25 12:03:50 +0200354 return wrapper;
355 }
356
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200357 int ObservedFingerprint() {
358 // This works correctly only if there is only one sample value
359 // that has been counted.
360 // Returns -1 for "not found".
361 return webrtc::metrics::MinSample(kUsagePatternMetric);
362 }
363
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700364 // The PeerConnection's port allocator is tied to the PeerConnection's
365 // lifetime and expects the underlying NetworkManager to outlive it. That
366 // prevents us from having the PeerConnectionWrapper own the fake network.
367 // Therefore, the test fixture will own all the fake networks even though
368 // tests should access the fake network through the PeerConnectionWrapper.
369 rtc::FakeNetworkManager* NewFakeNetwork() {
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200370 fake_networks_.emplace_back(std::make_unique<rtc::FakeNetworkManager>());
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700371 return fake_networks_.back().get();
372 }
373
374 rtc::SocketAddress NextLocalAddress() {
375 RTC_DCHECK(next_local_address_ < (int)arraysize(kLocalAddrs));
376 return kLocalAddrs[next_local_address_++];
377 }
378
379 std::vector<std::unique_ptr<rtc::FakeNetworkManager>> fake_networks_;
380 int next_local_address_ = 0;
Harald Alvestrand19793842018-06-25 12:03:50 +0200381 std::unique_ptr<rtc::VirtualSocketServer> vss_;
382 rtc::AutoSocketServerThread main_;
383};
384
385TEST_F(PeerConnectionUsageHistogramTest, UsageFingerprintHistogramFromTimeout) {
386 auto pc = CreatePeerConnectionWithImmediateReport();
387
Harald Alvestrand19793842018-06-25 12:03:50 +0200388 int expected_fingerprint = MakeUsageFingerprint({});
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200389 ASSERT_EQ_WAIT(1, webrtc::metrics::NumSamples(kUsagePatternMetric),
390 kDefaultTimeout);
391 EXPECT_EQ(
392 1, webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint));
Harald Alvestrand19793842018-06-25 12:03:50 +0200393}
394
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200395#ifndef WEBRTC_ANDROID
396// These tests do not work on Android. Why is unclear.
397// https://bugs.webrtc.org/9461
398
399// Test getting the usage fingerprint for an audio/video connection.
400TEST_F(PeerConnectionUsageHistogramTest, FingerprintAudioVideo) {
401 auto caller = CreatePeerConnection();
402 auto callee = CreatePeerConnection();
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200403 caller->AddAudioTrack("audio");
404 caller->AddVideoTrack("video");
Harald Alvestrand42386282018-07-12 07:56:05 +0200405 ASSERT_TRUE(caller->ConnectTo(callee.get()));
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200406 caller->pc()->Close();
407 callee->pc()->Close();
408 int expected_fingerprint = MakeUsageFingerprint(
409 {PeerConnection::UsageEvent::AUDIO_ADDED,
410 PeerConnection::UsageEvent::VIDEO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700411 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
412 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200413 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700414 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200415 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700416 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700417 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200418 PeerConnection::UsageEvent::CLOSE_CALLED});
Harald Alvestrand056d8112018-07-16 19:18:58 +0200419 // In this case, we may or may not have PRIVATE_CANDIDATE_COLLECTED,
420 // depending on the machine configuration.
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200421 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
Harald Alvestrand056d8112018-07-16 19:18:58 +0200422 EXPECT_TRUE(
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200423 webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint) ==
424 2 ||
Harald Alvestrand056d8112018-07-16 19:18:58 +0200425 webrtc::metrics::NumEvents(
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200426 kUsagePatternMetric,
Harald Alvestrand056d8112018-07-16 19:18:58 +0200427 expected_fingerprint |
428 static_cast<int>(
429 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
430 2);
431}
432
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700433// Test getting the usage fingerprint when the caller collects an mDNS
434// candidate.
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700435TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithMdnsCaller) {
436 RTCConfiguration config;
437
438 // Enable hostname candidates with mDNS names.
439 auto caller = CreatePeerConnectionWithMdns(config);
440 auto callee = CreatePeerConnection(config);
441
442 caller->AddAudioTrack("audio");
443 caller->AddVideoTrack("video");
444 ASSERT_TRUE(caller->ConnectTo(callee.get()));
445 caller->pc()->Close();
446 callee->pc()->Close();
447
448 int expected_fingerprint_caller = MakeUsageFingerprint(
449 {PeerConnection::UsageEvent::AUDIO_ADDED,
450 PeerConnection::UsageEvent::VIDEO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700451 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
452 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700453 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
454 PeerConnection::UsageEvent::MDNS_CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700455 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700456 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700457 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700458 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700459 PeerConnection::UsageEvent::CLOSE_CALLED});
460
Qingsi Wangcc46b102019-09-12 11:19:01 -0700461 // Without a resolver, the callee cannot resolve the received mDNS candidate
462 // but can still connect with the caller via a prflx candidate. As a result,
463 // the bit for the direct connection should not be logged.
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700464 int expected_fingerprint_callee = MakeUsageFingerprint(
465 {PeerConnection::UsageEvent::AUDIO_ADDED,
466 PeerConnection::UsageEvent::VIDEO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700467 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
468 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700469 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700470 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700471 PeerConnection::UsageEvent::REMOTE_MDNS_CANDIDATE_ADDED,
472 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700473 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700474 PeerConnection::UsageEvent::CLOSE_CALLED});
475
476 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
477 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
478 expected_fingerprint_caller));
479 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
480 expected_fingerprint_callee));
481}
482
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700483// Test getting the usage fingerprint when the callee collects an mDNS
484// candidate.
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700485TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithMdnsCallee) {
486 RTCConfiguration config;
487
488 // Enable hostname candidates with mDNS names.
489 auto caller = CreatePeerConnection(config);
490 auto callee = CreatePeerConnectionWithMdns(config);
491
492 caller->AddAudioTrack("audio");
493 caller->AddVideoTrack("video");
494 ASSERT_TRUE(caller->ConnectTo(callee.get()));
495 caller->pc()->Close();
496 callee->pc()->Close();
497
Qingsi Wangcc46b102019-09-12 11:19:01 -0700498 // Similar to the test above, the caller connects with the callee via a prflx
499 // candidate.
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700500 int expected_fingerprint_caller = MakeUsageFingerprint(
501 {PeerConnection::UsageEvent::AUDIO_ADDED,
502 PeerConnection::UsageEvent::VIDEO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700503 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
504 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700505 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700506 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700507 PeerConnection::UsageEvent::REMOTE_MDNS_CANDIDATE_ADDED,
508 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700509 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700510 PeerConnection::UsageEvent::CLOSE_CALLED});
511
512 int expected_fingerprint_callee = MakeUsageFingerprint(
513 {PeerConnection::UsageEvent::AUDIO_ADDED,
514 PeerConnection::UsageEvent::VIDEO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700515 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
516 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700517 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
518 PeerConnection::UsageEvent::MDNS_CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700519 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700520 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700521 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700522 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700523 PeerConnection::UsageEvent::CLOSE_CALLED});
524
525 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
526 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
527 expected_fingerprint_caller));
528 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
529 expected_fingerprint_callee));
530}
531
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200532#ifdef HAVE_SCTP
533TEST_F(PeerConnectionUsageHistogramTest, FingerprintDataOnly) {
534 auto caller = CreatePeerConnection();
535 auto callee = CreatePeerConnection();
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200536 caller->CreateDataChannel("foodata");
Harald Alvestrand42386282018-07-12 07:56:05 +0200537 ASSERT_TRUE(caller->ConnectTo(callee.get()));
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200538 ASSERT_TRUE_WAIT(callee->HaveDataChannel(), kDefaultTimeout);
539 caller->pc()->Close();
540 callee->pc()->Close();
541 int expected_fingerprint = MakeUsageFingerprint(
542 {PeerConnection::UsageEvent::DATA_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700543 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
544 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200545 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700546 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200547 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700548 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700549 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200550 PeerConnection::UsageEvent::CLOSE_CALLED});
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200551 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
Harald Alvestrand056d8112018-07-16 19:18:58 +0200552 EXPECT_TRUE(
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200553 webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint) ==
554 2 ||
Harald Alvestrand056d8112018-07-16 19:18:58 +0200555 webrtc::metrics::NumEvents(
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200556 kUsagePatternMetric,
Harald Alvestrand056d8112018-07-16 19:18:58 +0200557 expected_fingerprint |
558 static_cast<int>(
559 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
560 2);
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200561}
562#endif // HAVE_SCTP
563#endif // WEBRTC_ANDROID
564
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200565TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurn) {
566 RTCConfiguration configuration;
567 PeerConnection::IceServer server;
568 server.urls = {"stun:dummy.stun.server/"};
569 configuration.servers.push_back(server);
570 server.urls = {"turn:dummy.turn.server/"};
571 server.username = "username";
572 server.password = "password";
573 configuration.servers.push_back(server);
574 auto caller = CreatePeerConnection(configuration);
575 ASSERT_TRUE(caller);
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200576 caller->pc()->Close();
577 int expected_fingerprint =
578 MakeUsageFingerprint({PeerConnection::UsageEvent::STUN_SERVER_ADDED,
579 PeerConnection::UsageEvent::TURN_SERVER_ADDED,
580 PeerConnection::UsageEvent::CLOSE_CALLED});
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200581 EXPECT_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
582 EXPECT_EQ(
583 1, webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint));
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200584}
585
586TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurnInReconfiguration) {
587 RTCConfiguration configuration;
588 PeerConnection::IceServer server;
589 server.urls = {"stun:dummy.stun.server/"};
590 configuration.servers.push_back(server);
591 server.urls = {"turn:dummy.turn.server/"};
592 server.username = "username";
593 server.password = "password";
594 configuration.servers.push_back(server);
595 auto caller = CreatePeerConnection();
596 ASSERT_TRUE(caller);
Niels Möller2579f0c2019-08-19 09:58:17 +0200597 ASSERT_TRUE(caller->pc()->SetConfiguration(configuration).ok());
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200598 caller->pc()->Close();
599 int expected_fingerprint =
600 MakeUsageFingerprint({PeerConnection::UsageEvent::STUN_SERVER_ADDED,
601 PeerConnection::UsageEvent::TURN_SERVER_ADDED,
602 PeerConnection::UsageEvent::CLOSE_CALLED});
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200603 EXPECT_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
604 EXPECT_EQ(
605 1, webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint));
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200606}
607
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700608TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithPrivateIPCaller) {
Harald Alvestrand056d8112018-07-16 19:18:58 +0200609 auto caller = CreatePeerConnectionWithPrivateLocalAddresses();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700610 auto callee = CreatePeerConnection();
Harald Alvestrand056d8112018-07-16 19:18:58 +0200611 caller->AddAudioTrack("audio");
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700612 ASSERT_TRUE(caller->ConnectTo(callee.get()));
Harald Alvestrand056d8112018-07-16 19:18:58 +0200613 caller->pc()->Close();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700614 callee->pc()->Close();
615
616 int expected_fingerprint_caller = MakeUsageFingerprint(
Harald Alvestrand056d8112018-07-16 19:18:58 +0200617 {PeerConnection::UsageEvent::AUDIO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700618 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
619 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Harald Alvestrand056d8112018-07-16 19:18:58 +0200620 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700621 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700622 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700623 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700624 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700625 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700626 PeerConnection::UsageEvent::CLOSE_CALLED});
627
628 int expected_fingerprint_callee = MakeUsageFingerprint(
629 {PeerConnection::UsageEvent::AUDIO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700630 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
631 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700632 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700633 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700634 PeerConnection::UsageEvent::REMOTE_PRIVATE_CANDIDATE_ADDED,
635 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700636 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700637 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700638 PeerConnection::UsageEvent::CLOSE_CALLED});
639
640 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
641 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
642 expected_fingerprint_caller));
643 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
644 expected_fingerprint_callee));
645}
646
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700647TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithPrivateIpv6Callee) {
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700648 auto caller = CreatePeerConnection();
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700649 auto callee = CreatePeerConnectionWithPrivateIpv6LocalAddresses();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700650 caller->AddAudioTrack("audio");
651 ASSERT_TRUE(caller->ConnectTo(callee.get()));
652 caller->pc()->Close();
653 callee->pc()->Close();
654
655 int expected_fingerprint_caller = MakeUsageFingerprint(
656 {PeerConnection::UsageEvent::AUDIO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700657 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
658 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700659 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700660 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700661 PeerConnection::UsageEvent::REMOTE_PRIVATE_CANDIDATE_ADDED,
662 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700663 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
664 PeerConnection::UsageEvent::REMOTE_IPV6_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700665 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700666 PeerConnection::UsageEvent::CLOSE_CALLED});
667
668 int expected_fingerprint_callee = MakeUsageFingerprint(
669 {PeerConnection::UsageEvent::AUDIO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700670 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
671 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700672 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
673 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700674 PeerConnection::UsageEvent::IPV6_CANDIDATE_COLLECTED,
675 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700676 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
677 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700678 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700679 PeerConnection::UsageEvent::CLOSE_CALLED});
680
681 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
682 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
683 expected_fingerprint_caller));
684 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
685 expected_fingerprint_callee));
Harald Alvestrand056d8112018-07-16 19:18:58 +0200686}
687
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200688#ifndef WEBRTC_ANDROID
689#ifdef HAVE_SCTP
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700690// Test that the usage pattern bits for adding remote (private IPv6) candidates
691// are set when the remote candidates are retrieved from the Offer SDP instead
692// of trickled ICE messages.
693TEST_F(PeerConnectionUsageHistogramTest,
694 AddRemoteCandidatesFromRemoteDescription) {
695 // We construct the following data-channel-only scenario. The caller collects
696 // IPv6 private local candidates and appends them in the Offer as in
697 // non-trickled sessions. The callee collects mDNS candidates that are not
698 // contained in the Answer as in Trickle ICE. Only the Offer and Answer are
699 // signaled and we expect a connection with prflx remote candidates at the
700 // caller side.
701 auto caller = CreatePeerConnectionWithPrivateIpv6LocalAddresses();
702 auto callee = CreatePeerConnectionWithMdns(RTCConfiguration());
703 caller->CreateDataChannel("test_channel");
704 ASSERT_TRUE(caller->SetLocalDescription(caller->CreateOffer()));
705 // Wait until the gathering completes so that the session description would
706 // have contained ICE candidates.
707 EXPECT_EQ_WAIT(webrtc::PeerConnectionInterface::kIceGatheringComplete,
708 caller->ice_gathering_state(), kDefaultTimeout);
709 EXPECT_TRUE(caller->observer()->candidate_gathered());
710 // Get the current offer that contains candidates and pass it to the callee.
711 //
712 // Note that we cannot use CloneSessionDescription on |cur_offer| to obtain an
713 // SDP with candidates. The method above does not strictly copy everything, in
714 // particular, not copying the ICE candidates.
715 // TODO(qingsi): Technically, this is a bug. Fix it.
716 auto cur_offer = caller->pc()->local_description();
717 ASSERT_TRUE(cur_offer);
718 std::string sdp_with_candidates_str;
719 cur_offer->ToString(&sdp_with_candidates_str);
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200720 auto offer = std::make_unique<JsepSessionDescription>(SdpType::kOffer);
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700721 ASSERT_TRUE(SdpDeserialize(sdp_with_candidates_str, offer.get(),
722 nullptr /* error */));
723 ASSERT_TRUE(callee->SetRemoteDescription(std::move(offer)));
724
725 // By default, the Answer created does not contain ICE candidates.
726 auto answer = callee->CreateAnswer();
727 callee->SetLocalDescription(CloneSessionDescription(answer.get()));
728 caller->SetRemoteDescription(std::move(answer));
729 EXPECT_TRUE_WAIT(caller->IsConnected(), kDefaultTimeout);
730 EXPECT_TRUE_WAIT(callee->IsConnected(), kDefaultTimeout);
731 // The callee needs to process the open message to have the data channel open.
732 EXPECT_TRUE_WAIT(callee->observer()->last_datachannel_ != nullptr,
733 kDefaultTimeout);
734 caller->pc()->Close();
735 callee->pc()->Close();
736
737 // The caller should not have added any remote candidate either via
Qingsi Wangcc46b102019-09-12 11:19:01 -0700738 // AddIceCandidate or from the remote description. Also, the caller connects
739 // with the callee via a prflx candidate and hence no direct connection bit
740 // should be set.
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700741 int expected_fingerprint_caller = MakeUsageFingerprint(
742 {PeerConnection::UsageEvent::DATA_ADDED,
743 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
744 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
745 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
746 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED,
747 PeerConnection::UsageEvent::IPV6_CANDIDATE_COLLECTED,
748 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
749 PeerConnection::UsageEvent::CLOSE_CALLED});
750
751 int expected_fingerprint_callee = MakeUsageFingerprint(
752 {PeerConnection::UsageEvent::DATA_ADDED,
753 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
754 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
755 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
756 PeerConnection::UsageEvent::MDNS_CANDIDATE_COLLECTED,
757 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
758 PeerConnection::UsageEvent::REMOTE_PRIVATE_CANDIDATE_ADDED,
759 PeerConnection::UsageEvent::REMOTE_IPV6_CANDIDATE_ADDED,
760 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700761 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700762 PeerConnection::UsageEvent::CLOSE_CALLED});
763
764 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
765 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
766 expected_fingerprint_caller));
767 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
768 expected_fingerprint_callee));
769}
770
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200771TEST_F(PeerConnectionUsageHistogramTest, NotableUsageNoted) {
772 auto caller = CreatePeerConnection();
773 caller->CreateDataChannel("foo");
774 caller->GenerateOfferAndCollectCandidates();
775 caller->pc()->Close();
776 int expected_fingerprint = MakeUsageFingerprint(
777 {PeerConnection::UsageEvent::DATA_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700778 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200779 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
780 PeerConnection::UsageEvent::CLOSE_CALLED});
781 EXPECT_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
782 EXPECT_TRUE(expected_fingerprint == ObservedFingerprint() ||
783 (expected_fingerprint |
784 static_cast<int>(
785 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
786 ObservedFingerprint());
787 EXPECT_EQ(absl::make_optional(ObservedFingerprint()),
788 caller->observer()->interesting_usage_detected());
789}
Harald Alvestrand7a1c7f72018-08-01 10:50:16 +0200790
791TEST_F(PeerConnectionUsageHistogramTest, NotableUsageOnEventFiring) {
792 auto caller = CreatePeerConnection();
793 caller->CreateDataChannel("foo");
794 caller->GenerateOfferAndCollectCandidates();
795 int expected_fingerprint = MakeUsageFingerprint(
796 {PeerConnection::UsageEvent::DATA_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700797 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
Harald Alvestrand7a1c7f72018-08-01 10:50:16 +0200798 PeerConnection::UsageEvent::CANDIDATE_COLLECTED});
799 EXPECT_EQ(0, webrtc::metrics::NumSamples(kUsagePatternMetric));
800 caller->GetInternalPeerConnection()->RequestUsagePatternReportForTesting();
801 EXPECT_EQ_WAIT(1, webrtc::metrics::NumSamples(kUsagePatternMetric),
802 kDefaultTimeout);
803 EXPECT_TRUE(expected_fingerprint == ObservedFingerprint() ||
804 (expected_fingerprint |
805 static_cast<int>(
806 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
807 ObservedFingerprint());
808 EXPECT_EQ(absl::make_optional(ObservedFingerprint()),
809 caller->observer()->interesting_usage_detected());
810}
811
812TEST_F(PeerConnectionUsageHistogramTest,
813 NoNotableUsageOnEventFiringAfterClose) {
814 auto caller = CreatePeerConnection();
815 caller->CreateDataChannel("foo");
816 caller->GenerateOfferAndCollectCandidates();
817 int expected_fingerprint = MakeUsageFingerprint(
818 {PeerConnection::UsageEvent::DATA_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700819 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
Harald Alvestrand7a1c7f72018-08-01 10:50:16 +0200820 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
821 PeerConnection::UsageEvent::CLOSE_CALLED});
822 EXPECT_EQ(0, webrtc::metrics::NumSamples(kUsagePatternMetric));
823 caller->pc()->Close();
824 EXPECT_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
825 caller->GetInternalPeerConnection()->RequestUsagePatternReportForTesting();
826 caller->observer()->ClearInterestingUsageDetector();
827 EXPECT_EQ_WAIT(2, webrtc::metrics::NumSamples(kUsagePatternMetric),
828 kDefaultTimeout);
829 EXPECT_TRUE(expected_fingerprint == ObservedFingerprint() ||
830 (expected_fingerprint |
831 static_cast<int>(
832 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
833 ObservedFingerprint());
834 // After close, the usage-detection callback should NOT have been called.
835 EXPECT_FALSE(caller->observer()->interesting_usage_detected());
836}
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200837#endif
838#endif
839
Harald Alvestrand19793842018-06-25 12:03:50 +0200840} // namespace webrtc