Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(57)

Side by Side Diff: webrtc/api/rtcstatscollector_unittest.cc

Issue 2243123002: RTCCertificateStats added. (Closed) Base URL: https://chromium.googlesource.com/external/webrtc.git@master
Patch Set: Addressed comments Created 4 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 /* 1 /*
2 * Copyright 2016 The WebRTC project authors. All Rights Reserved. 2 * Copyright 2016 The WebRTC project authors. All Rights Reserved.
3 * 3 *
4 * Use of this source code is governed by a BSD-style license 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 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 6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may 7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree. 8 * be found in the AUTHORS file in the root of the source tree.
9 */ 9 */
10 10
11 #include "webrtc/api/rtcstatscollector.h" 11 #include "webrtc/api/rtcstatscollector.h"
12 12
13 #include <memory> 13 #include <memory>
14 #include <string> 14 #include <string>
15 #include <vector> 15 #include <vector>
16 16
17 #include "webrtc/api/jsepsessiondescription.h" 17 #include "webrtc/api/jsepsessiondescription.h"
18 #include "webrtc/api/stats/rtcstats_objects.h" 18 #include "webrtc/api/stats/rtcstats_objects.h"
19 #include "webrtc/api/stats/rtcstatsreport.h" 19 #include "webrtc/api/stats/rtcstatsreport.h"
20 #include "webrtc/api/test/mock_datachannel.h" 20 #include "webrtc/api/test/mock_datachannel.h"
21 #include "webrtc/api/test/mock_peerconnection.h" 21 #include "webrtc/api/test/mock_peerconnection.h"
22 #include "webrtc/api/test/mock_webrtcsession.h" 22 #include "webrtc/api/test/mock_webrtcsession.h"
23 #include "webrtc/base/checks.h" 23 #include "webrtc/base/checks.h"
24 #include "webrtc/base/fakeclock.h" 24 #include "webrtc/base/fakeclock.h"
25 #include "webrtc/base/fakesslidentity.h"
25 #include "webrtc/base/gunit.h" 26 #include "webrtc/base/gunit.h"
26 #include "webrtc/base/logging.h" 27 #include "webrtc/base/logging.h"
27 #include "webrtc/base/thread_checker.h" 28 #include "webrtc/base/thread_checker.h"
28 #include "webrtc/base/timedelta.h" 29 #include "webrtc/base/timedelta.h"
29 #include "webrtc/base/timeutils.h" 30 #include "webrtc/base/timeutils.h"
30 #include "webrtc/media/base/fakemediaengine.h" 31 #include "webrtc/media/base/fakemediaengine.h"
31 32
33 using testing::_;
34 using testing::Invoke;
32 using testing::Return; 35 using testing::Return;
33 using testing::ReturnRef; 36 using testing::ReturnRef;
34 37
35 namespace webrtc { 38 namespace webrtc {
36 39
37 namespace { 40 namespace {
38 41
39 const int64_t kGetStatsReportTimeoutMs = 1000; 42 const int64_t kGetStatsReportTimeoutMs = 1000;
40 43
44 struct CertificateInfo {
45 rtc::scoped_refptr<rtc::RTCCertificate> certificate;
46 std::vector<std::string> ders;
47 std::vector<std::string> pems;
48 std::vector<std::string> fingerprints;
49 };
50
51 std::unique_ptr<CertificateInfo> CreateFakeCertificateAndInfoFromDers(
52 const std::vector<std::string>& ders) {
53 RTC_CHECK(!ders.empty());
54 std::unique_ptr<CertificateInfo> info(new CertificateInfo());
55 info->ders = ders;
56 for (const std::string& der : ders) {
57 info->pems.push_back(rtc::SSLIdentity::DerToPem(
58 "CERTIFICATE",
59 reinterpret_cast<const unsigned char*>(der.c_str()),
60 der.length()));
61 }
62 info->certificate =
63 rtc::RTCCertificate::Create(std::unique_ptr<rtc::FakeSSLIdentity>(
64 new rtc::FakeSSLIdentity(rtc::FakeSSLCertificate(info->pems))));
65 // Strip header/footer and newline characters of PEM strings.
66 for (size_t i = 0; i < info->pems.size(); ++i) {
67 rtc::replace_substrs("-----BEGIN CERTIFICATE-----", 27,
68 "", 0, &info->pems[i]);
69 rtc::replace_substrs("-----END CERTIFICATE-----", 25,
70 "", 0, &info->pems[i]);
71 rtc::replace_substrs("\n", 1,
72 "", 0, &info->pems[i]);
73 }
74 // Fingerprint of leaf certificate.
75 std::unique_ptr<rtc::SSLFingerprint> fp(
76 rtc::SSLFingerprint::Create("sha-1",
77 &info->certificate->ssl_certificate()));
78 EXPECT_TRUE(fp);
79 info->fingerprints.push_back(fp->GetRfc4572Fingerprint());
80 // Fingerprints of the rest of the chain.
81 std::unique_ptr<rtc::SSLCertChain> chain =
82 info->certificate->ssl_certificate().GetChain();
83 if (chain) {
84 for (size_t i = 0; i < chain->GetSize(); i++) {
85 fp.reset(rtc::SSLFingerprint::Create("sha-1", &chain->Get(i)));
86 EXPECT_TRUE(fp);
87 info->fingerprints.push_back(fp->GetRfc4572Fingerprint());
88 }
89 }
90 EXPECT_EQ(info->ders.size(), info->fingerprints.size());
91 return info;
92 }
93
41 class RTCStatsCollectorTestHelper : public SetSessionDescriptionObserver { 94 class RTCStatsCollectorTestHelper : public SetSessionDescriptionObserver {
42 public: 95 public:
43 RTCStatsCollectorTestHelper() 96 RTCStatsCollectorTestHelper()
44 : worker_thread_(rtc::Thread::Current()), 97 : worker_thread_(rtc::Thread::Current()),
45 network_thread_(rtc::Thread::Current()), 98 network_thread_(rtc::Thread::Current()),
46 channel_manager_(new cricket::ChannelManager( 99 channel_manager_(new cricket::ChannelManager(
47 new cricket::FakeMediaEngine(), 100 new cricket::FakeMediaEngine(),
48 worker_thread_, 101 worker_thread_,
49 network_thread_)), 102 network_thread_)),
50 media_controller_( 103 media_controller_(
51 MediaControllerInterface::Create(cricket::MediaConfig(), 104 MediaControllerInterface::Create(cricket::MediaConfig(),
52 worker_thread_, 105 worker_thread_,
53 channel_manager_.get())), 106 channel_manager_.get())),
54 session_(media_controller_.get()), 107 session_(media_controller_.get()),
55 pc_() { 108 pc_() {
56 EXPECT_CALL(pc_, session()).WillRepeatedly(Return(&session_)); 109 EXPECT_CALL(pc_, session()).WillRepeatedly(Return(&session_));
57 EXPECT_CALL(pc_, sctp_data_channels()).WillRepeatedly( 110 EXPECT_CALL(pc_, sctp_data_channels()).WillRepeatedly(
58 ReturnRef(data_channels_)); 111 ReturnRef(data_channels_));
112 // Default return values for mocks.
hta-webrtc 2016/10/03 14:51:36 Move comment before all the EXPECT_CALL lines?
hbos 2016/10/03 15:38:49 Done.
113 EXPECT_CALL(session_, GetTransportStats(_)).WillRepeatedly(Return(false));
59 } 114 }
60 115
61 rtc::ScopedFakeClock& fake_clock() { return fake_clock_; } 116 rtc::ScopedFakeClock& fake_clock() { return fake_clock_; }
62 MockWebRtcSession& session() { return session_; } 117 MockWebRtcSession& session() { return session_; }
63 MockPeerConnection& pc() { return pc_; } 118 MockPeerConnection& pc() { return pc_; }
64 std::vector<rtc::scoped_refptr<DataChannel>>& data_channels() { 119 std::vector<rtc::scoped_refptr<DataChannel>>& data_channels() {
65 return data_channels_; 120 return data_channels_;
66 } 121 }
67 122
68 // SetSessionDescriptionObserver overrides. 123 // SetSessionDescriptionObserver overrides.
(...skipping 177 matching lines...) Expand 10 before | Expand all | Expand 10 after
246 &test_->pc(), 50 * rtc::kNumMicrosecsPerMillisec)) { 301 &test_->pc(), 50 * rtc::kNumMicrosecsPerMillisec)) {
247 } 302 }
248 303
249 rtc::scoped_refptr<const RTCStatsReport> GetStatsReport() { 304 rtc::scoped_refptr<const RTCStatsReport> GetStatsReport() {
250 rtc::scoped_refptr<StatsCallback> callback = StatsCallback::Create(); 305 rtc::scoped_refptr<StatsCallback> callback = StatsCallback::Create();
251 collector_->GetStatsReport(callback); 306 collector_->GetStatsReport(callback);
252 EXPECT_TRUE_WAIT(callback->report(), kGetStatsReportTimeoutMs); 307 EXPECT_TRUE_WAIT(callback->report(), kGetStatsReportTimeoutMs);
253 return callback->report(); 308 return callback->report();
254 } 309 }
255 310
311 void ExpectReportContainsCertificateInfo(
312 const rtc::scoped_refptr<const RTCStatsReport>& report,
313 const CertificateInfo& cert_info) {
314 for (size_t i = 0; i < cert_info.fingerprints.size(); ++i) {
315 const RTCStats* stats = report->Get(
316 "RTCCertificate_" + cert_info.fingerprints[i]);
317 EXPECT_TRUE(stats);
318 const RTCCertificateStats& cert_stats =
319 stats->cast_to<const RTCCertificateStats>();
320 EXPECT_EQ(*cert_stats.fingerprint, cert_info.fingerprints[i]);
321 EXPECT_EQ(*cert_stats.fingerprint_algorithm, "sha-1");
322 EXPECT_EQ(*cert_stats.base64_certificate, cert_info.pems[i]);
323 if (i + 1 < cert_info.fingerprints.size()) {
324 EXPECT_EQ(*cert_stats.issuer_certificate_id,
325 "RTCCertificate_" + cert_info.fingerprints[i + 1]);
326 } else {
327 EXPECT_FALSE(cert_stats.issuer_certificate_id.is_defined());
328 }
329 }
330 }
331
256 protected: 332 protected:
257 rtc::scoped_refptr<RTCStatsCollectorTestHelper> test_; 333 rtc::scoped_refptr<RTCStatsCollectorTestHelper> test_;
258 rtc::scoped_refptr<RTCStatsCollector> collector_; 334 rtc::scoped_refptr<RTCStatsCollector> collector_;
259 }; 335 };
260 336
261 TEST_F(RTCStatsCollectorTest, SingleCallback) { 337 TEST_F(RTCStatsCollectorTest, SingleCallback) {
262 rtc::scoped_refptr<const RTCStatsReport> result; 338 rtc::scoped_refptr<const RTCStatsReport> result;
263 collector_->GetStatsReport(StatsCallback::Create(&result)); 339 collector_->GetStatsReport(StatsCallback::Create(&result));
264 EXPECT_TRUE_WAIT(result, kGetStatsReportTimeoutMs); 340 EXPECT_TRUE_WAIT(result, kGetStatsReportTimeoutMs);
265 } 341 }
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after
305 collector_->GetStatsReport(StatsCallback::Create(&c)); 381 collector_->GetStatsReport(StatsCallback::Create(&c));
306 EXPECT_TRUE_WAIT(a, kGetStatsReportTimeoutMs); 382 EXPECT_TRUE_WAIT(a, kGetStatsReportTimeoutMs);
307 EXPECT_TRUE_WAIT(b, kGetStatsReportTimeoutMs); 383 EXPECT_TRUE_WAIT(b, kGetStatsReportTimeoutMs);
308 EXPECT_TRUE_WAIT(c, kGetStatsReportTimeoutMs); 384 EXPECT_TRUE_WAIT(c, kGetStatsReportTimeoutMs);
309 EXPECT_EQ(a.get(), b.get()); 385 EXPECT_EQ(a.get(), b.get());
310 // The act of doing |AdvanceTime| processes all messages. If this was not the 386 // The act of doing |AdvanceTime| processes all messages. If this was not the
311 // case we might not require |c| to be fresher than |b|. 387 // case we might not require |c| to be fresher than |b|.
312 EXPECT_NE(c.get(), b.get()); 388 EXPECT_NE(c.get(), b.get());
313 } 389 }
314 390
391 TEST_F(RTCStatsCollectorTest, CollectRTCCertificateStatsSingle) {
392 std::unique_ptr<CertificateInfo> local(new CertificateInfo());
393 local->ders.push_back("(local) single certificate");
394 local = CreateFakeCertificateAndInfoFromDers(local->ders);
hta-webrtc 2016/10/03 14:51:36 Here you are creating an unique_ptr with a generic
hbos 2016/10/03 15:38:49 Constructing using a vector instead.
395 std::unique_ptr<CertificateInfo> remote(new CertificateInfo());
396 remote->ders.push_back("(remote) single certificate");
397 remote = CreateFakeCertificateAndInfoFromDers(remote->ders);
398
399 // Mock the session to return the local and remote certificates.
400 EXPECT_CALL(test_->session(), GetTransportStats(_)).WillRepeatedly(Invoke(
401 [this](SessionStats* stats) {
402 stats->transport_stats["transport"].transport_name = "transport";
403 return true;
404 }));
405 EXPECT_CALL(test_->session(), GetLocalCertificate(_, _)).WillRepeatedly(
406 Invoke([this, &local](const std::string& transport_name,
407 rtc::scoped_refptr<rtc::RTCCertificate>* certificate) {
408 if (transport_name == "transport") {
409 *certificate = local->certificate;
410 return true;
411 }
412 return false;
413 }));
414 EXPECT_CALL(test_->session(),
415 GetRemoteSSLCertificate_ReturnsRawPointer(_)).WillRepeatedly(Invoke(
416 [this, &remote](const std::string& transport_name) {
417 if (transport_name == "transport")
418 return remote->certificate->ssl_certificate().GetReference();
419 return static_cast<rtc::SSLCertificate*>(nullptr);
420 }));
421
422 rtc::scoped_refptr<const RTCStatsReport> report = GetStatsReport();
423 ExpectReportContainsCertificateInfo(report, *local.get());
424 ExpectReportContainsCertificateInfo(report, *remote.get());
425 }
426
427 TEST_F(RTCStatsCollectorTest, CollectRTCCertificateStatsMultiple) {
428 std::unique_ptr<CertificateInfo> audio_local(new CertificateInfo());
429 audio_local->ders.push_back("(local) audio");
430 audio_local = CreateFakeCertificateAndInfoFromDers(audio_local->ders);
431 std::unique_ptr<CertificateInfo> audio_remote(new CertificateInfo());
432 audio_remote->ders.push_back("(remote) audio");
433 audio_remote = CreateFakeCertificateAndInfoFromDers(audio_remote->ders);
434
435 std::unique_ptr<CertificateInfo> video_local(new CertificateInfo());
436 video_local->ders.push_back("(local) video");
437 video_local = CreateFakeCertificateAndInfoFromDers(video_local->ders);
438 std::unique_ptr<CertificateInfo> video_remote(new CertificateInfo());
439 video_remote->ders.push_back("(remote) video");
440 video_remote = CreateFakeCertificateAndInfoFromDers(video_remote->ders);
441
442 // Mock the session to return the local and remote certificates.
443 EXPECT_CALL(test_->session(), GetTransportStats(_)).WillRepeatedly(Invoke(
444 [this](SessionStats* stats) {
445 stats->transport_stats["audio"].transport_name = "audio";
446 stats->transport_stats["video"].transport_name = "video";
447 return true;
448 }));
449 EXPECT_CALL(test_->session(), GetLocalCertificate(_, _)).WillRepeatedly(
450 Invoke([this, &audio_local, &video_local](
451 const std::string& transport_name,
452 rtc::scoped_refptr<rtc::RTCCertificate>* certificate) {
453 if (transport_name == "audio") {
454 *certificate = audio_local->certificate;
455 return true;
456 }
457 if (transport_name == "video") {
458 *certificate = video_local->certificate;
459 return true;
460 }
461 return false;
462 }));
463 EXPECT_CALL(test_->session(),
464 GetRemoteSSLCertificate_ReturnsRawPointer(_)).WillRepeatedly(Invoke(
465 [this, &audio_remote, &video_remote](const std::string& transport_name) {
466 if (transport_name == "audio")
467 return audio_remote->certificate->ssl_certificate().GetReference();
468 if (transport_name == "video")
469 return video_remote->certificate->ssl_certificate().GetReference();
470 return static_cast<rtc::SSLCertificate*>(nullptr);
471 }));
472
473 rtc::scoped_refptr<const RTCStatsReport> report = GetStatsReport();
474 ExpectReportContainsCertificateInfo(report, *audio_local.get());
475 ExpectReportContainsCertificateInfo(report, *audio_remote.get());
476 ExpectReportContainsCertificateInfo(report, *video_local.get());
477 ExpectReportContainsCertificateInfo(report, *video_remote.get());
hta-webrtc 2016/10/03 14:51:36 Good rename. More readable.
hbos 2016/10/03 15:38:49 Acknowledged.
478 }
479
480 TEST_F(RTCStatsCollectorTest, CollectRTCCertificateStatsChain) {
481 std::unique_ptr<CertificateInfo> local(new CertificateInfo());
482 local->ders.push_back("(local) this");
483 local->ders.push_back("(local) is");
484 local->ders.push_back("(local) a");
485 local->ders.push_back("(local) chain");
486 local = CreateFakeCertificateAndInfoFromDers(local->ders);
487 std::unique_ptr<CertificateInfo> remote(new CertificateInfo());
488 remote->ders.push_back("(remote) this");
489 remote->ders.push_back("(remote) is");
490 remote->ders.push_back("(remote) another");
491 remote->ders.push_back("(remote) chain");
492 remote = CreateFakeCertificateAndInfoFromDers(remote->ders);
493
494 // Mock the session to return the local and remote certificates.
495 EXPECT_CALL(test_->session(), GetTransportStats(_)).WillRepeatedly(Invoke(
496 [this](SessionStats* stats) {
497 stats->transport_stats["transport"].transport_name = "transport";
498 return true;
499 }));
500 EXPECT_CALL(test_->session(), GetLocalCertificate(_, _)).WillRepeatedly(
501 Invoke([this, &local](const std::string& transport_name,
502 rtc::scoped_refptr<rtc::RTCCertificate>* certificate) {
503 if (transport_name == "transport") {
504 *certificate = local->certificate;
505 return true;
506 }
507 return false;
508 }));
509 EXPECT_CALL(test_->session(),
510 GetRemoteSSLCertificate_ReturnsRawPointer(_)).WillRepeatedly(Invoke(
511 [this, &remote](const std::string& transport_name) {
512 if (transport_name == "transport")
513 return remote->certificate->ssl_certificate().GetReference();
514 return static_cast<rtc::SSLCertificate*>(nullptr);
515 }));
516
517 rtc::scoped_refptr<const RTCStatsReport> report = GetStatsReport();
518 ExpectReportContainsCertificateInfo(report, *local.get());
519 ExpectReportContainsCertificateInfo(report, *remote.get());
520 }
521
315 TEST_F(RTCStatsCollectorTest, CollectRTCPeerConnectionStats) { 522 TEST_F(RTCStatsCollectorTest, CollectRTCPeerConnectionStats) {
316 int64_t before = rtc::TimeUTCMicros(); 523 int64_t before = rtc::TimeUTCMicros();
317 rtc::scoped_refptr<const RTCStatsReport> report = GetStatsReport(); 524 rtc::scoped_refptr<const RTCStatsReport> report = GetStatsReport();
318 int64_t after = rtc::TimeUTCMicros(); 525 int64_t after = rtc::TimeUTCMicros();
319 EXPECT_EQ(report->GetStatsOfType<RTCPeerConnectionStats>().size(), 526 EXPECT_EQ(report->GetStatsOfType<RTCPeerConnectionStats>().size(),
320 static_cast<size_t>(1)) << "Expecting 1 RTCPeerConnectionStats."; 527 static_cast<size_t>(1)) << "Expecting 1 RTCPeerConnectionStats.";
321 const RTCStats* stats = report->Get("RTCPeerConnection"); 528 const RTCStats* stats = report->Get("RTCPeerConnection");
322 EXPECT_TRUE(stats); 529 EXPECT_TRUE(stats);
323 EXPECT_LE(before, stats->timestamp_us()); 530 EXPECT_LE(before, stats->timestamp_us());
324 EXPECT_LE(stats->timestamp_us(), after); 531 EXPECT_LE(stats->timestamp_us(), after);
(...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after
371 rtc::scoped_refptr<FakeRTCStatsCollector> collector_; 578 rtc::scoped_refptr<FakeRTCStatsCollector> collector_;
372 }; 579 };
373 580
374 TEST_F(RTCStatsCollectorTestWithFakeCollector, ThreadUsageAndResultsMerging) { 581 TEST_F(RTCStatsCollectorTestWithFakeCollector, ThreadUsageAndResultsMerging) {
375 collector_->VerifyThreadUsageAndResultsMerging(); 582 collector_->VerifyThreadUsageAndResultsMerging();
376 } 583 }
377 584
378 } // namespace 585 } // namespace
379 586
380 } // namespace webrtc 587 } // namespace webrtc
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698