diff src/services/anilist.cpp @ 9:5c0397762b53

INCOMPLETE: megacommit :)
author Paper <mrpapersonic@gmail.com>
date Sun, 10 Sep 2023 03:59:16 -0400 (16 months ago)
parents
children 4b198a111713
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/services/anilist.cpp	Sun Sep 10 03:59:16 2023 -0400
@@ -0,0 +1,337 @@
+#include "services/anilist.h"
+#include "core/anime.h"
+#include "core/config.h"
+#include "core/json.h"
+#include "core/session.h"
+#include "core/strings.h"
+#include <QDesktopServices>
+#include <QInputDialog>
+#include <QLineEdit>
+#include <QMessageBox>
+#include <chrono>
+#include <curl/curl.h>
+#include <exception>
+#include <format>
+#define CLIENT_ID "13706"
+
+namespace Services::AniList {
+
+class Account {
+	public:
+		std::string Username() const { return session.anilist.username; }
+		void SetUsername(std::string const& username) { session.anilist.username = username; }
+
+		int UserId() const { return session.anilist.user_id; }
+		void SetUserId(const int id) { session.anilist.user_id = id; }
+
+		std::string AuthToken() const { return session.anilist.auth_token; }
+		void SetAuthToken(std::string const& auth_token) { session.anilist.auth_token = auth_token; }
+
+		bool Authenticated() const { return !AuthToken().empty(); }
+}
+
+static Account account;
+
+static size_t CurlWriteCallback(void* contents, size_t size, size_t nmemb, void* userdata) {
+	((std::string*)userdata)->append((char*)contents, size * nmemb);
+	return size * nmemb;
+}
+
+/* A wrapper around cURL to send requests to AniList */
+std::string SendRequest(std::string data) {
+	struct curl_slist* list = NULL;
+	std::string userdata;
+	CURL* curl = curl_easy_init();
+	if (curl) {
+		list = curl_slist_append(list, "Accept: application/json");
+		list = curl_slist_append(list, "Content-Type: application/json");
+		std::string bearer = "Authorization: Bearer " + account.AuthToken();
+		list = curl_slist_append(list, bearer.c_str());
+		curl_easy_setopt(curl, CURLOPT_URL, "https://graphql.anilist.co");
+		curl_easy_setopt(curl, CURLOPT_POSTFIELDS, data.c_str());
+		curl_easy_setopt(curl, CURLOPT_HTTPHEADER, list);
+		curl_easy_setopt(curl, CURLOPT_WRITEDATA, &userdata);
+		curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &CurlWriteCallback);
+		/* Use system certs... useful on Windows. */
+		curl_easy_setopt(curl, CURLOPT_SSL_OPTIONS, CURLSSLOPT_NATIVE_CA);
+		CURLcode res = curl_easy_perform(curl);
+		curl_slist_free_all(list);
+		curl_easy_cleanup(curl);
+		if (res != CURLE_OK) {
+			QMessageBox box(QMessageBox::Icon::Critical, "",
+							QString("curl_easy_perform(curl) failed!: ") + QString(curl_easy_strerror(res)));
+			box.exec();
+			return "";
+		}
+		return userdata;
+	}
+	return "";
+}
+
+/* Maps to convert string forms to our internal enums */
+
+std::map<std::string, enum AnimeWatchingStatus> StringToAnimeWatchingMap = {
+	{"CURRENT",	CURRENT  },
+	  {"PLANNING",  PLANNING },
+	  {"COMPLETED", COMPLETED},
+	{"DROPPED",	DROPPED  },
+	  {"PAUSED",	 PAUSED   },
+	  {"REPEATING", REPEATING}
+};
+
+std::map<enum AnimeWatchingStatus, std::string> AnimeWatchingToStringMap = {
+	{CURRENT,	  "CURRENT"  },
+	  {PLANNING,	 "PLANNING" },
+	  {COMPLETED, "COMPLETED"},
+	{DROPPED,	  "DROPPED"  },
+	  {PAUSED,	   "PAUSED"   },
+	  {REPEATING, "REPEATING"}
+};
+
+std::map<std::string, enum AnimeAiringStatus> StringToAnimeAiringMap = {
+	{"FINISHED",		 FINISHED		 },
+	{"RELEASING",		  RELEASING	   },
+	{"NOT_YET_RELEASED", NOT_YET_RELEASED},
+	{"CANCELLED",		  CANCELLED	   },
+	{"HIATUS",		   HIATUS			 }
+};
+
+std::map<std::string, enum AnimeSeason> StringToAnimeSeasonMap = {
+	{"WINTER", WINTER},
+	{"SPRING", SPRING},
+	{"SUMMER", SUMMER},
+	{"FALL",	 FALL	 }
+};
+
+std::map<std::string, enum AnimeFormat> StringToAnimeFormatMap = {
+	{"TV",	   TV		 },
+	  {"TV_SHORT", TV_SHORT},
+	  {"MOVIE",	MOVIE	 },
+	{"SPECIAL",	SPECIAL },
+	  {"OVA",	  OVA	 },
+	{"ONA",		ONA	   },
+	  {"MUSIC",	MUSIC	 },
+	  {"MANGA",	MANGA	 },
+	{"NOVEL",	  NOVEL   },
+	  {"ONE_SHOT", ONE_SHOT}
+};
+
+void ParseDate(const nlohmann::json& json, Date& date) {
+	if (json.contains("/year"_json_pointer) && json["/year"_json_pointer].is_number())
+		date.SetYear(JSON::GetInt(json, "/year"_json_pointer));
+	else
+		date.VoidYear();
+
+	if (json.contains("/month"_json_pointer) && json["/month"_json_pointer].is_number())
+		date.SetMonth(JSON::GetInt(json, "/month"_json_pointer));
+	else
+		date.VoidMonth();
+
+	if (json.contains("/day"_json_pointer) && json["/day"_json_pointer].is_number())
+		date.SetDay(JSON::GetInt(json, "/day"_json_pointer));
+	else
+		date.VoidDay();
+}
+
+void ParseTitle(const nlohmann::json& json, Anime::Anime& anime) {
+	anime.SetNativeTitle(JSON::GetString(json, "/native"_json_pointer));
+	anime.SetEnglishTitle(JSON::GetString(json, "/english"_json_pointer));
+	anime.SetRomajiTitle(JSON::GetString(json, "/romaji"_json_pointer));
+}
+
+int ParseMediaJson(const nlohmann::json& json) {
+	int id = JSON::GetInt(json, "/id"_json_pointer);
+	if (!id)
+		return 0;
+	Anime::Anime& anime = Anime::db.items[id];
+	anime.SetId(id);
+
+	ParseTitle(json["/title"_json_pointer], anime);
+
+	anime.SetEpisodes(JSON::GetInt(json, "/episodes"_json_pointer));
+	anime.SetFormat(AniListStringToAnimeFormatMap[JSON::GetString(json, "/format"_json_pointer)]);
+
+	anime.SetListStatus(AniListStringToAnimeAiringMap[JSON::GetString(json, "/status"_json_pointer)]);
+
+	ParseDate(json["/startDate"_json_pointer], anime.air_date);
+
+	anime.SetAudienceScore(JSON::GetInt(json, "/averageScore"_json_pointer));
+	anime.SetSeason(AniListStringToAnimeSeasonMap[JSON::GetString(json, "/season"_json_pointer)]);
+	anime.SetDuration(JSON::GetInt(json, "/duration"_json_pointer));
+	anime.SetSynopsis(StringUtils::TextifySynopsis(JSON::GetString(json, "/description"_json_pointer)));
+
+	if (json.contains("/genres"_json_pointer) && json["/genres"_json_pointer].is_array())
+		anime.SetGenres(json["/genres"_json_pointer].get<std::vector<std::string>>());
+	if (json.contains("/synonyms"_json_pointer) && json["/synonyms"_json_pointer].is_array())
+		anime.SetSynonyms(json["/synonyms"_json_pointer].get<std::vector<std::string>>());
+	return 1;
+}
+
+int ParseListItem(const nlohmann::json& json, Anime::Anime& anime) {
+	anime.SetScore(JSON::GetInt(entry.value(), "/score"_json_pointer));
+	anime.SetProgress(JSON::GetInt(entry.value(), "/progress"_json_pointer));
+	anime.SetStatus(AniListStringToAnimeWatchingMap[JSON::GetString(entry.value(), "/status"_json_pointer)]);
+	anime.SetNotes(JSON::GetString(entry.value(), "/notes"_json_pointer));
+
+	ParseDate(json["/startedAt"_json_pointer], anime.started);
+	ParseDate(json["/completedAt"_json_pointer], anime.completed);
+
+	anime.SetUpdated(JSON::GetInt(entry.value(), "/updatedAt"_json_pointer));
+
+	return ParseMediaJson(json["media"], anime);
+}
+
+int ParseList(const nlohmann::json& json) {
+	for (const auto& entry : json["entries"].items()) {
+		ParseListItem(entry.value());
+	}
+}
+
+int GetAnimeList(int id) {
+	/* NOTE: these should be in the qrc file */
+	const std::string query = "query ($id: Int) {\n"
+							  "  MediaListCollection (userId: $id, type: ANIME) {\n"
+							  "    lists {\n"
+							  "      name\n"
+							  "      entries {\n"
+							  "        score\n"
+							  "        notes\n"
+							  "        progress\n"
+							  "        startedAt {\n"
+							  "          year\n"
+							  "          month\n"
+							  "          day\n"
+							  "        }\n"
+							  "        completedAt {\n"
+							  "          year\n"
+							  "          month\n"
+							  "          day\n"
+							  "        }\n"
+							  "        updatedAt\n"
+							  "        media {\n"
+							  "          id\n"
+							  "          title {\n"
+							  "            romaji\n"
+							  "            english\n"
+							  "            native\n"
+							  "          }\n"
+							  "          format\n"
+							  "          status\n"
+							  "          averageScore\n"
+							  "          season\n"
+							  "          startDate {\n"
+							  "            year\n"
+							  "            month\n"
+							  "            day\n"
+							  "          }\n"
+							  "          genres\n"
+							  "          episodes\n"
+							  "          duration\n"
+							  "          synonyms\n"
+							  "          description(asHtml: false)\n"
+							  "        }\n"
+							  "      }\n"
+							  "    }\n"
+							  "  }\n"
+							  "}\n";
+	// clang-format off
+	nlohmann::json json = {
+		{"query", query},
+		{"variables", {
+			{"id", id}
+		}}
+	};
+	// clang-format on
+	/* TODO: do a try catch here, catch any json errors and then call
+	   Authorize() if needed */
+	auto res = nlohmann::json::parse(SendRequest(json.dump()));
+	/* TODO: make sure that we actually need the wstring converter and see
+	   if we can just get wide strings back from nlohmann::json */
+	for (const auto& list : res["data"]["MediaListCollection"]["lists"].items()) {
+
+		ParseList(list.entry());
+	}
+	return 1;
+}
+
+int UpdateAnimeEntry(const Anime& anime) {
+	/**
+	 * possible values:
+	 * 
+	 * int mediaId,
+	 * MediaListStatus status,
+	 * float score,
+	 * int scoreRaw,
+	 * int progress,
+	 * int progressVolumes,
+	 * int repeat,
+	 * int priority,
+	 * bool private,
+	 * string notes,
+	 * bool hiddenFromStatusLists,
+	 * string[] customLists,
+	 * float[] advancedScores,
+	 * Date startedAt,
+	 * Date completedAt
+	**/
+	const std::string query =
+		"mutation ($media_id: Int, $progress: Int, $status: MediaListStatus, $score: Int, $notes: String) {\n"
+		"  SaveMediaListEntry (mediaId: $media_id, progress: $progress, status: $status, scoreRaw: $score, notes: "
+		"$notes) {\n"
+		"    id\n"
+		"  }\n"
+		"}\n";
+	// clang-format off
+	nlohmann::json json = {
+		{"query", query},
+		{"variables", {
+			{"media_id", anime.id},
+			{"progress", anime.progress},
+			{"status",   AnimeWatchingToStringMap[anime.status]},
+			{"score",    anime.score},
+			{"notes",    anime.notes}
+		}}
+	};
+	// clang-format on
+	SendRequest(json.dump());
+	return 1;
+}
+
+int ParseUser(const nlohmann::json& json) {
+	account.SetUsername(JSON::GetString(json, "/name"_json_pointer));
+	account.SetUserId(JSON::GetInt(json, "/id"_json_pointer));
+	account.SetAuthenticated(true);
+}
+
+int AuthorizeUser() {
+	/* Prompt for PIN */
+	QDesktopServices::openUrl(
+		QUrl("https://anilist.co/api/v2/oauth/authorize?client_id=" CLIENT_ID "&response_type=token"));
+	bool ok;
+	QString token = QInputDialog::getText(
+		0, "Credentials needed!", "Please enter the code given to you after logging in to AniList:", QLineEdit::Normal,
+		"", &ok);
+	if (ok && !token.isEmpty())
+		account.SetAuthToken(token.toStdString());
+	else { // fail
+		account.SetAuthenticated(false);
+		return 0;
+	}
+	const std::string query = "query {\n"
+							  "  Viewer {\n"
+							  "    id\n"
+							  "    name\n"
+							  "    mediaListOptions {\n"
+							  "      scoreFormat\n"
+							  "    }\n"
+							  "  }\n"
+							  "}\n";
+	nlohmann::json json = {
+		{"query", query}
+	};
+	auto ret = nlohmann::json::parse(SendRequest(json.dump()));
+	ParseUser(json["Viewer"]) account.SetAuthenticated(true);
+	return 1;
+}
+
+} // namespace Services::AniList