blob: f734e50e9abacb593bed1c5b1eaecac9713f3363 [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 }
Qingsi Wang32913c12019-10-30 16:03:46 -0700170 void BufferIceCandidate(const webrtc::IceCandidateInterface* candidate) {
171 std::string sdp;
172 EXPECT_TRUE(candidate->ToString(&sdp));
173 std::unique_ptr<webrtc::IceCandidateInterface> candidate_copy(
174 CreateIceCandidate(candidate->sdp_mid(), candidate->sdp_mline_index(),
175 sdp, nullptr));
176 buffered_candidates_.push_back(std::move(candidate_copy));
Harald Alvestrand42386282018-07-12 07:56:05 +0200177 }
Harald Alvestrand056d8112018-07-16 19:18:58 +0200178
Harald Alvestrand42386282018-07-12 07:56:05 +0200179 void AddBufferedIceCandidates() {
180 for (const auto& candidate : buffered_candidates_) {
181 EXPECT_TRUE(pc()->AddIceCandidate(candidate.get()));
182 }
183 buffered_candidates_.clear();
184 }
Harald Alvestrand056d8112018-07-16 19:18:58 +0200185
Qingsi Wang32913c12019-10-30 16:03:46 -0700186 // This method performs the following actions in sequence:
187 // 1. Exchange Offer and Answer.
188 // 2. Exchange ICE candidates after both caller and callee complete
189 // gathering.
190 // 3. Wait for ICE to connect.
191 //
192 // This guarantees a deterministic sequence of events and also rules out the
193 // occurrence of prflx candidates if the offer/answer signaling and the
194 // candidate trickling race in order. In case prflx candidates need to be
195 // simulated, see the approach used by tests below for that.
Harald Alvestrand42386282018-07-12 07:56:05 +0200196 bool ConnectTo(PeerConnectionWrapperForUsageHistogramTest* callee) {
197 PrepareToExchangeCandidates(callee);
Harald Alvestrand056d8112018-07-16 19:18:58 +0200198 if (!ExchangeOfferAnswerWith(callee)) {
199 return false;
200 }
Qingsi Wang32913c12019-10-30 16:03:46 -0700201 // Wait until the gathering completes before we signal the candidate.
202 WAIT(observer()->ice_gathering_complete_, kDefaultTimeout);
203 WAIT(callee->observer()->ice_gathering_complete_, kDefaultTimeout);
Harald Alvestrand42386282018-07-12 07:56:05 +0200204 AddBufferedIceCandidates();
205 callee->AddBufferedIceCandidates();
Harald Alvestrand056d8112018-07-16 19:18:58 +0200206 WAIT(IsConnected(), kDefaultTimeout);
207 WAIT(callee->IsConnected(), kDefaultTimeout);
Harald Alvestrand42386282018-07-12 07:56:05 +0200208 return IsConnected() && callee->IsConnected();
209 }
210
Harald Alvestrand056d8112018-07-16 19:18:58 +0200211 bool GenerateOfferAndCollectCandidates() {
212 auto offer = CreateOffer(RTCOfferAnswerOptions());
213 if (!offer) {
214 return false;
215 }
216 bool set_local_offer =
217 SetLocalDescription(CloneSessionDescription(offer.get()));
218 EXPECT_TRUE(set_local_offer);
219 if (!set_local_offer) {
220 return false;
221 }
222 EXPECT_TRUE_WAIT(observer()->ice_gathering_complete_, kDefaultTimeout);
223 return true;
224 }
225
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700226 webrtc::PeerConnectionInterface::IceGatheringState ice_gathering_state() {
227 return pc()->ice_gathering_state();
228 }
229
Harald Alvestrand42386282018-07-12 07:56:05 +0200230 private:
231 // Candidates that have been sent but not yet configured
232 std::vector<std::unique_ptr<webrtc::IceCandidateInterface>>
233 buffered_candidates_;
Harald Alvestrand19793842018-06-25 12:03:50 +0200234};
235
Qingsi Wang32913c12019-10-30 16:03:46 -0700236// Buffers candidates until we add them via AddBufferedIceCandidates.
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200237void ObserverForUsageHistogramTest::OnIceCandidate(
238 const webrtc::IceCandidateInterface* candidate) {
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700239 // If target is not set, ignore. This happens in one-ended unit tests.
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200240 if (candidate_target_) {
Qingsi Wang32913c12019-10-30 16:03:46 -0700241 this->candidate_target_->BufferIceCandidate(candidate);
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200242 }
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700243 candidate_gathered_ = true;
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200244}
245
Harald Alvestrand19793842018-06-25 12:03:50 +0200246class PeerConnectionUsageHistogramTest : public ::testing::Test {
247 protected:
248 typedef std::unique_ptr<PeerConnectionWrapperForUsageHistogramTest>
249 WrapperPtr;
250
251 PeerConnectionUsageHistogramTest()
252 : vss_(new rtc::VirtualSocketServer()), main_(vss_.get()) {
Qingsi Wang7fc821d2018-07-12 12:54:53 -0700253 webrtc::metrics::Reset();
Harald Alvestrand19793842018-06-25 12:03:50 +0200254 }
255
256 WrapperPtr CreatePeerConnection() {
Harald Alvestrand056d8112018-07-16 19:18:58 +0200257 return CreatePeerConnection(RTCConfiguration(),
258 PeerConnectionFactoryInterface::Options(),
259 nullptr, false);
Harald Alvestrand19793842018-06-25 12:03:50 +0200260 }
261
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200262 WrapperPtr CreatePeerConnection(const RTCConfiguration& config) {
263 return CreatePeerConnection(
Harald Alvestrand056d8112018-07-16 19:18:58 +0200264 config, PeerConnectionFactoryInterface::Options(), nullptr, false);
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200265 }
266
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700267 WrapperPtr CreatePeerConnectionWithMdns(const RTCConfiguration& config) {
268 auto resolver_factory =
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200269 std::make_unique<NiceMock<webrtc::MockAsyncResolverFactory>>();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700270
271 webrtc::PeerConnectionDependencies deps(nullptr /* observer_in */);
272
273 auto fake_network = NewFakeNetwork();
Qingsi Wangecd30542019-05-22 14:34:56 -0700274 fake_network->set_mdns_responder(
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200275 std::make_unique<webrtc::FakeMdnsResponder>(rtc::Thread::Current()));
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700276 fake_network->AddInterface(NextLocalAddress());
277
278 std::unique_ptr<cricket::BasicPortAllocator> port_allocator(
279 new cricket::BasicPortAllocator(fake_network));
280
281 deps.async_resolver_factory = std::move(resolver_factory);
282 deps.allocator = std::move(port_allocator);
283
284 return CreatePeerConnection(config,
285 PeerConnectionFactoryInterface::Options(),
286 std::move(deps), false);
287 }
288
Harald Alvestrand19793842018-06-25 12:03:50 +0200289 WrapperPtr CreatePeerConnectionWithImmediateReport() {
Harald Alvestrand056d8112018-07-16 19:18:58 +0200290 return CreatePeerConnection(RTCConfiguration(),
291 PeerConnectionFactoryInterface::Options(),
292 nullptr, true);
293 }
294
295 WrapperPtr CreatePeerConnectionWithPrivateLocalAddresses() {
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700296 auto* fake_network = NewFakeNetwork();
297 fake_network->AddInterface(NextLocalAddress());
298 fake_network->AddInterface(kPrivateLocalAddress);
299
300 auto port_allocator =
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200301 std::make_unique<cricket::BasicPortAllocator>(fake_network);
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700302
Harald Alvestrand056d8112018-07-16 19:18:58 +0200303 return CreatePeerConnection(RTCConfiguration(),
304 PeerConnectionFactoryInterface::Options(),
305 std::move(port_allocator), false);
Harald Alvestrand19793842018-06-25 12:03:50 +0200306 }
307
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700308 WrapperPtr CreatePeerConnectionWithPrivateIpv6LocalAddresses() {
309 auto* fake_network = NewFakeNetwork();
310 fake_network->AddInterface(NextLocalAddress());
311 fake_network->AddInterface(kPrivateIpv6LocalAddress);
312
313 auto port_allocator =
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200314 std::make_unique<cricket::BasicPortAllocator>(fake_network);
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700315
316 return CreatePeerConnection(RTCConfiguration(),
317 PeerConnectionFactoryInterface::Options(),
318 std::move(port_allocator), false);
319 }
320
Harald Alvestrand19793842018-06-25 12:03:50 +0200321 WrapperPtr CreatePeerConnection(
322 const RTCConfiguration& config,
323 const PeerConnectionFactoryInterface::Options factory_options,
Harald Alvestrand056d8112018-07-16 19:18:58 +0200324 std::unique_ptr<cricket::PortAllocator> allocator,
Harald Alvestrand19793842018-06-25 12:03:50 +0200325 bool immediate_report) {
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700326 PeerConnectionDependencies deps(nullptr);
327 deps.allocator = std::move(allocator);
328
329 return CreatePeerConnection(config, factory_options, std::move(deps),
330 immediate_report);
331 }
332
333 WrapperPtr CreatePeerConnection(
334 const RTCConfiguration& config,
335 const PeerConnectionFactoryInterface::Options factory_options,
336 PeerConnectionDependencies deps,
337 bool immediate_report) {
Harald Alvestrand19793842018-06-25 12:03:50 +0200338 rtc::scoped_refptr<PeerConnectionFactoryForUsageHistogramTest> pc_factory(
339 new PeerConnectionFactoryForUsageHistogramTest());
340 pc_factory->SetOptions(factory_options);
341 RTC_CHECK(pc_factory->Initialize());
342 if (immediate_report) {
343 pc_factory->ReturnHistogramVeryQuickly();
344 }
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700345
346 // If no allocator is provided, one will be created using a network manager
347 // that uses the host network. This doesn't work on all trybots.
348 if (!deps.allocator) {
349 auto fake_network = NewFakeNetwork();
350 fake_network->AddInterface(NextLocalAddress());
351 deps.allocator =
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200352 std::make_unique<cricket::BasicPortAllocator>(fake_network);
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700353 }
354
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200355 auto observer = std::make_unique<ObserverForUsageHistogramTest>();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700356 deps.observer = observer.get();
357
358 auto pc = pc_factory->CreatePeerConnection(config, std::move(deps));
Harald Alvestrand19793842018-06-25 12:03:50 +0200359 if (!pc) {
360 return nullptr;
361 }
362
Yves Gerey4e933292018-10-31 15:36:05 +0100363 observer->SetPeerConnectionInterface(pc.get());
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200364 auto wrapper = std::make_unique<PeerConnectionWrapperForUsageHistogramTest>(
365 pc_factory, pc, std::move(observer));
Harald Alvestrand19793842018-06-25 12:03:50 +0200366 return wrapper;
367 }
368
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200369 int ObservedFingerprint() {
370 // This works correctly only if there is only one sample value
371 // that has been counted.
372 // Returns -1 for "not found".
373 return webrtc::metrics::MinSample(kUsagePatternMetric);
374 }
375
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700376 // The PeerConnection's port allocator is tied to the PeerConnection's
377 // lifetime and expects the underlying NetworkManager to outlive it. That
378 // prevents us from having the PeerConnectionWrapper own the fake network.
379 // Therefore, the test fixture will own all the fake networks even though
380 // tests should access the fake network through the PeerConnectionWrapper.
381 rtc::FakeNetworkManager* NewFakeNetwork() {
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200382 fake_networks_.emplace_back(std::make_unique<rtc::FakeNetworkManager>());
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700383 return fake_networks_.back().get();
384 }
385
386 rtc::SocketAddress NextLocalAddress() {
387 RTC_DCHECK(next_local_address_ < (int)arraysize(kLocalAddrs));
388 return kLocalAddrs[next_local_address_++];
389 }
390
391 std::vector<std::unique_ptr<rtc::FakeNetworkManager>> fake_networks_;
392 int next_local_address_ = 0;
Harald Alvestrand19793842018-06-25 12:03:50 +0200393 std::unique_ptr<rtc::VirtualSocketServer> vss_;
394 rtc::AutoSocketServerThread main_;
395};
396
397TEST_F(PeerConnectionUsageHistogramTest, UsageFingerprintHistogramFromTimeout) {
398 auto pc = CreatePeerConnectionWithImmediateReport();
399
Harald Alvestrand19793842018-06-25 12:03:50 +0200400 int expected_fingerprint = MakeUsageFingerprint({});
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200401 ASSERT_EQ_WAIT(1, webrtc::metrics::NumSamples(kUsagePatternMetric),
402 kDefaultTimeout);
403 EXPECT_EQ(
404 1, webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint));
Harald Alvestrand19793842018-06-25 12:03:50 +0200405}
406
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200407#ifndef WEBRTC_ANDROID
408// These tests do not work on Android. Why is unclear.
409// https://bugs.webrtc.org/9461
410
411// Test getting the usage fingerprint for an audio/video connection.
412TEST_F(PeerConnectionUsageHistogramTest, FingerprintAudioVideo) {
413 auto caller = CreatePeerConnection();
414 auto callee = CreatePeerConnection();
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200415 caller->AddAudioTrack("audio");
416 caller->AddVideoTrack("video");
Harald Alvestrand42386282018-07-12 07:56:05 +0200417 ASSERT_TRUE(caller->ConnectTo(callee.get()));
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200418 caller->pc()->Close();
419 callee->pc()->Close();
420 int expected_fingerprint = MakeUsageFingerprint(
421 {PeerConnection::UsageEvent::AUDIO_ADDED,
422 PeerConnection::UsageEvent::VIDEO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700423 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
424 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200425 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700426 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200427 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700428 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700429 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200430 PeerConnection::UsageEvent::CLOSE_CALLED});
Harald Alvestrand056d8112018-07-16 19:18:58 +0200431 // In this case, we may or may not have PRIVATE_CANDIDATE_COLLECTED,
432 // depending on the machine configuration.
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200433 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
Harald Alvestrand056d8112018-07-16 19:18:58 +0200434 EXPECT_TRUE(
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200435 webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint) ==
436 2 ||
Harald Alvestrand056d8112018-07-16 19:18:58 +0200437 webrtc::metrics::NumEvents(
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200438 kUsagePatternMetric,
Harald Alvestrand056d8112018-07-16 19:18:58 +0200439 expected_fingerprint |
440 static_cast<int>(
441 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
442 2);
443}
444
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700445// Test getting the usage fingerprint when the caller collects an mDNS
446// candidate.
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700447TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithMdnsCaller) {
448 RTCConfiguration config;
449
450 // Enable hostname candidates with mDNS names.
451 auto caller = CreatePeerConnectionWithMdns(config);
452 auto callee = CreatePeerConnection(config);
453
454 caller->AddAudioTrack("audio");
455 caller->AddVideoTrack("video");
456 ASSERT_TRUE(caller->ConnectTo(callee.get()));
457 caller->pc()->Close();
458 callee->pc()->Close();
459
460 int expected_fingerprint_caller = MakeUsageFingerprint(
461 {PeerConnection::UsageEvent::AUDIO_ADDED,
462 PeerConnection::UsageEvent::VIDEO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700463 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
464 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700465 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
466 PeerConnection::UsageEvent::MDNS_CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700467 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700468 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700469 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700470 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700471 PeerConnection::UsageEvent::CLOSE_CALLED});
472
Qingsi Wangcc46b102019-09-12 11:19:01 -0700473 // Without a resolver, the callee cannot resolve the received mDNS candidate
474 // but can still connect with the caller via a prflx candidate. As a result,
475 // the bit for the direct connection should not be logged.
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700476 int expected_fingerprint_callee = MakeUsageFingerprint(
477 {PeerConnection::UsageEvent::AUDIO_ADDED,
478 PeerConnection::UsageEvent::VIDEO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700479 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
480 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700481 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700482 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700483 PeerConnection::UsageEvent::REMOTE_MDNS_CANDIDATE_ADDED,
484 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700485 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700486 PeerConnection::UsageEvent::CLOSE_CALLED});
487
488 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
489 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
490 expected_fingerprint_caller));
491 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
492 expected_fingerprint_callee));
493}
494
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700495// Test getting the usage fingerprint when the callee collects an mDNS
496// candidate.
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700497TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithMdnsCallee) {
498 RTCConfiguration config;
499
500 // Enable hostname candidates with mDNS names.
501 auto caller = CreatePeerConnection(config);
502 auto callee = CreatePeerConnectionWithMdns(config);
503
504 caller->AddAudioTrack("audio");
505 caller->AddVideoTrack("video");
506 ASSERT_TRUE(caller->ConnectTo(callee.get()));
507 caller->pc()->Close();
508 callee->pc()->Close();
509
Qingsi Wangcc46b102019-09-12 11:19:01 -0700510 // Similar to the test above, the caller connects with the callee via a prflx
511 // candidate.
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700512 int expected_fingerprint_caller = 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,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700518 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700519 PeerConnection::UsageEvent::REMOTE_MDNS_CANDIDATE_ADDED,
520 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700521 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700522 PeerConnection::UsageEvent::CLOSE_CALLED});
523
524 int expected_fingerprint_callee = MakeUsageFingerprint(
525 {PeerConnection::UsageEvent::AUDIO_ADDED,
526 PeerConnection::UsageEvent::VIDEO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700527 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
528 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700529 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
530 PeerConnection::UsageEvent::MDNS_CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700531 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700532 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700533 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700534 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700535 PeerConnection::UsageEvent::CLOSE_CALLED});
536
537 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
538 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
539 expected_fingerprint_caller));
540 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
541 expected_fingerprint_callee));
542}
543
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200544#ifdef HAVE_SCTP
545TEST_F(PeerConnectionUsageHistogramTest, FingerprintDataOnly) {
546 auto caller = CreatePeerConnection();
547 auto callee = CreatePeerConnection();
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200548 caller->CreateDataChannel("foodata");
Harald Alvestrand42386282018-07-12 07:56:05 +0200549 ASSERT_TRUE(caller->ConnectTo(callee.get()));
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200550 ASSERT_TRUE_WAIT(callee->HaveDataChannel(), kDefaultTimeout);
551 caller->pc()->Close();
552 callee->pc()->Close();
553 int expected_fingerprint = MakeUsageFingerprint(
554 {PeerConnection::UsageEvent::DATA_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700555 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
556 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200557 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700558 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200559 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700560 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700561 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200562 PeerConnection::UsageEvent::CLOSE_CALLED});
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200563 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
Harald Alvestrand056d8112018-07-16 19:18:58 +0200564 EXPECT_TRUE(
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200565 webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint) ==
566 2 ||
Harald Alvestrand056d8112018-07-16 19:18:58 +0200567 webrtc::metrics::NumEvents(
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200568 kUsagePatternMetric,
Harald Alvestrand056d8112018-07-16 19:18:58 +0200569 expected_fingerprint |
570 static_cast<int>(
571 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
572 2);
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200573}
574#endif // HAVE_SCTP
575#endif // WEBRTC_ANDROID
576
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200577TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurn) {
578 RTCConfiguration configuration;
579 PeerConnection::IceServer server;
580 server.urls = {"stun:dummy.stun.server/"};
581 configuration.servers.push_back(server);
582 server.urls = {"turn:dummy.turn.server/"};
583 server.username = "username";
584 server.password = "password";
585 configuration.servers.push_back(server);
586 auto caller = CreatePeerConnection(configuration);
587 ASSERT_TRUE(caller);
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200588 caller->pc()->Close();
589 int expected_fingerprint =
590 MakeUsageFingerprint({PeerConnection::UsageEvent::STUN_SERVER_ADDED,
591 PeerConnection::UsageEvent::TURN_SERVER_ADDED,
592 PeerConnection::UsageEvent::CLOSE_CALLED});
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200593 EXPECT_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
594 EXPECT_EQ(
595 1, webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint));
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200596}
597
598TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurnInReconfiguration) {
599 RTCConfiguration configuration;
600 PeerConnection::IceServer server;
601 server.urls = {"stun:dummy.stun.server/"};
602 configuration.servers.push_back(server);
603 server.urls = {"turn:dummy.turn.server/"};
604 server.username = "username";
605 server.password = "password";
606 configuration.servers.push_back(server);
607 auto caller = CreatePeerConnection();
608 ASSERT_TRUE(caller);
Niels Möller2579f0c2019-08-19 09:58:17 +0200609 ASSERT_TRUE(caller->pc()->SetConfiguration(configuration).ok());
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200610 caller->pc()->Close();
611 int expected_fingerprint =
612 MakeUsageFingerprint({PeerConnection::UsageEvent::STUN_SERVER_ADDED,
613 PeerConnection::UsageEvent::TURN_SERVER_ADDED,
614 PeerConnection::UsageEvent::CLOSE_CALLED});
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200615 EXPECT_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
616 EXPECT_EQ(
617 1, webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint));
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200618}
619
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700620TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithPrivateIPCaller) {
Harald Alvestrand056d8112018-07-16 19:18:58 +0200621 auto caller = CreatePeerConnectionWithPrivateLocalAddresses();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700622 auto callee = CreatePeerConnection();
Harald Alvestrand056d8112018-07-16 19:18:58 +0200623 caller->AddAudioTrack("audio");
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700624 ASSERT_TRUE(caller->ConnectTo(callee.get()));
Harald Alvestrand056d8112018-07-16 19:18:58 +0200625 caller->pc()->Close();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700626 callee->pc()->Close();
627
628 int expected_fingerprint_caller = MakeUsageFingerprint(
Harald Alvestrand056d8112018-07-16 19:18:58 +0200629 {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,
Harald Alvestrand056d8112018-07-16 19:18:58 +0200632 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700633 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700634 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700635 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 int expected_fingerprint_callee = MakeUsageFingerprint(
641 {PeerConnection::UsageEvent::AUDIO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700642 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
643 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700644 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700645 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700646 PeerConnection::UsageEvent::REMOTE_PRIVATE_CANDIDATE_ADDED,
647 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700648 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700649 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700650 PeerConnection::UsageEvent::CLOSE_CALLED});
651
652 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
653 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
654 expected_fingerprint_caller));
655 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
656 expected_fingerprint_callee));
657}
658
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700659TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithPrivateIpv6Callee) {
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700660 auto caller = CreatePeerConnection();
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700661 auto callee = CreatePeerConnectionWithPrivateIpv6LocalAddresses();
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700662 caller->AddAudioTrack("audio");
663 ASSERT_TRUE(caller->ConnectTo(callee.get()));
664 caller->pc()->Close();
665 callee->pc()->Close();
666
667 int expected_fingerprint_caller = MakeUsageFingerprint(
668 {PeerConnection::UsageEvent::AUDIO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700669 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
670 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700671 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700672 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700673 PeerConnection::UsageEvent::REMOTE_PRIVATE_CANDIDATE_ADDED,
674 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700675 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
676 PeerConnection::UsageEvent::REMOTE_IPV6_CANDIDATE_ADDED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700677 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700678 PeerConnection::UsageEvent::CLOSE_CALLED});
679
680 int expected_fingerprint_callee = MakeUsageFingerprint(
681 {PeerConnection::UsageEvent::AUDIO_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700682 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
683 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700684 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
685 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700686 PeerConnection::UsageEvent::IPV6_CANDIDATE_COLLECTED,
687 PeerConnection::UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700688 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
689 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700690 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Jeroen de Borstaf242c82019-04-24 13:13:48 -0700691 PeerConnection::UsageEvent::CLOSE_CALLED});
692
693 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
694 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
695 expected_fingerprint_caller));
696 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
697 expected_fingerprint_callee));
Harald Alvestrand056d8112018-07-16 19:18:58 +0200698}
699
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200700#ifndef WEBRTC_ANDROID
701#ifdef HAVE_SCTP
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700702// Test that the usage pattern bits for adding remote (private IPv6) candidates
703// are set when the remote candidates are retrieved from the Offer SDP instead
704// of trickled ICE messages.
705TEST_F(PeerConnectionUsageHistogramTest,
706 AddRemoteCandidatesFromRemoteDescription) {
707 // We construct the following data-channel-only scenario. The caller collects
708 // IPv6 private local candidates and appends them in the Offer as in
709 // non-trickled sessions. The callee collects mDNS candidates that are not
710 // contained in the Answer as in Trickle ICE. Only the Offer and Answer are
711 // signaled and we expect a connection with prflx remote candidates at the
712 // caller side.
713 auto caller = CreatePeerConnectionWithPrivateIpv6LocalAddresses();
714 auto callee = CreatePeerConnectionWithMdns(RTCConfiguration());
715 caller->CreateDataChannel("test_channel");
716 ASSERT_TRUE(caller->SetLocalDescription(caller->CreateOffer()));
717 // Wait until the gathering completes so that the session description would
718 // have contained ICE candidates.
719 EXPECT_EQ_WAIT(webrtc::PeerConnectionInterface::kIceGatheringComplete,
720 caller->ice_gathering_state(), kDefaultTimeout);
721 EXPECT_TRUE(caller->observer()->candidate_gathered());
722 // Get the current offer that contains candidates and pass it to the callee.
723 //
724 // Note that we cannot use CloneSessionDescription on |cur_offer| to obtain an
725 // SDP with candidates. The method above does not strictly copy everything, in
726 // particular, not copying the ICE candidates.
727 // TODO(qingsi): Technically, this is a bug. Fix it.
728 auto cur_offer = caller->pc()->local_description();
729 ASSERT_TRUE(cur_offer);
730 std::string sdp_with_candidates_str;
731 cur_offer->ToString(&sdp_with_candidates_str);
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200732 auto offer = std::make_unique<JsepSessionDescription>(SdpType::kOffer);
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700733 ASSERT_TRUE(SdpDeserialize(sdp_with_candidates_str, offer.get(),
734 nullptr /* error */));
735 ASSERT_TRUE(callee->SetRemoteDescription(std::move(offer)));
736
737 // By default, the Answer created does not contain ICE candidates.
738 auto answer = callee->CreateAnswer();
739 callee->SetLocalDescription(CloneSessionDescription(answer.get()));
740 caller->SetRemoteDescription(std::move(answer));
741 EXPECT_TRUE_WAIT(caller->IsConnected(), kDefaultTimeout);
742 EXPECT_TRUE_WAIT(callee->IsConnected(), kDefaultTimeout);
743 // The callee needs to process the open message to have the data channel open.
744 EXPECT_TRUE_WAIT(callee->observer()->last_datachannel_ != nullptr,
745 kDefaultTimeout);
746 caller->pc()->Close();
747 callee->pc()->Close();
748
749 // The caller should not have added any remote candidate either via
Qingsi Wangcc46b102019-09-12 11:19:01 -0700750 // AddIceCandidate or from the remote description. Also, the caller connects
751 // with the callee via a prflx candidate and hence no direct connection bit
752 // should be set.
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700753 int expected_fingerprint_caller = MakeUsageFingerprint(
754 {PeerConnection::UsageEvent::DATA_ADDED,
755 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
756 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
757 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
758 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED,
759 PeerConnection::UsageEvent::IPV6_CANDIDATE_COLLECTED,
760 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
761 PeerConnection::UsageEvent::CLOSE_CALLED});
762
763 int expected_fingerprint_callee = MakeUsageFingerprint(
764 {PeerConnection::UsageEvent::DATA_ADDED,
765 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
766 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
767 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
768 PeerConnection::UsageEvent::MDNS_CANDIDATE_COLLECTED,
769 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
770 PeerConnection::UsageEvent::REMOTE_PRIVATE_CANDIDATE_ADDED,
771 PeerConnection::UsageEvent::REMOTE_IPV6_CANDIDATE_ADDED,
772 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
Qingsi Wangcc46b102019-09-12 11:19:01 -0700773 PeerConnection::UsageEvent::DIRECT_CONNECTION_SELECTED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700774 PeerConnection::UsageEvent::CLOSE_CALLED});
775
776 EXPECT_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
777 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
778 expected_fingerprint_caller));
779 EXPECT_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
780 expected_fingerprint_callee));
781}
782
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200783TEST_F(PeerConnectionUsageHistogramTest, NotableUsageNoted) {
784 auto caller = CreatePeerConnection();
785 caller->CreateDataChannel("foo");
786 caller->GenerateOfferAndCollectCandidates();
787 caller->pc()->Close();
788 int expected_fingerprint = MakeUsageFingerprint(
789 {PeerConnection::UsageEvent::DATA_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700790 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200791 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
792 PeerConnection::UsageEvent::CLOSE_CALLED});
793 EXPECT_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
794 EXPECT_TRUE(expected_fingerprint == ObservedFingerprint() ||
795 (expected_fingerprint |
796 static_cast<int>(
797 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
798 ObservedFingerprint());
799 EXPECT_EQ(absl::make_optional(ObservedFingerprint()),
800 caller->observer()->interesting_usage_detected());
801}
Harald Alvestrand7a1c7f72018-08-01 10:50:16 +0200802
803TEST_F(PeerConnectionUsageHistogramTest, NotableUsageOnEventFiring) {
804 auto caller = CreatePeerConnection();
805 caller->CreateDataChannel("foo");
806 caller->GenerateOfferAndCollectCandidates();
807 int expected_fingerprint = MakeUsageFingerprint(
808 {PeerConnection::UsageEvent::DATA_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700809 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
Harald Alvestrand7a1c7f72018-08-01 10:50:16 +0200810 PeerConnection::UsageEvent::CANDIDATE_COLLECTED});
811 EXPECT_EQ(0, webrtc::metrics::NumSamples(kUsagePatternMetric));
812 caller->GetInternalPeerConnection()->RequestUsagePatternReportForTesting();
813 EXPECT_EQ_WAIT(1, webrtc::metrics::NumSamples(kUsagePatternMetric),
814 kDefaultTimeout);
815 EXPECT_TRUE(expected_fingerprint == ObservedFingerprint() ||
816 (expected_fingerprint |
817 static_cast<int>(
818 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
819 ObservedFingerprint());
820 EXPECT_EQ(absl::make_optional(ObservedFingerprint()),
821 caller->observer()->interesting_usage_detected());
822}
823
824TEST_F(PeerConnectionUsageHistogramTest,
825 NoNotableUsageOnEventFiringAfterClose) {
826 auto caller = CreatePeerConnection();
827 caller->CreateDataChannel("foo");
828 caller->GenerateOfferAndCollectCandidates();
829 int expected_fingerprint = MakeUsageFingerprint(
830 {PeerConnection::UsageEvent::DATA_ADDED,
Qingsi Wang1ba5dec2019-08-19 11:57:17 -0700831 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
Harald Alvestrand7a1c7f72018-08-01 10:50:16 +0200832 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
833 PeerConnection::UsageEvent::CLOSE_CALLED});
834 EXPECT_EQ(0, webrtc::metrics::NumSamples(kUsagePatternMetric));
835 caller->pc()->Close();
836 EXPECT_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
837 caller->GetInternalPeerConnection()->RequestUsagePatternReportForTesting();
838 caller->observer()->ClearInterestingUsageDetector();
839 EXPECT_EQ_WAIT(2, webrtc::metrics::NumSamples(kUsagePatternMetric),
840 kDefaultTimeout);
841 EXPECT_TRUE(expected_fingerprint == ObservedFingerprint() ||
842 (expected_fingerprint |
843 static_cast<int>(
844 PeerConnection::UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
845 ObservedFingerprint());
846 // After close, the usage-detection callback should NOT have been called.
847 EXPECT_FALSE(caller->observer()->interesting_usage_detected());
848}
Harald Alvestrandc0e97252018-07-26 10:39:55 +0200849#endif
850#endif
851
Harald Alvestrand19793842018-06-25 12:03:50 +0200852} // namespace webrtc