blob: 049d4c5665a68a75b088df3d80b1cb581c117932 [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 Alvestrand42386282018-07-12 07:56:05 +020014#include "api/jsep.h"
Harald Alvestrand19793842018-06-25 12:03:50 +020015#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"
Qingsi Wang7fc821d2018-07-12 12:54:53 -070025#include "system_wrappers/include/metrics_default.h"
Harald Alvestrand19793842018-06-25 12:03:50 +020026
27namespace webrtc {
28
29using RTCConfiguration = PeerConnectionInterface::RTCConfiguration;
30using RTCOfferAnswerOptions = PeerConnectionInterface::RTCOfferAnswerOptions;
31using ::testing::Values;
32
33static constexpr int kDefaultTimeout = 10000;
34
35int MakeUsageFingerprint(std::set<PeerConnection::UsageEvent> events) {
36 int signature = 0;
37 for (const auto it : events) {
38 signature |= static_cast<int>(it);
39 }
40 return signature;
41}
42
43class PeerConnectionFactoryForUsageHistogramTest
44 : public rtc::RefCountedObject<PeerConnectionFactory> {
45 public:
46 PeerConnectionFactoryForUsageHistogramTest()
47 : rtc::RefCountedObject<PeerConnectionFactory>(
48 rtc::Thread::Current(),
49 rtc::Thread::Current(),
50 rtc::Thread::Current(),
Karl Wiberg918f50c2018-07-05 11:40:33 +020051 absl::make_unique<cricket::FakeMediaEngine>(),
Harald Alvestrand19793842018-06-25 12:03:50 +020052 CreateCallFactory(),
53 nullptr) {}
54
55 void ActionsBeforeInitializeForTesting(PeerConnectionInterface* pc) override {
56 PeerConnection* internal_pc = static_cast<PeerConnection*>(pc);
57 if (return_histogram_very_quickly_) {
58 internal_pc->ReturnHistogramVeryQuicklyForTesting();
59 }
60 }
61
62 void ReturnHistogramVeryQuickly() { return_histogram_very_quickly_ = true; }
63
64 private:
Harald Alvestrand183e09d2018-06-28 12:04:41 +020065 bool return_histogram_very_quickly_ = false;
66};
67
68class PeerConnectionWrapperForUsageHistogramTest;
69typedef PeerConnectionWrapperForUsageHistogramTest* RawWrapperPtr;
70
71class ObserverForUsageHistogramTest : public MockPeerConnectionObserver {
72 public:
73 void OnIceCandidate(const webrtc::IceCandidateInterface* candidate) override;
74 void PrepareToExchangeCandidates(RawWrapperPtr other) {
75 candidate_target_ = other;
76 }
77
78 bool HaveDataChannel() { return last_datachannel_; }
79
80 private:
81 RawWrapperPtr candidate_target_; // Note: Not thread-safe against deletions.
Harald Alvestrand19793842018-06-25 12:03:50 +020082};
83
84class PeerConnectionWrapperForUsageHistogramTest
85 : public PeerConnectionWrapper {
86 public:
87 using PeerConnectionWrapper::PeerConnectionWrapper;
88
89 PeerConnection* GetInternalPeerConnection() {
90 auto* pci =
91 static_cast<PeerConnectionProxyWithInternal<PeerConnectionInterface>*>(
92 pc());
93 return static_cast<PeerConnection*>(pci->internal());
94 }
Harald Alvestrand183e09d2018-06-28 12:04:41 +020095
96 void PrepareToExchangeCandidates(
97 PeerConnectionWrapperForUsageHistogramTest* other) {
98 static_cast<ObserverForUsageHistogramTest*>(observer())
99 ->PrepareToExchangeCandidates(other);
100 static_cast<ObserverForUsageHistogramTest*>(other->observer())
101 ->PrepareToExchangeCandidates(this);
102 }
103
104 bool IsConnected() {
105 return pc()->ice_connection_state() ==
106 PeerConnectionInterface::kIceConnectionConnected ||
107 pc()->ice_connection_state() ==
108 PeerConnectionInterface::kIceConnectionCompleted;
109 }
110
111 bool HaveDataChannel() {
112 return static_cast<ObserverForUsageHistogramTest*>(observer())
113 ->HaveDataChannel();
114 }
Harald Alvestrand42386282018-07-12 07:56:05 +0200115 void AddOrBufferIceCandidate(const webrtc::IceCandidateInterface* candidate) {
116 if (!pc()->AddIceCandidate(candidate)) {
117 std::string sdp;
118 EXPECT_TRUE(candidate->ToString(&sdp));
119 std::unique_ptr<webrtc::IceCandidateInterface> candidate_copy(
120 CreateIceCandidate(candidate->sdp_mid(), candidate->sdp_mline_index(),
121 sdp, nullptr));
122 buffered_candidates_.push_back(std::move(candidate_copy));
123 }
124 }
125 void AddBufferedIceCandidates() {
126 for (const auto& candidate : buffered_candidates_) {
127 EXPECT_TRUE(pc()->AddIceCandidate(candidate.get()));
128 }
129 buffered_candidates_.clear();
130 }
131 bool ConnectTo(PeerConnectionWrapperForUsageHistogramTest* callee) {
132 PrepareToExchangeCandidates(callee);
133 EXPECT_TRUE(ExchangeOfferAnswerWith(callee));
134 AddBufferedIceCandidates();
135 callee->AddBufferedIceCandidates();
136 EXPECT_TRUE_WAIT(IsConnected(), kDefaultTimeout);
137 EXPECT_TRUE_WAIT(callee->IsConnected(), kDefaultTimeout);
138 return IsConnected() && callee->IsConnected();
139 }
140
141 private:
142 // Candidates that have been sent but not yet configured
143 std::vector<std::unique_ptr<webrtc::IceCandidateInterface>>
144 buffered_candidates_;
Harald Alvestrand19793842018-06-25 12:03:50 +0200145};
146
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200147void ObserverForUsageHistogramTest::OnIceCandidate(
148 const webrtc::IceCandidateInterface* candidate) {
149 if (candidate_target_) {
Harald Alvestrand42386282018-07-12 07:56:05 +0200150 this->candidate_target_->AddOrBufferIceCandidate(candidate);
151 } else {
152 FAIL() << "Early candidate detected";
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200153 }
154}
155
Harald Alvestrand19793842018-06-25 12:03:50 +0200156class PeerConnectionUsageHistogramTest : public ::testing::Test {
157 protected:
158 typedef std::unique_ptr<PeerConnectionWrapperForUsageHistogramTest>
159 WrapperPtr;
160
161 PeerConnectionUsageHistogramTest()
162 : vss_(new rtc::VirtualSocketServer()), main_(vss_.get()) {
Qingsi Wang7fc821d2018-07-12 12:54:53 -0700163 webrtc::metrics::Reset();
Harald Alvestrand19793842018-06-25 12:03:50 +0200164 }
165
166 WrapperPtr CreatePeerConnection() {
167 return CreatePeerConnection(
168 RTCConfiguration(), PeerConnectionFactoryInterface::Options(), false);
169 }
170
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200171 WrapperPtr CreatePeerConnection(const RTCConfiguration& config) {
172 return CreatePeerConnection(
173 config, PeerConnectionFactoryInterface::Options(), false);
174 }
175
Harald Alvestrand19793842018-06-25 12:03:50 +0200176 WrapperPtr CreatePeerConnectionWithImmediateReport() {
177 return CreatePeerConnection(
178 RTCConfiguration(), PeerConnectionFactoryInterface::Options(), true);
179 }
180
181 WrapperPtr CreatePeerConnection(
182 const RTCConfiguration& config,
183 const PeerConnectionFactoryInterface::Options factory_options,
184 bool immediate_report) {
185 rtc::scoped_refptr<PeerConnectionFactoryForUsageHistogramTest> pc_factory(
186 new PeerConnectionFactoryForUsageHistogramTest());
187 pc_factory->SetOptions(factory_options);
188 RTC_CHECK(pc_factory->Initialize());
189 if (immediate_report) {
190 pc_factory->ReturnHistogramVeryQuickly();
191 }
Karl Wiberg918f50c2018-07-05 11:40:33 +0200192 auto observer = absl::make_unique<ObserverForUsageHistogramTest>();
Harald Alvestrand19793842018-06-25 12:03:50 +0200193 auto pc = pc_factory->CreatePeerConnection(config, nullptr, nullptr,
194 observer.get());
195 if (!pc) {
196 return nullptr;
197 }
198
Karl Wiberg918f50c2018-07-05 11:40:33 +0200199 auto wrapper =
200 absl::make_unique<PeerConnectionWrapperForUsageHistogramTest>(
201 pc_factory, pc, std::move(observer));
Harald Alvestrand19793842018-06-25 12:03:50 +0200202 return wrapper;
203 }
204
205 std::unique_ptr<rtc::VirtualSocketServer> vss_;
206 rtc::AutoSocketServerThread main_;
207};
208
209TEST_F(PeerConnectionUsageHistogramTest, UsageFingerprintHistogramFromTimeout) {
210 auto pc = CreatePeerConnectionWithImmediateReport();
211
Harald Alvestrand19793842018-06-25 12:03:50 +0200212 int expected_fingerprint = MakeUsageFingerprint({});
Qingsi Wang7fc821d2018-07-12 12:54:53 -0700213 ASSERT_TRUE_WAIT(
214 1u == webrtc::metrics::NumSamples("WebRTC.PeerConnection.UsagePattern"),
215 kDefaultTimeout);
216 EXPECT_EQ(1, webrtc::metrics::NumEvents("WebRTC.PeerConnection.UsagePattern",
217 expected_fingerprint));
Harald Alvestrand19793842018-06-25 12:03:50 +0200218}
219
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200220#ifndef WEBRTC_ANDROID
221// These tests do not work on Android. Why is unclear.
222// https://bugs.webrtc.org/9461
223
224// Test getting the usage fingerprint for an audio/video connection.
225TEST_F(PeerConnectionUsageHistogramTest, FingerprintAudioVideo) {
226 auto caller = CreatePeerConnection();
227 auto callee = CreatePeerConnection();
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200228 caller->AddAudioTrack("audio");
229 caller->AddVideoTrack("video");
Harald Alvestrand42386282018-07-12 07:56:05 +0200230 ASSERT_TRUE(caller->ConnectTo(callee.get()));
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200231 caller->pc()->Close();
232 callee->pc()->Close();
233 int expected_fingerprint = MakeUsageFingerprint(
234 {PeerConnection::UsageEvent::AUDIO_ADDED,
235 PeerConnection::UsageEvent::VIDEO_ADDED,
236 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_CALLED,
237 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_CALLED,
238 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
239 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
240 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
241 PeerConnection::UsageEvent::CLOSE_CALLED});
Qingsi Wang7fc821d2018-07-12 12:54:53 -0700242 EXPECT_EQ(2,
243 webrtc::metrics::NumSamples("WebRTC.PeerConnection.UsagePattern"));
244 EXPECT_EQ(2, webrtc::metrics::NumEvents("WebRTC.PeerConnection.UsagePattern",
245 expected_fingerprint));
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200246}
247
248#ifdef HAVE_SCTP
249TEST_F(PeerConnectionUsageHistogramTest, FingerprintDataOnly) {
250 auto caller = CreatePeerConnection();
251 auto callee = CreatePeerConnection();
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200252 caller->CreateDataChannel("foodata");
Harald Alvestrand42386282018-07-12 07:56:05 +0200253 ASSERT_TRUE(caller->ConnectTo(callee.get()));
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200254 ASSERT_TRUE_WAIT(callee->HaveDataChannel(), kDefaultTimeout);
255 caller->pc()->Close();
256 callee->pc()->Close();
257 int expected_fingerprint = MakeUsageFingerprint(
258 {PeerConnection::UsageEvent::DATA_ADDED,
259 PeerConnection::UsageEvent::SET_LOCAL_DESCRIPTION_CALLED,
260 PeerConnection::UsageEvent::SET_REMOTE_DESCRIPTION_CALLED,
261 PeerConnection::UsageEvent::CANDIDATE_COLLECTED,
262 PeerConnection::UsageEvent::REMOTE_CANDIDATE_ADDED,
263 PeerConnection::UsageEvent::ICE_STATE_CONNECTED,
264 PeerConnection::UsageEvent::CLOSE_CALLED});
Qingsi Wang7fc821d2018-07-12 12:54:53 -0700265 EXPECT_EQ(2,
266 webrtc::metrics::NumSamples("WebRTC.PeerConnection.UsagePattern"));
267 EXPECT_EQ(2, webrtc::metrics::NumEvents("WebRTC.PeerConnection.UsagePattern",
268 expected_fingerprint));
Harald Alvestrand183e09d2018-06-28 12:04:41 +0200269}
270#endif // HAVE_SCTP
271#endif // WEBRTC_ANDROID
272
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200273TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurn) {
274 RTCConfiguration configuration;
275 PeerConnection::IceServer server;
276 server.urls = {"stun:dummy.stun.server/"};
277 configuration.servers.push_back(server);
278 server.urls = {"turn:dummy.turn.server/"};
279 server.username = "username";
280 server.password = "password";
281 configuration.servers.push_back(server);
282 auto caller = CreatePeerConnection(configuration);
283 ASSERT_TRUE(caller);
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200284 caller->pc()->Close();
285 int expected_fingerprint =
286 MakeUsageFingerprint({PeerConnection::UsageEvent::STUN_SERVER_ADDED,
287 PeerConnection::UsageEvent::TURN_SERVER_ADDED,
288 PeerConnection::UsageEvent::CLOSE_CALLED});
Qingsi Wang7fc821d2018-07-12 12:54:53 -0700289 EXPECT_EQ(1,
290 webrtc::metrics::NumSamples("WebRTC.PeerConnection.UsagePattern"));
291 EXPECT_EQ(1, webrtc::metrics::NumEvents("WebRTC.PeerConnection.UsagePattern",
292 expected_fingerprint));
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200293}
294
295TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurnInReconfiguration) {
296 RTCConfiguration configuration;
297 PeerConnection::IceServer server;
298 server.urls = {"stun:dummy.stun.server/"};
299 configuration.servers.push_back(server);
300 server.urls = {"turn:dummy.turn.server/"};
301 server.username = "username";
302 server.password = "password";
303 configuration.servers.push_back(server);
304 auto caller = CreatePeerConnection();
305 ASSERT_TRUE(caller);
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200306 RTCError error;
307 caller->pc()->SetConfiguration(configuration, &error);
308 ASSERT_TRUE(error.ok());
309 caller->pc()->Close();
310 int expected_fingerprint =
311 MakeUsageFingerprint({PeerConnection::UsageEvent::STUN_SERVER_ADDED,
312 PeerConnection::UsageEvent::TURN_SERVER_ADDED,
313 PeerConnection::UsageEvent::CLOSE_CALLED});
Qingsi Wang7fc821d2018-07-12 12:54:53 -0700314 EXPECT_EQ(1,
315 webrtc::metrics::NumSamples("WebRTC.PeerConnection.UsagePattern"));
316 EXPECT_EQ(1, webrtc::metrics::NumEvents("WebRTC.PeerConnection.UsagePattern",
317 expected_fingerprint));
Harald Alvestrandb2a74782018-06-28 13:54:07 +0200318}
319
Harald Alvestrand19793842018-06-25 12:03:50 +0200320} // namespace webrtc