Mercurial > minori
annotate src/services/anilist.cc @ 371:c87dcf967cff
dep/animone: add mpc-qt posix and x11
author | Paper <paper@tflc.us> |
---|---|
date | Fri, 25 Jul 2025 11:01:57 -0400 |
parents | 47c9f8502269 |
children |
rev | line source |
---|---|
9 | 1 #include "services/anilist.h" |
2 #include "core/anime.h" | |
10 | 3 #include "core/anime_db.h" |
369 | 4 #include "core/config.h" |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
5 #include "core/date.h" |
76 | 6 #include "core/http.h" |
9 | 7 #include "core/json.h" |
8 #include "core/session.h" | |
9 #include "core/strings.h" | |
15 | 10 #include "gui/translate/anilist.h" |
187
9613d72b097e
*: multiple performance improvements
Paper <mrpapersonic@gmail.com>
parents:
185
diff
changeset
|
11 |
258 | 12 #include <QByteArray> |
187
9613d72b097e
*: multiple performance improvements
Paper <mrpapersonic@gmail.com>
parents:
185
diff
changeset
|
13 #include <QDate> |
9 | 14 #include <QDesktopServices> |
15 #include <QInputDialog> | |
16 #include <QLineEdit> | |
17 #include <QMessageBox> | |
10 | 18 #include <QUrl> |
187
9613d72b097e
*: multiple performance improvements
Paper <mrpapersonic@gmail.com>
parents:
185
diff
changeset
|
19 |
9 | 20 #include <chrono> |
21 #include <exception> | |
291 | 22 #include <string_view> |
175 | 23 |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
24 #include <fmt/core.h> |
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
25 |
175 | 26 #include <iostream> |
27 | |
63 | 28 namespace Services { |
29 namespace AniList { | |
9 | 30 |
291 | 31 static constexpr std::string_view CLIENT_ID = "13706"; |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
32 |
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
33 /* This is used in multiple queries, so just put it here I guess. */ |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
34 #define MEDIA_FIELDS \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
35 "coverImage {\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
36 " large\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
37 "}\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
38 "id\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
39 "title {\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
40 " romaji\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
41 " english\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
42 " native\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
43 "}\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
44 "format\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
45 "status\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
46 "averageScore\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
47 "season\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
48 "startDate {\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
49 " year\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
50 " month\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
51 " day\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
52 "}\n" \ |
324
5d3c9b31aa6e
anime: add completed date member
Paper <paper@paper.us.eu.org>
parents:
323
diff
changeset
|
53 "endDate {\n" \ |
5d3c9b31aa6e
anime: add completed date member
Paper <paper@paper.us.eu.org>
parents:
323
diff
changeset
|
54 " year\n" \ |
5d3c9b31aa6e
anime: add completed date member
Paper <paper@paper.us.eu.org>
parents:
323
diff
changeset
|
55 " month\n" \ |
5d3c9b31aa6e
anime: add completed date member
Paper <paper@paper.us.eu.org>
parents:
323
diff
changeset
|
56 " day\n" \ |
5d3c9b31aa6e
anime: add completed date member
Paper <paper@paper.us.eu.org>
parents:
323
diff
changeset
|
57 "}\n" \ |
321
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
58 "studios {\n" \ |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
59 " edges {\n" \ |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
60 " node {\n" \ |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
61 " name\n" \ |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
62 " }\n" \ |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
63 " }\n" \ |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
64 "}\n" \ |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
65 "genres\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
66 "episodes\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
67 "duration\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
68 "synonyms\n" \ |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
69 "description(asHtml: false)\n" |
185
62e336597bb7
anime list: add support for different score formats
Paper <mrpapersonic@gmail.com>
parents:
184
diff
changeset
|
70 |
317
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
71 /* FIXME: why is this here */ |
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
72 |
369 | 73 static std::optional<nlohmann::json> SendJSONRequest(const nlohmann::json &data) |
74 { | |
365
f81bed4e04ac
*: megacommit that probably breaks things
Paper <paper@paper.us.eu.org>
parents:
334
diff
changeset
|
75 std::vector<std::string> headers = { |
369 | 76 "Accept: application/json", |
77 "Content-Type: application/json", | |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
78 }; |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
79 |
365
f81bed4e04ac
*: megacommit that probably breaks things
Paper <paper@paper.us.eu.org>
parents:
334
diff
changeset
|
80 if (!session.config.auth.anilist.auth_token.empty()) |
f81bed4e04ac
*: megacommit that probably breaks things
Paper <paper@paper.us.eu.org>
parents:
334
diff
changeset
|
81 headers.push_back("Authorization: Bearer " + session.config.auth.anilist.auth_token); |
f81bed4e04ac
*: megacommit that probably breaks things
Paper <paper@paper.us.eu.org>
parents:
334
diff
changeset
|
82 |
369 | 83 const std::string response = |
84 Strings::ToUtf8String(HTTP::Request("https://graphql.anilist.co", headers, data.dump(), HTTP::Type::Post)); | |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
85 if (response.empty()) { |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
86 session.SetStatusBar(Strings::Translate("AniList: JSON request returned an empty result!")); |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
87 return std::nullopt; |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
88 } |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
89 |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
90 nlohmann::json out; |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
91 |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
92 try { |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
93 out = nlohmann::json::parse(response); |
369 | 94 } catch (const std::exception &ex) { |
95 session.SetStatusBar( | |
96 fmt::format(Strings::Translate("AniList: Failed to parse request JSON with error \"{}\"!"), ex.what())); | |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
97 return std::nullopt; |
175 | 98 } |
99 | |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
100 if (out.contains("/errors"_json_pointer) && out.at("/errors"_json_pointer).is_array()) { |
369 | 101 for (const auto &error : out.at("/errors"_json_pointer)) |
317
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
102 std::cerr << "AniList: Received an error in response: " |
369 | 103 << JSON::GetString<std::string>(error, "/message"_json_pointer, "") << std::endl; |
175 | 104 |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
105 session.SetStatusBar(Strings::Translate("AniList: Received an error in response!")); |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
106 return std::nullopt; |
175 | 107 } |
108 | |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
109 return out; |
175 | 110 } |
111 | |
369 | 112 static void ParseListStatus(std::string status, Anime::Anime &anime) |
113 { | |
187
9613d72b097e
*: multiple performance improvements
Paper <mrpapersonic@gmail.com>
parents:
185
diff
changeset
|
114 static const std::unordered_map<std::string, Anime::ListStatus> map = { |
369 | 115 {"CURRENT", Anime::ListStatus::Current }, |
116 {"PLANNING", Anime::ListStatus::Planning }, | |
117 {"COMPLETED", Anime::ListStatus::Completed}, | |
118 {"DROPPED", Anime::ListStatus::Dropped }, | |
119 {"PAUSED", Anime::ListStatus::Paused } | |
120 }; | |
9 | 121 |
15 | 122 if (status == "REPEATING") { |
123 anime.SetUserIsRewatching(true); | |
279 | 124 anime.SetUserStatus(Anime::ListStatus::Current); |
15 | 125 return; |
126 } | |
9 | 127 |
47
d8eb763e6661
information.cpp: add widgets to the list tab, and add an
Paper <mrpapersonic@gmail.com>
parents:
44
diff
changeset
|
128 if (map.find(status) == map.end()) { |
279 | 129 anime.SetUserStatus(Anime::ListStatus::NotInList); |
15 | 130 return; |
131 } | |
9 | 132 |
187
9613d72b097e
*: multiple performance improvements
Paper <mrpapersonic@gmail.com>
parents:
185
diff
changeset
|
133 anime.SetUserStatus(map.at(status)); |
15 | 134 } |
9 | 135 |
369 | 136 static std::string ListStatusToString(const Anime::Anime &anime) |
137 { | |
279 | 138 if (anime.GetUserIsRewatching() && anime.GetUserStatus() == Anime::ListStatus::Current) |
15 | 139 return "REWATCHING"; |
140 | |
70
64e5f427c6a2
services/anilist: remove unordered_map usage for enum classes
Paper <mrpapersonic@gmail.com>
parents:
66
diff
changeset
|
141 switch (anime.GetUserStatus()) { |
279 | 142 case Anime::ListStatus::Planning: return "PLANNING"; |
143 case Anime::ListStatus::Completed: return "COMPLETED"; | |
144 case Anime::ListStatus::Dropped: return "DROPPED"; | |
145 case Anime::ListStatus::Paused: return "PAUSED"; | |
76 | 146 default: break; |
70
64e5f427c6a2
services/anilist: remove unordered_map usage for enum classes
Paper <mrpapersonic@gmail.com>
parents:
66
diff
changeset
|
147 } |
64e5f427c6a2
services/anilist: remove unordered_map usage for enum classes
Paper <mrpapersonic@gmail.com>
parents:
66
diff
changeset
|
148 return "CURRENT"; |
15 | 149 } |
9 | 150 |
369 | 151 static void ParseTitle(const nlohmann::json &json, Anime::Anime &anime) |
152 { | |
305
91ac90a34003
core/time: remove Duration class, use regular functions instead
Paper <paper@paper.us.eu.org>
parents:
304
diff
changeset
|
153 static const std::unordered_map<Anime::TitleLanguage, nlohmann::json::json_pointer> map = { |
369 | 154 {Anime::TitleLanguage::Native, "/native"_json_pointer }, |
155 {Anime::TitleLanguage::English, "/english"_json_pointer}, | |
156 {Anime::TitleLanguage::Romaji, "/romaji"_json_pointer }, | |
305
91ac90a34003
core/time: remove Duration class, use regular functions instead
Paper <paper@paper.us.eu.org>
parents:
304
diff
changeset
|
157 }; |
284
e66ffc338d82
anime: refactor title structure to a map
Paper <paper@paper.us.eu.org>
parents:
279
diff
changeset
|
158 |
369 | 159 for (const auto &[language, ptr] : map) |
305
91ac90a34003
core/time: remove Duration class, use regular functions instead
Paper <paper@paper.us.eu.org>
parents:
304
diff
changeset
|
160 if (json.contains(ptr) && json[ptr].is_string()) |
91ac90a34003
core/time: remove Duration class, use regular functions instead
Paper <paper@paper.us.eu.org>
parents:
304
diff
changeset
|
161 anime.SetTitle(language, json[ptr]); |
9 | 162 } |
163 | |
369 | 164 static int ParseMediaJson(const nlohmann::json &json) |
165 { | |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
166 if (!json.contains("/id"_json_pointer) || !json["/id"_json_pointer].is_number()) { |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
167 session.SetStatusBar(Strings::Translate("AniList: Failed to parse anime object!")); |
9 | 168 return 0; |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
169 } |
175 | 170 |
317
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
171 std::string service_id = Strings::ToUtf8String(json["/id"_json_pointer].get<int>()); |
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
172 |
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
173 int id = Anime::db.LookupServiceIdOrUnused(Anime::Service::AniList, service_id); |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
174 if (!id) { |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
175 session.SetStatusBar(Strings::Translate("AniList: Failed to parse anime object!")); |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
176 return 0; |
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
177 } |
317
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
178 |
369 | 179 Anime::Anime &anime = Anime::db.items[id]; |
9 | 180 anime.SetId(id); |
317
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
181 anime.SetServiceId(Anime::Service::AniList, service_id); |
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
182 |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
183 if (json.contains("/idMal"_json_pointer) && json["/idMal"_json_pointer].is_number()) |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
184 anime.SetServiceId(Anime::Service::MyAnimeList, Strings::ToUtf8String(json["/idMal"_json_pointer].get<int>())); |
9 | 185 |
11 | 186 ParseTitle(json.at("/title"_json_pointer), anime); |
9 | 187 |
175 | 188 anime.SetEpisodes(JSON::GetNumber(json, "/episodes"_json_pointer, 0)); |
189 anime.SetFormat(Translate::AniList::ToSeriesFormat(JSON::GetString<std::string>(json, "/format"_json_pointer, ""))); | |
9 | 190 |
258 | 191 anime.SetAiringStatus( |
369 | 192 Translate::AniList::ToSeriesStatus(JSON::GetString<std::string>(json, "/status"_json_pointer, ""))); |
9 | 193 |
324
5d3c9b31aa6e
anime: add completed date member
Paper <paper@paper.us.eu.org>
parents:
323
diff
changeset
|
194 if (json.contains("/startDate"_json_pointer) && json["/startDate"_json_pointer].is_object()) |
5d3c9b31aa6e
anime: add completed date member
Paper <paper@paper.us.eu.org>
parents:
323
diff
changeset
|
195 anime.SetStartedDate(Date(json["/startDate"_json_pointer])); |
5d3c9b31aa6e
anime: add completed date member
Paper <paper@paper.us.eu.org>
parents:
323
diff
changeset
|
196 |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
197 anime.SetCompletedDate(json.contains("/endDate"_json_pointer) && json["/endDate"_json_pointer].is_object() |
369 | 198 ? Date(json["/endDate"_json_pointer]) |
199 : anime.GetStartedDate()); | |
9 | 200 |
175 | 201 anime.SetPosterUrl(JSON::GetString<std::string>(json, "/coverImage/large"_json_pointer, "")); |
66
6481c5aed3e1
posters: add poster widget...
Paper <mrpapersonic@gmail.com>
parents:
65
diff
changeset
|
202 |
175 | 203 anime.SetAudienceScore(JSON::GetNumber(json, "/averageScore"_json_pointer, 0)); |
369 | 204 // anime.SetSeason(Translate::AniList::ToSeriesSeason(JSON::GetString<std::string>(json, "/season"_json_pointer, |
205 // ""))); | |
175 | 206 anime.SetDuration(JSON::GetNumber(json, "/duration"_json_pointer, 0)); |
260
dd211ff68b36
pages/seasons: add initial functionality
Paper <paper@paper.us.eu.org>
parents:
258
diff
changeset
|
207 |
dd211ff68b36
pages/seasons: add initial functionality
Paper <paper@paper.us.eu.org>
parents:
258
diff
changeset
|
208 std::string synopsis = JSON::GetString<std::string>(json, "/description"_json_pointer, ""); |
dd211ff68b36
pages/seasons: add initial functionality
Paper <paper@paper.us.eu.org>
parents:
258
diff
changeset
|
209 Strings::TextifySynopsis(synopsis); |
dd211ff68b36
pages/seasons: add initial functionality
Paper <paper@paper.us.eu.org>
parents:
258
diff
changeset
|
210 anime.SetSynopsis(synopsis); |
9 | 211 |
175 | 212 anime.SetGenres(JSON::GetArray<std::vector<std::string>>(json, "/genres"_json_pointer, {})); |
213 anime.SetTitleSynonyms(JSON::GetArray<std::vector<std::string>>(json, "/synonyms"_json_pointer, {})); | |
214 | |
321
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
215 { |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
216 std::vector<std::string> producers; |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
217 |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
218 if (json.contains("/studios/edges"_json_pointer) && json["/studios/edges"_json_pointer].is_array()) |
369 | 219 for (const auto &edge : json["/studios/edges"_json_pointer]) |
321
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
220 if (edge.contains("/node/name"_json_pointer) && edge["/node/name"_json_pointer].is_string()) |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
221 producers.push_back(edge["/node/name"_json_pointer].get<std::string>()); |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
222 |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
223 anime.SetProducers(producers); |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
224 } |
8141f409d52c
services/anilist: fix getting producers
Paper <paper@paper.us.eu.org>
parents:
320
diff
changeset
|
225 |
10 | 226 return id; |
9 | 227 } |
228 | |
369 | 229 static int ParseListItem(const nlohmann::json &json) |
230 { | |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
231 int id = ParseMediaJson(json["/media"_json_pointer]); |
317
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
232 if (!id) |
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
233 return 0; |
10 | 234 |
369 | 235 Anime::Anime &anime = Anime::db.items[id]; |
10 | 236 |
237 anime.AddToUserList(); | |
9 | 238 |
175 | 239 anime.SetUserScore(JSON::GetNumber(json, "/score"_json_pointer, 0)); |
240 anime.SetUserProgress(JSON::GetNumber(json, "/progress"_json_pointer, 0)); | |
241 ParseListStatus(JSON::GetString<std::string>(json, "/status"_json_pointer, ""), anime); | |
242 anime.SetUserNotes(JSON::GetString<std::string>(json, "/notes"_json_pointer, "")); | |
9 | 243 |
175 | 244 anime.SetUserDateStarted(Date(json["/startedAt"_json_pointer])); |
245 anime.SetUserDateCompleted(Date(json["/completedAt"_json_pointer])); | |
9 | 246 |
175 | 247 anime.SetUserTimeUpdated(JSON::GetNumber(json, "/updatedAt"_json_pointer, 0)); |
10 | 248 |
249 return id; | |
9 | 250 } |
251 | |
369 | 252 static bool ParseList(const nlohmann::json &json) |
253 { | |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
254 bool success = true; |
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
255 |
369 | 256 for (const auto &entry : json["entries"].items()) |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
257 if (!ParseListItem(entry.value())) |
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
258 success = false; |
317
b1f4d1867ab1
services: VERY initial Kitsu support
Paper <paper@paper.us.eu.org>
parents:
315
diff
changeset
|
259 |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
260 return success; |
9 | 261 } |
262 | |
369 | 263 int GetAnimeList() |
264 { | |
265 auto &auth = session.config.auth.anilist; | |
175 | 266 |
369 | 267 static constexpr std::string_view query = "query ($id: Int) {\n" |
268 " MediaListCollection (userId: $id, type: ANIME) {\n" | |
269 " lists {\n" | |
270 " name\n" | |
271 " entries {\n" | |
272 " score\n" | |
273 " notes\n" | |
274 " status\n" | |
275 " progress\n" | |
276 " startedAt {\n" | |
277 " year\n" | |
278 " month\n" | |
279 " day\n" | |
280 " }\n" | |
281 " completedAt {\n" | |
282 " year\n" | |
283 " month\n" | |
284 " day\n" | |
285 " }\n" | |
286 " updatedAt\n" | |
287 " media {\n" MEDIA_FIELDS " }\n" | |
288 " }\n" | |
289 " }\n" | |
290 " }\n" | |
291 "}\n"; | |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
292 |
9 | 293 // clang-format off |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
294 nlohmann::json request = { |
9 | 295 {"query", query}, |
296 {"variables", { | |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
297 {"id", auth.user_id} |
9 | 298 }} |
299 }; | |
300 // clang-format on | |
175 | 301 |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
302 session.SetStatusBar(Strings::Translate("AniList: Parsing anime list...")); |
175 | 303 |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
304 const std::optional<nlohmann::json> response = SendJSONRequest(request); |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
305 if (!response) |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
306 return 0; |
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
307 |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
308 Anime::db.RemoveAllUserData(); |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
309 |
369 | 310 const nlohmann::json &json = response.value(); |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
311 |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
312 bool success = true; |
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
313 |
369 | 314 for (const auto &list : json["data"]["MediaListCollection"]["lists"].items()) |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
315 if (!ParseList(list.value())) |
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
316 success = false; |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
317 |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
318 if (success) |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
319 session.SetStatusBar(Strings::Translate("AniList: Retrieved anime list successfully!")); |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
320 |
9 | 321 return 1; |
322 } | |
323 | |
250 | 324 /* return is a vector of anime ids */ |
369 | 325 std::vector<int> Search(const std::string &search) |
326 { | |
327 static constexpr std::string_view query = "query ($search: String) {\n" | |
328 " Page (page: 1, perPage: 50) {\n" | |
329 " media (search: $search, type: ANIME) {\n" MEDIA_FIELDS " }\n" | |
330 " }\n" | |
331 "}\n"; | |
250 | 332 |
333 // clang-format off | |
334 nlohmann::json json = { | |
335 {"query", query}, | |
336 {"variables", { | |
337 {"search", search} | |
338 }} | |
339 }; | |
340 // clang-format on | |
341 | |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
342 const std::optional<nlohmann::json> response = SendJSONRequest(json); |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
343 if (!response) |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
344 return {}; |
250 | 345 |
369 | 346 const nlohmann::json &result = response.value(); |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
347 |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
348 /* FIXME: error handling here */ |
250 | 349 std::vector<int> ret; |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
350 ret.reserve(result["/data/Page/media"_json_pointer].size()); |
250 | 351 |
369 | 352 for (const auto &media : result["/data/Page/media"_json_pointer].items()) |
250 | 353 ret.push_back(ParseMediaJson(media.value())); |
354 | |
355 return ret; | |
356 } | |
357 | |
369 | 358 bool GetSeason(Anime::Season season) |
359 { | |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
360 static constexpr std::string_view query = |
369 | 361 "query ($season: MediaSeason!, $season_year: Int!, $page: Int) {\n" |
362 " Page(page: $page) {\n" | |
363 " media(season: $season, seasonYear: $season_year, type: ANIME, sort: START_DATE) {\n" MEDIA_FIELDS " }\n" | |
364 " pageInfo {\n" | |
365 " total\n" | |
366 " perPage\n" | |
367 " currentPage\n" | |
368 " lastPage\n" | |
369 " hasNextPage\n" | |
370 " }\n" | |
371 " }\n" | |
372 "}\n"; | |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
373 |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
374 int page = 0; |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
375 bool has_next_page = true; |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
376 |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
377 while (has_next_page) { |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
378 nlohmann::json json = { |
369 | 379 {"query", query}, |
380 {"variables", | |
381 { | |
382 {"season", Translate::AniList::ToString(season.season)}, | |
383 {"season_year", Strings::ToUtf8String(season.year)}, | |
384 {"page", page}, | |
385 } }, | |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
386 }; |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
387 |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
388 const std::optional<nlohmann::json> res = SendJSONRequest(json); |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
389 if (!res) |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
390 return false; |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
391 |
369 | 392 const nlohmann::json &result = res.value(); |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
393 |
369 | 394 for (const auto &media : result["/data/Page/media"_json_pointer].items()) |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
395 ParseMediaJson(media.value()); |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
396 |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
397 has_next_page = JSON::GetBoolean(result, "/data/Page/pageInfo/hasNextPage"_json_pointer, false); |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
398 if (has_next_page) |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
399 page++; |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
400 } |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
401 |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
402 return true; |
304
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
403 } |
2115488eb302
*: add very early season searcher
Paper <paper@paper.us.eu.org>
parents:
291
diff
changeset
|
404 |
369 | 405 int UpdateAnimeEntry(int id) |
406 { | |
407 Anime::Anime &anime = Anime::db.items[id]; | |
184
09492158bcc5
anime: etc. comments and changes
Paper <mrpapersonic@gmail.com>
parents:
183
diff
changeset
|
408 if (!anime.IsInUserList()) |
09492158bcc5
anime: etc. comments and changes
Paper <mrpapersonic@gmail.com>
parents:
183
diff
changeset
|
409 return 0; |
09492158bcc5
anime: etc. comments and changes
Paper <mrpapersonic@gmail.com>
parents:
183
diff
changeset
|
410 |
291 | 411 std::optional<std::string> service_id = anime.GetServiceId(Anime::Service::AniList); |
412 if (!service_id) | |
413 return 0; | |
414 | |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
415 static constexpr std::string_view query = |
369 | 416 "mutation ($media_id: Int, $progress: Int, $status: MediaListStatus, $score: Int, $notes: String," |
417 " $start: FuzzyDateInput, $comp: FuzzyDateInput, $repeat: Int) {\n" | |
418 " SaveMediaListEntry (mediaId: $media_id, progress: $progress, status: $status, scoreRaw: $score," | |
419 " notes: $notes, startedAt: $start, completedAt: $comp, repeat: $repeat) {\n" | |
420 " id\n" | |
421 " }\n" | |
422 "}\n"; | |
9 | 423 // clang-format off |
424 nlohmann::json json = { | |
425 {"query", query}, | |
426 {"variables", { | |
291 | 427 {"media_id", Strings::ToInt<int64_t>(service_id.value())}, |
10 | 428 {"progress", anime.GetUserProgress()}, |
15 | 429 {"status", ListStatusToString(anime)}, |
10 | 430 {"score", anime.GetUserScore()}, |
77 | 431 {"notes", anime.GetUserNotes()}, |
432 {"start", anime.GetUserDateStarted().GetAsAniListJson()}, | |
184
09492158bcc5
anime: etc. comments and changes
Paper <mrpapersonic@gmail.com>
parents:
183
diff
changeset
|
433 {"comp", anime.GetUserDateCompleted().GetAsAniListJson()}, |
09492158bcc5
anime: etc. comments and changes
Paper <mrpapersonic@gmail.com>
parents:
183
diff
changeset
|
434 {"repeat", anime.GetUserRewatchedTimes()} |
9 | 435 }} |
436 }; | |
437 // clang-format on | |
175 | 438 |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
439 const std::optional<nlohmann::json> res = SendJSONRequest(json); |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
440 if (!res) |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
441 return 0; |
175 | 442 |
369 | 443 const nlohmann::json &result = res.value(); |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
444 |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
445 session.SetStatusBar(Strings::Translate("AniList: Anime entry updated successfully!")); |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
446 |
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
447 return JSON::GetNumber(result, "/data/SaveMediaListEntry/id"_json_pointer, 0); |
9 | 448 } |
449 | |
369 | 450 static int ParseUser(const nlohmann::json &json) |
451 { | |
452 auto &auth = session.config.auth.anilist; | |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
453 |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
454 return auth.user_id = JSON::GetNumber(json, "/id"_json_pointer, 0); |
9 | 455 } |
456 | |
369 | 457 bool AuthorizeUser() |
458 { | |
459 auto &auth = session.config.auth.anilist; | |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
460 |
9 | 461 /* Prompt for PIN */ |
369 | 462 QDesktopServices::openUrl(QUrl(Strings::ToQString( |
463 "https://anilist.co/api/v2/oauth/authorize?client_id=" + std::string(CLIENT_ID) + "&response_type=token"))); | |
175 | 464 |
9 | 465 bool ok; |
466 QString token = QInputDialog::getText( | |
369 | 467 0, "Credentials needed!", "Please enter the code given to you after logging in to AniList:", QLineEdit::Normal, |
468 "", &ok); | |
175 | 469 |
470 if (!ok || token.isEmpty()) | |
44
619cbd6e69f9
filesystem: fix CreateDirectories function
Paper <mrpapersonic@gmail.com>
parents:
36
diff
changeset
|
471 return false; |
175 | 472 |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
473 auth.auth_token = Strings::ToUtf8String(token); |
175 | 474 |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
475 session.SetStatusBar(Strings::Translate("AniList: Requesting user ID...")); |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
476 |
369 | 477 static constexpr std::string_view query = "query {\n" |
478 " Viewer {\n" | |
479 " id\n" | |
480 " }\n" | |
481 "}\n"; | |
325
78929794e7d8
pages/seasons: run seasons search in a separate thread
Paper <paper@paper.us.eu.org>
parents:
324
diff
changeset
|
482 |
9 | 483 nlohmann::json json = { |
369 | 484 {"query", query} |
485 }; | |
175 | 486 |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
487 /* SendJSONRequest handles status errors */ |
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
488 const std::optional<nlohmann::json> ret = SendJSONRequest(json); |
315
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
489 if (!ret) |
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
490 return 0; |
34347fd2a2de
session: allow printing status messages
Paper <paper@paper.us.eu.org>
parents:
305
diff
changeset
|
491 |
369 | 492 const nlohmann::json &result = ret.value(); |
323
1686fac290c5
services/anilist: refactor HTTP requests...
Paper <paper@paper.us.eu.org>
parents:
321
diff
changeset
|
493 |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
494 if (ParseUser(result["data"]["Viewer"])) |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
495 session.SetStatusBar(Strings::Translate("AniList: Successfully retrieved user data!")); |
319
d928ec7b6a0d
services/kitsu: implement GetAnimeList()
Paper <paper@paper.us.eu.org>
parents:
317
diff
changeset
|
496 else |
334
948955c3ba81
services: use fmt for setting the status bar
Paper <paper@paper.us.eu.org>
parents:
327
diff
changeset
|
497 session.SetStatusBar(Strings::Translate("AniList: Failed to retrieve user ID!")); |
369 | 498 |
44
619cbd6e69f9
filesystem: fix CreateDirectories function
Paper <mrpapersonic@gmail.com>
parents:
36
diff
changeset
|
499 return true; |
9 | 500 } |
501 | |
63 | 502 } // namespace AniList |
503 } // namespace Services |