Mercurial > minori
comparison src/services/kitsu.cc @ 317:b1f4d1867ab1
services: VERY initial Kitsu support
it only supports user authentication for now, but it's definitely
a start.
author | Paper <paper@paper.us.eu.org> |
---|---|
date | Wed, 12 Jun 2024 04:07:10 -0400 (7 months ago) |
parents | |
children | d928ec7b6a0d |
comparison
equal
deleted
inserted
replaced
316:180714442770 | 317:b1f4d1867ab1 |
---|---|
1 #include "services/anilist.h" | |
2 #include "core/anime.h" | |
3 #include "core/anime_db.h" | |
4 #include "core/date.h" | |
5 #include "core/config.h" | |
6 #include "core/http.h" | |
7 #include "core/json.h" | |
8 #include "core/session.h" | |
9 #include "core/strings.h" | |
10 #include "core/time.h" | |
11 #include "gui/translate/anilist.h" | |
12 | |
13 #include <QByteArray> | |
14 #include <QDate> | |
15 #include <QDesktopServices> | |
16 #include <QInputDialog> | |
17 #include <QLineEdit> | |
18 #include <QMessageBox> | |
19 #include <QUrl> | |
20 | |
21 #include <chrono> | |
22 #include <exception> | |
23 #include <string_view> | |
24 | |
25 #include <iostream> | |
26 | |
27 using namespace nlohmann::literals::json_literals; | |
28 | |
29 static constexpr std::string_view CLIENT_ID = "dd031b32d2f56c990b1425efe6c42ad847e7fe3ab46bf1299f05ecd856bdb7dd"; | |
30 static constexpr std::string_view CLIENT_SECRET = "54d7307928f63414defd96399fc31ba847961ceaecef3a5fd93144e960c0e151"; | |
31 | |
32 static constexpr std::string_view BASE_API_PATH = "https://kitsu.io/api/edge"; | |
33 static constexpr std::string_view OAUTH_PATH = "https://kitsu.io/api/oauth/token"; | |
34 | |
35 namespace Services { | |
36 namespace Kitsu { | |
37 | |
38 /* This nifty little function basically handles authentication AND reauthentication. */ | |
39 static bool SendAuthRequest(const nlohmann::json& data) { | |
40 static const std::vector<std::string> headers = { | |
41 {"Content-Type: application/json"} | |
42 }; | |
43 | |
44 const std::string ret = Strings::ToUtf8String(HTTP::Request(std::string(OAUTH_PATH), headers, data.dump(), HTTP::Type::Post)); | |
45 if (ret.empty()) { | |
46 session.SetStatusBar("Kitsu: Request returned empty data!"); | |
47 return false; | |
48 } | |
49 | |
50 nlohmann::json result; | |
51 try { | |
52 result = nlohmann::json::parse(ret, nullptr, false); | |
53 } catch (const std::exception& ex) { | |
54 session.SetStatusBar(std::string("Kitsu: Failed to parse authorization data with error \"") + ex.what() + "\"!"); | |
55 return false; | |
56 } | |
57 | |
58 if (result.contains("/error"_json_pointer)) { | |
59 std::string status = "Kitsu: Failed with error \""; | |
60 status += result["/error"_json_pointer].get<std::string>(); | |
61 | |
62 if (result.contains("/error_description"_json_pointer)) { | |
63 status += "\" and description \""; | |
64 status += result["/error_description"_json_pointer].get<std::string>(); | |
65 } | |
66 | |
67 status += "\"!"; | |
68 | |
69 session.SetStatusBar(status); | |
70 return false; | |
71 } | |
72 | |
73 const std::vector<nlohmann::json::json_pointer> required = { | |
74 "/access_token"_json_pointer, | |
75 "/created_at"_json_pointer, | |
76 "/expires_in"_json_pointer, | |
77 "/refresh_token"_json_pointer, | |
78 "/scope"_json_pointer, | |
79 "/token_type"_json_pointer | |
80 }; | |
81 | |
82 for (const auto& ptr : required) { | |
83 if (!result.contains(ptr)) { | |
84 session.SetStatusBar("Kitsu: Authorization request returned bad data!"); | |
85 return false; | |
86 } | |
87 } | |
88 | |
89 session.config.auth.kitsu.access_token = result["/access_token"_json_pointer].get<std::string>(); | |
90 session.config.auth.kitsu.access_token_expiration | |
91 = result["/created_at"_json_pointer].get<Time::Timestamp>(); | |
92 + result["/expires_in"_json_pointer].get<Time::Timestamp>(); | |
93 session.config.auth.kitsu.refresh_token = result["/refresh_token"_json_pointer].get<std::string>(); | |
94 | |
95 /* the next two are not that important */ | |
96 | |
97 return true; | |
98 } | |
99 | |
100 static bool RefreshAccessToken(std::string& access_token, const std::string& refresh_token) { | |
101 const nlohmann::json request = { | |
102 {"grant_type", "refresh_token"}, | |
103 {"refresh_token", refresh_token} | |
104 }; | |
105 | |
106 if (!SendAuthRequest(request)) | |
107 return false; | |
108 | |
109 return true; | |
110 } | |
111 | |
112 /* ----------------------------------------------------------------------------- */ | |
113 | |
114 static std::optional<std::string> AccountAccessToken() { | |
115 auto& auth = session.config.auth.kitsu; | |
116 | |
117 if (Time::GetSystemTime() >= session.config.auth.kitsu.access_token_expiration) | |
118 if (!RefreshAccessToken(auth.access_token, auth.refresh_token)) | |
119 return std::nullopt; | |
120 | |
121 return auth.access_token; | |
122 } | |
123 | |
124 /* ----------------------------------------------------------------------------- */ | |
125 | |
126 static std::optional<std::string> SendRequest(const std::string& path, const std::map<std::string, std::string>& params) { | |
127 std::optional<std::string> token = AccountAccessToken(); | |
128 if (!token) | |
129 return std::nullopt; | |
130 | |
131 const std::vector<std::string> headers = { | |
132 "Accept: application/vnd.api+json", | |
133 "Authorization: Bearer " + token.value(), | |
134 "Content-Type: application/vnd.api+json" | |
135 }; | |
136 | |
137 const std::string url = HTTP::EncodeParamsList(std::string(BASE_API_PATH) + path, params); | |
138 | |
139 return Strings::ToUtf8String(HTTP::Request(url, headers, "", HTTP::Type::Get)); | |
140 } | |
141 | |
142 static void ParseTitleJson(Anime::Anime& anime, const nlohmann::json& json) { | |
143 static const std::map<std::string, Anime::TitleLanguage> lookup = { | |
144 {"en", Anime::TitleLanguage::English}, | |
145 {"en_jp", Anime::TitleLanguage::Romaji}, | |
146 {"ja_jp", Anime::TitleLanguage::Native} | |
147 }; | |
148 | |
149 for (const auto& [string, title] : lookup) | |
150 if (json.contains(string)) | |
151 anime.SetTitle(title, json[string].get<std::string>()); | |
152 } | |
153 | |
154 static void ParseSubtype(Anime::Anime& anime, const std::string& str) { | |
155 static const std::map<std::string, Anime::SeriesFormat> lookup = { | |
156 {"ONA", Anime::SeriesFormat::Ona}, | |
157 {"OVA", Anime::SeriesFormat::Ova}, | |
158 {"TV", Anime::SeriesFormat::Tv}, | |
159 {"movie", Anime::SeriesFormat::Movie}, | |
160 {"music", Anime::SeriesFormat::Music}, | |
161 {"special", Anime::SeriesFormat::Special} | |
162 }; | |
163 | |
164 if (lookup.find(str) == lookup.end()) | |
165 return; | |
166 | |
167 anime.SetFormat(lookup.at(str)); | |
168 } | |
169 | |
170 static const std::string FAILED_TO_PARSE = "Kitsu: Failed to parse anime object!"; | |
171 | |
172 static int ParseAnimeJson(const nlohmann::json& json) { | |
173 const std::string service_id = json["/id"_json_pointer].get<std::string>(); | |
174 if (service_id.empty()) { | |
175 session.SetStatusBar(FAILED_TO_PARSE); | |
176 return 0; | |
177 } | |
178 | |
179 if (!json.contains("/attributes"_json_pointer)) { | |
180 session.SetStatusBar(FAILED_TO_PARSE); | |
181 return 0; | |
182 } | |
183 | |
184 const auto& attributes = json["/attributes"_json_pointer]; | |
185 | |
186 int id = Anime::db.LookupServiceIdOrUnused(Anime::Service::Kitsu, service_id); | |
187 if (!id) { | |
188 session.SetStatusBar(FAILED_TO_PARSE); | |
189 return 0; | |
190 } | |
191 | |
192 Anime::Anime& anime = Anime::db.items[id]; | |
193 | |
194 anime.SetId(id); | |
195 anime.SetServiceId(Anime::Service::Kitsu, service_id); | |
196 anime.SetSynopsis(attributes["/synopsis"_json_pointer].get<std::string>()); | |
197 ParseTitleJson(anime, attributes["/titles"_json_pointer]); | |
198 | |
199 // FIXME: parse abbreviatedTitles for synonyms?? | |
200 | |
201 anime.SetAudienceScore(JSON::GetNumber<double>(attributes, "/averageRating"_json_pointer)); | |
202 | |
203 if (attributes.contains("/startDate"_json_pointer)) | |
204 anime.SetAirDate(attributes["/startDate"_json_pointer].get<std::string>()); | |
205 | |
206 // TODO: endDate | |
207 | |
208 if (attributes.contains("/subtype"_json_pointer)) | |
209 ParseSubtype(anime, attributes["/subtype"_json_pointer].get<std::string>()); | |
210 | |
211 anime.SetPosterUrl(attributes["/posterImage/original"_json_pointer].get<std::string>()); | |
212 anime.SetEpisodes(attributes["/episodeCount"_json_pointer].get<int>()); | |
213 anime.SetDuration(attributes["/episodeLength"_json_pointer].get<int>()); | |
214 | |
215 return id; | |
216 } | |
217 | |
218 static int ParseLibraryJson(const nlohmann::json& json) { | |
219 if (!json.contains("/relationships/anime/data"_json_pointer) | |
220 || !json.contains("/attributes"_json_pointer) | |
221 || !json.contains("/id"_json_pointer)) { | |
222 session.SetStatusBar("Kitsu: Failed to parse library object!"); | |
223 return 0; | |
224 } | |
225 | |
226 int id = ParseAnimeJson(json["/relationships/anime/data"_json_pointer]); | |
227 if (!id) | |
228 return 0; | |
229 | |
230 const auto& attributes = json["/attributes"_json_pointer]; | |
231 | |
232 const std::string library_id = json["/id"_json_pointer].get<std::string>(); | |
233 | |
234 Anime::Anime& anime = Anime::db.items[id]; | |
235 | |
236 anime.AddToUserList(); | |
237 | |
238 anime.SetUserId(library_id); | |
239 anime.SetUserDateStarted(Date(attributes["/startedAt"_json_pointer].get<std::string>())); | |
240 anime.SetUserDateCompleted(Date(attributes["/finishedAt"_json_pointer].get<std::string>())); | |
241 anime.SetUserNotes(attributes["/notes"_json_pointer].get<std::string>()); | |
242 anime.SetUserProgress(attributes["/progress"_json_pointer].get<int>()); | |
243 anime.SetUserScore(attributes["/ratingTwenty"_json_pointer].get<int>() * 5); | |
244 anime.SetUserIsPrivate(attributes["/private"_json_pointer].get<bool>()); | |
245 anime.SetUserRewatchedTimes(attributes["/reconsumeCount"_json_pointer].get<int>()); | |
246 anime.SetUserIsRewatching(attributes["/reconsuming"_json_pointer].get<bool>()); /* "reconsuming". really? */ | |
247 // anime.SetUserStatus(); | |
248 // anime.SetUserLastUpdated(); | |
249 | |
250 return id; | |
251 } | |
252 | |
253 int GetAnimeList() { | |
254 return 0; | |
255 } | |
256 | |
257 /* unimplemented for now */ | |
258 std::vector<int> Search(const std::string& search) { | |
259 return {}; | |
260 } | |
261 | |
262 std::vector<int> GetSeason(Anime::SeriesSeason season, Date::Year year) { | |
263 return {}; | |
264 } | |
265 | |
266 int UpdateAnimeEntry(int id) { | |
267 return 0; | |
268 } | |
269 | |
270 bool AuthorizeUser(const std::string& email, const std::string& password) { | |
271 const nlohmann::json body = { | |
272 {"grant_type", "password"}, | |
273 {"username", email}, | |
274 {"password", HTTP::UrlEncode(password)} | |
275 }; | |
276 | |
277 if (!SendAuthRequest(body)) | |
278 return false; | |
279 | |
280 static const std::map<std::string, std::string> params = { | |
281 {"filter[self]", "true"} | |
282 }; | |
283 | |
284 std::optional<std::string> response = SendRequest("/users", params); | |
285 if (!response) | |
286 return false; // whuh? | |
287 | |
288 nlohmann::json json; | |
289 try { | |
290 json = nlohmann::json::parse(response.value()); | |
291 } catch (const std::exception& ex) { | |
292 session.SetStatusBar(std::string("Kitsu: Failed to parse user data with error \"") + ex.what() + "\"!"); | |
293 return false; | |
294 } | |
295 | |
296 if (!json.contains("/data/0/id"_json_pointer)) { | |
297 session.SetStatusBar("Kitsu: Failed to retrieve user ID!"); | |
298 return false; | |
299 } | |
300 | |
301 session.SetStatusBar("Kitsu: Successfully retrieved user data!"); | |
302 session.config.auth.kitsu.user_id = json["/data/0/id"_json_pointer].get<std::string>(); | |
303 | |
304 return true; | |
305 } | |
306 | |
307 } // namespace Kitsu | |
308 } // namespace Services |