hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2016 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 | |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 11 | #include "api/stats/rtc_stats.h" |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 12 | |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 13 | #include <cmath> |
Yves Gerey | 3e70781 | 2018-11-28 16:47:49 +0100 | [diff] [blame] | 14 | #include <cstdint> |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 15 | #include <cstring> |
Yves Gerey | 3e70781 | 2018-11-28 16:47:49 +0100 | [diff] [blame] | 16 | #include <iostream> |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 17 | |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 18 | #include "rtc_base/checks.h" |
Sam Zackrisson | b45bdb5 | 2018-10-02 16:25:59 +0200 | [diff] [blame] | 19 | #include "rtc_base/strings/json.h" |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 20 | #include "stats/test/rtc_test_stats.h" |
Yves Gerey | 3e70781 | 2018-11-28 16:47:49 +0100 | [diff] [blame] | 21 | #include "test/gtest.h" |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 22 | |
| 23 | namespace webrtc { |
| 24 | |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 25 | namespace { |
| 26 | |
| 27 | // JSON stores numbers as floating point numbers with 53 significant bits, which |
| 28 | // amounts to about 15.95 decimal digits. Thus, when comparing large numbers |
| 29 | // processed by JSON, that's all the precision we should expect. |
| 30 | const double JSON_EPSILON = 1e-15; |
| 31 | |
| 32 | // We do this since Google Test doesn't support relative error. |
| 33 | // This is computed as follows: |
| 34 | // If |a - b| / |a| < EPS, then |a - b| < |a| * EPS, so |a| * EPS is the |
| 35 | // maximum expected error. |
| 36 | double GetExpectedError(const double expected_value) { |
| 37 | return JSON_EPSILON * fabs(expected_value); |
| 38 | } |
| 39 | |
| 40 | } // namespace |
| 41 | |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 42 | class RTCChildStats : public RTCStats { |
| 43 | public: |
hbos | fc5e050 | 2016-10-06 02:06:10 -0700 | [diff] [blame] | 44 | WEBRTC_RTCSTATS_DECL(); |
| 45 | |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 46 | RTCChildStats(const std::string& id, Timestamp timestamp) |
| 47 | : RTCStats(id, timestamp), child_int("childInt") {} |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 48 | |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 49 | RTCStatsMember<int32_t> child_int; |
| 50 | }; |
| 51 | |
Mirko Bonadei | c4dd730 | 2019-02-25 09:12:02 +0100 | [diff] [blame] | 52 | WEBRTC_RTCSTATS_IMPL(RTCChildStats, RTCStats, "child-stats", &child_int) |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 53 | |
| 54 | class RTCGrandChildStats : public RTCChildStats { |
| 55 | public: |
hbos | fc5e050 | 2016-10-06 02:06:10 -0700 | [diff] [blame] | 56 | WEBRTC_RTCSTATS_DECL(); |
| 57 | |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 58 | RTCGrandChildStats(const std::string& id, Timestamp timestamp) |
| 59 | : RTCChildStats(id, timestamp), grandchild_int("grandchildInt") {} |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 60 | |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 61 | RTCStatsMember<int32_t> grandchild_int; |
| 62 | }; |
| 63 | |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 64 | WEBRTC_RTCSTATS_IMPL(RTCGrandChildStats, |
| 65 | RTCChildStats, |
| 66 | "grandchild-stats", |
Mirko Bonadei | c4dd730 | 2019-02-25 09:12:02 +0100 | [diff] [blame] | 67 | &grandchild_int) |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 68 | |
| 69 | TEST(RTCStatsTest, RTCStatsAndMembers) { |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 70 | RTCTestStats stats("testId", Timestamp::Micros(42)); |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 71 | EXPECT_EQ(stats.id(), "testId"); |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 72 | EXPECT_EQ(stats.timestamp().us(), static_cast<int64_t>(42)); |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 73 | std::vector<const RTCStatsMemberInterface*> members = stats.Members(); |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 74 | EXPECT_EQ(members.size(), static_cast<size_t>(16)); |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 75 | for (const RTCStatsMemberInterface* member : members) { |
| 76 | EXPECT_FALSE(member->is_defined()); |
| 77 | } |
hbos | b20f387 | 2016-10-04 14:37:11 -0700 | [diff] [blame] | 78 | stats.m_bool = true; |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 79 | stats.m_int32 = 123; |
| 80 | stats.m_uint32 = 123; |
| 81 | stats.m_int64 = 123; |
| 82 | stats.m_uint64 = 123; |
| 83 | stats.m_double = 123.0; |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 84 | stats.m_string = std::string("123"); |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 85 | |
hbos | b20f387 | 2016-10-04 14:37:11 -0700 | [diff] [blame] | 86 | std::vector<bool> sequence_bool; |
| 87 | sequence_bool.push_back(true); |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 88 | std::vector<int32_t> sequence_int32; |
| 89 | sequence_int32.push_back(static_cast<int32_t>(1)); |
| 90 | std::vector<uint32_t> sequence_uint32; |
| 91 | sequence_uint32.push_back(static_cast<uint32_t>(2)); |
| 92 | std::vector<int64_t> sequence_int64; |
| 93 | sequence_int64.push_back(static_cast<int64_t>(3)); |
| 94 | std::vector<uint64_t> sequence_uint64; |
| 95 | sequence_uint64.push_back(static_cast<uint64_t>(4)); |
| 96 | std::vector<double> sequence_double; |
| 97 | sequence_double.push_back(5.0); |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 98 | std::vector<std::string> sequence_string; |
hbos | 8faf9e0 | 2016-09-15 06:52:43 -0700 | [diff] [blame] | 99 | sequence_string.push_back(std::string("six")); |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 100 | |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 101 | std::map<std::string, uint64_t> map_string_uint64{{"seven", 8}}; |
| 102 | std::map<std::string, double> map_string_double{{"nine", 10.0}}; |
| 103 | |
hbos | b20f387 | 2016-10-04 14:37:11 -0700 | [diff] [blame] | 104 | stats.m_sequence_bool = sequence_bool; |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 105 | stats.m_sequence_int32 = sequence_int32; |
| 106 | stats.m_sequence_uint32 = sequence_uint32; |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 107 | EXPECT_FALSE(stats.m_sequence_int64.is_defined()); |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 108 | stats.m_sequence_int64 = sequence_int64; |
| 109 | stats.m_sequence_uint64 = sequence_uint64; |
| 110 | stats.m_sequence_double = sequence_double; |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 111 | stats.m_sequence_string = sequence_string; |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 112 | stats.m_map_string_uint64 = map_string_uint64; |
| 113 | stats.m_map_string_double = map_string_double; |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 114 | for (const RTCStatsMemberInterface* member : members) { |
| 115 | EXPECT_TRUE(member->is_defined()); |
| 116 | } |
hbos | b20f387 | 2016-10-04 14:37:11 -0700 | [diff] [blame] | 117 | EXPECT_EQ(*stats.m_bool, true); |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 118 | EXPECT_EQ(*stats.m_int32, static_cast<int32_t>(123)); |
| 119 | EXPECT_EQ(*stats.m_uint32, static_cast<uint32_t>(123)); |
| 120 | EXPECT_EQ(*stats.m_int64, static_cast<int64_t>(123)); |
| 121 | EXPECT_EQ(*stats.m_uint64, static_cast<uint64_t>(123)); |
| 122 | EXPECT_EQ(*stats.m_double, 123.0); |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 123 | EXPECT_EQ(*stats.m_string, std::string("123")); |
hbos | b20f387 | 2016-10-04 14:37:11 -0700 | [diff] [blame] | 124 | EXPECT_EQ(*stats.m_sequence_bool, sequence_bool); |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 125 | EXPECT_EQ(*stats.m_sequence_int32, sequence_int32); |
| 126 | EXPECT_EQ(*stats.m_sequence_uint32, sequence_uint32); |
| 127 | EXPECT_EQ(*stats.m_sequence_int64, sequence_int64); |
| 128 | EXPECT_EQ(*stats.m_sequence_uint64, sequence_uint64); |
| 129 | EXPECT_EQ(*stats.m_sequence_double, sequence_double); |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 130 | EXPECT_EQ(*stats.m_sequence_string, sequence_string); |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 131 | EXPECT_EQ(*stats.m_map_string_uint64, map_string_uint64); |
| 132 | EXPECT_EQ(*stats.m_map_string_double, map_string_double); |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 133 | |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 134 | int32_t numbers[] = {4, 8, 15, 16, 23, 42}; |
hbos | fdafab8 | 2016-09-14 06:02:13 -0700 | [diff] [blame] | 135 | std::vector<int32_t> numbers_sequence(&numbers[0], &numbers[6]); |
| 136 | stats.m_sequence_int32->clear(); |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 137 | stats.m_sequence_int32->insert(stats.m_sequence_int32->end(), |
| 138 | numbers_sequence.begin(), |
| 139 | numbers_sequence.end()); |
| 140 | EXPECT_EQ(*stats.m_sequence_int32, numbers_sequence); |
| 141 | } |
| 142 | |
hbos | 67c8bc4 | 2016-10-25 04:31:23 -0700 | [diff] [blame] | 143 | TEST(RTCStatsTest, EqualityOperator) { |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 144 | RTCTestStats empty_stats("testId", Timestamp::Micros(123)); |
hbos | 67c8bc4 | 2016-10-25 04:31:23 -0700 | [diff] [blame] | 145 | EXPECT_EQ(empty_stats, empty_stats); |
| 146 | |
| 147 | RTCTestStats stats_with_all_values = empty_stats; |
| 148 | stats_with_all_values.m_bool = true; |
| 149 | stats_with_all_values.m_int32 = 123; |
| 150 | stats_with_all_values.m_uint32 = 123; |
| 151 | stats_with_all_values.m_int64 = 123; |
| 152 | stats_with_all_values.m_uint64 = 123; |
| 153 | stats_with_all_values.m_double = 123.0; |
| 154 | stats_with_all_values.m_string = "123"; |
| 155 | stats_with_all_values.m_sequence_bool = std::vector<bool>(); |
| 156 | stats_with_all_values.m_sequence_int32 = std::vector<int32_t>(); |
| 157 | stats_with_all_values.m_sequence_uint32 = std::vector<uint32_t>(); |
| 158 | stats_with_all_values.m_sequence_int64 = std::vector<int64_t>(); |
| 159 | stats_with_all_values.m_sequence_uint64 = std::vector<uint64_t>(); |
| 160 | stats_with_all_values.m_sequence_double = std::vector<double>(); |
| 161 | stats_with_all_values.m_sequence_string = std::vector<std::string>(); |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 162 | stats_with_all_values.m_map_string_uint64 = std::map<std::string, uint64_t>(); |
| 163 | stats_with_all_values.m_map_string_double = std::map<std::string, double>(); |
hbos | 67c8bc4 | 2016-10-25 04:31:23 -0700 | [diff] [blame] | 164 | EXPECT_NE(stats_with_all_values, empty_stats); |
| 165 | EXPECT_EQ(stats_with_all_values, stats_with_all_values); |
| 166 | EXPECT_NE(stats_with_all_values.m_int32, stats_with_all_values.m_uint32); |
| 167 | |
| 168 | RTCTestStats one_member_different[] = { |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 169 | stats_with_all_values, stats_with_all_values, stats_with_all_values, |
| 170 | stats_with_all_values, stats_with_all_values, stats_with_all_values, |
| 171 | stats_with_all_values, stats_with_all_values, stats_with_all_values, |
| 172 | stats_with_all_values, stats_with_all_values, stats_with_all_values, |
| 173 | stats_with_all_values, stats_with_all_values, |
hbos | 67c8bc4 | 2016-10-25 04:31:23 -0700 | [diff] [blame] | 174 | }; |
| 175 | for (size_t i = 0; i < 14; ++i) { |
| 176 | EXPECT_EQ(stats_with_all_values, one_member_different[i]); |
| 177 | } |
| 178 | one_member_different[0].m_bool = false; |
| 179 | one_member_different[1].m_int32 = 321; |
| 180 | one_member_different[2].m_uint32 = 321; |
| 181 | one_member_different[3].m_int64 = 321; |
| 182 | one_member_different[4].m_uint64 = 321; |
| 183 | one_member_different[5].m_double = 321.0; |
| 184 | one_member_different[6].m_string = "321"; |
| 185 | one_member_different[7].m_sequence_bool->push_back(false); |
| 186 | one_member_different[8].m_sequence_int32->push_back(321); |
| 187 | one_member_different[9].m_sequence_uint32->push_back(321); |
| 188 | one_member_different[10].m_sequence_int64->push_back(321); |
| 189 | one_member_different[11].m_sequence_uint64->push_back(321); |
| 190 | one_member_different[12].m_sequence_double->push_back(321.0); |
| 191 | one_member_different[13].m_sequence_string->push_back("321"); |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 192 | (*one_member_different[13].m_map_string_uint64)["321"] = 321; |
| 193 | (*one_member_different[13].m_map_string_double)["321"] = 321.0; |
hbos | 67c8bc4 | 2016-10-25 04:31:23 -0700 | [diff] [blame] | 194 | for (size_t i = 0; i < 14; ++i) { |
| 195 | EXPECT_NE(stats_with_all_values, one_member_different[i]); |
| 196 | } |
| 197 | |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 198 | RTCTestStats empty_stats_different_id("testId2", Timestamp::Micros(123)); |
hbos | 67c8bc4 | 2016-10-25 04:31:23 -0700 | [diff] [blame] | 199 | EXPECT_NE(empty_stats, empty_stats_different_id); |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 200 | RTCTestStats empty_stats_different_timestamp("testId", |
| 201 | Timestamp::Micros(321)); |
hbos | 0583b28 | 2016-11-30 01:50:14 -0800 | [diff] [blame] | 202 | EXPECT_EQ(empty_stats, empty_stats_different_timestamp); |
hbos | 67c8bc4 | 2016-10-25 04:31:23 -0700 | [diff] [blame] | 203 | |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 204 | RTCChildStats child("childId", Timestamp::Micros(42)); |
| 205 | RTCGrandChildStats grandchild("grandchildId", Timestamp::Micros(42)); |
hbos | 67c8bc4 | 2016-10-25 04:31:23 -0700 | [diff] [blame] | 206 | EXPECT_NE(child, grandchild); |
hbos | 2874796 | 2016-11-21 09:17:41 -0800 | [diff] [blame] | 207 | |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 208 | RTCChildStats stats_with_defined_member("leId", Timestamp::Micros(0)); |
hbos | 2874796 | 2016-11-21 09:17:41 -0800 | [diff] [blame] | 209 | stats_with_defined_member.child_int = 0; |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 210 | RTCChildStats stats_with_undefined_member("leId", Timestamp::Micros(0)); |
hbos | 2874796 | 2016-11-21 09:17:41 -0800 | [diff] [blame] | 211 | EXPECT_NE(stats_with_defined_member, stats_with_undefined_member); |
| 212 | EXPECT_NE(stats_with_undefined_member, stats_with_defined_member); |
hbos | 67c8bc4 | 2016-10-25 04:31:23 -0700 | [diff] [blame] | 213 | } |
| 214 | |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 215 | TEST(RTCStatsTest, RTCStatsGrandChild) { |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 216 | RTCGrandChildStats stats("grandchild", Timestamp::Micros(0.0)); |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 217 | stats.child_int = 1; |
| 218 | stats.grandchild_int = 2; |
| 219 | int32_t sum = 0; |
| 220 | for (const RTCStatsMemberInterface* member : stats.Members()) { |
| 221 | sum += *member->cast_to<const RTCStatsMember<int32_t>>(); |
| 222 | } |
| 223 | EXPECT_EQ(sum, static_cast<int32_t>(3)); |
| 224 | |
| 225 | std::unique_ptr<RTCStats> copy_ptr = stats.copy(); |
| 226 | const RTCGrandChildStats& copy = copy_ptr->cast_to<RTCGrandChildStats>(); |
| 227 | EXPECT_EQ(*copy.child_int, *stats.child_int); |
| 228 | EXPECT_EQ(*copy.grandchild_int, *stats.grandchild_int); |
| 229 | } |
| 230 | |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 231 | TEST(RTCStatsTest, RTCStatsPrintsValidJson) { |
| 232 | std::string id = "statsId"; |
| 233 | int timestamp = 42; |
| 234 | bool m_bool = true; |
| 235 | int m_int32 = 123; |
| 236 | int64_t m_int64 = 1234567890123456499L; |
| 237 | double m_double = 123.4567890123456499; |
| 238 | std::string m_string = "123"; |
| 239 | |
| 240 | std::vector<bool> sequence_bool; |
| 241 | std::vector<int32_t> sequence_int32; |
| 242 | sequence_int32.push_back(static_cast<int32_t>(1)); |
| 243 | std::vector<int64_t> sequence_int64; |
| 244 | sequence_int64.push_back(static_cast<int64_t>(-1234567890123456499L)); |
| 245 | sequence_int64.push_back(static_cast<int64_t>(1)); |
| 246 | sequence_int64.push_back(static_cast<int64_t>(1234567890123456499L)); |
| 247 | std::vector<double> sequence_double; |
| 248 | sequence_double.push_back(123.4567890123456499); |
| 249 | sequence_double.push_back(1234567890123.456499); |
| 250 | std::vector<std::string> sequence_string; |
| 251 | sequence_string.push_back(std::string("four")); |
| 252 | |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 253 | std::map<std::string, uint64_t> map_string_uint64{ |
| 254 | {"long", static_cast<uint64_t>(1234567890123456499L)}}; |
| 255 | std::map<std::string, double> map_string_double{ |
| 256 | {"three", 123.4567890123456499}, {"thirteen", 123.4567890123456499}}; |
| 257 | |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 258 | RTCTestStats stats(id, Timestamp::Micros(timestamp)); |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 259 | stats.m_bool = m_bool; |
| 260 | stats.m_int32 = m_int32; |
| 261 | stats.m_int64 = m_int64; |
| 262 | stats.m_double = m_double; |
| 263 | stats.m_string = m_string; |
| 264 | stats.m_sequence_bool = sequence_bool; |
| 265 | stats.m_sequence_int32 = sequence_int32; |
| 266 | stats.m_sequence_int64 = sequence_int64; |
| 267 | stats.m_sequence_double = sequence_double; |
| 268 | stats.m_sequence_string = sequence_string; |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 269 | stats.m_map_string_uint64 = map_string_uint64; |
| 270 | stats.m_map_string_double = map_string_double; |
Mirko Bonadei | e99f687 | 2021-06-24 15:24:59 +0200 | [diff] [blame] | 271 | std::string json_stats = stats.ToJson(); |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 272 | |
Mirko Bonadei | e99f687 | 2021-06-24 15:24:59 +0200 | [diff] [blame] | 273 | Json::CharReaderBuilder builder; |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 274 | Json::Value json_output; |
Mirko Bonadei | e99f687 | 2021-06-24 15:24:59 +0200 | [diff] [blame] | 275 | std::unique_ptr<Json::CharReader> json_reader(builder.newCharReader()); |
| 276 | EXPECT_TRUE(json_reader->parse(json_stats.c_str(), |
| 277 | json_stats.c_str() + json_stats.size(), |
| 278 | &json_output, nullptr)); |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 279 | |
| 280 | EXPECT_TRUE(rtc::GetStringFromJsonObject(json_output, "id", &id)); |
| 281 | EXPECT_TRUE(rtc::GetIntFromJsonObject(json_output, "timestamp", ×tamp)); |
| 282 | EXPECT_TRUE(rtc::GetBoolFromJsonObject(json_output, "mBool", &m_bool)); |
| 283 | EXPECT_TRUE(rtc::GetIntFromJsonObject(json_output, "mInt32", &m_int32)); |
| 284 | EXPECT_TRUE(rtc::GetDoubleFromJsonObject(json_output, "mDouble", &m_double)); |
| 285 | EXPECT_TRUE(rtc::GetStringFromJsonObject(json_output, "mString", &m_string)); |
| 286 | |
| 287 | Json::Value json_array; |
| 288 | |
| 289 | EXPECT_TRUE( |
| 290 | rtc::GetValueFromJsonObject(json_output, "mSequenceBool", &json_array)); |
| 291 | EXPECT_TRUE(rtc::JsonArrayToBoolVector(json_array, &sequence_bool)); |
| 292 | |
| 293 | EXPECT_TRUE( |
| 294 | rtc::GetValueFromJsonObject(json_output, "mSequenceInt32", &json_array)); |
| 295 | EXPECT_TRUE(rtc::JsonArrayToIntVector(json_array, &sequence_int32)); |
| 296 | |
| 297 | EXPECT_TRUE( |
| 298 | rtc::GetValueFromJsonObject(json_output, "mSequenceDouble", &json_array)); |
| 299 | EXPECT_TRUE(rtc::JsonArrayToDoubleVector(json_array, &sequence_double)); |
| 300 | |
| 301 | EXPECT_TRUE( |
| 302 | rtc::GetValueFromJsonObject(json_output, "mSequenceString", &json_array)); |
| 303 | EXPECT_TRUE(rtc::JsonArrayToStringVector(json_array, &sequence_string)); |
| 304 | |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 305 | Json::Value json_map; |
| 306 | EXPECT_TRUE( |
| 307 | rtc::GetValueFromJsonObject(json_output, "mMapStringDouble", &json_map)); |
| 308 | for (const auto& entry : map_string_double) { |
| 309 | double double_output = 0.0; |
| 310 | EXPECT_TRUE( |
| 311 | rtc::GetDoubleFromJsonObject(json_map, entry.first, &double_output)); |
| 312 | EXPECT_NEAR(double_output, entry.second, GetExpectedError(entry.second)); |
| 313 | } |
| 314 | |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 315 | EXPECT_EQ(id, stats.id()); |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 316 | EXPECT_EQ(timestamp, stats.timestamp().us()); |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 317 | EXPECT_EQ(m_bool, *stats.m_bool); |
| 318 | EXPECT_EQ(m_int32, *stats.m_int32); |
| 319 | EXPECT_EQ(m_string, *stats.m_string); |
| 320 | EXPECT_EQ(sequence_bool, *stats.m_sequence_bool); |
| 321 | EXPECT_EQ(sequence_int32, *stats.m_sequence_int32); |
| 322 | EXPECT_EQ(sequence_string, *stats.m_sequence_string); |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 323 | EXPECT_EQ(map_string_double, *stats.m_map_string_double); |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 324 | |
| 325 | EXPECT_NEAR(m_double, *stats.m_double, GetExpectedError(*stats.m_double)); |
| 326 | |
| 327 | EXPECT_EQ(sequence_double.size(), stats.m_sequence_double->size()); |
| 328 | for (size_t i = 0; i < stats.m_sequence_double->size(); ++i) { |
| 329 | EXPECT_NEAR(sequence_double[i], stats.m_sequence_double->at(i), |
| 330 | GetExpectedError(stats.m_sequence_double->at(i))); |
| 331 | } |
| 332 | |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 333 | EXPECT_EQ(map_string_double.size(), stats.m_map_string_double->size()); |
| 334 | for (const auto& entry : map_string_double) { |
| 335 | auto it = stats.m_map_string_double->find(entry.first); |
| 336 | EXPECT_NE(it, stats.m_map_string_double->end()); |
| 337 | EXPECT_NEAR(entry.second, it->second, GetExpectedError(it->second)); |
| 338 | } |
| 339 | |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 340 | // We read mInt64 as double since JSON stores all numbers as doubles, so there |
| 341 | // is not enough precision to represent large numbers. |
| 342 | double m_int64_as_double; |
| 343 | std::vector<double> sequence_int64_as_double; |
| 344 | |
| 345 | EXPECT_TRUE( |
| 346 | rtc::GetDoubleFromJsonObject(json_output, "mInt64", &m_int64_as_double)); |
| 347 | |
| 348 | EXPECT_TRUE( |
| 349 | rtc::GetValueFromJsonObject(json_output, "mSequenceInt64", &json_array)); |
| 350 | EXPECT_TRUE( |
| 351 | rtc::JsonArrayToDoubleVector(json_array, &sequence_int64_as_double)); |
| 352 | |
| 353 | double stats_m_int64_as_double = static_cast<double>(*stats.m_int64); |
| 354 | EXPECT_NEAR(m_int64_as_double, stats_m_int64_as_double, |
| 355 | GetExpectedError(stats_m_int64_as_double)); |
| 356 | |
| 357 | EXPECT_EQ(sequence_int64_as_double.size(), stats.m_sequence_int64->size()); |
| 358 | for (size_t i = 0; i < stats.m_sequence_int64->size(); ++i) { |
| 359 | const double stats_value_as_double = |
| 360 | static_cast<double>((*stats.m_sequence_int64)[i]); |
| 361 | EXPECT_NEAR(sequence_int64_as_double[i], stats_value_as_double, |
| 362 | GetExpectedError(stats_value_as_double)); |
| 363 | } |
| 364 | |
Byoungchan Lee | 0a52ede | 2021-05-22 08:41:02 +0900 | [diff] [blame] | 365 | // Similarly, read Uint64 as double |
| 366 | EXPECT_TRUE( |
| 367 | rtc::GetValueFromJsonObject(json_output, "mMapStringUint64", &json_map)); |
| 368 | for (const auto& entry : map_string_uint64) { |
| 369 | const double stats_value_as_double = |
| 370 | static_cast<double>((*stats.m_map_string_uint64)[entry.first]); |
| 371 | double double_output = 0.0; |
| 372 | EXPECT_TRUE( |
| 373 | rtc::GetDoubleFromJsonObject(json_map, entry.first, &double_output)); |
| 374 | EXPECT_NEAR(double_output, stats_value_as_double, |
| 375 | GetExpectedError(stats_value_as_double)); |
| 376 | } |
| 377 | |
ehmaldonado | 35a872c | 2017-07-28 07:29:12 -0700 | [diff] [blame] | 378 | // Neither stats.m_uint32 nor stats.m_uint64 are defined, so "mUint64" and |
| 379 | // "mUint32" should not be part of the generated JSON object. |
| 380 | int m_uint32; |
| 381 | int m_uint64; |
| 382 | EXPECT_FALSE(stats.m_uint32.is_defined()); |
| 383 | EXPECT_FALSE(stats.m_uint64.is_defined()); |
| 384 | EXPECT_FALSE(rtc::GetIntFromJsonObject(json_output, "mUint32", &m_uint32)); |
| 385 | EXPECT_FALSE(rtc::GetIntFromJsonObject(json_output, "mUint64", &m_uint64)); |
| 386 | |
| 387 | std::cout << stats.ToJson() << std::endl; |
| 388 | } |
| 389 | |
Taylor Brandstetter | e275174 | 2018-06-25 13:42:44 -0700 | [diff] [blame] | 390 | TEST(RTCStatsTest, IsStandardized) { |
| 391 | RTCStatsMember<int32_t> standardized("standardized"); |
| 392 | RTCNonStandardStatsMember<int32_t> unstandardized("unstandardized"); |
| 393 | EXPECT_TRUE(standardized.is_standardized()); |
| 394 | EXPECT_FALSE(unstandardized.is_standardized()); |
| 395 | } |
| 396 | |
Evan Shrubsole | 22e0e61 | 2022-10-11 13:18:38 +0000 | [diff] [blame] | 397 | TEST(RTCStatsTest, IsSequence) { |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 398 | RTCTestStats stats("statsId", Timestamp::Micros(42)); |
Evan Shrubsole | 22e0e61 | 2022-10-11 13:18:38 +0000 | [diff] [blame] | 399 | EXPECT_FALSE(stats.m_bool.is_sequence()); |
| 400 | EXPECT_FALSE(stats.m_int32.is_sequence()); |
| 401 | EXPECT_FALSE(stats.m_uint32.is_sequence()); |
| 402 | EXPECT_FALSE(stats.m_int64.is_sequence()); |
| 403 | EXPECT_FALSE(stats.m_uint64.is_sequence()); |
| 404 | EXPECT_FALSE(stats.m_double.is_sequence()); |
| 405 | EXPECT_FALSE(stats.m_string.is_sequence()); |
| 406 | EXPECT_TRUE(stats.m_sequence_bool.is_sequence()); |
| 407 | EXPECT_TRUE(stats.m_sequence_int32.is_sequence()); |
| 408 | EXPECT_TRUE(stats.m_sequence_uint32.is_sequence()); |
| 409 | EXPECT_TRUE(stats.m_sequence_int64.is_sequence()); |
| 410 | EXPECT_TRUE(stats.m_sequence_uint64.is_sequence()); |
| 411 | EXPECT_TRUE(stats.m_sequence_double.is_sequence()); |
| 412 | EXPECT_TRUE(stats.m_sequence_string.is_sequence()); |
| 413 | EXPECT_FALSE(stats.m_map_string_uint64.is_sequence()); |
| 414 | EXPECT_FALSE(stats.m_map_string_double.is_sequence()); |
| 415 | } |
| 416 | |
| 417 | TEST(RTCStatsTest, Type) { |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 418 | RTCTestStats stats("statsId", Timestamp::Micros(42)); |
Evan Shrubsole | 22e0e61 | 2022-10-11 13:18:38 +0000 | [diff] [blame] | 419 | EXPECT_EQ(RTCStatsMemberInterface::kBool, stats.m_bool.type()); |
| 420 | EXPECT_EQ(RTCStatsMemberInterface::kInt32, stats.m_int32.type()); |
| 421 | EXPECT_EQ(RTCStatsMemberInterface::kUint32, stats.m_uint32.type()); |
| 422 | EXPECT_EQ(RTCStatsMemberInterface::kInt64, stats.m_int64.type()); |
| 423 | EXPECT_EQ(RTCStatsMemberInterface::kUint64, stats.m_uint64.type()); |
| 424 | EXPECT_EQ(RTCStatsMemberInterface::kDouble, stats.m_double.type()); |
| 425 | EXPECT_EQ(RTCStatsMemberInterface::kString, stats.m_string.type()); |
| 426 | EXPECT_EQ(RTCStatsMemberInterface::kSequenceBool, |
| 427 | stats.m_sequence_bool.type()); |
| 428 | EXPECT_EQ(RTCStatsMemberInterface::kSequenceInt32, |
| 429 | stats.m_sequence_int32.type()); |
| 430 | EXPECT_EQ(RTCStatsMemberInterface::kSequenceUint32, |
| 431 | stats.m_sequence_uint32.type()); |
| 432 | EXPECT_EQ(RTCStatsMemberInterface::kSequenceInt64, |
| 433 | stats.m_sequence_int64.type()); |
| 434 | EXPECT_EQ(RTCStatsMemberInterface::kSequenceUint64, |
| 435 | stats.m_sequence_uint64.type()); |
| 436 | EXPECT_EQ(RTCStatsMemberInterface::kSequenceDouble, |
| 437 | stats.m_sequence_double.type()); |
| 438 | EXPECT_EQ(RTCStatsMemberInterface::kSequenceString, |
| 439 | stats.m_sequence_string.type()); |
| 440 | EXPECT_EQ(RTCStatsMemberInterface::kMapStringUint64, |
| 441 | stats.m_map_string_uint64.type()); |
| 442 | EXPECT_EQ(RTCStatsMemberInterface::kMapStringDouble, |
| 443 | stats.m_map_string_double.type()); |
| 444 | } |
| 445 | |
| 446 | TEST(RTCStatsTest, IsString) { |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 447 | RTCTestStats stats("statsId", Timestamp::Micros(42)); |
Evan Shrubsole | 22e0e61 | 2022-10-11 13:18:38 +0000 | [diff] [blame] | 448 | EXPECT_TRUE(stats.m_string.is_string()); |
| 449 | EXPECT_FALSE(stats.m_bool.is_string()); |
| 450 | EXPECT_FALSE(stats.m_int32.is_string()); |
| 451 | EXPECT_FALSE(stats.m_uint32.is_string()); |
| 452 | EXPECT_FALSE(stats.m_int64.is_string()); |
| 453 | EXPECT_FALSE(stats.m_uint64.is_string()); |
| 454 | EXPECT_FALSE(stats.m_double.is_string()); |
| 455 | EXPECT_FALSE(stats.m_sequence_bool.is_string()); |
| 456 | EXPECT_FALSE(stats.m_sequence_int32.is_string()); |
| 457 | EXPECT_FALSE(stats.m_sequence_uint32.is_string()); |
| 458 | EXPECT_FALSE(stats.m_sequence_int64.is_string()); |
| 459 | EXPECT_FALSE(stats.m_sequence_uint64.is_string()); |
| 460 | EXPECT_FALSE(stats.m_sequence_double.is_string()); |
| 461 | EXPECT_FALSE(stats.m_sequence_string.is_string()); |
| 462 | EXPECT_FALSE(stats.m_map_string_uint64.is_string()); |
| 463 | EXPECT_FALSE(stats.m_map_string_double.is_string()); |
| 464 | } |
| 465 | |
| 466 | TEST(RTCStatsTest, ValueToString) { |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 467 | RTCTestStats stats("statsId", Timestamp::Micros(42)); |
Evan Shrubsole | 22e0e61 | 2022-10-11 13:18:38 +0000 | [diff] [blame] | 468 | stats.m_bool = true; |
| 469 | EXPECT_EQ("true", stats.m_bool.ValueToString()); |
| 470 | |
| 471 | stats.m_string = "foo"; |
| 472 | EXPECT_EQ("foo", stats.m_string.ValueToString()); |
| 473 | stats.m_int32 = -32; |
| 474 | EXPECT_EQ("-32", stats.m_int32.ValueToString()); |
| 475 | stats.m_uint32 = 32; |
| 476 | EXPECT_EQ("32", stats.m_uint32.ValueToString()); |
| 477 | stats.m_int64 = -64; |
| 478 | EXPECT_EQ("-64", stats.m_int64.ValueToString()); |
| 479 | stats.m_uint64 = 64; |
| 480 | EXPECT_EQ("64", stats.m_uint64.ValueToString()); |
| 481 | stats.m_double = 0.5; |
| 482 | EXPECT_EQ("0.5", stats.m_double.ValueToString()); |
| 483 | stats.m_sequence_bool = {true, false}; |
| 484 | EXPECT_EQ("[true,false]", stats.m_sequence_bool.ValueToString()); |
| 485 | stats.m_sequence_int32 = {-32, 32}; |
| 486 | EXPECT_EQ("[-32,32]", stats.m_sequence_int32.ValueToString()); |
| 487 | stats.m_sequence_uint32 = {64, 32}; |
| 488 | EXPECT_EQ("[64,32]", stats.m_sequence_uint32.ValueToString()); |
| 489 | stats.m_sequence_int64 = {-64, 32}; |
| 490 | EXPECT_EQ("[-64,32]", stats.m_sequence_int64.ValueToString()); |
| 491 | stats.m_sequence_uint64 = {16, 32}; |
| 492 | EXPECT_EQ("[16,32]", stats.m_sequence_uint64.ValueToString()); |
| 493 | stats.m_sequence_double = {0.5, 0.25}; |
| 494 | EXPECT_EQ("[0.5,0.25]", stats.m_sequence_double.ValueToString()); |
| 495 | stats.m_sequence_string = {"foo", "bar"}; |
| 496 | EXPECT_EQ("[\"foo\",\"bar\"]", stats.m_sequence_string.ValueToString()); |
| 497 | stats.m_map_string_uint64 = std::map<std::string, uint64_t>(); |
| 498 | stats.m_map_string_uint64->emplace("foo", 32); |
| 499 | stats.m_map_string_uint64->emplace("bar", 64); |
| 500 | EXPECT_EQ("{bar:64,foo:32}", stats.m_map_string_uint64.ValueToString()); |
| 501 | stats.m_map_string_double = std::map<std::string, double>(); |
| 502 | stats.m_map_string_double->emplace("foo", 0.5); |
| 503 | stats.m_map_string_double->emplace("bar", 0.25); |
| 504 | EXPECT_EQ("{bar:0.25,foo:0.5}", stats.m_map_string_double.ValueToString()); |
| 505 | } |
| 506 | |
Evan Shrubsole | 6c733ee | 2022-10-12 15:11:14 +0000 | [diff] [blame] | 507 | TEST(RTCStatsTest, RestrictedStatsTest) { |
| 508 | RTCStatsMember<bool> unrestricted("unrestricted"); |
| 509 | EXPECT_EQ(unrestricted.exposure_criteria(), StatExposureCriteria::kAlways); |
| 510 | RTCRestrictedStatsMember<bool, StatExposureCriteria::kHardwareCapability> |
| 511 | restricted("restricted"); |
| 512 | EXPECT_EQ(restricted.exposure_criteria(), |
| 513 | StatExposureCriteria::kHardwareCapability); |
| 514 | |
| 515 | unrestricted = true; |
| 516 | restricted = true; |
| 517 | EXPECT_NE(unrestricted, restricted) |
| 518 | << "These can not be equal as they have different exposure criteria."; |
| 519 | } |
| 520 | |
Jakob Ivarsson | 758d946 | 2019-03-19 15:38:49 +0100 | [diff] [blame] | 521 | TEST(RTCStatsTest, NonStandardGroupId) { |
Jakob Ivarsson | aa023e2 | 2019-03-27 10:17:31 +0100 | [diff] [blame] | 522 | auto group_id = NonStandardGroupId::kGroupIdForTesting; |
Jakob Ivarsson | 758d946 | 2019-03-19 15:38:49 +0100 | [diff] [blame] | 523 | RTCNonStandardStatsMember<int32_t> with_group_id("stat", {group_id}); |
| 524 | std::vector<NonStandardGroupId> expected_ids({group_id}); |
| 525 | EXPECT_EQ(expected_ids, with_group_id.group_ids()); |
| 526 | |
| 527 | RTCNonStandardStatsMember<int32_t> without_group_id("stat"); |
| 528 | EXPECT_TRUE(without_group_id.group_ids().empty()); |
| 529 | } |
| 530 | |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 531 | // Death tests. |
| 532 | // Disabled on Android because death tests misbehave on Android, see |
| 533 | // base/test/gtest_util.h. |
| 534 | #if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID) |
| 535 | |
| 536 | TEST(RTCStatsDeathTest, ValueOfUndefinedMember) { |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 537 | RTCTestStats stats("testId", Timestamp::Micros(0)); |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 538 | EXPECT_FALSE(stats.m_int32.is_defined()); |
| 539 | EXPECT_DEATH(*stats.m_int32, ""); |
| 540 | } |
| 541 | |
| 542 | TEST(RTCStatsDeathTest, InvalidCasting) { |
Philipp Hancke | b81823a | 2023-01-04 15:17:42 +0100 | [diff] [blame] | 543 | RTCGrandChildStats stats("grandchild", Timestamp::Micros(0.0)); |
hbos | 615d301 | 2016-08-24 01:33:13 -0700 | [diff] [blame] | 544 | EXPECT_DEATH(stats.cast_to<RTCChildStats>(), ""); |
| 545 | } |
| 546 | |
| 547 | #endif // RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID) |
| 548 | |
| 549 | } // namespace webrtc |