2017-06-27 23:01:49 -04:00
|
|
|
// Copyright 2017 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2017-08-26 19:02:03 -04:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <winsock.h>
|
|
|
|
#endif
|
|
|
|
|
2017-08-24 19:27:13 -04:00
|
|
|
#include <cstdlib>
|
|
|
|
#include <thread>
|
2017-06-27 23:18:52 -04:00
|
|
|
#include <cpr/cpr.h>
|
2017-10-31 10:02:42 +01:00
|
|
|
#include "common/announce_multiplayer_room.h"
|
2017-06-27 23:18:52 -04:00
|
|
|
#include "common/logging/log.h"
|
2017-06-27 23:01:49 -04:00
|
|
|
#include "web_service/web_backend.h"
|
|
|
|
|
|
|
|
namespace WebService {
|
|
|
|
|
2017-07-09 18:37:14 -04:00
|
|
|
static constexpr char API_VERSION[]{"1"};
|
2017-06-27 23:18:52 -04:00
|
|
|
|
2017-08-26 19:02:03 -04:00
|
|
|
static std::unique_ptr<cpr::Session> g_session;
|
2017-08-24 19:27:13 -04:00
|
|
|
|
2017-09-19 03:18:26 +02:00
|
|
|
void Win32WSAStartup() {
|
|
|
|
#ifdef _WIN32
|
|
|
|
// On Windows, CPR/libcurl does not properly initialize Winsock. The below code is used to
|
|
|
|
// initialize Winsock globally, which fixes this problem. Without this, only the first CPR
|
|
|
|
// session will properly be created, and subsequent ones will fail.
|
|
|
|
WSADATA wsa_data;
|
|
|
|
const int wsa_result{WSAStartup(MAKEWORD(2, 2), &wsa_data)};
|
|
|
|
if (wsa_result) {
|
|
|
|
LOG_CRITICAL(WebService, "WSAStartup failed: %d", wsa_result);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-10-31 10:02:42 +01:00
|
|
|
std::future<Common::WebResult> PostJson(const std::string& url, const std::string& data,
|
|
|
|
bool allow_anonymous, const std::string& username,
|
|
|
|
const std::string& token) {
|
2017-08-23 21:09:34 -04:00
|
|
|
if (url.empty()) {
|
|
|
|
LOG_ERROR(WebService, "URL is invalid");
|
2017-10-31 10:02:42 +01:00
|
|
|
return std::async(std::launch::async, []() {
|
|
|
|
return Common::WebResult{Common::WebResult::Code::InvalidURL, "URL is invalid"};
|
|
|
|
});
|
2017-06-27 23:18:52 -04:00
|
|
|
}
|
|
|
|
|
2017-08-23 21:09:34 -04:00
|
|
|
const bool are_credentials_provided{!token.empty() && !username.empty()};
|
|
|
|
if (!allow_anonymous && !are_credentials_provided) {
|
|
|
|
LOG_ERROR(WebService, "Credentials must be provided for authenticated requests");
|
2017-10-31 10:02:42 +01:00
|
|
|
return std::async(std::launch::async, []() {
|
|
|
|
return Common::WebResult{Common::WebResult::Code::CredentialsMissing,
|
|
|
|
"Credentials needed"};
|
|
|
|
});
|
2017-06-27 23:18:52 -04:00
|
|
|
}
|
|
|
|
|
2017-09-19 03:18:26 +02:00
|
|
|
Win32WSAStartup();
|
2017-08-26 19:02:03 -04:00
|
|
|
|
|
|
|
// Built request header
|
|
|
|
cpr::Header header;
|
2017-08-23 21:09:34 -04:00
|
|
|
if (are_credentials_provided) {
|
|
|
|
// Authenticated request if credentials are provided
|
2017-08-26 19:02:03 -04:00
|
|
|
header = {{"Content-Type", "application/json"},
|
|
|
|
{"x-username", username.c_str()},
|
|
|
|
{"x-token", token.c_str()},
|
|
|
|
{"api-version", API_VERSION}};
|
2017-08-23 21:09:34 -04:00
|
|
|
} else {
|
|
|
|
// Otherwise, anonymous request
|
2017-08-26 19:02:03 -04:00
|
|
|
header = cpr::Header{{"Content-Type", "application/json"}, {"api-version", API_VERSION}};
|
2017-06-27 23:18:52 -04:00
|
|
|
}
|
2017-08-26 19:02:03 -04:00
|
|
|
|
|
|
|
// Post JSON asynchronously
|
2017-10-31 10:02:42 +01:00
|
|
|
return cpr::PostCallback(
|
2017-09-19 03:18:26 +02:00
|
|
|
[](cpr::Response r) {
|
|
|
|
if (r.error) {
|
2017-10-31 10:02:42 +01:00
|
|
|
LOG_ERROR(WebService, "POST to %s returned cpr error: %u:%s", r.url.c_str(),
|
2017-09-19 03:18:26 +02:00
|
|
|
static_cast<u32>(r.error.code), r.error.message.c_str());
|
2017-10-31 10:02:42 +01:00
|
|
|
return Common::WebResult{Common::WebResult::Code::CprError, r.error.message};
|
2017-09-19 03:18:26 +02:00
|
|
|
}
|
|
|
|
if (r.status_code >= 400) {
|
2017-10-31 10:02:42 +01:00
|
|
|
LOG_ERROR(WebService, "POST to %s returned error status code: %u", r.url.c_str(),
|
|
|
|
r.status_code);
|
|
|
|
return Common::WebResult{Common::WebResult::Code::HttpError,
|
|
|
|
std::to_string(r.status_code)};
|
2017-09-19 03:18:26 +02:00
|
|
|
}
|
|
|
|
if (r.header["content-type"].find("application/json") == std::string::npos) {
|
2017-10-31 10:02:42 +01:00
|
|
|
LOG_ERROR(WebService, "POST to %s returned wrong content: %s", r.url.c_str(),
|
2017-09-19 03:18:26 +02:00
|
|
|
r.header["content-type"].c_str());
|
2017-10-31 10:02:42 +01:00
|
|
|
return Common::WebResult{Common::WebResult::Code::WrongContent,
|
|
|
|
r.header["content-type"]};
|
2017-09-19 03:18:26 +02:00
|
|
|
}
|
2017-10-31 10:02:42 +01:00
|
|
|
return Common::WebResult{Common::WebResult::Code::Success, ""};
|
2017-09-19 03:18:26 +02:00
|
|
|
},
|
|
|
|
cpr::Url{url}, cpr::Body{data}, header);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
std::future<T> GetJson(std::function<T(const std::string&)> func, const std::string& url,
|
|
|
|
bool allow_anonymous, const std::string& username,
|
|
|
|
const std::string& token) {
|
|
|
|
if (url.empty()) {
|
|
|
|
LOG_ERROR(WebService, "URL is invalid");
|
|
|
|
return std::async(std::launch::async, [func{std::move(func)}]() { return func(""); });
|
|
|
|
}
|
|
|
|
|
|
|
|
const bool are_credentials_provided{!token.empty() && !username.empty()};
|
|
|
|
if (!allow_anonymous && !are_credentials_provided) {
|
|
|
|
LOG_ERROR(WebService, "Credentials must be provided for authenticated requests");
|
|
|
|
return std::async(std::launch::async, [func{std::move(func)}]() { return func(""); });
|
|
|
|
}
|
|
|
|
|
|
|
|
Win32WSAStartup();
|
|
|
|
|
|
|
|
// Built request header
|
|
|
|
cpr::Header header;
|
|
|
|
if (are_credentials_provided) {
|
|
|
|
// Authenticated request if credentials are provided
|
|
|
|
header = {{"Content-Type", "application/json"},
|
|
|
|
{"x-username", username.c_str()},
|
|
|
|
{"x-token", token.c_str()},
|
|
|
|
{"api-version", API_VERSION}};
|
|
|
|
} else {
|
|
|
|
// Otherwise, anonymous request
|
|
|
|
header = cpr::Header{{"Content-Type", "application/json"}, {"api-version", API_VERSION}};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get JSON asynchronously
|
|
|
|
return cpr::GetCallback(
|
|
|
|
[func{std::move(func)}](cpr::Response r) {
|
|
|
|
if (r.error) {
|
2017-10-31 10:02:42 +01:00
|
|
|
LOG_ERROR(WebService, "GET to %s returned cpr error: %u:%s", r.url.c_str(),
|
2017-09-19 03:18:26 +02:00
|
|
|
static_cast<u32>(r.error.code), r.error.message.c_str());
|
|
|
|
return func("");
|
|
|
|
}
|
|
|
|
if (r.status_code >= 400) {
|
2017-10-31 10:02:42 +01:00
|
|
|
LOG_ERROR(WebService, "GET to %s returned error code: %u", r.url.c_str(),
|
|
|
|
r.status_code);
|
2017-09-19 03:18:26 +02:00
|
|
|
return func("");
|
|
|
|
}
|
|
|
|
if (r.header["content-type"].find("application/json") == std::string::npos) {
|
2017-10-31 10:02:42 +01:00
|
|
|
LOG_ERROR(WebService, "GET to %s returned wrong content: %s", r.url.c_str(),
|
2017-09-19 03:18:26 +02:00
|
|
|
r.header["content-type"].c_str());
|
|
|
|
return func("");
|
|
|
|
}
|
|
|
|
return func(r.text);
|
|
|
|
},
|
|
|
|
cpr::Url{url}, header);
|
2017-06-27 23:18:52 -04:00
|
|
|
}
|
|
|
|
|
2017-09-19 03:18:26 +02:00
|
|
|
template std::future<bool> GetJson(std::function<bool(const std::string&)> func,
|
|
|
|
const std::string& url, bool allow_anonymous,
|
|
|
|
const std::string& username, const std::string& token);
|
2017-10-31 10:02:42 +01:00
|
|
|
template std::future<AnnounceMultiplayerRoom::RoomList> GetJson(
|
|
|
|
std::function<AnnounceMultiplayerRoom::RoomList(const std::string&)> func,
|
|
|
|
const std::string& url, bool allow_anonymous, const std::string& username,
|
|
|
|
const std::string& token);
|
|
|
|
|
|
|
|
void DeleteJson(const std::string& url, const std::string& data, const std::string& username,
|
|
|
|
const std::string& token) {
|
|
|
|
if (url.empty()) {
|
|
|
|
LOG_ERROR(WebService, "URL is invalid");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (token.empty() || username.empty()) {
|
|
|
|
LOG_ERROR(WebService, "Credentials must be provided for authenticated requests");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Win32WSAStartup();
|
|
|
|
|
|
|
|
// Built request header
|
|
|
|
cpr::Header header = {{"Content-Type", "application/json"},
|
|
|
|
{"x-username", username.c_str()},
|
|
|
|
{"x-token", token.c_str()},
|
|
|
|
{"api-version", API_VERSION}};
|
|
|
|
|
|
|
|
// Delete JSON asynchronously
|
|
|
|
static std::future<void> future;
|
|
|
|
future = cpr::DeleteCallback(
|
|
|
|
[](cpr::Response r) {
|
|
|
|
if (r.error) {
|
|
|
|
LOG_ERROR(WebService, "Delete to %s returned cpr error: %u:%s", r.url.c_str(),
|
|
|
|
static_cast<u32>(r.error.code), r.error.message.c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (r.status_code >= 400) {
|
|
|
|
LOG_ERROR(WebService, "Delete to %s returned error status code: %u", r.url.c_str(),
|
|
|
|
r.status_code);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (r.header["content-type"].find("application/json") == std::string::npos) {
|
|
|
|
LOG_ERROR(WebService, "Delete to %s returned wrong content: %s", r.url.c_str(),
|
|
|
|
r.header["content-type"].c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
cpr::Url{url}, cpr::Body{data}, header);
|
|
|
|
}
|
2017-09-19 03:18:26 +02:00
|
|
|
|
2017-06-27 23:01:49 -04:00
|
|
|
} // namespace WebService
|