diff options
Diffstat (limited to 'src/web_service')
| -rw-r--r-- | src/web_service/telemetry_json.cpp | 3 | ||||
| -rw-r--r-- | src/web_service/telemetry_json.h | 7 | ||||
| -rw-r--r-- | src/web_service/web_backend.cpp | 67 | ||||
| -rw-r--r-- | src/web_service/web_backend.h | 18 | 
4 files changed, 51 insertions, 44 deletions
diff --git a/src/web_service/telemetry_json.cpp b/src/web_service/telemetry_json.cpp index a2d007e77..6ad2ffcd4 100644 --- a/src/web_service/telemetry_json.cpp +++ b/src/web_service/telemetry_json.cpp @@ -3,7 +3,6 @@  // Refer to the license.txt file included.  #include "common/assert.h" -#include "core/settings.h"  #include "web_service/telemetry_json.h"  #include "web_service/web_backend.h" @@ -81,7 +80,7 @@ void TelemetryJson::Complete() {      SerializeSection(Telemetry::FieldType::UserFeedback, "UserFeedback");      SerializeSection(Telemetry::FieldType::UserConfig, "UserConfig");      SerializeSection(Telemetry::FieldType::UserSystem, "UserSystem"); -    PostJson(Settings::values.telemetry_endpoint_url, TopSection().dump()); +    PostJson(endpoint_url, TopSection().dump(), true, username, token);  }  } // namespace WebService diff --git a/src/web_service/telemetry_json.h b/src/web_service/telemetry_json.h index 39038b4f9..9e78c6803 100644 --- a/src/web_service/telemetry_json.h +++ b/src/web_service/telemetry_json.h @@ -17,7 +17,9 @@ namespace WebService {   */  class TelemetryJson : public Telemetry::VisitorInterface {  public: -    TelemetryJson() = default; +    TelemetryJson(const std::string& endpoint_url, const std::string& username, +                  const std::string& token) +        : endpoint_url(endpoint_url), username(username), token(token) {}      ~TelemetryJson() = default;      void Visit(const Telemetry::Field<bool>& field) override; @@ -49,6 +51,9 @@ private:      nlohmann::json output;      std::array<nlohmann::json, 7> sections; +    std::string endpoint_url; +    std::string username; +    std::string token;  };  } // namespace WebService diff --git a/src/web_service/web_backend.cpp b/src/web_service/web_backend.cpp index 13e4555ac..d28a3f757 100644 --- a/src/web_service/web_backend.cpp +++ b/src/web_service/web_backend.cpp @@ -2,51 +2,62 @@  // Licensed under GPLv2 or any later version  // Refer to the license.txt file included. +#ifdef _WIN32 +#include <winsock.h> +#endif + +#include <cstdlib> +#include <thread>  #include <cpr/cpr.h> -#include <stdlib.h>  #include "common/logging/log.h"  #include "web_service/web_backend.h"  namespace WebService {  static constexpr char API_VERSION[]{"1"}; -static constexpr char ENV_VAR_USERNAME[]{"CITRA_WEB_SERVICES_USERNAME"}; -static constexpr char ENV_VAR_TOKEN[]{"CITRA_WEB_SERVICES_TOKEN"}; - -static std::string GetEnvironmentVariable(const char* name) { -    const char* value{getenv(name)}; -    if (value) { -        return value; -    } -    return {}; -} - -const std::string& GetUsername() { -    static const std::string username{GetEnvironmentVariable(ENV_VAR_USERNAME)}; -    return username; -} -const std::string& GetToken() { -    static const std::string token{GetEnvironmentVariable(ENV_VAR_TOKEN)}; -    return token; -} +static std::unique_ptr<cpr::Session> g_session; -void PostJson(const std::string& url, const std::string& data) { +void PostJson(const std::string& url, const std::string& data, bool allow_anonymous, +              const std::string& username, const std::string& token) {      if (url.empty()) {          LOG_ERROR(WebService, "URL is invalid");          return;      } -    if (GetUsername().empty() || GetToken().empty()) { -        LOG_ERROR(WebService, "Environment variables %s and %s must be set to POST JSON", -                  ENV_VAR_USERNAME, ENV_VAR_TOKEN); +    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;      } -    cpr::PostAsync(cpr::Url{url}, cpr::Body{data}, cpr::Header{{"Content-Type", "application/json"}, -                                                               {"x-username", GetUsername()}, -                                                               {"x-token", GetToken()}, -                                                               {"api-version", API_VERSION}}); +#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 + +    // 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}}; +    } + +    // Post JSON asynchronously +    static cpr::AsyncResponse future; +    future = cpr::PostAsync(cpr::Url{url.c_str()}, cpr::Body{data.c_str()}, header);  }  } // namespace WebService diff --git a/src/web_service/web_backend.h b/src/web_service/web_backend.h index 2753d3b68..d17100398 100644 --- a/src/web_service/web_backend.h +++ b/src/web_service/web_backend.h @@ -10,22 +10,14 @@  namespace WebService {  /** - * Gets the current username for accessing services.citra-emu.org. - * @returns Username as a string, empty if not set. - */ -const std::string& GetUsername(); - -/** - * Gets the current token for accessing services.citra-emu.org. - * @returns Token as a string, empty if not set. - */ -const std::string& GetToken(); - -/**   * Posts JSON to services.citra-emu.org.   * @param url URL of the services.citra-emu.org endpoint to post data to.   * @param data String of JSON data to use for the body of the POST request. + * @param allow_anonymous If true, allow anonymous unauthenticated requests. + * @param username Citra username to use for authentication. + * @param token Citra token to use for authentication.   */ -void PostJson(const std::string& url, const std::string& data); +void PostJson(const std::string& url, const std::string& data, bool allow_anonymous, +              const std::string& username = {}, const std::string& token = {});  } // namespace WebService  | 
