Mercurial > minori
comparison src/services/anilist.cc @ 175:9b10175be389
dep/json: update to v3.11.3
anime/db: save anime list to database, very much untested and likely won't work as intended
| author | Paper <mrpapersonic@gmail.com> |
|---|---|
| date | Thu, 30 Nov 2023 13:52:26 -0500 |
| parents | 275da698697d |
| children | 01d259b9c89f |
comparison
equal
deleted
inserted
replaced
| 174:f88eda79c60a | 175:9b10175be389 |
|---|---|
| 13 #include <QLineEdit> | 13 #include <QLineEdit> |
| 14 #include <QMessageBox> | 14 #include <QMessageBox> |
| 15 #include <QUrl> | 15 #include <QUrl> |
| 16 #include <chrono> | 16 #include <chrono> |
| 17 #include <exception> | 17 #include <exception> |
| 18 | |
| 19 #include <iostream> | |
| 20 | |
| 18 #define CLIENT_ID "13706" | 21 #define CLIENT_ID "13706" |
| 19 | 22 |
| 20 using namespace nlohmann::literals::json_literals; | 23 using namespace nlohmann::literals::json_literals; |
| 21 | 24 |
| 22 namespace Services { | 25 namespace Services { |
| 32 | 35 |
| 33 std::string AuthToken() const { return session.config.auth.anilist.auth_token; } | 36 std::string AuthToken() const { return session.config.auth.anilist.auth_token; } |
| 34 void SetAuthToken(std::string const& auth_token) { session.config.auth.anilist.auth_token = auth_token; } | 37 void SetAuthToken(std::string const& auth_token) { session.config.auth.anilist.auth_token = auth_token; } |
| 35 | 38 |
| 36 bool Authenticated() const { return !AuthToken().empty(); } | 39 bool Authenticated() const { return !AuthToken().empty(); } |
| 40 bool IsValid() const { return UserId() && Authenticated(); } | |
| 37 }; | 41 }; |
| 38 | 42 |
| 39 static Account account; | 43 static Account account; |
| 40 | 44 |
| 41 std::string SendRequest(std::string data) { | 45 std::string SendRequest(std::string data) { |
| 42 std::vector<std::string> headers = {"Authorization: Bearer " + account.AuthToken(), "Accept: application/json", | 46 std::vector<std::string> headers = {"Authorization: Bearer " + account.AuthToken(), "Accept: application/json", |
| 43 "Content-Type: application/json"}; | 47 "Content-Type: application/json"}; |
| 44 return Strings::ToUtf8String(HTTP::Post("https://graphql.anilist.co", data, headers)); | 48 return Strings::ToUtf8String(HTTP::Post("https://graphql.anilist.co", data, headers)); |
| 49 } | |
| 50 | |
| 51 nlohmann::json SendJSONRequest(nlohmann::json data) { | |
| 52 std::string request = SendRequest(data.dump()); | |
| 53 if (request.empty()) { | |
| 54 std::cerr << "[AniList] JSON Request returned an empty result!" << std::endl; | |
| 55 return {}; | |
| 56 } | |
| 57 | |
| 58 auto ret = nlohmann::json::parse(request, nullptr, false); | |
| 59 if (ret.is_discarded()) { | |
| 60 std::cerr << "[AniList] Failed to parse request JSON!" << std::endl; | |
| 61 return {}; | |
| 62 } | |
| 63 | |
| 64 if (ret.contains("/errors"_json_pointer) && ret.at("/errors"_json_pointer).is_array()) { | |
| 65 for (const auto& error : ret.at("/errors"_json_pointer)) | |
| 66 std::cerr << "[AniList] Received an error in response: " << JSON::GetString<std::string>(error, "/message"_json_pointer, "") << std::endl; | |
| 67 | |
| 68 return {}; | |
| 69 } | |
| 70 | |
| 71 return ret; | |
| 45 } | 72 } |
| 46 | 73 |
| 47 void ParseListStatus(std::string status, Anime::Anime& anime) { | 74 void ParseListStatus(std::string status, Anime::Anime& anime) { |
| 48 std::unordered_map<std::string, Anime::ListStatus> map = { | 75 std::unordered_map<std::string, Anime::ListStatus> map = { |
| 49 {"CURRENT", Anime::ListStatus::CURRENT }, | 76 {"CURRENT", Anime::ListStatus::CURRENT }, |
| 50 {"PLANNING", Anime::ListStatus::PLANNING }, | 77 {"PLANNING", Anime::ListStatus::PLANNING }, |
| 51 {"COMPLETED", Anime::ListStatus::COMPLETED}, | 78 {"COMPLETED", Anime::ListStatus::COMPLETED}, |
| 52 {"DROPPED", Anime::ListStatus::DROPPED }, | 79 {"DROPPED", Anime::ListStatus::DROPPED }, |
| 53 {"PAUSED", Anime::ListStatus::PAUSED } | 80 {"PAUSED", Anime::ListStatus::PAUSED } |
| 54 }; | 81 }; |
| 55 | 82 |
| 56 if (status == "REPEATING") { | 83 if (status == "REPEATING") { |
| 57 anime.SetUserIsRewatching(true); | 84 anime.SetUserIsRewatching(true); |
| 58 anime.SetUserStatus(Anime::ListStatus::CURRENT); | 85 anime.SetUserStatus(Anime::ListStatus::CURRENT); |
| 59 return; | 86 return; |
| 79 default: break; | 106 default: break; |
| 80 } | 107 } |
| 81 return "CURRENT"; | 108 return "CURRENT"; |
| 82 } | 109 } |
| 83 | 110 |
| 84 Date ParseDate(const nlohmann::json& json) { | |
| 85 Date date; | |
| 86 /* JSON for Modern C++ warns here. I'm not too sure why, this code works when I set the | |
| 87 standard to C++17 :/ */ | |
| 88 if (json.contains("/year"_json_pointer) && json.at("/year"_json_pointer).is_number()) | |
| 89 date.SetYear(JSON::GetInt(json, "/year"_json_pointer)); | |
| 90 else | |
| 91 date.VoidYear(); | |
| 92 | |
| 93 if (json.contains("/month"_json_pointer) && json.at("/month"_json_pointer).is_number()) | |
| 94 date.SetMonth(JSON::GetInt(json, "/month"_json_pointer)); | |
| 95 else | |
| 96 date.VoidMonth(); | |
| 97 | |
| 98 if (json.contains("/day"_json_pointer) && json.at("/day"_json_pointer).is_number()) | |
| 99 date.SetDay(JSON::GetInt(json, "/day"_json_pointer)); | |
| 100 else | |
| 101 date.VoidDay(); | |
| 102 return date; | |
| 103 } | |
| 104 | |
| 105 void ParseTitle(const nlohmann::json& json, Anime::Anime& anime) { | 111 void ParseTitle(const nlohmann::json& json, Anime::Anime& anime) { |
| 106 anime.SetNativeTitle(JSON::GetString(json, "/native"_json_pointer)); | 112 anime.SetNativeTitle(JSON::GetString<std::string>(json, "/native"_json_pointer, "")); |
| 107 anime.SetEnglishTitle(JSON::GetString(json, "/english"_json_pointer)); | 113 anime.SetEnglishTitle(JSON::GetString<std::string>(json, "/english"_json_pointer, "")); |
| 108 anime.SetRomajiTitle(JSON::GetString(json, "/romaji"_json_pointer)); | 114 anime.SetRomajiTitle(JSON::GetString<std::string>(json, "/romaji"_json_pointer, "")); |
| 109 } | 115 } |
| 110 | 116 |
| 111 int ParseMediaJson(const nlohmann::json& json) { | 117 int ParseMediaJson(const nlohmann::json& json) { |
| 112 int id = JSON::GetInt(json, "/id"_json_pointer); | 118 int id = JSON::GetNumber(json, "/id"_json_pointer); |
| 113 if (!id) | 119 if (!id) |
| 114 return 0; | 120 return 0; |
| 121 | |
| 115 Anime::Anime& anime = Anime::db.items[id]; | 122 Anime::Anime& anime = Anime::db.items[id]; |
| 116 anime.SetId(id); | 123 anime.SetId(id); |
| 117 | 124 |
| 118 ParseTitle(json.at("/title"_json_pointer), anime); | 125 ParseTitle(json.at("/title"_json_pointer), anime); |
| 119 | 126 |
| 120 anime.SetEpisodes(JSON::GetInt(json, "/episodes"_json_pointer)); | 127 anime.SetEpisodes(JSON::GetNumber(json, "/episodes"_json_pointer, 0)); |
| 121 anime.SetFormat(Translate::AniList::ToSeriesFormat(JSON::GetString(json, "/format"_json_pointer))); | 128 anime.SetFormat(Translate::AniList::ToSeriesFormat(JSON::GetString<std::string>(json, "/format"_json_pointer, ""))); |
| 122 | 129 |
| 123 anime.SetAiringStatus(Translate::AniList::ToSeriesStatus(JSON::GetString(json, "/status"_json_pointer))); | 130 anime.SetAiringStatus(Translate::AniList::ToSeriesStatus(JSON::GetString<std::string>(json, "/status"_json_pointer, ""))); |
| 124 | 131 |
| 125 anime.SetAirDate(ParseDate(json["/startDate"_json_pointer])); | 132 anime.SetAirDate(Date(json["/startDate"_json_pointer])); |
| 126 | 133 |
| 127 anime.SetPosterUrl(JSON::GetString(json, "/coverImage/large"_json_pointer)); | 134 anime.SetPosterUrl(JSON::GetString<std::string>(json, "/coverImage/large"_json_pointer, "")); |
| 128 | 135 |
| 129 anime.SetAudienceScore(JSON::GetInt(json, "/averageScore"_json_pointer)); | 136 anime.SetAudienceScore(JSON::GetNumber(json, "/averageScore"_json_pointer, 0)); |
| 130 anime.SetSeason(Translate::AniList::ToSeriesSeason(JSON::GetString(json, "/season"_json_pointer))); | 137 anime.SetSeason(Translate::AniList::ToSeriesSeason(JSON::GetString<std::string>(json, "/season"_json_pointer, ""))); |
| 131 anime.SetDuration(JSON::GetInt(json, "/duration"_json_pointer)); | 138 anime.SetDuration(JSON::GetNumber(json, "/duration"_json_pointer, 0)); |
| 132 anime.SetSynopsis(Strings::TextifySynopsis(JSON::GetString(json, "/description"_json_pointer))); | 139 anime.SetSynopsis(Strings::TextifySynopsis(JSON::GetString<std::string>(json, "/description"_json_pointer, ""))); |
| 133 | 140 |
| 134 if (json.contains("/genres"_json_pointer) && json["/genres"_json_pointer].is_array()) | 141 anime.SetGenres(JSON::GetArray<std::vector<std::string>>(json, "/genres"_json_pointer, {})); |
| 135 anime.SetGenres(json["/genres"_json_pointer].get<std::vector<std::string>>()); | 142 anime.SetTitleSynonyms(JSON::GetArray<std::vector<std::string>>(json, "/synonyms"_json_pointer, {})); |
| 136 if (json.contains("/synonyms"_json_pointer) && json["/synonyms"_json_pointer].is_array()) | 143 |
| 137 anime.SetTitleSynonyms(json["/synonyms"_json_pointer].get<std::vector<std::string>>()); | |
| 138 return id; | 144 return id; |
| 139 } | 145 } |
| 140 | 146 |
| 141 int ParseListItem(const nlohmann::json& json) { | 147 int ParseListItem(const nlohmann::json& json) { |
| 142 int id = ParseMediaJson(json["media"]); | 148 int id = ParseMediaJson(json["media"]); |
| 143 | 149 |
| 144 Anime::Anime& anime = Anime::db.items[id]; | 150 Anime::Anime& anime = Anime::db.items[id]; |
| 145 | 151 |
| 146 anime.AddToUserList(); | 152 anime.AddToUserList(); |
| 147 | 153 |
| 148 anime.SetUserScore(JSON::GetInt(json, "/score"_json_pointer)); | 154 anime.SetUserScore(JSON::GetNumber(json, "/score"_json_pointer, 0)); |
| 149 anime.SetUserProgress(JSON::GetInt(json, "/progress"_json_pointer)); | 155 anime.SetUserProgress(JSON::GetNumber(json, "/progress"_json_pointer, 0)); |
| 150 ParseListStatus(JSON::GetString(json, "/status"_json_pointer), anime); | 156 ParseListStatus(JSON::GetString<std::string>(json, "/status"_json_pointer, ""), anime); |
| 151 anime.SetUserNotes(JSON::GetString(json, "/notes"_json_pointer)); | 157 anime.SetUserNotes(JSON::GetString<std::string>(json, "/notes"_json_pointer, "")); |
| 152 | 158 |
| 153 anime.SetUserDateStarted(ParseDate(json["/startedAt"_json_pointer])); | 159 anime.SetUserDateStarted(Date(json["/startedAt"_json_pointer])); |
| 154 anime.SetUserDateCompleted(ParseDate(json["/completedAt"_json_pointer])); | 160 anime.SetUserDateCompleted(Date(json["/completedAt"_json_pointer])); |
| 155 | 161 |
| 156 anime.SetUserTimeUpdated(JSON::GetInt(json, "/updatedAt"_json_pointer)); | 162 anime.SetUserTimeUpdated(JSON::GetNumber(json, "/updatedAt"_json_pointer, 0)); |
| 157 | 163 |
| 158 return id; | 164 return id; |
| 159 } | 165 } |
| 160 | 166 |
| 161 int ParseList(const nlohmann::json& json) { | 167 int ParseList(const nlohmann::json& json) { |
| 164 } | 170 } |
| 165 return 1; | 171 return 1; |
| 166 } | 172 } |
| 167 | 173 |
| 168 int GetAnimeList() { | 174 int GetAnimeList() { |
| 175 if (!account.IsValid()) { | |
| 176 std::cerr << "AniList: Account isn't valid!" << std::endl; | |
| 177 return 0; | |
| 178 } | |
| 179 | |
| 169 /* NOTE: these should be in the qrc file */ | 180 /* NOTE: these should be in the qrc file */ |
| 170 const std::string query = "query ($id: Int) {\n" | 181 const std::string query = "query ($id: Int) {\n" |
| 171 " MediaListCollection (userId: $id, type: ANIME) {\n" | 182 " MediaListCollection (userId: $id, type: ANIME) {\n" |
| 172 " lists {\n" | 183 " lists {\n" |
| 173 " name\n" | 184 " name\n" |
| 174 " entries {\n" | 185 " entries {\n" |
| 175 " score\n" | 186 " score\n" |
| 176 " notes\n" | 187 " notes\n" |
| 177 " status\n" | 188 " status\n" |
| 178 " progress\n" | 189 " progress\n" |
| 179 " startedAt {\n" | 190 " startedAt {\n" |
| 180 " year\n" | 191 " year\n" |
| 181 " month\n" | 192 " month\n" |
| 182 " day\n" | 193 " day\n" |
| 183 " }\n" | 194 " }\n" |
| 184 " completedAt {\n" | 195 " completedAt {\n" |
| 185 " year\n" | 196 " year\n" |
| 186 " month\n" | 197 " month\n" |
| 187 " day\n" | 198 " day\n" |
| 188 " }\n" | 199 " }\n" |
| 189 " updatedAt\n" | 200 " updatedAt\n" |
| 190 " media {\n" | 201 " media {\n" |
| 191 " coverImage {\n" | 202 " coverImage {\n" |
| 192 " large\n" | 203 " large\n" |
| 193 " }\n" | 204 " }\n" |
| 194 " id\n" | 205 " id\n" |
| 195 " title {\n" | 206 " title {\n" |
| 196 " romaji\n" | 207 " romaji\n" |
| 197 " english\n" | 208 " english\n" |
| 198 " native\n" | 209 " native\n" |
| 199 " }\n" | 210 " }\n" |
| 200 " format\n" | 211 " format\n" |
| 201 " status\n" | 212 " status\n" |
| 202 " averageScore\n" | 213 " averageScore\n" |
| 203 " season\n" | 214 " season\n" |
| 204 " startDate {\n" | 215 " startDate {\n" |
| 205 " year\n" | 216 " year\n" |
| 206 " month\n" | 217 " month\n" |
| 207 " day\n" | 218 " day\n" |
| 208 " }\n" | 219 " }\n" |
| 209 " genres\n" | 220 " genres\n" |
| 210 " episodes\n" | 221 " episodes\n" |
| 211 " duration\n" | 222 " duration\n" |
| 212 " synonyms\n" | 223 " synonyms\n" |
| 213 " description(asHtml: false)\n" | 224 " description(asHtml: false)\n" |
| 214 " }\n" | 225 " }\n" |
| 215 " }\n" | 226 " }\n" |
| 216 " }\n" | 227 " }\n" |
| 217 " }\n" | 228 " }\n" |
| 218 "}\n"; | 229 "}\n"; |
| 219 // clang-format off | 230 // clang-format off |
| 220 nlohmann::json json = { | 231 nlohmann::json json = { |
| 221 {"query", query}, | 232 {"query", query}, |
| 222 {"variables", { | 233 {"variables", { |
| 223 {"id", account.UserId()} | 234 {"id", account.UserId()} |
| 224 }} | 235 }} |
| 225 }; | 236 }; |
| 226 // clang-format on | 237 // clang-format on |
| 227 /* TODO: do a try catch here, catch any json errors and then call | 238 |
| 228 Authorize() if needed */ | 239 auto res = SendJSONRequest(json); |
| 229 auto res = nlohmann::json::parse(SendRequest(json.dump())); | 240 |
| 230 /* TODO: make sure that we actually need the wstring converter and see | |
| 231 if we can just get wide strings back from nlohmann::json */ | |
| 232 for (const auto& list : res["data"]["MediaListCollection"]["lists"].items()) { | 241 for (const auto& list : res["data"]["MediaListCollection"]["lists"].items()) { |
| 233 ParseList(list.value()); | 242 ParseList(list.value()); |
| 234 } | 243 } |
| 235 return 1; | 244 return 1; |
| 236 } | 245 } |
| 255 * Date startedAt, | 264 * Date startedAt, |
| 256 * Date completedAt | 265 * Date completedAt |
| 257 **/ | 266 **/ |
| 258 Anime::Anime& anime = Anime::db.items[id]; | 267 Anime::Anime& anime = Anime::db.items[id]; |
| 259 const std::string query = "mutation ($media_id: Int, $progress: Int, $status: MediaListStatus, $score: Int, " | 268 const std::string query = "mutation ($media_id: Int, $progress: Int, $status: MediaListStatus, $score: Int, " |
| 260 "$notes: String, $start: FuzzyDateInput, $comp: FuzzyDateInput) {\n" | 269 "$notes: String, $start: FuzzyDateInput, $comp: FuzzyDateInput) {\n" |
| 261 " SaveMediaListEntry (mediaId: $media_id, progress: $progress, status: $status, " | 270 " SaveMediaListEntry (mediaId: $media_id, progress: $progress, status: $status, " |
| 262 "scoreRaw: $score, notes: $notes, startedAt: $start, completedAt: $comp) {\n" | 271 "scoreRaw: $score, notes: $notes, startedAt: $start, completedAt: $comp) {\n" |
| 263 " id\n" | 272 " id\n" |
| 264 " }\n" | 273 " }\n" |
| 265 "}\n"; | 274 "}\n"; |
| 266 // clang-format off | 275 // clang-format off |
| 267 nlohmann::json json = { | 276 nlohmann::json json = { |
| 268 {"query", query}, | 277 {"query", query}, |
| 269 {"variables", { | 278 {"variables", { |
| 270 {"media_id", anime.GetId()}, | 279 {"media_id", anime.GetId()}, |
| 275 {"start", anime.GetUserDateStarted().GetAsAniListJson()}, | 284 {"start", anime.GetUserDateStarted().GetAsAniListJson()}, |
| 276 {"comp", anime.GetUserDateCompleted().GetAsAniListJson()} | 285 {"comp", anime.GetUserDateCompleted().GetAsAniListJson()} |
| 277 }} | 286 }} |
| 278 }; | 287 }; |
| 279 // clang-format on | 288 // clang-format on |
| 280 auto ret = nlohmann::json::parse(SendRequest(json.dump())); | 289 |
| 281 return JSON::GetInt(ret, "/data/SaveMediaListEntry/id"_json_pointer); | 290 auto ret = SendJSONRequest(json); |
| 291 | |
| 292 return JSON::GetNumber(ret, "/data/SaveMediaListEntry/id"_json_pointer, 0); | |
| 282 } | 293 } |
| 283 | 294 |
| 284 int ParseUser(const nlohmann::json& json) { | 295 int ParseUser(const nlohmann::json& json) { |
| 285 account.SetUsername(JSON::GetString(json, "/name"_json_pointer)); | 296 account.SetUsername(JSON::GetString<std::string>(json, "/name"_json_pointer, "")); |
| 286 account.SetUserId(JSON::GetInt(json, "/id"_json_pointer)); | 297 account.SetUserId(JSON::GetNumber(json, "/id"_json_pointer, 0)); |
| 287 return account.UserId(); | 298 return account.UserId(); |
| 288 } | 299 } |
| 289 | 300 |
| 290 bool AuthorizeUser() { | 301 bool AuthorizeUser() { |
| 291 /* Prompt for PIN */ | 302 /* Prompt for PIN */ |
| 292 QDesktopServices::openUrl( | 303 QDesktopServices::openUrl( |
| 293 QUrl("https://anilist.co/api/v2/oauth/authorize?client_id=" CLIENT_ID "&response_type=token")); | 304 QUrl("https://anilist.co/api/v2/oauth/authorize?client_id=" CLIENT_ID "&response_type=token")); |
| 305 | |
| 294 bool ok; | 306 bool ok; |
| 295 QString token = QInputDialog::getText( | 307 QString token = QInputDialog::getText( |
| 296 0, "Credentials needed!", "Please enter the code given to you after logging in to AniList:", QLineEdit::Normal, | 308 0, "Credentials needed!", "Please enter the code given to you after logging in to AniList:", QLineEdit::Normal, |
| 297 "", &ok); | 309 "", &ok); |
| 298 if (ok && !token.isEmpty()) | 310 |
| 299 account.SetAuthToken(Strings::ToUtf8String(token)); | 311 if (!ok || token.isEmpty()) |
| 300 else // fail | |
| 301 return false; | 312 return false; |
| 313 | |
| 314 account.SetAuthToken(Strings::ToUtf8String(token)); | |
| 315 | |
| 302 const std::string query = "query {\n" | 316 const std::string query = "query {\n" |
| 303 " Viewer {\n" | 317 " Viewer {\n" |
| 304 " id\n" | 318 " id\n" |
| 305 " name\n" | 319 " name\n" |
| 306 " mediaListOptions {\n" | 320 " mediaListOptions {\n" |
| 307 " scoreFormat\n" | 321 " scoreFormat\n" // this will be used... eventually |
| 308 " }\n" | 322 " }\n" |
| 309 " }\n" | 323 " }\n" |
| 310 "}\n"; | 324 "}\n"; |
| 311 nlohmann::json json = { | 325 nlohmann::json json = { |
| 312 {"query", query} | 326 {"query", query} |
| 313 }; | 327 }; |
| 314 auto ret = nlohmann::json::parse(SendRequest(json.dump())); | 328 |
| 329 auto ret = SendJSONRequest(json); | |
| 330 | |
| 315 ParseUser(ret["data"]["Viewer"]); | 331 ParseUser(ret["data"]["Viewer"]); |
| 316 return true; | 332 return true; |
| 317 } | 333 } |
| 318 | 334 |
| 319 } // namespace AniList | 335 } // namespace AniList |
