2020-08-14 16:58:22 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2016 The WebRTC Project Authors. All rights reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
|
* tree. An additional intellectual property rights grant can be found
|
|
|
|
* in the file PATENTS. All contributing project authors may
|
|
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "api/stats/rtc_stats.h"
|
|
|
|
|
|
|
|
#include <cstdio>
|
|
|
|
|
|
|
|
#include "rtc_base/arraysize.h"
|
|
|
|
#include "rtc_base/string_encode.h"
|
|
|
|
#include "rtc_base/strings/string_builder.h"
|
|
|
|
|
|
|
|
namespace webrtc {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Produces "[a,b,c]". Works for non-vector |RTCStatsMemberInterface::Type|
|
|
|
|
// types.
|
|
|
|
template <typename T>
|
|
|
|
std::string VectorToString(const std::vector<T>& vector) {
|
|
|
|
rtc::StringBuilder sb;
|
|
|
|
sb << "[";
|
|
|
|
const char* separator = "";
|
|
|
|
for (const T& element : vector) {
|
|
|
|
sb << separator << rtc::ToString(element);
|
|
|
|
separator = ",";
|
|
|
|
}
|
|
|
|
sb << "]";
|
|
|
|
return sb.Release();
|
|
|
|
}
|
|
|
|
|
2020-12-23 07:48:30 +00:00
|
|
|
// This overload is required because std::vector<bool> range loops don't
|
|
|
|
// return references but objects, causing -Wrange-loop-analysis diagnostics.
|
|
|
|
std::string VectorToString(const std::vector<bool>& vector) {
|
|
|
|
rtc::StringBuilder sb;
|
|
|
|
sb << "[";
|
|
|
|
const char* separator = "";
|
|
|
|
for (bool element : vector) {
|
|
|
|
sb << separator << rtc::ToString(element);
|
|
|
|
separator = ",";
|
|
|
|
}
|
|
|
|
sb << "]";
|
|
|
|
return sb.Release();
|
|
|
|
}
|
|
|
|
|
2020-08-14 16:58:22 +00:00
|
|
|
// Produces "[\"a\",\"b\",\"c\"]". Works for vectors of both const char* and
|
|
|
|
// std::string element types.
|
|
|
|
template <typename T>
|
|
|
|
std::string VectorOfStringsToString(const std::vector<T>& strings) {
|
|
|
|
rtc::StringBuilder sb;
|
|
|
|
sb << "[";
|
|
|
|
const char* separator = "";
|
|
|
|
for (const T& element : strings) {
|
|
|
|
sb << separator << "\"" << rtc::ToString(element) << "\"";
|
|
|
|
separator = ",";
|
|
|
|
}
|
|
|
|
sb << "]";
|
|
|
|
return sb.Release();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
std::string ToStringAsDouble(const T value) {
|
|
|
|
// JSON represents numbers as floating point numbers with about 15 decimal
|
|
|
|
// digits of precision.
|
|
|
|
char buf[32];
|
|
|
|
const int len = std::snprintf(&buf[0], arraysize(buf), "%.16g",
|
|
|
|
static_cast<double>(value));
|
|
|
|
RTC_DCHECK_LE(len, arraysize(buf));
|
|
|
|
return std::string(&buf[0], len);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
std::string VectorToStringAsDouble(const std::vector<T>& vector) {
|
|
|
|
rtc::StringBuilder sb;
|
|
|
|
sb << "[";
|
|
|
|
const char* separator = "";
|
|
|
|
for (const T& element : vector) {
|
|
|
|
sb << separator << ToStringAsDouble<T>(element);
|
|
|
|
separator = ",";
|
|
|
|
}
|
|
|
|
sb << "]";
|
|
|
|
return sb.Release();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
bool RTCStats::operator==(const RTCStats& other) const {
|
|
|
|
if (type() != other.type() || id() != other.id())
|
|
|
|
return false;
|
|
|
|
std::vector<const RTCStatsMemberInterface*> members = Members();
|
|
|
|
std::vector<const RTCStatsMemberInterface*> other_members = other.Members();
|
|
|
|
RTC_DCHECK_EQ(members.size(), other_members.size());
|
|
|
|
for (size_t i = 0; i < members.size(); ++i) {
|
|
|
|
const RTCStatsMemberInterface* member = members[i];
|
|
|
|
const RTCStatsMemberInterface* other_member = other_members[i];
|
|
|
|
RTC_DCHECK_EQ(member->type(), other_member->type());
|
|
|
|
RTC_DCHECK_EQ(member->name(), other_member->name());
|
|
|
|
if (*member != *other_member)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RTCStats::operator!=(const RTCStats& other) const {
|
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string RTCStats::ToJson() const {
|
|
|
|
rtc::StringBuilder sb;
|
|
|
|
sb << "{\"type\":\"" << type()
|
|
|
|
<< "\","
|
|
|
|
"\"id\":\""
|
|
|
|
<< id_
|
|
|
|
<< "\","
|
|
|
|
"\"timestamp\":"
|
|
|
|
<< timestamp_us_;
|
|
|
|
for (const RTCStatsMemberInterface* member : Members()) {
|
|
|
|
if (member->is_defined()) {
|
|
|
|
sb << ",\"" << member->name() << "\":";
|
|
|
|
if (member->is_string())
|
|
|
|
sb << "\"" << member->ValueToJson() << "\"";
|
|
|
|
else
|
|
|
|
sb << member->ValueToJson();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sb << "}";
|
|
|
|
return sb.Release();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<const RTCStatsMemberInterface*> RTCStats::Members() const {
|
|
|
|
return MembersOfThisObjectAndAncestors(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<const RTCStatsMemberInterface*>
|
|
|
|
RTCStats::MembersOfThisObjectAndAncestors(size_t additional_capacity) const {
|
|
|
|
std::vector<const RTCStatsMemberInterface*> members;
|
|
|
|
members.reserve(additional_capacity);
|
|
|
|
return members;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define WEBRTC_DEFINE_RTCSTATSMEMBER(T, type, is_seq, is_str, to_str, to_json) \
|
|
|
|
template <> \
|
|
|
|
RTCStatsMemberInterface::Type RTCStatsMember<T>::StaticType() { \
|
|
|
|
return type; \
|
|
|
|
} \
|
|
|
|
template <> \
|
|
|
|
bool RTCStatsMember<T>::is_sequence() const { \
|
|
|
|
return is_seq; \
|
|
|
|
} \
|
|
|
|
template <> \
|
|
|
|
bool RTCStatsMember<T>::is_string() const { \
|
|
|
|
return is_str; \
|
|
|
|
} \
|
|
|
|
template <> \
|
|
|
|
std::string RTCStatsMember<T>::ValueToString() const { \
|
|
|
|
RTC_DCHECK(is_defined_); \
|
|
|
|
return to_str; \
|
|
|
|
} \
|
|
|
|
template <> \
|
|
|
|
std::string RTCStatsMember<T>::ValueToJson() const { \
|
|
|
|
RTC_DCHECK(is_defined_); \
|
|
|
|
return to_json; \
|
|
|
|
} \
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT) RTCStatsMember<T>
|
|
|
|
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(bool,
|
|
|
|
kBool,
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
rtc::ToString(value_),
|
|
|
|
rtc::ToString(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(int32_t,
|
|
|
|
kInt32,
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
rtc::ToString(value_),
|
|
|
|
rtc::ToString(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(uint32_t,
|
|
|
|
kUint32,
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
rtc::ToString(value_),
|
|
|
|
rtc::ToString(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(int64_t,
|
|
|
|
kInt64,
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
rtc::ToString(value_),
|
|
|
|
ToStringAsDouble(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(uint64_t,
|
|
|
|
kUint64,
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
rtc::ToString(value_),
|
|
|
|
ToStringAsDouble(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(double,
|
|
|
|
kDouble,
|
|
|
|
false,
|
|
|
|
false,
|
|
|
|
rtc::ToString(value_),
|
|
|
|
ToStringAsDouble(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(std::string, kString, false, true, value_, value_);
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(std::vector<bool>,
|
|
|
|
kSequenceBool,
|
|
|
|
true,
|
|
|
|
false,
|
|
|
|
VectorToString(value_),
|
|
|
|
VectorToString(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(std::vector<int32_t>,
|
|
|
|
kSequenceInt32,
|
|
|
|
true,
|
|
|
|
false,
|
|
|
|
VectorToString(value_),
|
|
|
|
VectorToString(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(std::vector<uint32_t>,
|
|
|
|
kSequenceUint32,
|
|
|
|
true,
|
|
|
|
false,
|
|
|
|
VectorToString(value_),
|
|
|
|
VectorToString(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(std::vector<int64_t>,
|
|
|
|
kSequenceInt64,
|
|
|
|
true,
|
|
|
|
false,
|
|
|
|
VectorToString(value_),
|
|
|
|
VectorToStringAsDouble(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(std::vector<uint64_t>,
|
|
|
|
kSequenceUint64,
|
|
|
|
true,
|
|
|
|
false,
|
|
|
|
VectorToString(value_),
|
|
|
|
VectorToStringAsDouble(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(std::vector<double>,
|
|
|
|
kSequenceDouble,
|
|
|
|
true,
|
|
|
|
false,
|
|
|
|
VectorToString(value_),
|
|
|
|
VectorToStringAsDouble(value_));
|
|
|
|
WEBRTC_DEFINE_RTCSTATSMEMBER(std::vector<std::string>,
|
|
|
|
kSequenceString,
|
|
|
|
true,
|
|
|
|
false,
|
|
|
|
VectorOfStringsToString(value_),
|
|
|
|
VectorOfStringsToString(value_));
|
|
|
|
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<bool>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<int32_t>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<uint32_t>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<int64_t>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<uint64_t>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<double>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<std::string>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<std::vector<bool>>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<std::vector<int32_t>>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<std::vector<uint32_t>>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<std::vector<int64_t>>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<std::vector<uint64_t>>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<std::vector<double>>;
|
|
|
|
template class RTC_EXPORT_TEMPLATE_DEFINE(RTC_EXPORT)
|
|
|
|
RTCNonStandardStatsMember<std::vector<std::string>>;
|
|
|
|
|
|
|
|
} // namespace webrtc
|