2018-09-16 14:05:51 -04:00
|
|
|
// Copyright 2017 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2020-05-08 17:09:29 -04:00
|
|
|
#include <nlohmann/json.hpp>
|
2018-09-16 14:05:51 -04:00
|
|
|
#include "common/detached_tasks.h"
|
|
|
|
#include "web_service/telemetry_json.h"
|
|
|
|
#include "web_service/web_backend.h"
|
2020-08-22 19:23:22 -04:00
|
|
|
#include "web_service/web_result.h"
|
2018-09-16 14:05:51 -04:00
|
|
|
|
|
|
|
namespace WebService {
|
|
|
|
|
2020-08-18 14:21:50 -04:00
|
|
|
namespace Telemetry = Common::Telemetry;
|
|
|
|
|
2018-10-10 21:04:19 -04:00
|
|
|
struct TelemetryJson::Impl {
|
|
|
|
Impl(std::string host, std::string username, std::string token)
|
|
|
|
: host{std::move(host)}, username{std::move(username)}, token{std::move(token)} {}
|
2018-09-17 11:16:01 -04:00
|
|
|
|
2018-10-10 21:04:19 -04:00
|
|
|
nlohmann::json& TopSection() {
|
|
|
|
return sections[static_cast<u8>(Telemetry::FieldType::None)];
|
|
|
|
}
|
2018-09-16 14:05:51 -04:00
|
|
|
|
2018-10-10 21:04:19 -04:00
|
|
|
const nlohmann::json& TopSection() const {
|
|
|
|
return sections[static_cast<u8>(Telemetry::FieldType::None)];
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
void Serialize(Telemetry::FieldType type, const std::string& name, T value) {
|
|
|
|
sections[static_cast<u8>(type)][name] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SerializeSection(Telemetry::FieldType type, const std::string& name) {
|
|
|
|
TopSection()[name] = sections[static_cast<unsigned>(type)];
|
|
|
|
}
|
|
|
|
|
|
|
|
nlohmann::json output;
|
|
|
|
std::array<nlohmann::json, 7> sections;
|
|
|
|
std::string host;
|
|
|
|
std::string username;
|
|
|
|
std::string token;
|
|
|
|
};
|
|
|
|
|
|
|
|
TelemetryJson::TelemetryJson(std::string host, std::string username, std::string token)
|
|
|
|
: impl{std::make_unique<Impl>(std::move(host), std::move(username), std::move(token))} {}
|
|
|
|
TelemetryJson::~TelemetryJson() = default;
|
2018-09-16 14:05:51 -04:00
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<bool>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<double>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<float>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<u8>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<u16>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<u32>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<u64>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<s8>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<s16>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<s32>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<s64>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<std::string>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<const char*>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), std::string(field.GetValue()));
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Visit(const Telemetry::Field<std::chrono::microseconds>& field) {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->Serialize(field.GetType(), field.GetName(), field.GetValue().count());
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TelemetryJson::Complete() {
|
2018-10-10 21:04:19 -04:00
|
|
|
impl->SerializeSection(Telemetry::FieldType::App, "App");
|
|
|
|
impl->SerializeSection(Telemetry::FieldType::Session, "Session");
|
|
|
|
impl->SerializeSection(Telemetry::FieldType::Performance, "Performance");
|
|
|
|
impl->SerializeSection(Telemetry::FieldType::UserConfig, "UserConfig");
|
|
|
|
impl->SerializeSection(Telemetry::FieldType::UserSystem, "UserSystem");
|
|
|
|
|
|
|
|
auto content = impl->TopSection().dump();
|
2018-09-16 14:05:51 -04:00
|
|
|
// Send the telemetry async but don't handle the errors since they were written to the log
|
2018-10-26 10:21:45 -04:00
|
|
|
Common::DetachedTasks::AddTask([host{impl->host}, content]() {
|
|
|
|
Client{host, "", ""}.PostJson("/telemetry", content, true);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TelemetryJson::SubmitTestcase() {
|
|
|
|
impl->SerializeSection(Telemetry::FieldType::App, "App");
|
|
|
|
impl->SerializeSection(Telemetry::FieldType::Session, "Session");
|
|
|
|
impl->SerializeSection(Telemetry::FieldType::UserFeedback, "UserFeedback");
|
|
|
|
impl->SerializeSection(Telemetry::FieldType::UserSystem, "UserSystem");
|
2020-01-21 14:38:36 -05:00
|
|
|
impl->SerializeSection(Telemetry::FieldType::UserConfig, "UserConfig");
|
2018-10-26 10:21:45 -04:00
|
|
|
|
|
|
|
auto content = impl->TopSection().dump();
|
|
|
|
Client client(impl->host, impl->username, impl->token);
|
|
|
|
auto value = client.PostJson("/gamedb/testcase", content, false);
|
|
|
|
|
2020-08-22 19:23:22 -04:00
|
|
|
return value.result_code == WebResult::Code::Success;
|
2018-09-16 14:05:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace WebService
|