blob: b9983727d9acd1c73795ea6f09a80079f82fc21d [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
11#include <tuple>
12
Karl Wiberg918f50c2018-07-05 11:40:33 +020013#include "absl/memory/memory.h"
Harald Alvestrand19793842018-06-25 12:03:50 +020014#include "api/fakemetricsobserver.h"
15#include "api/peerconnectionproxy.h"
16#include "media/base/fakemediaengine.h"
17#include "pc/mediasession.h"
18#include "pc/peerconnection.h"
19#include "pc/peerconnectionfactory.h"
20#include "pc/peerconnectionwrapper.h"
21#include "pc/sdputils.h"
Harald Alvestrand19793842018-06-25 12:03:50 +020022#include "pc/test/fakesctptransport.h"
23#include "rtc_base/gunit.h"
Harald Alvestrand19793842018-06-25 12:03:50 +020024#include "rtc_base/virtualsocketserver.h"
25
26namespace webrtc {
27
28using RTCConfiguration = PeerConnectionInterface::RTCConfiguration;
29using RTCOfferAnswerOptions = PeerConnectionInterface::RTCOfferAnswerOptions;
30using ::testing::Values;
31
32static constexpr int kDefaultTimeout = 10000;
33
34int MakeUsageFingerprint(std::set<PeerConnection::UsageEvent> events) {
35 int signature = 0;
36 for (const auto it : events) {
37 signature |= static_cast<int>(it);
38 }
39 return signature;
40}
41
42class PeerConnectionFactoryForUsageHistogramTest
43 : public rtc::RefCountedObject<PeerConnectionFactory> {
44 public:
45 PeerConnectionFactoryForUsageHistogramTest()
46 : rtc::RefCountedObject<PeerConnectionFactory>(
47 rtc::Thread::Current(),
48 rtc::Thread::Current(),
49 rtc::Thread::Current(),
Karl Wiberg918f50c2018-07-05 11:40:33 +020050 absl::make_unique<cricket::FakeMediaEngine>(),
Harald Alvestrand19793842018-06-25 12:03:50 +020051 CreateCallFactory(),
52 nullptr) {}
53
54 void ActionsBeforeInitializeForTesting(PeerConnectionInterface* pc) override {
55 PeerConnection* internal_pc = static_cast<PeerConnection*>(pc);
56 if (return_histogram_very_quickly_) {
57 internal_pc->ReturnHistogramVeryQuicklyForTesting();
58 }
59 }
60
61 void ReturnHistogramVeryQuickly() { return_histogram_very_quickly_ = true; }
62
63 private:
Harald Alvestrand183e09d2018-06-28 12:04:41 +020064 bool return_histogram_very_quickly_ = false;
65};
66
67class PeerConnectionWrapperForUsageHistogramTest;
68typedef PeerConnectionWrapperForUsageHistogramTest* RawWrapperPtr;
69
70class ObserverForUsageHistogramTest : public MockPeerConnectionObserver {
71 public:
72 void OnIceCandidate(const webrtc::IceCandidateInterface* candidate) override;
73 void PrepareToExchangeCandidates(RawWrapperPtr other) {
74 candidate_target_ = other;
75 }
76
77 bool HaveDataChannel() { return last_datachannel_; }
78
79 private:
80 RawWrapperPtr candidate_target_; // Note: Not thread-safe against deletions.
Harald Alvestrand19793842018-06-25 12:03:50 +020081};
82
83class PeerConnectionWrapperForUsageHistogramTest
84 : public PeerConnectionWrapper {
85 public:
86 using PeerConnectionWrapper::PeerConnectionWrapper;
87
88 PeerConnection* GetInternalPeerConnection() {
89 auto* pci =
90 static_cast<PeerConnectionProxyWithInternal<PeerConnectionInterface>*>(
91 pc());
92 return static_cast<PeerConnection*>(pci->internal());
93 }
Harald Alvestrand183e09d2018-06-28 12:04:41 +020094
95 void PrepareToExchangeCandidates(
96 PeerConnectionWrapperForUsageHistogramTest* other) {
97 static_cast<ObserverForUsageHistogramTest*>(observer())
98 ->PrepareToExchangeCandidates(other);
99 static_cast<ObserverForUsageHistogramTest*>(other->observer())
100 ->PrepareToExchangeCandidates(this);
101 }
102
103 bool IsConnected() {
104 return pc()->ice_connection_state() ==
105 PeerConnectionInterface::kIceConnectionConnected ||
106 pc()->ice_connection_state() ==
107 PeerConnectionInterface::kIceConnectionCompleted;
108 }
109
110 bool HaveDataChannel() {
111 return static_cast<ObserverForUsageHistogramTest*>(observer())
112 ->HaveDataChannel();
113 }
Harald Alvestrand19793842018-06-25 12:03:50 +0200114};
115
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200116void ObserverForUsageHistogramTest::OnIceCandidate(
117 const webrtc::IceCandidateInterface* candidate) {
118 if (candidate_target_) {
119 candidate_target_->pc()->AddIceCandidate(candidate);
120 }
121}
122
Harald Alvestrand19793842018-06-25 12:03:50 +0200123class PeerConnectionUsageHistogramTest : public ::testing::Test {
124 protected:
125 typedef std::unique_ptr<PeerConnectionWrapperForUsageHistogramTest>
126 WrapperPtr;
127
128 PeerConnectionUsageHistogramTest()
129 : vss_(new rtc::VirtualSocketServer()), main_(vss_.get()) {
Harald Alvestrand19793842018-06-25 12:03:50 +0200130 }
131
132 WrapperPtr CreatePeerConnection() {
133 return CreatePeerConnection(
134 RTCConfiguration(), PeerConnectionFactoryInterface::Options(), false);
135 }
136
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200137 WrapperPtr CreatePeerConnection(const RTCConfiguration& config) {
138 return CreatePeerConnection(
139 config, PeerConnectionFactoryInterface::Options(), false);
140 }
141
Harald Alvestrand19793842018-06-25 12:03:50 +0200142 WrapperPtr CreatePeerConnectionWithImmediateReport() {
143 return CreatePeerConnection(
144 RTCConfiguration(), PeerConnectionFactoryInterface::Options(), true);
145 }
146
147 WrapperPtr CreatePeerConnection(
148 const RTCConfiguration& config,
149 const PeerConnectionFactoryInterface::Options factory_options,
150 bool immediate_report) {
151 rtc::scoped_refptr<PeerConnectionFactoryForUsageHistogramTest> pc_factory(
152 new PeerConnectionFactoryForUsageHistogramTest());
153 pc_factory->SetOptions(factory_options);
154 RTC_CHECK(pc_factory->Initialize());
155 if (immediate_report) {
156 pc_factory->ReturnHistogramVeryQuickly();
157 }
Karl Wiberg918f50c2018-07-05 11:40:33 +0200158 auto observer = absl::make_unique<ObserverForUsageHistogramTest>();
Harald Alvestrand19793842018-06-25 12:03:50 +0200159 auto pc = pc_factory->CreatePeerConnection(config, nullptr, nullptr,
160 observer.get());
161 if (!pc) {
162 return nullptr;
163 }
164
Karl Wiberg918f50c2018-07-05 11:40:33 +0200165 auto wrapper =
166 absl::make_unique<PeerConnectionWrapperForUsageHistogramTest>(
167 pc_factory, pc, std::move(observer));
Harald Alvestrand19793842018-06-25 12:03:50 +0200168 return wrapper;
169 }
170
171 std::unique_ptr<rtc::VirtualSocketServer> vss_;
172 rtc::AutoSocketServerThread main_;
173};
174
175TEST_F(PeerConnectionUsageHistogramTest, UsageFingerprintHistogramFromTimeout) {
176 auto pc = CreatePeerConnectionWithImmediateReport();
177
178 // Register UMA observer before signaling begins.
179 rtc::scoped_refptr<webrtc::FakeMetricsObserver> caller_observer =
180 new rtc::RefCountedObject<webrtc::FakeMetricsObserver>();
181 pc->GetInternalPeerConnection()->RegisterUMAObserver(caller_observer);
182 int expected_fingerprint = MakeUsageFingerprint({});
183 ASSERT_TRUE_WAIT(caller_observer->ExpectOnlySingleEnumCount(
184 webrtc::kEnumCounterUsagePattern, expected_fingerprint),
185 kDefaultTimeout);
186}
187
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200188#ifndef WEBRTC_ANDROID
189// These tests do not work on Android. Why is unclear.
190// https://bugs.webrtc.org/9461
191
192// Test getting the usage fingerprint for an audio/video connection.
193TEST_F(PeerConnectionUsageHistogramTest, FingerprintAudioVideo) {
194 auto caller = CreatePeerConnection();
195 auto callee = CreatePeerConnection();
196 // Register UMA observer before signaling begins.
197 auto caller_observer = caller->RegisterFakeMetricsObserver();
198 auto callee_observer = callee->RegisterFakeMetricsObserver();
199 caller->AddAudioTrack("audio");
200 caller->AddVideoTrack("video");
201 caller->PrepareToExchangeCandidates(callee.get());
202 ASSERT_TRUE(caller->ExchangeOfferAnswerWith(callee.get()));
203 ASSERT_TRUE_WAIT(caller->IsConnected(), kDefaultTimeout);
204 ASSERT_TRUE_WAIT(callee->IsConnected(), kDefaultTimeout);
205 caller->pc()->Close();
206 callee->pc()->Close();
207 int expected_fingerprint = MakeUsageFingerprint(
208 {PeerConnection::UsageEvent::AUDIO_ADDED,
209 PeerConnection::UsageEvent::VIDEO_ADDED,
210 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_CALLED,
211 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_CALLED,
212 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
213 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
214 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
215 PeerConnection::UsageEvent::CLOSE_CALLED});
216 EXPECT_TRUE(caller_observer->ExpectOnlySingleEnumCount(
217 webrtc::kEnumCounterUsagePattern, expected_fingerprint));
218 EXPECT_TRUE(callee_observer->ExpectOnlySingleEnumCount(
219 webrtc::kEnumCounterUsagePattern, expected_fingerprint));
220}
221
222#ifdef HAVE_SCTP
223TEST_F(PeerConnectionUsageHistogramTest, FingerprintDataOnly) {
224 auto caller = CreatePeerConnection();
225 auto callee = CreatePeerConnection();
226 // Register UMA observer before signaling begins.
227 auto caller_observer = caller->RegisterFakeMetricsObserver();
228 auto callee_observer = callee->RegisterFakeMetricsObserver();
229 caller->CreateDataChannel("foodata");
230 caller->PrepareToExchangeCandidates(callee.get());
231 ASSERT_TRUE(caller->ExchangeOfferAnswerWith(callee.get()));
232 ASSERT_TRUE_WAIT(callee->HaveDataChannel(), kDefaultTimeout);
233 caller->pc()->Close();
234 callee->pc()->Close();
235 int expected_fingerprint = MakeUsageFingerprint(
236 {PeerConnection::UsageEvent::DATA_ADDED,
237 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_CALLED,
238 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_CALLED,
239 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
240 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
241 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
242 PeerConnection::UsageEvent::CLOSE_CALLED});
243 EXPECT_TRUE(caller_observer->ExpectOnlySingleEnumCount(
244 webrtc::kEnumCounterUsagePattern, expected_fingerprint));
245 EXPECT_TRUE(callee_observer->ExpectOnlySingleEnumCount(
246 webrtc::kEnumCounterUsagePattern, expected_fingerprint));
247}
248#endif // HAVE_SCTP
249#endif // WEBRTC_ANDROID
250
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200251TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurn) {
252 RTCConfiguration configuration;
253 PeerConnection::IceServer server;
254 server.urls = {"stun:dummy.stun.server/"};
255 configuration.servers.push_back(server);
256 server.urls = {"turn:dummy.turn.server/"};
257 server.username = "username";
258 server.password = "password";
259 configuration.servers.push_back(server);
260 auto caller = CreatePeerConnection(configuration);
261 ASSERT_TRUE(caller);
262 auto caller_observer = caller->RegisterFakeMetricsObserver();
263 caller->pc()->Close();
264 int expected_fingerprint =
265 MakeUsageFingerprint({PeerConnection::UsageEvent::STUN_SERVER_ADDED,
266 PeerConnection::UsageEvent::TURN_SERVER_ADDED,
267 PeerConnection::UsageEvent::CLOSE_CALLED});
268 EXPECT_TRUE(caller_observer->ExpectOnlySingleEnumCount(
269 webrtc::kEnumCounterUsagePattern, expected_fingerprint));
270}
271
272TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurnInReconfiguration) {
273 RTCConfiguration configuration;
274 PeerConnection::IceServer server;
275 server.urls = {"stun:dummy.stun.server/"};
276 configuration.servers.push_back(server);
277 server.urls = {"turn:dummy.turn.server/"};
278 server.username = "username";
279 server.password = "password";
280 configuration.servers.push_back(server);
281 auto caller = CreatePeerConnection();
282 ASSERT_TRUE(caller);
283 auto caller_observer = caller->RegisterFakeMetricsObserver();
284 RTCError error;
285 caller->pc()->SetConfiguration(configuration, &error);
286 ASSERT_TRUE(error.ok());
287 caller->pc()->Close();
288 int expected_fingerprint =
289 MakeUsageFingerprint({PeerConnection::UsageEvent::STUN_SERVER_ADDED,
290 PeerConnection::UsageEvent::TURN_SERVER_ADDED,
291 PeerConnection::UsageEvent::CLOSE_CALLED});
292 EXPECT_TRUE(caller_observer->ExpectOnlySingleEnumCount(
293 webrtc::kEnumCounterUsagePattern, expected_fingerprint));
294}
295
Harald Alvestrand19793842018-06-25 12:03:50 +0200296} // namespace webrtc