xref: /aosp_15_r20/external/webrtc/pc/peer_connection_histogram_unittest.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright 2017 The WebRTC project authors. All Rights Reserved.
3*d9f75844SAndroid Build Coastguard Worker  *
4*d9f75844SAndroid Build Coastguard Worker  *  Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker  *  that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker  *  tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker  *  in the file PATENTS.  All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker  *  be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker  */
10*d9f75844SAndroid Build Coastguard Worker 
11*d9f75844SAndroid Build Coastguard Worker #include <memory>
12*d9f75844SAndroid Build Coastguard Worker #include <set>
13*d9f75844SAndroid Build Coastguard Worker #include <string>
14*d9f75844SAndroid Build Coastguard Worker #include <utility>
15*d9f75844SAndroid Build Coastguard Worker #include <vector>
16*d9f75844SAndroid Build Coastguard Worker 
17*d9f75844SAndroid Build Coastguard Worker #include "absl/types/optional.h"
18*d9f75844SAndroid Build Coastguard Worker #include "api/async_resolver_factory.h"
19*d9f75844SAndroid Build Coastguard Worker #include "api/call/call_factory_interface.h"
20*d9f75844SAndroid Build Coastguard Worker #include "api/jsep.h"
21*d9f75844SAndroid Build Coastguard Worker #include "api/jsep_session_description.h"
22*d9f75844SAndroid Build Coastguard Worker #include "api/peer_connection_interface.h"
23*d9f75844SAndroid Build Coastguard Worker #include "api/rtc_error.h"
24*d9f75844SAndroid Build Coastguard Worker #include "api/scoped_refptr.h"
25*d9f75844SAndroid Build Coastguard Worker #include "api/task_queue/default_task_queue_factory.h"
26*d9f75844SAndroid Build Coastguard Worker #include "api/task_queue/task_queue_factory.h"
27*d9f75844SAndroid Build Coastguard Worker #include "media/base/fake_media_engine.h"
28*d9f75844SAndroid Build Coastguard Worker #include "media/base/media_engine.h"
29*d9f75844SAndroid Build Coastguard Worker #include "p2p/base/mock_async_resolver.h"
30*d9f75844SAndroid Build Coastguard Worker #include "p2p/base/port_allocator.h"
31*d9f75844SAndroid Build Coastguard Worker #include "p2p/client/basic_port_allocator.h"
32*d9f75844SAndroid Build Coastguard Worker #include "pc/peer_connection.h"
33*d9f75844SAndroid Build Coastguard Worker #include "pc/peer_connection_factory.h"
34*d9f75844SAndroid Build Coastguard Worker #include "pc/peer_connection_proxy.h"
35*d9f75844SAndroid Build Coastguard Worker #include "pc/peer_connection_wrapper.h"
36*d9f75844SAndroid Build Coastguard Worker #include "pc/sdp_utils.h"
37*d9f75844SAndroid Build Coastguard Worker #include "pc/test/mock_peer_connection_observers.h"
38*d9f75844SAndroid Build Coastguard Worker #include "pc/usage_pattern.h"
39*d9f75844SAndroid Build Coastguard Worker #include "pc/webrtc_sdp.h"
40*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/arraysize.h"
41*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/checks.h"
42*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/fake_mdns_responder.h"
43*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/fake_network.h"
44*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/gunit.h"
45*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/mdns_responder_interface.h"
46*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/socket_address.h"
47*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/thread.h"
48*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/virtual_socket_server.h"
49*d9f75844SAndroid Build Coastguard Worker #include "system_wrappers/include/metrics.h"
50*d9f75844SAndroid Build Coastguard Worker #include "test/gmock.h"
51*d9f75844SAndroid Build Coastguard Worker #include "test/gtest.h"
52*d9f75844SAndroid Build Coastguard Worker 
53*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
54*d9f75844SAndroid Build Coastguard Worker 
55*d9f75844SAndroid Build Coastguard Worker using RTCConfiguration = PeerConnectionInterface::RTCConfiguration;
56*d9f75844SAndroid Build Coastguard Worker using RTCOfferAnswerOptions = PeerConnectionInterface::RTCOfferAnswerOptions;
57*d9f75844SAndroid Build Coastguard Worker using ::testing::NiceMock;
58*d9f75844SAndroid Build Coastguard Worker using ::testing::Values;
59*d9f75844SAndroid Build Coastguard Worker 
60*d9f75844SAndroid Build Coastguard Worker static const char kUsagePatternMetric[] = "WebRTC.PeerConnection.UsagePattern";
61*d9f75844SAndroid Build Coastguard Worker static constexpr int kDefaultTimeout = 10000;
62*d9f75844SAndroid Build Coastguard Worker static const rtc::SocketAddress kLocalAddrs[2] = {
63*d9f75844SAndroid Build Coastguard Worker     rtc::SocketAddress("1.1.1.1", 0), rtc::SocketAddress("2.2.2.2", 0)};
64*d9f75844SAndroid Build Coastguard Worker static const rtc::SocketAddress kPrivateLocalAddress("10.1.1.1", 0);
65*d9f75844SAndroid Build Coastguard Worker static const rtc::SocketAddress kPrivateIpv6LocalAddress("fd12:3456:789a:1::1",
66*d9f75844SAndroid Build Coastguard Worker                                                          0);
67*d9f75844SAndroid Build Coastguard Worker 
MakeUsageFingerprint(std::set<UsageEvent> events)68*d9f75844SAndroid Build Coastguard Worker int MakeUsageFingerprint(std::set<UsageEvent> events) {
69*d9f75844SAndroid Build Coastguard Worker   int signature = 0;
70*d9f75844SAndroid Build Coastguard Worker   for (const auto it : events) {
71*d9f75844SAndroid Build Coastguard Worker     signature |= static_cast<int>(it);
72*d9f75844SAndroid Build Coastguard Worker   }
73*d9f75844SAndroid Build Coastguard Worker   return signature;
74*d9f75844SAndroid Build Coastguard Worker }
75*d9f75844SAndroid Build Coastguard Worker 
76*d9f75844SAndroid Build Coastguard Worker class PeerConnectionFactoryForUsageHistogramTest
77*d9f75844SAndroid Build Coastguard Worker     : public PeerConnectionFactory {
78*d9f75844SAndroid Build Coastguard Worker  public:
PeerConnectionFactoryForUsageHistogramTest()79*d9f75844SAndroid Build Coastguard Worker   PeerConnectionFactoryForUsageHistogramTest()
80*d9f75844SAndroid Build Coastguard Worker       : PeerConnectionFactory([] {
81*d9f75844SAndroid Build Coastguard Worker           PeerConnectionFactoryDependencies dependencies;
82*d9f75844SAndroid Build Coastguard Worker           dependencies.network_thread = rtc::Thread::Current();
83*d9f75844SAndroid Build Coastguard Worker           dependencies.worker_thread = rtc::Thread::Current();
84*d9f75844SAndroid Build Coastguard Worker           dependencies.signaling_thread = rtc::Thread::Current();
85*d9f75844SAndroid Build Coastguard Worker           dependencies.task_queue_factory = CreateDefaultTaskQueueFactory();
86*d9f75844SAndroid Build Coastguard Worker           dependencies.media_engine =
87*d9f75844SAndroid Build Coastguard Worker               std::make_unique<cricket::FakeMediaEngine>();
88*d9f75844SAndroid Build Coastguard Worker           dependencies.call_factory = CreateCallFactory();
89*d9f75844SAndroid Build Coastguard Worker           return dependencies;
90*d9f75844SAndroid Build Coastguard Worker         }()) {}
91*d9f75844SAndroid Build Coastguard Worker };
92*d9f75844SAndroid Build Coastguard Worker 
93*d9f75844SAndroid Build Coastguard Worker class PeerConnectionWrapperForUsageHistogramTest;
94*d9f75844SAndroid Build Coastguard Worker 
95*d9f75844SAndroid Build Coastguard Worker typedef PeerConnectionWrapperForUsageHistogramTest* RawWrapperPtr;
96*d9f75844SAndroid Build Coastguard Worker 
97*d9f75844SAndroid Build Coastguard Worker class ObserverForUsageHistogramTest : public MockPeerConnectionObserver {
98*d9f75844SAndroid Build Coastguard Worker  public:
99*d9f75844SAndroid Build Coastguard Worker   void OnIceCandidate(const webrtc::IceCandidateInterface* candidate) override;
100*d9f75844SAndroid Build Coastguard Worker 
OnInterestingUsage(int usage_pattern)101*d9f75844SAndroid Build Coastguard Worker   void OnInterestingUsage(int usage_pattern) override {
102*d9f75844SAndroid Build Coastguard Worker     interesting_usage_detected_ = usage_pattern;
103*d9f75844SAndroid Build Coastguard Worker   }
104*d9f75844SAndroid Build Coastguard Worker 
PrepareToExchangeCandidates(RawWrapperPtr other)105*d9f75844SAndroid Build Coastguard Worker   void PrepareToExchangeCandidates(RawWrapperPtr other) {
106*d9f75844SAndroid Build Coastguard Worker     candidate_target_ = other;
107*d9f75844SAndroid Build Coastguard Worker   }
108*d9f75844SAndroid Build Coastguard Worker 
HaveDataChannel()109*d9f75844SAndroid Build Coastguard Worker   bool HaveDataChannel() { return last_datachannel_ != nullptr; }
110*d9f75844SAndroid Build Coastguard Worker 
interesting_usage_detected()111*d9f75844SAndroid Build Coastguard Worker   absl::optional<int> interesting_usage_detected() {
112*d9f75844SAndroid Build Coastguard Worker     return interesting_usage_detected_;
113*d9f75844SAndroid Build Coastguard Worker   }
114*d9f75844SAndroid Build Coastguard Worker 
ClearInterestingUsageDetector()115*d9f75844SAndroid Build Coastguard Worker   void ClearInterestingUsageDetector() {
116*d9f75844SAndroid Build Coastguard Worker     interesting_usage_detected_ = absl::optional<int>();
117*d9f75844SAndroid Build Coastguard Worker   }
118*d9f75844SAndroid Build Coastguard Worker 
candidate_gathered() const119*d9f75844SAndroid Build Coastguard Worker   bool candidate_gathered() const { return candidate_gathered_; }
120*d9f75844SAndroid Build Coastguard Worker 
121*d9f75844SAndroid Build Coastguard Worker  private:
122*d9f75844SAndroid Build Coastguard Worker   absl::optional<int> interesting_usage_detected_;
123*d9f75844SAndroid Build Coastguard Worker   bool candidate_gathered_ = false;
124*d9f75844SAndroid Build Coastguard Worker   RawWrapperPtr candidate_target_;  // Note: Not thread-safe against deletions.
125*d9f75844SAndroid Build Coastguard Worker };
126*d9f75844SAndroid Build Coastguard Worker 
127*d9f75844SAndroid Build Coastguard Worker class PeerConnectionWrapperForUsageHistogramTest
128*d9f75844SAndroid Build Coastguard Worker     : public PeerConnectionWrapper {
129*d9f75844SAndroid Build Coastguard Worker  public:
130*d9f75844SAndroid Build Coastguard Worker   using PeerConnectionWrapper::PeerConnectionWrapper;
131*d9f75844SAndroid Build Coastguard Worker 
GetInternalPeerConnection()132*d9f75844SAndroid Build Coastguard Worker   PeerConnection* GetInternalPeerConnection() {
133*d9f75844SAndroid Build Coastguard Worker     auto* pci =
134*d9f75844SAndroid Build Coastguard Worker         static_cast<PeerConnectionProxyWithInternal<PeerConnectionInterface>*>(
135*d9f75844SAndroid Build Coastguard Worker             pc());
136*d9f75844SAndroid Build Coastguard Worker     return static_cast<PeerConnection*>(pci->internal());
137*d9f75844SAndroid Build Coastguard Worker   }
138*d9f75844SAndroid Build Coastguard Worker 
139*d9f75844SAndroid Build Coastguard Worker   // Override with different return type
observer()140*d9f75844SAndroid Build Coastguard Worker   ObserverForUsageHistogramTest* observer() {
141*d9f75844SAndroid Build Coastguard Worker     return static_cast<ObserverForUsageHistogramTest*>(
142*d9f75844SAndroid Build Coastguard Worker         PeerConnectionWrapper::observer());
143*d9f75844SAndroid Build Coastguard Worker   }
144*d9f75844SAndroid Build Coastguard Worker 
PrepareToExchangeCandidates(PeerConnectionWrapperForUsageHistogramTest * other)145*d9f75844SAndroid Build Coastguard Worker   void PrepareToExchangeCandidates(
146*d9f75844SAndroid Build Coastguard Worker       PeerConnectionWrapperForUsageHistogramTest* other) {
147*d9f75844SAndroid Build Coastguard Worker     observer()->PrepareToExchangeCandidates(other);
148*d9f75844SAndroid Build Coastguard Worker     other->observer()->PrepareToExchangeCandidates(this);
149*d9f75844SAndroid Build Coastguard Worker   }
150*d9f75844SAndroid Build Coastguard Worker 
IsConnected()151*d9f75844SAndroid Build Coastguard Worker   bool IsConnected() {
152*d9f75844SAndroid Build Coastguard Worker     return pc()->ice_connection_state() ==
153*d9f75844SAndroid Build Coastguard Worker                PeerConnectionInterface::kIceConnectionConnected ||
154*d9f75844SAndroid Build Coastguard Worker            pc()->ice_connection_state() ==
155*d9f75844SAndroid Build Coastguard Worker                PeerConnectionInterface::kIceConnectionCompleted;
156*d9f75844SAndroid Build Coastguard Worker   }
157*d9f75844SAndroid Build Coastguard Worker 
HaveDataChannel()158*d9f75844SAndroid Build Coastguard Worker   bool HaveDataChannel() {
159*d9f75844SAndroid Build Coastguard Worker     return static_cast<ObserverForUsageHistogramTest*>(observer())
160*d9f75844SAndroid Build Coastguard Worker         ->HaveDataChannel();
161*d9f75844SAndroid Build Coastguard Worker   }
BufferIceCandidate(const webrtc::IceCandidateInterface * candidate)162*d9f75844SAndroid Build Coastguard Worker   void BufferIceCandidate(const webrtc::IceCandidateInterface* candidate) {
163*d9f75844SAndroid Build Coastguard Worker     std::string sdp;
164*d9f75844SAndroid Build Coastguard Worker     EXPECT_TRUE(candidate->ToString(&sdp));
165*d9f75844SAndroid Build Coastguard Worker     std::unique_ptr<webrtc::IceCandidateInterface> candidate_copy(
166*d9f75844SAndroid Build Coastguard Worker         CreateIceCandidate(candidate->sdp_mid(), candidate->sdp_mline_index(),
167*d9f75844SAndroid Build Coastguard Worker                            sdp, nullptr));
168*d9f75844SAndroid Build Coastguard Worker     buffered_candidates_.push_back(std::move(candidate_copy));
169*d9f75844SAndroid Build Coastguard Worker   }
170*d9f75844SAndroid Build Coastguard Worker 
AddBufferedIceCandidates()171*d9f75844SAndroid Build Coastguard Worker   void AddBufferedIceCandidates() {
172*d9f75844SAndroid Build Coastguard Worker     for (const auto& candidate : buffered_candidates_) {
173*d9f75844SAndroid Build Coastguard Worker       EXPECT_TRUE(pc()->AddIceCandidate(candidate.get()));
174*d9f75844SAndroid Build Coastguard Worker     }
175*d9f75844SAndroid Build Coastguard Worker     buffered_candidates_.clear();
176*d9f75844SAndroid Build Coastguard Worker   }
177*d9f75844SAndroid Build Coastguard Worker 
178*d9f75844SAndroid Build Coastguard Worker   // This method performs the following actions in sequence:
179*d9f75844SAndroid Build Coastguard Worker   // 1. Exchange Offer and Answer.
180*d9f75844SAndroid Build Coastguard Worker   // 2. Exchange ICE candidates after both caller and callee complete
181*d9f75844SAndroid Build Coastguard Worker   // gathering.
182*d9f75844SAndroid Build Coastguard Worker   // 3. Wait for ICE to connect.
183*d9f75844SAndroid Build Coastguard Worker   //
184*d9f75844SAndroid Build Coastguard Worker   // This guarantees a deterministic sequence of events and also rules out the
185*d9f75844SAndroid Build Coastguard Worker   // occurrence of prflx candidates if the offer/answer signaling and the
186*d9f75844SAndroid Build Coastguard Worker   // candidate trickling race in order. In case prflx candidates need to be
187*d9f75844SAndroid Build Coastguard Worker   // simulated, see the approach used by tests below for that.
ConnectTo(PeerConnectionWrapperForUsageHistogramTest * callee)188*d9f75844SAndroid Build Coastguard Worker   bool ConnectTo(PeerConnectionWrapperForUsageHistogramTest* callee) {
189*d9f75844SAndroid Build Coastguard Worker     PrepareToExchangeCandidates(callee);
190*d9f75844SAndroid Build Coastguard Worker     if (!ExchangeOfferAnswerWith(callee)) {
191*d9f75844SAndroid Build Coastguard Worker       return false;
192*d9f75844SAndroid Build Coastguard Worker     }
193*d9f75844SAndroid Build Coastguard Worker     // Wait until the gathering completes before we signal the candidate.
194*d9f75844SAndroid Build Coastguard Worker     WAIT(observer()->ice_gathering_complete_, kDefaultTimeout);
195*d9f75844SAndroid Build Coastguard Worker     WAIT(callee->observer()->ice_gathering_complete_, kDefaultTimeout);
196*d9f75844SAndroid Build Coastguard Worker     AddBufferedIceCandidates();
197*d9f75844SAndroid Build Coastguard Worker     callee->AddBufferedIceCandidates();
198*d9f75844SAndroid Build Coastguard Worker     WAIT(IsConnected(), kDefaultTimeout);
199*d9f75844SAndroid Build Coastguard Worker     WAIT(callee->IsConnected(), kDefaultTimeout);
200*d9f75844SAndroid Build Coastguard Worker     return IsConnected() && callee->IsConnected();
201*d9f75844SAndroid Build Coastguard Worker   }
202*d9f75844SAndroid Build Coastguard Worker 
GenerateOfferAndCollectCandidates()203*d9f75844SAndroid Build Coastguard Worker   bool GenerateOfferAndCollectCandidates() {
204*d9f75844SAndroid Build Coastguard Worker     auto offer = CreateOffer(RTCOfferAnswerOptions());
205*d9f75844SAndroid Build Coastguard Worker     if (!offer) {
206*d9f75844SAndroid Build Coastguard Worker       return false;
207*d9f75844SAndroid Build Coastguard Worker     }
208*d9f75844SAndroid Build Coastguard Worker     bool set_local_offer =
209*d9f75844SAndroid Build Coastguard Worker         SetLocalDescription(CloneSessionDescription(offer.get()));
210*d9f75844SAndroid Build Coastguard Worker     EXPECT_TRUE(set_local_offer);
211*d9f75844SAndroid Build Coastguard Worker     if (!set_local_offer) {
212*d9f75844SAndroid Build Coastguard Worker       return false;
213*d9f75844SAndroid Build Coastguard Worker     }
214*d9f75844SAndroid Build Coastguard Worker     EXPECT_TRUE_WAIT(observer()->ice_gathering_complete_, kDefaultTimeout);
215*d9f75844SAndroid Build Coastguard Worker     return true;
216*d9f75844SAndroid Build Coastguard Worker   }
217*d9f75844SAndroid Build Coastguard Worker 
ice_gathering_state()218*d9f75844SAndroid Build Coastguard Worker   webrtc::PeerConnectionInterface::IceGatheringState ice_gathering_state() {
219*d9f75844SAndroid Build Coastguard Worker     return pc()->ice_gathering_state();
220*d9f75844SAndroid Build Coastguard Worker   }
221*d9f75844SAndroid Build Coastguard Worker 
222*d9f75844SAndroid Build Coastguard Worker  private:
223*d9f75844SAndroid Build Coastguard Worker   // Candidates that have been sent but not yet configured
224*d9f75844SAndroid Build Coastguard Worker   std::vector<std::unique_ptr<webrtc::IceCandidateInterface>>
225*d9f75844SAndroid Build Coastguard Worker       buffered_candidates_;
226*d9f75844SAndroid Build Coastguard Worker };
227*d9f75844SAndroid Build Coastguard Worker 
228*d9f75844SAndroid Build Coastguard Worker // Buffers candidates until we add them via AddBufferedIceCandidates.
OnIceCandidate(const webrtc::IceCandidateInterface * candidate)229*d9f75844SAndroid Build Coastguard Worker void ObserverForUsageHistogramTest::OnIceCandidate(
230*d9f75844SAndroid Build Coastguard Worker     const webrtc::IceCandidateInterface* candidate) {
231*d9f75844SAndroid Build Coastguard Worker   // If target is not set, ignore. This happens in one-ended unit tests.
232*d9f75844SAndroid Build Coastguard Worker   if (candidate_target_) {
233*d9f75844SAndroid Build Coastguard Worker     this->candidate_target_->BufferIceCandidate(candidate);
234*d9f75844SAndroid Build Coastguard Worker   }
235*d9f75844SAndroid Build Coastguard Worker   candidate_gathered_ = true;
236*d9f75844SAndroid Build Coastguard Worker }
237*d9f75844SAndroid Build Coastguard Worker 
238*d9f75844SAndroid Build Coastguard Worker class PeerConnectionUsageHistogramTest : public ::testing::Test {
239*d9f75844SAndroid Build Coastguard Worker  protected:
240*d9f75844SAndroid Build Coastguard Worker   typedef std::unique_ptr<PeerConnectionWrapperForUsageHistogramTest>
241*d9f75844SAndroid Build Coastguard Worker       WrapperPtr;
242*d9f75844SAndroid Build Coastguard Worker 
PeerConnectionUsageHistogramTest()243*d9f75844SAndroid Build Coastguard Worker   PeerConnectionUsageHistogramTest()
244*d9f75844SAndroid Build Coastguard Worker       : vss_(new rtc::VirtualSocketServer()), main_(vss_.get()) {
245*d9f75844SAndroid Build Coastguard Worker     webrtc::metrics::Reset();
246*d9f75844SAndroid Build Coastguard Worker   }
247*d9f75844SAndroid Build Coastguard Worker 
CreatePeerConnection()248*d9f75844SAndroid Build Coastguard Worker   WrapperPtr CreatePeerConnection() {
249*d9f75844SAndroid Build Coastguard Worker     RTCConfiguration config;
250*d9f75844SAndroid Build Coastguard Worker     config.sdp_semantics = webrtc::SdpSemantics::kUnifiedPlan;
251*d9f75844SAndroid Build Coastguard Worker     return CreatePeerConnection(
252*d9f75844SAndroid Build Coastguard Worker         config, PeerConnectionFactoryInterface::Options(), nullptr);
253*d9f75844SAndroid Build Coastguard Worker   }
254*d9f75844SAndroid Build Coastguard Worker 
CreatePeerConnection(const RTCConfiguration & config)255*d9f75844SAndroid Build Coastguard Worker   WrapperPtr CreatePeerConnection(const RTCConfiguration& config) {
256*d9f75844SAndroid Build Coastguard Worker     return CreatePeerConnection(
257*d9f75844SAndroid Build Coastguard Worker         config, PeerConnectionFactoryInterface::Options(), nullptr);
258*d9f75844SAndroid Build Coastguard Worker   }
259*d9f75844SAndroid Build Coastguard Worker 
CreatePeerConnectionWithMdns(const RTCConfiguration & config)260*d9f75844SAndroid Build Coastguard Worker   WrapperPtr CreatePeerConnectionWithMdns(const RTCConfiguration& config) {
261*d9f75844SAndroid Build Coastguard Worker     auto resolver_factory =
262*d9f75844SAndroid Build Coastguard Worker         std::make_unique<NiceMock<webrtc::MockAsyncResolverFactory>>();
263*d9f75844SAndroid Build Coastguard Worker 
264*d9f75844SAndroid Build Coastguard Worker     webrtc::PeerConnectionDependencies deps(nullptr /* observer_in */);
265*d9f75844SAndroid Build Coastguard Worker 
266*d9f75844SAndroid Build Coastguard Worker     auto fake_network = NewFakeNetwork();
267*d9f75844SAndroid Build Coastguard Worker     fake_network->set_mdns_responder(
268*d9f75844SAndroid Build Coastguard Worker         std::make_unique<webrtc::FakeMdnsResponder>(rtc::Thread::Current()));
269*d9f75844SAndroid Build Coastguard Worker     fake_network->AddInterface(NextLocalAddress());
270*d9f75844SAndroid Build Coastguard Worker 
271*d9f75844SAndroid Build Coastguard Worker     std::unique_ptr<cricket::BasicPortAllocator> port_allocator(
272*d9f75844SAndroid Build Coastguard Worker         new cricket::BasicPortAllocator(
273*d9f75844SAndroid Build Coastguard Worker             fake_network,
274*d9f75844SAndroid Build Coastguard Worker             std::make_unique<rtc::BasicPacketSocketFactory>(vss_.get())));
275*d9f75844SAndroid Build Coastguard Worker 
276*d9f75844SAndroid Build Coastguard Worker     deps.async_resolver_factory = std::move(resolver_factory);
277*d9f75844SAndroid Build Coastguard Worker     deps.allocator = std::move(port_allocator);
278*d9f75844SAndroid Build Coastguard Worker 
279*d9f75844SAndroid Build Coastguard Worker     return CreatePeerConnection(
280*d9f75844SAndroid Build Coastguard Worker         config, PeerConnectionFactoryInterface::Options(), std::move(deps));
281*d9f75844SAndroid Build Coastguard Worker   }
282*d9f75844SAndroid Build Coastguard Worker 
CreatePeerConnectionWithImmediateReport()283*d9f75844SAndroid Build Coastguard Worker   WrapperPtr CreatePeerConnectionWithImmediateReport() {
284*d9f75844SAndroid Build Coastguard Worker     RTCConfiguration configuration;
285*d9f75844SAndroid Build Coastguard Worker     configuration.sdp_semantics = webrtc::SdpSemantics::kUnifiedPlan;
286*d9f75844SAndroid Build Coastguard Worker     configuration.report_usage_pattern_delay_ms = 0;
287*d9f75844SAndroid Build Coastguard Worker     return CreatePeerConnection(
288*d9f75844SAndroid Build Coastguard Worker         configuration, PeerConnectionFactoryInterface::Options(), nullptr);
289*d9f75844SAndroid Build Coastguard Worker   }
290*d9f75844SAndroid Build Coastguard Worker 
CreatePeerConnectionWithPrivateLocalAddresses()291*d9f75844SAndroid Build Coastguard Worker   WrapperPtr CreatePeerConnectionWithPrivateLocalAddresses() {
292*d9f75844SAndroid Build Coastguard Worker     auto* fake_network = NewFakeNetwork();
293*d9f75844SAndroid Build Coastguard Worker     fake_network->AddInterface(NextLocalAddress());
294*d9f75844SAndroid Build Coastguard Worker     fake_network->AddInterface(kPrivateLocalAddress);
295*d9f75844SAndroid Build Coastguard Worker 
296*d9f75844SAndroid Build Coastguard Worker     auto port_allocator = std::make_unique<cricket::BasicPortAllocator>(
297*d9f75844SAndroid Build Coastguard Worker         fake_network,
298*d9f75844SAndroid Build Coastguard Worker         std::make_unique<rtc::BasicPacketSocketFactory>(vss_.get()));
299*d9f75844SAndroid Build Coastguard Worker     RTCConfiguration config;
300*d9f75844SAndroid Build Coastguard Worker     config.sdp_semantics = SdpSemantics::kUnifiedPlan;
301*d9f75844SAndroid Build Coastguard Worker     return CreatePeerConnection(config,
302*d9f75844SAndroid Build Coastguard Worker                                 PeerConnectionFactoryInterface::Options(),
303*d9f75844SAndroid Build Coastguard Worker                                 std::move(port_allocator));
304*d9f75844SAndroid Build Coastguard Worker   }
305*d9f75844SAndroid Build Coastguard Worker 
CreatePeerConnectionWithPrivateIpv6LocalAddresses()306*d9f75844SAndroid Build Coastguard Worker   WrapperPtr CreatePeerConnectionWithPrivateIpv6LocalAddresses() {
307*d9f75844SAndroid Build Coastguard Worker     auto* fake_network = NewFakeNetwork();
308*d9f75844SAndroid Build Coastguard Worker     fake_network->AddInterface(NextLocalAddress());
309*d9f75844SAndroid Build Coastguard Worker     fake_network->AddInterface(kPrivateIpv6LocalAddress);
310*d9f75844SAndroid Build Coastguard Worker 
311*d9f75844SAndroid Build Coastguard Worker     auto port_allocator = std::make_unique<cricket::BasicPortAllocator>(
312*d9f75844SAndroid Build Coastguard Worker         fake_network,
313*d9f75844SAndroid Build Coastguard Worker         std::make_unique<rtc::BasicPacketSocketFactory>(vss_.get()));
314*d9f75844SAndroid Build Coastguard Worker 
315*d9f75844SAndroid Build Coastguard Worker     RTCConfiguration config;
316*d9f75844SAndroid Build Coastguard Worker     config.sdp_semantics = SdpSemantics::kUnifiedPlan;
317*d9f75844SAndroid Build Coastguard Worker     return CreatePeerConnection(config,
318*d9f75844SAndroid Build Coastguard Worker                                 PeerConnectionFactoryInterface::Options(),
319*d9f75844SAndroid Build Coastguard Worker                                 std::move(port_allocator));
320*d9f75844SAndroid Build Coastguard Worker   }
321*d9f75844SAndroid Build Coastguard Worker 
CreatePeerConnection(const RTCConfiguration & config,const PeerConnectionFactoryInterface::Options factory_options,std::unique_ptr<cricket::PortAllocator> allocator)322*d9f75844SAndroid Build Coastguard Worker   WrapperPtr CreatePeerConnection(
323*d9f75844SAndroid Build Coastguard Worker       const RTCConfiguration& config,
324*d9f75844SAndroid Build Coastguard Worker       const PeerConnectionFactoryInterface::Options factory_options,
325*d9f75844SAndroid Build Coastguard Worker       std::unique_ptr<cricket::PortAllocator> allocator) {
326*d9f75844SAndroid Build Coastguard Worker     PeerConnectionDependencies deps(nullptr);
327*d9f75844SAndroid Build Coastguard Worker     deps.allocator = std::move(allocator);
328*d9f75844SAndroid Build Coastguard Worker 
329*d9f75844SAndroid Build Coastguard Worker     return CreatePeerConnection(config, factory_options, std::move(deps));
330*d9f75844SAndroid Build Coastguard Worker   }
331*d9f75844SAndroid Build Coastguard Worker 
CreatePeerConnection(const RTCConfiguration & config,const PeerConnectionFactoryInterface::Options factory_options,PeerConnectionDependencies deps)332*d9f75844SAndroid Build Coastguard Worker   WrapperPtr CreatePeerConnection(
333*d9f75844SAndroid Build Coastguard Worker       const RTCConfiguration& config,
334*d9f75844SAndroid Build Coastguard Worker       const PeerConnectionFactoryInterface::Options factory_options,
335*d9f75844SAndroid Build Coastguard Worker       PeerConnectionDependencies deps) {
336*d9f75844SAndroid Build Coastguard Worker     auto pc_factory =
337*d9f75844SAndroid Build Coastguard Worker         rtc::make_ref_counted<PeerConnectionFactoryForUsageHistogramTest>();
338*d9f75844SAndroid Build Coastguard Worker     pc_factory->SetOptions(factory_options);
339*d9f75844SAndroid Build Coastguard Worker 
340*d9f75844SAndroid Build Coastguard Worker     // If no allocator is provided, one will be created using a network manager
341*d9f75844SAndroid Build Coastguard Worker     // that uses the host network. This doesn't work on all trybots.
342*d9f75844SAndroid Build Coastguard Worker     if (!deps.allocator) {
343*d9f75844SAndroid Build Coastguard Worker       auto fake_network = NewFakeNetwork();
344*d9f75844SAndroid Build Coastguard Worker       fake_network->AddInterface(NextLocalAddress());
345*d9f75844SAndroid Build Coastguard Worker       deps.allocator = std::make_unique<cricket::BasicPortAllocator>(
346*d9f75844SAndroid Build Coastguard Worker           fake_network,
347*d9f75844SAndroid Build Coastguard Worker           std::make_unique<rtc::BasicPacketSocketFactory>(vss_.get()));
348*d9f75844SAndroid Build Coastguard Worker     }
349*d9f75844SAndroid Build Coastguard Worker 
350*d9f75844SAndroid Build Coastguard Worker     auto observer = std::make_unique<ObserverForUsageHistogramTest>();
351*d9f75844SAndroid Build Coastguard Worker     deps.observer = observer.get();
352*d9f75844SAndroid Build Coastguard Worker 
353*d9f75844SAndroid Build Coastguard Worker     auto result =
354*d9f75844SAndroid Build Coastguard Worker         pc_factory->CreatePeerConnectionOrError(config, std::move(deps));
355*d9f75844SAndroid Build Coastguard Worker     if (!result.ok()) {
356*d9f75844SAndroid Build Coastguard Worker       return nullptr;
357*d9f75844SAndroid Build Coastguard Worker     }
358*d9f75844SAndroid Build Coastguard Worker 
359*d9f75844SAndroid Build Coastguard Worker     observer->SetPeerConnectionInterface(result.value().get());
360*d9f75844SAndroid Build Coastguard Worker     auto wrapper = std::make_unique<PeerConnectionWrapperForUsageHistogramTest>(
361*d9f75844SAndroid Build Coastguard Worker         pc_factory, result.MoveValue(), std::move(observer));
362*d9f75844SAndroid Build Coastguard Worker     return wrapper;
363*d9f75844SAndroid Build Coastguard Worker   }
364*d9f75844SAndroid Build Coastguard Worker 
ObservedFingerprint()365*d9f75844SAndroid Build Coastguard Worker   int ObservedFingerprint() {
366*d9f75844SAndroid Build Coastguard Worker     // This works correctly only if there is only one sample value
367*d9f75844SAndroid Build Coastguard Worker     // that has been counted.
368*d9f75844SAndroid Build Coastguard Worker     // Returns -1 for "not found".
369*d9f75844SAndroid Build Coastguard Worker     return webrtc::metrics::MinSample(kUsagePatternMetric);
370*d9f75844SAndroid Build Coastguard Worker   }
371*d9f75844SAndroid Build Coastguard Worker 
372*d9f75844SAndroid Build Coastguard Worker   // The PeerConnection's port allocator is tied to the PeerConnection's
373*d9f75844SAndroid Build Coastguard Worker   // lifetime and expects the underlying NetworkManager to outlive it.  That
374*d9f75844SAndroid Build Coastguard Worker   // prevents us from having the PeerConnectionWrapper own the fake network.
375*d9f75844SAndroid Build Coastguard Worker   // Therefore, the test fixture will own all the fake networks even though
376*d9f75844SAndroid Build Coastguard Worker   // tests should access the fake network through the PeerConnectionWrapper.
NewFakeNetwork()377*d9f75844SAndroid Build Coastguard Worker   rtc::FakeNetworkManager* NewFakeNetwork() {
378*d9f75844SAndroid Build Coastguard Worker     fake_networks_.emplace_back(std::make_unique<rtc::FakeNetworkManager>());
379*d9f75844SAndroid Build Coastguard Worker     return fake_networks_.back().get();
380*d9f75844SAndroid Build Coastguard Worker   }
381*d9f75844SAndroid Build Coastguard Worker 
NextLocalAddress()382*d9f75844SAndroid Build Coastguard Worker   rtc::SocketAddress NextLocalAddress() {
383*d9f75844SAndroid Build Coastguard Worker     RTC_DCHECK(next_local_address_ < (int)arraysize(kLocalAddrs));
384*d9f75844SAndroid Build Coastguard Worker     return kLocalAddrs[next_local_address_++];
385*d9f75844SAndroid Build Coastguard Worker   }
386*d9f75844SAndroid Build Coastguard Worker 
387*d9f75844SAndroid Build Coastguard Worker   std::vector<std::unique_ptr<rtc::FakeNetworkManager>> fake_networks_;
388*d9f75844SAndroid Build Coastguard Worker   int next_local_address_ = 0;
389*d9f75844SAndroid Build Coastguard Worker   std::unique_ptr<rtc::VirtualSocketServer> vss_;
390*d9f75844SAndroid Build Coastguard Worker   rtc::AutoSocketServerThread main_;
391*d9f75844SAndroid Build Coastguard Worker };
392*d9f75844SAndroid Build Coastguard Worker 
TEST_F(PeerConnectionUsageHistogramTest,UsageFingerprintHistogramFromTimeout)393*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, UsageFingerprintHistogramFromTimeout) {
394*d9f75844SAndroid Build Coastguard Worker   auto pc = CreatePeerConnectionWithImmediateReport();
395*d9f75844SAndroid Build Coastguard Worker 
396*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint = MakeUsageFingerprint({});
397*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ_WAIT(1, webrtc::metrics::NumSamples(kUsagePatternMetric),
398*d9f75844SAndroid Build Coastguard Worker                         kDefaultTimeout);
399*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(
400*d9f75844SAndroid Build Coastguard Worker       1, webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint));
401*d9f75844SAndroid Build Coastguard Worker }
402*d9f75844SAndroid Build Coastguard Worker 
403*d9f75844SAndroid Build Coastguard Worker #ifndef WEBRTC_ANDROID
404*d9f75844SAndroid Build Coastguard Worker // These tests do not work on Android. Why is unclear.
405*d9f75844SAndroid Build Coastguard Worker // https://bugs.webrtc.org/9461
406*d9f75844SAndroid Build Coastguard Worker 
407*d9f75844SAndroid Build Coastguard Worker // Test getting the usage fingerprint for an audio/video connection.
TEST_F(PeerConnectionUsageHistogramTest,FingerprintAudioVideo)408*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, FingerprintAudioVideo) {
409*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnection();
410*d9f75844SAndroid Build Coastguard Worker   auto callee = CreatePeerConnection();
411*d9f75844SAndroid Build Coastguard Worker   caller->AddAudioTrack("audio");
412*d9f75844SAndroid Build Coastguard Worker   caller->AddVideoTrack("video");
413*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller->ConnectTo(callee.get()));
414*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
415*d9f75844SAndroid Build Coastguard Worker   callee->pc()->Close();
416*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint = MakeUsageFingerprint(
417*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::AUDIO_ADDED, UsageEvent::VIDEO_ADDED,
418*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
419*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
420*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
421*d9f75844SAndroid Build Coastguard Worker        UsageEvent::ICE_STATE_CONNECTED, UsageEvent::REMOTE_CANDIDATE_ADDED,
422*d9f75844SAndroid Build Coastguard Worker        UsageEvent::DIRECT_CONNECTION_SELECTED, UsageEvent::CLOSE_CALLED});
423*d9f75844SAndroid Build Coastguard Worker   // In this case, we may or may not have PRIVATE_CANDIDATE_COLLECTED,
424*d9f75844SAndroid Build Coastguard Worker   // depending on the machine configuration.
425*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
426*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_TRUE(
427*d9f75844SAndroid Build Coastguard Worker       webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint) ==
428*d9f75844SAndroid Build Coastguard Worker           2 ||
429*d9f75844SAndroid Build Coastguard Worker       webrtc::metrics::NumEvents(
430*d9f75844SAndroid Build Coastguard Worker           kUsagePatternMetric,
431*d9f75844SAndroid Build Coastguard Worker           expected_fingerprint |
432*d9f75844SAndroid Build Coastguard Worker               static_cast<int>(UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) == 2);
433*d9f75844SAndroid Build Coastguard Worker }
434*d9f75844SAndroid Build Coastguard Worker 
435*d9f75844SAndroid Build Coastguard Worker // Test getting the usage fingerprint when the caller collects an mDNS
436*d9f75844SAndroid Build Coastguard Worker // candidate.
TEST_F(PeerConnectionUsageHistogramTest,FingerprintWithMdnsCaller)437*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithMdnsCaller) {
438*d9f75844SAndroid Build Coastguard Worker   RTCConfiguration config;
439*d9f75844SAndroid Build Coastguard Worker   config.sdp_semantics = SdpSemantics::kUnifiedPlan;
440*d9f75844SAndroid Build Coastguard Worker 
441*d9f75844SAndroid Build Coastguard Worker   // Enable hostname candidates with mDNS names.
442*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnectionWithMdns(config);
443*d9f75844SAndroid Build Coastguard Worker   auto callee = CreatePeerConnection(config);
444*d9f75844SAndroid Build Coastguard Worker 
445*d9f75844SAndroid Build Coastguard Worker   caller->AddAudioTrack("audio");
446*d9f75844SAndroid Build Coastguard Worker   caller->AddVideoTrack("video");
447*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller->ConnectTo(callee.get()));
448*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
449*d9f75844SAndroid Build Coastguard Worker   callee->pc()->Close();
450*d9f75844SAndroid Build Coastguard Worker 
451*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_caller = MakeUsageFingerprint(
452*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::AUDIO_ADDED, UsageEvent::VIDEO_ADDED,
453*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
454*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
455*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::MDNS_CANDIDATE_COLLECTED,
456*d9f75844SAndroid Build Coastguard Worker        UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED, UsageEvent::ICE_STATE_CONNECTED,
457*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_CANDIDATE_ADDED,
458*d9f75844SAndroid Build Coastguard Worker        UsageEvent::DIRECT_CONNECTION_SELECTED, UsageEvent::CLOSE_CALLED});
459*d9f75844SAndroid Build Coastguard Worker 
460*d9f75844SAndroid Build Coastguard Worker   // Without a resolver, the callee cannot resolve the received mDNS candidate
461*d9f75844SAndroid Build Coastguard Worker   // but can still connect with the caller via a prflx candidate. As a result,
462*d9f75844SAndroid Build Coastguard Worker   // the bit for the direct connection should not be logged.
463*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_callee = MakeUsageFingerprint(
464*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::AUDIO_ADDED, UsageEvent::VIDEO_ADDED,
465*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
466*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
467*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
468*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_MDNS_CANDIDATE_ADDED, UsageEvent::ICE_STATE_CONNECTED,
469*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_CANDIDATE_ADDED, UsageEvent::CLOSE_CALLED});
470*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
471*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
472*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_caller));
473*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
474*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_callee));
475*d9f75844SAndroid Build Coastguard Worker }
476*d9f75844SAndroid Build Coastguard Worker 
477*d9f75844SAndroid Build Coastguard Worker // Test getting the usage fingerprint when the callee collects an mDNS
478*d9f75844SAndroid Build Coastguard Worker // candidate.
TEST_F(PeerConnectionUsageHistogramTest,FingerprintWithMdnsCallee)479*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithMdnsCallee) {
480*d9f75844SAndroid Build Coastguard Worker   RTCConfiguration config;
481*d9f75844SAndroid Build Coastguard Worker   config.sdp_semantics = SdpSemantics::kUnifiedPlan;
482*d9f75844SAndroid Build Coastguard Worker 
483*d9f75844SAndroid Build Coastguard Worker   // Enable hostname candidates with mDNS names.
484*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnection(config);
485*d9f75844SAndroid Build Coastguard Worker   auto callee = CreatePeerConnectionWithMdns(config);
486*d9f75844SAndroid Build Coastguard Worker 
487*d9f75844SAndroid Build Coastguard Worker   caller->AddAudioTrack("audio");
488*d9f75844SAndroid Build Coastguard Worker   caller->AddVideoTrack("video");
489*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller->ConnectTo(callee.get()));
490*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
491*d9f75844SAndroid Build Coastguard Worker   callee->pc()->Close();
492*d9f75844SAndroid Build Coastguard Worker 
493*d9f75844SAndroid Build Coastguard Worker   // Similar to the test above, the caller connects with the callee via a prflx
494*d9f75844SAndroid Build Coastguard Worker   // candidate.
495*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_caller = MakeUsageFingerprint(
496*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::AUDIO_ADDED, UsageEvent::VIDEO_ADDED,
497*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
498*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
499*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
500*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_MDNS_CANDIDATE_ADDED, UsageEvent::ICE_STATE_CONNECTED,
501*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_CANDIDATE_ADDED, UsageEvent::CLOSE_CALLED});
502*d9f75844SAndroid Build Coastguard Worker 
503*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_callee = MakeUsageFingerprint(
504*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::AUDIO_ADDED, UsageEvent::VIDEO_ADDED,
505*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
506*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
507*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::MDNS_CANDIDATE_COLLECTED,
508*d9f75844SAndroid Build Coastguard Worker        UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED, UsageEvent::ICE_STATE_CONNECTED,
509*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_CANDIDATE_ADDED,
510*d9f75844SAndroid Build Coastguard Worker        UsageEvent::DIRECT_CONNECTION_SELECTED, UsageEvent::CLOSE_CALLED});
511*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
512*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
513*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_caller));
514*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
515*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_callee));
516*d9f75844SAndroid Build Coastguard Worker }
517*d9f75844SAndroid Build Coastguard Worker 
518*d9f75844SAndroid Build Coastguard Worker #ifdef WEBRTC_HAVE_SCTP
TEST_F(PeerConnectionUsageHistogramTest,FingerprintDataOnly)519*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, FingerprintDataOnly) {
520*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnection();
521*d9f75844SAndroid Build Coastguard Worker   auto callee = CreatePeerConnection();
522*d9f75844SAndroid Build Coastguard Worker   caller->CreateDataChannel("foodata");
523*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller->ConnectTo(callee.get()));
524*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE_WAIT(callee->HaveDataChannel(), kDefaultTimeout);
525*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
526*d9f75844SAndroid Build Coastguard Worker   callee->pc()->Close();
527*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint = MakeUsageFingerprint(
528*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::DATA_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
529*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
530*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
531*d9f75844SAndroid Build Coastguard Worker        UsageEvent::ICE_STATE_CONNECTED, UsageEvent::REMOTE_CANDIDATE_ADDED,
532*d9f75844SAndroid Build Coastguard Worker        UsageEvent::DIRECT_CONNECTION_SELECTED, UsageEvent::CLOSE_CALLED});
533*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
534*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_TRUE(
535*d9f75844SAndroid Build Coastguard Worker       webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint) ==
536*d9f75844SAndroid Build Coastguard Worker           2 ||
537*d9f75844SAndroid Build Coastguard Worker       webrtc::metrics::NumEvents(
538*d9f75844SAndroid Build Coastguard Worker           kUsagePatternMetric,
539*d9f75844SAndroid Build Coastguard Worker           expected_fingerprint |
540*d9f75844SAndroid Build Coastguard Worker               static_cast<int>(UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) == 2);
541*d9f75844SAndroid Build Coastguard Worker }
542*d9f75844SAndroid Build Coastguard Worker #endif  // WEBRTC_HAVE_SCTP
543*d9f75844SAndroid Build Coastguard Worker #endif  // WEBRTC_ANDROID
544*d9f75844SAndroid Build Coastguard Worker 
TEST_F(PeerConnectionUsageHistogramTest,FingerprintStunTurn)545*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurn) {
546*d9f75844SAndroid Build Coastguard Worker   RTCConfiguration configuration;
547*d9f75844SAndroid Build Coastguard Worker   configuration.sdp_semantics = SdpSemantics::kUnifiedPlan;
548*d9f75844SAndroid Build Coastguard Worker   PeerConnection::IceServer server;
549*d9f75844SAndroid Build Coastguard Worker   server.urls = {"stun:dummy.stun.server"};
550*d9f75844SAndroid Build Coastguard Worker   configuration.servers.push_back(server);
551*d9f75844SAndroid Build Coastguard Worker   server.urls = {"turn:dummy.turn.server"};
552*d9f75844SAndroid Build Coastguard Worker   server.username = "username";
553*d9f75844SAndroid Build Coastguard Worker   server.password = "password";
554*d9f75844SAndroid Build Coastguard Worker   configuration.servers.push_back(server);
555*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnection(configuration);
556*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller);
557*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
558*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint = MakeUsageFingerprint(
559*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::STUN_SERVER_ADDED, UsageEvent::TURN_SERVER_ADDED,
560*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CLOSE_CALLED});
561*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
562*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(
563*d9f75844SAndroid Build Coastguard Worker       1, webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint));
564*d9f75844SAndroid Build Coastguard Worker }
565*d9f75844SAndroid Build Coastguard Worker 
TEST_F(PeerConnectionUsageHistogramTest,FingerprintStunTurnInReconfiguration)566*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, FingerprintStunTurnInReconfiguration) {
567*d9f75844SAndroid Build Coastguard Worker   RTCConfiguration configuration;
568*d9f75844SAndroid Build Coastguard Worker   configuration.sdp_semantics = SdpSemantics::kUnifiedPlan;
569*d9f75844SAndroid Build Coastguard Worker   PeerConnection::IceServer server;
570*d9f75844SAndroid Build Coastguard Worker   server.urls = {"stun:dummy.stun.server"};
571*d9f75844SAndroid Build Coastguard Worker   configuration.servers.push_back(server);
572*d9f75844SAndroid Build Coastguard Worker   server.urls = {"turn:dummy.turn.server"};
573*d9f75844SAndroid Build Coastguard Worker   server.username = "username";
574*d9f75844SAndroid Build Coastguard Worker   server.password = "password";
575*d9f75844SAndroid Build Coastguard Worker   configuration.servers.push_back(server);
576*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnection();
577*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller);
578*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller->pc()->SetConfiguration(configuration).ok());
579*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
580*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint = MakeUsageFingerprint(
581*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::STUN_SERVER_ADDED, UsageEvent::TURN_SERVER_ADDED,
582*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CLOSE_CALLED});
583*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
584*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(
585*d9f75844SAndroid Build Coastguard Worker       1, webrtc::metrics::NumEvents(kUsagePatternMetric, expected_fingerprint));
586*d9f75844SAndroid Build Coastguard Worker }
587*d9f75844SAndroid Build Coastguard Worker 
TEST_F(PeerConnectionUsageHistogramTest,FingerprintWithPrivateIPCaller)588*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithPrivateIPCaller) {
589*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnectionWithPrivateLocalAddresses();
590*d9f75844SAndroid Build Coastguard Worker   auto callee = CreatePeerConnection();
591*d9f75844SAndroid Build Coastguard Worker   caller->AddAudioTrack("audio");
592*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller->ConnectTo(callee.get()));
593*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
594*d9f75844SAndroid Build Coastguard Worker   callee->pc()->Close();
595*d9f75844SAndroid Build Coastguard Worker 
596*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_caller = MakeUsageFingerprint(
597*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::AUDIO_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
598*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
599*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::PRIVATE_CANDIDATE_COLLECTED,
600*d9f75844SAndroid Build Coastguard Worker        UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED, UsageEvent::ICE_STATE_CONNECTED,
601*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_CANDIDATE_ADDED,
602*d9f75844SAndroid Build Coastguard Worker        UsageEvent::DIRECT_CONNECTION_SELECTED, UsageEvent::CLOSE_CALLED});
603*d9f75844SAndroid Build Coastguard Worker 
604*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_callee = MakeUsageFingerprint(
605*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::AUDIO_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
606*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
607*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
608*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_PRIVATE_CANDIDATE_ADDED,
609*d9f75844SAndroid Build Coastguard Worker        UsageEvent::ICE_STATE_CONNECTED, UsageEvent::REMOTE_CANDIDATE_ADDED,
610*d9f75844SAndroid Build Coastguard Worker        UsageEvent::DIRECT_CONNECTION_SELECTED, UsageEvent::CLOSE_CALLED});
611*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
612*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
613*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_caller));
614*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
615*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_callee));
616*d9f75844SAndroid Build Coastguard Worker }
617*d9f75844SAndroid Build Coastguard Worker 
TEST_F(PeerConnectionUsageHistogramTest,FingerprintWithPrivateIpv6Callee)618*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, FingerprintWithPrivateIpv6Callee) {
619*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnection();
620*d9f75844SAndroid Build Coastguard Worker   auto callee = CreatePeerConnectionWithPrivateIpv6LocalAddresses();
621*d9f75844SAndroid Build Coastguard Worker   caller->AddAudioTrack("audio");
622*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller->ConnectTo(callee.get()));
623*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
624*d9f75844SAndroid Build Coastguard Worker   callee->pc()->Close();
625*d9f75844SAndroid Build Coastguard Worker 
626*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_caller = MakeUsageFingerprint(
627*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::AUDIO_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
628*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
629*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
630*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_PRIVATE_CANDIDATE_ADDED,
631*d9f75844SAndroid Build Coastguard Worker        UsageEvent::ICE_STATE_CONNECTED, UsageEvent::REMOTE_CANDIDATE_ADDED,
632*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_IPV6_CANDIDATE_ADDED,
633*d9f75844SAndroid Build Coastguard Worker        UsageEvent::DIRECT_CONNECTION_SELECTED, UsageEvent::CLOSE_CALLED});
634*d9f75844SAndroid Build Coastguard Worker 
635*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_callee = MakeUsageFingerprint(
636*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::AUDIO_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
637*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
638*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::PRIVATE_CANDIDATE_COLLECTED,
639*d9f75844SAndroid Build Coastguard Worker        UsageEvent::IPV6_CANDIDATE_COLLECTED,
640*d9f75844SAndroid Build Coastguard Worker        UsageEvent::ADD_ICE_CANDIDATE_SUCCEEDED,
641*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_CANDIDATE_ADDED, UsageEvent::ICE_STATE_CONNECTED,
642*d9f75844SAndroid Build Coastguard Worker        UsageEvent::DIRECT_CONNECTION_SELECTED, UsageEvent::CLOSE_CALLED});
643*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
644*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
645*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_caller));
646*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
647*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_callee));
648*d9f75844SAndroid Build Coastguard Worker }
649*d9f75844SAndroid Build Coastguard Worker 
650*d9f75844SAndroid Build Coastguard Worker #ifndef WEBRTC_ANDROID
651*d9f75844SAndroid Build Coastguard Worker #ifdef WEBRTC_HAVE_SCTP
652*d9f75844SAndroid Build Coastguard Worker // Test that the usage pattern bits for adding remote (private IPv6) candidates
653*d9f75844SAndroid Build Coastguard Worker // are set when the remote candidates are retrieved from the Offer SDP instead
654*d9f75844SAndroid Build Coastguard Worker // of trickled ICE messages.
TEST_F(PeerConnectionUsageHistogramTest,AddRemoteCandidatesFromRemoteDescription)655*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest,
656*d9f75844SAndroid Build Coastguard Worker        AddRemoteCandidatesFromRemoteDescription) {
657*d9f75844SAndroid Build Coastguard Worker   // We construct the following data-channel-only scenario. The caller collects
658*d9f75844SAndroid Build Coastguard Worker   // IPv6 private local candidates and appends them in the Offer as in
659*d9f75844SAndroid Build Coastguard Worker   // non-trickled sessions. The callee collects mDNS candidates that are not
660*d9f75844SAndroid Build Coastguard Worker   // contained in the Answer as in Trickle ICE. Only the Offer and Answer are
661*d9f75844SAndroid Build Coastguard Worker   // signaled and we expect a connection with prflx remote candidates at the
662*d9f75844SAndroid Build Coastguard Worker   // caller side.
663*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnectionWithPrivateIpv6LocalAddresses();
664*d9f75844SAndroid Build Coastguard Worker   RTCConfiguration config;
665*d9f75844SAndroid Build Coastguard Worker   config.sdp_semantics = SdpSemantics::kUnifiedPlan;
666*d9f75844SAndroid Build Coastguard Worker   auto callee = CreatePeerConnectionWithMdns(config);
667*d9f75844SAndroid Build Coastguard Worker   caller->CreateDataChannel("test_channel");
668*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(caller->SetLocalDescription(caller->CreateOffer()));
669*d9f75844SAndroid Build Coastguard Worker   // Wait until the gathering completes so that the session description would
670*d9f75844SAndroid Build Coastguard Worker   // have contained ICE candidates.
671*d9f75844SAndroid Build Coastguard Worker   EXPECT_EQ_WAIT(webrtc::PeerConnectionInterface::kIceGatheringComplete,
672*d9f75844SAndroid Build Coastguard Worker                  caller->ice_gathering_state(), kDefaultTimeout);
673*d9f75844SAndroid Build Coastguard Worker   EXPECT_TRUE(caller->observer()->candidate_gathered());
674*d9f75844SAndroid Build Coastguard Worker   // Get the current offer that contains candidates and pass it to the callee.
675*d9f75844SAndroid Build Coastguard Worker   //
676*d9f75844SAndroid Build Coastguard Worker   // Note that we cannot use CloneSessionDescription on `cur_offer` to obtain an
677*d9f75844SAndroid Build Coastguard Worker   // SDP with candidates. The method above does not strictly copy everything, in
678*d9f75844SAndroid Build Coastguard Worker   // particular, not copying the ICE candidates.
679*d9f75844SAndroid Build Coastguard Worker   // TODO(qingsi): Technically, this is a bug. Fix it.
680*d9f75844SAndroid Build Coastguard Worker   auto cur_offer = caller->pc()->local_description();
681*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(cur_offer);
682*d9f75844SAndroid Build Coastguard Worker   std::string sdp_with_candidates_str;
683*d9f75844SAndroid Build Coastguard Worker   cur_offer->ToString(&sdp_with_candidates_str);
684*d9f75844SAndroid Build Coastguard Worker   auto offer = std::make_unique<JsepSessionDescription>(SdpType::kOffer);
685*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(SdpDeserialize(sdp_with_candidates_str, offer.get(),
686*d9f75844SAndroid Build Coastguard Worker                              nullptr /* error */));
687*d9f75844SAndroid Build Coastguard Worker   ASSERT_TRUE(callee->SetRemoteDescription(std::move(offer)));
688*d9f75844SAndroid Build Coastguard Worker 
689*d9f75844SAndroid Build Coastguard Worker   // By default, the Answer created does not contain ICE candidates.
690*d9f75844SAndroid Build Coastguard Worker   auto answer = callee->CreateAnswer();
691*d9f75844SAndroid Build Coastguard Worker   callee->SetLocalDescription(CloneSessionDescription(answer.get()));
692*d9f75844SAndroid Build Coastguard Worker   caller->SetRemoteDescription(std::move(answer));
693*d9f75844SAndroid Build Coastguard Worker   EXPECT_TRUE_WAIT(caller->IsConnected(), kDefaultTimeout);
694*d9f75844SAndroid Build Coastguard Worker   EXPECT_TRUE_WAIT(callee->IsConnected(), kDefaultTimeout);
695*d9f75844SAndroid Build Coastguard Worker   // The callee needs to process the open message to have the data channel open.
696*d9f75844SAndroid Build Coastguard Worker   EXPECT_TRUE_WAIT(callee->observer()->last_datachannel_ != nullptr,
697*d9f75844SAndroid Build Coastguard Worker                    kDefaultTimeout);
698*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
699*d9f75844SAndroid Build Coastguard Worker   callee->pc()->Close();
700*d9f75844SAndroid Build Coastguard Worker 
701*d9f75844SAndroid Build Coastguard Worker   // The caller should not have added any remote candidate either via
702*d9f75844SAndroid Build Coastguard Worker   // AddIceCandidate or from the remote description. Also, the caller connects
703*d9f75844SAndroid Build Coastguard Worker   // with the callee via a prflx candidate and hence no direct connection bit
704*d9f75844SAndroid Build Coastguard Worker   // should be set.
705*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_caller = MakeUsageFingerprint(
706*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::DATA_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
707*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
708*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::PRIVATE_CANDIDATE_COLLECTED,
709*d9f75844SAndroid Build Coastguard Worker        UsageEvent::IPV6_CANDIDATE_COLLECTED, UsageEvent::ICE_STATE_CONNECTED,
710*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CLOSE_CALLED});
711*d9f75844SAndroid Build Coastguard Worker 
712*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint_callee = MakeUsageFingerprint(
713*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::DATA_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
714*d9f75844SAndroid Build Coastguard Worker        UsageEvent::SET_REMOTE_DESCRIPTION_SUCCEEDED,
715*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::MDNS_CANDIDATE_COLLECTED,
716*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_CANDIDATE_ADDED,
717*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_PRIVATE_CANDIDATE_ADDED,
718*d9f75844SAndroid Build Coastguard Worker        UsageEvent::REMOTE_IPV6_CANDIDATE_ADDED, UsageEvent::ICE_STATE_CONNECTED,
719*d9f75844SAndroid Build Coastguard Worker        UsageEvent::DIRECT_CONNECTION_SELECTED, UsageEvent::CLOSE_CALLED});
720*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(2, webrtc::metrics::NumSamples(kUsagePatternMetric));
721*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
722*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_caller));
723*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumEvents(kUsagePatternMetric,
724*d9f75844SAndroid Build Coastguard Worker                                                  expected_fingerprint_callee));
725*d9f75844SAndroid Build Coastguard Worker }
726*d9f75844SAndroid Build Coastguard Worker 
TEST_F(PeerConnectionUsageHistogramTest,NotableUsageNoted)727*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, NotableUsageNoted) {
728*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnection();
729*d9f75844SAndroid Build Coastguard Worker   caller->CreateDataChannel("foo");
730*d9f75844SAndroid Build Coastguard Worker   caller->GenerateOfferAndCollectCandidates();
731*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
732*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint = MakeUsageFingerprint(
733*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::DATA_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
734*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::CLOSE_CALLED});
735*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
736*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_TRUE(
737*d9f75844SAndroid Build Coastguard Worker       expected_fingerprint == ObservedFingerprint() ||
738*d9f75844SAndroid Build Coastguard Worker       (expected_fingerprint |
739*d9f75844SAndroid Build Coastguard Worker        static_cast<int>(UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
740*d9f75844SAndroid Build Coastguard Worker           ObservedFingerprint());
741*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(absl::make_optional(ObservedFingerprint()),
742*d9f75844SAndroid Build Coastguard Worker                    caller->observer()->interesting_usage_detected());
743*d9f75844SAndroid Build Coastguard Worker }
744*d9f75844SAndroid Build Coastguard Worker 
TEST_F(PeerConnectionUsageHistogramTest,NotableUsageOnEventFiring)745*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest, NotableUsageOnEventFiring) {
746*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnection();
747*d9f75844SAndroid Build Coastguard Worker   caller->CreateDataChannel("foo");
748*d9f75844SAndroid Build Coastguard Worker   caller->GenerateOfferAndCollectCandidates();
749*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint = MakeUsageFingerprint(
750*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::DATA_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
751*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED});
752*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(0, webrtc::metrics::NumSamples(kUsagePatternMetric));
753*d9f75844SAndroid Build Coastguard Worker   caller->GetInternalPeerConnection()->RequestUsagePatternReportForTesting();
754*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ_WAIT(1, webrtc::metrics::NumSamples(kUsagePatternMetric),
755*d9f75844SAndroid Build Coastguard Worker                         kDefaultTimeout);
756*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_TRUE(
757*d9f75844SAndroid Build Coastguard Worker       expected_fingerprint == ObservedFingerprint() ||
758*d9f75844SAndroid Build Coastguard Worker       (expected_fingerprint |
759*d9f75844SAndroid Build Coastguard Worker        static_cast<int>(UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
760*d9f75844SAndroid Build Coastguard Worker           ObservedFingerprint());
761*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(absl::make_optional(ObservedFingerprint()),
762*d9f75844SAndroid Build Coastguard Worker                    caller->observer()->interesting_usage_detected());
763*d9f75844SAndroid Build Coastguard Worker }
764*d9f75844SAndroid Build Coastguard Worker 
TEST_F(PeerConnectionUsageHistogramTest,NoNotableUsageOnEventFiringAfterClose)765*d9f75844SAndroid Build Coastguard Worker TEST_F(PeerConnectionUsageHistogramTest,
766*d9f75844SAndroid Build Coastguard Worker        NoNotableUsageOnEventFiringAfterClose) {
767*d9f75844SAndroid Build Coastguard Worker   auto caller = CreatePeerConnection();
768*d9f75844SAndroid Build Coastguard Worker   caller->CreateDataChannel("foo");
769*d9f75844SAndroid Build Coastguard Worker   caller->GenerateOfferAndCollectCandidates();
770*d9f75844SAndroid Build Coastguard Worker   int expected_fingerprint = MakeUsageFingerprint(
771*d9f75844SAndroid Build Coastguard Worker       {UsageEvent::DATA_ADDED, UsageEvent::SET_LOCAL_DESCRIPTION_SUCCEEDED,
772*d9f75844SAndroid Build Coastguard Worker        UsageEvent::CANDIDATE_COLLECTED, UsageEvent::CLOSE_CALLED});
773*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(0, webrtc::metrics::NumSamples(kUsagePatternMetric));
774*d9f75844SAndroid Build Coastguard Worker   caller->pc()->Close();
775*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ(1, webrtc::metrics::NumSamples(kUsagePatternMetric));
776*d9f75844SAndroid Build Coastguard Worker   caller->GetInternalPeerConnection()->RequestUsagePatternReportForTesting();
777*d9f75844SAndroid Build Coastguard Worker   caller->observer()->ClearInterestingUsageDetector();
778*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_EQ_WAIT(2, webrtc::metrics::NumSamples(kUsagePatternMetric),
779*d9f75844SAndroid Build Coastguard Worker                         kDefaultTimeout);
780*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_TRUE(
781*d9f75844SAndroid Build Coastguard Worker       expected_fingerprint == ObservedFingerprint() ||
782*d9f75844SAndroid Build Coastguard Worker       (expected_fingerprint |
783*d9f75844SAndroid Build Coastguard Worker        static_cast<int>(UsageEvent::PRIVATE_CANDIDATE_COLLECTED)) ==
784*d9f75844SAndroid Build Coastguard Worker           ObservedFingerprint());
785*d9f75844SAndroid Build Coastguard Worker   // After close, the usage-detection callback should NOT have been called.
786*d9f75844SAndroid Build Coastguard Worker   EXPECT_METRIC_FALSE(caller->observer()->interesting_usage_detected());
787*d9f75844SAndroid Build Coastguard Worker }
788*d9f75844SAndroid Build Coastguard Worker #endif
789*d9f75844SAndroid Build Coastguard Worker #endif
790*d9f75844SAndroid Build Coastguard Worker 
791*d9f75844SAndroid Build Coastguard Worker }  // namespace webrtc
792