blob: 2b74beed6fe48ba8da4aa4a7be182602eac4297a [file] [log] [blame]
Henrik Boströmb6199362018-03-12 10:27:55 +01001/*
2 * Copyright 2018 The WebRTC Project Authors. All rights reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#include "pc/rtcstatstraversal.h"
12
13#include <memory>
14#include <string>
15#include <utility>
16#include <vector>
17
18#include "api/stats/rtcstats_objects.h"
19#include "rtc_base/checks.h"
20
21namespace webrtc {
22
23namespace {
24
25void TraverseAndTakeVisitedStats(RTCStatsReport* report,
26 RTCStatsReport* visited_report,
27 const std::string& current_id) {
28 // Mark current stats object as visited by moving it |report| to
29 // |visited_report|.
30 std::unique_ptr<const RTCStats> current = report->Take(current_id);
31 if (!current) {
32 // This node has already been visited (or it is an invalid id).
33 return;
34 }
35 std::vector<const std::string*> neighbor_ids =
36 GetStatsReferencedIds(*current);
37 visited_report->AddStats(std::move(current));
38
39 // Recursively traverse all neighbors.
40 for (const auto* neighbor_id : neighbor_ids) {
41 TraverseAndTakeVisitedStats(report, visited_report, *neighbor_id);
42 }
43}
44
45void AddIdIfDefined(const RTCStatsMember<std::string>& id,
46 std::vector<const std::string*>* neighbor_ids) {
47 if (id.is_defined())
48 neighbor_ids->push_back(&(*id));
49}
50
51void AddIdsIfDefined(const RTCStatsMember<std::vector<std::string>>& ids,
52 std::vector<const std::string*>* neighbor_ids) {
53 if (ids.is_defined()) {
54 for (const std::string& id : *ids)
55 neighbor_ids->push_back(&id);
56 }
57}
58
59} // namespace
60
61rtc::scoped_refptr<RTCStatsReport> TakeReferencedStats(
62 rtc::scoped_refptr<RTCStatsReport> report,
63 const std::vector<std::string>& ids) {
64 rtc::scoped_refptr<RTCStatsReport> result = RTCStatsReport::Create();
65 for (const auto& id : ids) {
66 TraverseAndTakeVisitedStats(report.get(), result.get(), id);
67 }
68 return result;
69}
70
71std::vector<const std::string*> GetStatsReferencedIds(const RTCStats& stats) {
72 std::vector<const std::string*> neighbor_ids;
73 const char* type = stats.type();
74 if (type == RTCCertificateStats::kType) {
75 const auto& certificate = static_cast<const RTCCertificateStats&>(stats);
76 AddIdIfDefined(certificate.issuer_certificate_id, &neighbor_ids);
77 } else if (type == RTCCodecStats::kType) {
78 // RTCCodecStats does not have any neighbor references.
79 } else if (type == RTCDataChannelStats::kType) {
80 // RTCDataChannelStats does not have any neighbor references.
81 } else if (type == RTCIceCandidatePairStats::kType) {
82 const auto& candidate_pair =
83 static_cast<const RTCIceCandidatePairStats&>(stats);
84 AddIdIfDefined(candidate_pair.transport_id, &neighbor_ids);
85 AddIdIfDefined(candidate_pair.local_candidate_id, &neighbor_ids);
86 AddIdIfDefined(candidate_pair.remote_candidate_id, &neighbor_ids);
87 } else if (type == RTCLocalIceCandidateStats::kType ||
88 type == RTCRemoteIceCandidateStats::kType) {
89 const auto& local_or_remote_candidate =
90 static_cast<const RTCIceCandidateStats&>(stats);
91 AddIdIfDefined(local_or_remote_candidate.transport_id, &neighbor_ids);
92 } else if (type == RTCMediaStreamStats::kType) {
93 const auto& stream = static_cast<const RTCMediaStreamStats&>(stats);
94 AddIdsIfDefined(stream.track_ids, &neighbor_ids);
95 } else if (type == RTCMediaStreamTrackStats::kType) {
96 // RTCMediaStreamTrackStats does not have any neighbor references.
97 } else if (type == RTCPeerConnectionStats::kType) {
98 // RTCPeerConnectionStats does not have any neighbor references.
99 } else if (type == RTCInboundRTPStreamStats::kType ||
100 type == RTCOutboundRTPStreamStats::kType) {
101 const auto& rtp = static_cast<const RTCRTPStreamStats&>(stats);
102 AddIdIfDefined(rtp.associate_stats_id, &neighbor_ids);
103 AddIdIfDefined(rtp.track_id, &neighbor_ids);
104 AddIdIfDefined(rtp.transport_id, &neighbor_ids);
105 AddIdIfDefined(rtp.codec_id, &neighbor_ids);
106 } else if (type == RTCTransportStats::kType) {
107 // RTCTransportStats does not have any neighbor references.
108 const auto& transport = static_cast<const RTCTransportStats&>(stats);
109 AddIdIfDefined(transport.rtcp_transport_stats_id, &neighbor_ids);
110 AddIdIfDefined(transport.selected_candidate_pair_id, &neighbor_ids);
111 AddIdIfDefined(transport.local_certificate_id, &neighbor_ids);
112 AddIdIfDefined(transport.remote_certificate_id, &neighbor_ids);
113 } else {
114 RTC_NOTREACHED() << "Unrecognized type: " << type;
115 }
116 return neighbor_ids;
117}
118
119} // namespace webrtc