Index: webrtc/api/rtcstatscollector.cc |
diff --git a/webrtc/api/rtcstatscollector.cc b/webrtc/api/rtcstatscollector.cc |
index 71f4725ae0aec845174d5f55e66e3773008636e8..9052de448589aefa7b59bfd5d9bc45ce7a63d9f1 100644 |
--- a/webrtc/api/rtcstatscollector.cc |
+++ b/webrtc/api/rtcstatscollector.cc |
@@ -17,14 +17,30 @@ |
#include "webrtc/api/peerconnection.h" |
#include "webrtc/api/webrtcsession.h" |
#include "webrtc/base/checks.h" |
-#include "webrtc/base/sslidentity.h" |
#include "webrtc/p2p/base/candidate.h" |
+#include "webrtc/p2p/base/p2pconstants.h" |
#include "webrtc/p2p/base/port.h" |
namespace webrtc { |
namespace { |
+std::string RTCCertificateIDFromFingerprint(const std::string& fingerprint) { |
+ return "RTCCertificate_" + fingerprint; |
+} |
+ |
+std::string RTCIceCandidatePairStatsIDFromConnectionInfo( |
+ const cricket::ConnectionInfo& info) { |
+ return "RTCIceCandidatePair_" + info.local_candidate.id() + "_" + |
+ info.remote_candidate.id(); |
+} |
+ |
+std::string RTCTransportStatsIDFromTransportChannel( |
+ const std::string& transport_name, int channel_component) { |
+ return "RTCTransport_" + transport_name + "_" + |
+ rtc::ToString<>(channel_component); |
+} |
+ |
const char* CandidateTypeToRTCIceCandidateType(const std::string& type) { |
if (type == cricket::LOCAL_PORT_TYPE) |
return RTCIceCandidateType::kHost; |
@@ -128,9 +144,15 @@ void RTCStatsCollector::ProducePartialResultsOnSignalingThread( |
SessionStats session_stats; |
if (pc_->session()->GetTransportStats(&session_stats)) { |
- ProduceCertificateStats_s(timestamp_us, session_stats, report.get()); |
- ProduceIceCandidateAndPairStats_s(timestamp_us, session_stats, |
- report.get()); |
+ std::map<std::string, CertificateStatsPair> transport_cert_stats = |
+ PrepareTransportCertificateStats_s(session_stats); |
+ |
+ ProduceCertificateStats_s( |
+ timestamp_us, transport_cert_stats, report.get()); |
+ ProduceIceCandidateAndPairStats_s( |
+ timestamp_us, session_stats, report.get()); |
+ ProduceTransportStats_s( |
+ timestamp_us, session_stats, transport_cert_stats, report.get()); |
} |
ProduceDataChannelStats_s(timestamp_us, report.get()); |
ProducePeerConnectionStats_s(timestamp_us, report.get()); |
@@ -199,44 +221,38 @@ void RTCStatsCollector::DeliverCachedReport() { |
} |
void RTCStatsCollector::ProduceCertificateStats_s( |
- int64_t timestamp_us, const SessionStats& session_stats, |
+ int64_t timestamp_us, |
+ const std::map<std::string, CertificateStatsPair>& transport_cert_stats, |
RTCStatsReport* report) const { |
RTC_DCHECK(signaling_thread_->IsCurrent()); |
- for (const auto& transport_stats : session_stats.transport_stats) { |
- rtc::scoped_refptr<rtc::RTCCertificate> local_certificate; |
- if (pc_->session()->GetLocalCertificate( |
- transport_stats.second.transport_name, &local_certificate)) { |
- ProduceCertificateStatsFromSSLCertificateAndChain_s( |
- timestamp_us, local_certificate->ssl_certificate(), report); |
+ for (const auto& kvp : transport_cert_stats) { |
+ if (kvp.second.local) { |
+ ProduceCertificateStatsFromSSLCertificateStats_s( |
+ timestamp_us, *kvp.second.local.get(), report); |
} |
- std::unique_ptr<rtc::SSLCertificate> remote_certificate = |
- pc_->session()->GetRemoteSSLCertificate( |
- transport_stats.second.transport_name); |
- if (remote_certificate) { |
- ProduceCertificateStatsFromSSLCertificateAndChain_s( |
- timestamp_us, *remote_certificate.get(), report); |
+ if (kvp.second.remote) { |
+ ProduceCertificateStatsFromSSLCertificateStats_s( |
+ timestamp_us, *kvp.second.remote.get(), report); |
} |
} |
} |
-void RTCStatsCollector::ProduceCertificateStatsFromSSLCertificateAndChain_s( |
- int64_t timestamp_us, const rtc::SSLCertificate& certificate, |
+void RTCStatsCollector::ProduceCertificateStatsFromSSLCertificateStats_s( |
+ int64_t timestamp_us, const rtc::SSLCertificateStats& certificate_stats, |
RTCStatsReport* report) const { |
RTC_DCHECK(signaling_thread_->IsCurrent()); |
- std::unique_ptr<rtc::SSLCertificateStats> ssl_stats = |
- certificate.GetStats(); |
- RTCCertificateStats* prev_stats = nullptr; |
- for (rtc::SSLCertificateStats* s = ssl_stats.get(); s; |
+ RTCCertificateStats* prev_certificate_stats = nullptr; |
+ for (const rtc::SSLCertificateStats* s = &certificate_stats; s; |
s = s->issuer.get()) { |
- RTCCertificateStats* stats = new RTCCertificateStats( |
- "RTCCertificate_" + s->fingerprint, timestamp_us); |
- stats->fingerprint = s->fingerprint; |
- stats->fingerprint_algorithm = s->fingerprint_algorithm; |
- stats->base64_certificate = s->base64_certificate; |
- if (prev_stats) |
- prev_stats->issuer_certificate_id = stats->id(); |
- report->AddStats(std::unique_ptr<RTCCertificateStats>(stats)); |
- prev_stats = stats; |
+ RTCCertificateStats* certificate_stats = new RTCCertificateStats( |
+ RTCCertificateIDFromFingerprint(s->fingerprint), timestamp_us); |
+ certificate_stats->fingerprint = s->fingerprint; |
+ certificate_stats->fingerprint_algorithm = s->fingerprint_algorithm; |
+ certificate_stats->base64_certificate = s->base64_certificate; |
+ if (prev_certificate_stats) |
+ prev_certificate_stats->issuer_certificate_id = certificate_stats->id(); |
+ report->AddStats(std::unique_ptr<RTCCertificateStats>(certificate_stats)); |
+ prev_certificate_stats = certificate_stats; |
} |
} |
@@ -270,10 +286,10 @@ void RTCStatsCollector::ProduceIceCandidateAndPairStats_s( |
for (const auto& channel_stats : transport_stats.second.channel_stats) { |
for (const cricket::ConnectionInfo& info : |
channel_stats.connection_infos) { |
- const std::string& id = "RTCIceCandidatePair_" + |
- info.local_candidate.id() + "_" + info.remote_candidate.id(); |
std::unique_ptr<RTCIceCandidatePairStats> candidate_pair_stats( |
- new RTCIceCandidatePairStats(id, timestamp_us)); |
+ new RTCIceCandidatePairStats( |
+ RTCIceCandidatePairStatsIDFromConnectionInfo(info), |
+ timestamp_us)); |
// TODO(hbos): Set all of the |RTCIceCandidatePairStats|'s members, |
// crbug.com/633550. |
@@ -381,6 +397,98 @@ void RTCStatsCollector::ProducePeerConnectionStats_s( |
report->AddStats(std::move(stats)); |
} |
+void RTCStatsCollector::ProduceTransportStats_s( |
+ int64_t timestamp_us, const SessionStats& session_stats, |
+ const std::map<std::string, CertificateStatsPair>& transport_cert_stats, |
+ RTCStatsReport* report) const { |
+ RTC_DCHECK(signaling_thread_->IsCurrent()); |
+ for (const auto& transport : session_stats.transport_stats) { |
+ // Get reference to RTCP channel, if it exists. |
+ std::string rtcp_transport_stats_id; |
+ for (const auto& channel_stats : transport.second.channel_stats) { |
+ if (channel_stats.component == |
+ cricket::ICE_CANDIDATE_COMPONENT_RTCP) { |
+ rtcp_transport_stats_id = RTCTransportStatsIDFromTransportChannel( |
+ transport.second.transport_name, channel_stats.component); |
+ break; |
+ } |
+ } |
+ |
+ // Get reference to local and remote certificates of this transport, if they |
+ // exist. |
+ const auto& certificate_stats_it = transport_cert_stats.find( |
+ transport.second.transport_name); |
+ RTC_DCHECK(certificate_stats_it != transport_cert_stats.cend()); |
+ std::string local_certificate_id; |
+ if (certificate_stats_it->second.local) { |
+ local_certificate_id = RTCCertificateIDFromFingerprint( |
+ certificate_stats_it->second.local->fingerprint); |
+ } |
+ std::string remote_certificate_id; |
+ if (certificate_stats_it->second.remote) { |
+ remote_certificate_id = RTCCertificateIDFromFingerprint( |
+ certificate_stats_it->second.remote->fingerprint); |
+ } |
+ |
+ // There is one transport stats for each channel. |
+ for (const auto& channel_stats : transport.second.channel_stats) { |
+ std::unique_ptr<RTCTransportStats> transport_stats( |
+ new RTCTransportStats( |
+ RTCTransportStatsIDFromTransportChannel( |
+ transport.second.transport_name, channel_stats.component), |
+ timestamp_us)); |
+ transport_stats->bytes_sent = 0; |
+ transport_stats->bytes_received = 0; |
+ transport_stats->active_connection = false; |
+ for (const cricket::ConnectionInfo& info : |
+ channel_stats.connection_infos) { |
+ *transport_stats->bytes_sent += info.sent_total_bytes; |
+ *transport_stats->bytes_received += info.recv_total_bytes; |
+ if (info.best_connection) { |
+ transport_stats->active_connection = true; |
+ transport_stats->selected_candidate_pair_id = |
+ RTCIceCandidatePairStatsIDFromConnectionInfo(info); |
+ } |
+ } |
+ if (channel_stats.component != cricket::ICE_CANDIDATE_COMPONENT_RTCP && |
+ !rtcp_transport_stats_id.empty()) { |
+ transport_stats->rtcp_transport_stats_id = rtcp_transport_stats_id; |
+ } |
+ if (!local_certificate_id.empty()) |
+ transport_stats->local_certificate_id = local_certificate_id; |
+ if (!remote_certificate_id.empty()) |
+ transport_stats->remote_certificate_id = remote_certificate_id; |
+ report->AddStats(std::move(transport_stats)); |
+ } |
+ } |
+} |
+ |
+std::map<std::string, RTCStatsCollector::CertificateStatsPair> |
+RTCStatsCollector::PrepareTransportCertificateStats_s( |
+ const SessionStats& session_stats) const { |
+ RTC_DCHECK(signaling_thread_->IsCurrent()); |
+ std::map<std::string, CertificateStatsPair> transport_cert_stats; |
+ for (const auto& transport_stats : session_stats.transport_stats) { |
+ CertificateStatsPair certificate_stats_pair; |
+ rtc::scoped_refptr<rtc::RTCCertificate> local_certificate; |
+ if (pc_->session()->GetLocalCertificate( |
+ transport_stats.second.transport_name, &local_certificate)) { |
+ certificate_stats_pair.local = |
+ local_certificate->ssl_certificate().GetStats(); |
+ } |
+ std::unique_ptr<rtc::SSLCertificate> remote_certificate = |
+ pc_->session()->GetRemoteSSLCertificate( |
+ transport_stats.second.transport_name); |
+ if (remote_certificate) { |
+ certificate_stats_pair.remote = remote_certificate->GetStats(); |
+ } |
+ transport_cert_stats.insert( |
+ std::make_pair(transport_stats.second.transport_name, |
+ std::move(certificate_stats_pair))); |
+ } |
+ return transport_cert_stats; |
+} |
+ |
const char* CandidateTypeToRTCIceCandidateTypeForTesting( |
const std::string& type) { |
return CandidateTypeToRTCIceCandidateType(type); |