Mercurial > minori
changeset 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 | 180714442770 |
children | 3b355fa948c7 |
files | Makefile.am include/core/anime.h include/core/anime_db.h include/core/config.h include/core/date.h include/core/http.h include/core/json.h include/gui/dialog/settings.h include/gui/widgets/drop_list_widget.h include/services/kitsu.h include/services/services.h src/core/anime.cc src/core/anime_db.cc src/core/config.cc src/core/date.cc src/core/http.cc src/gui/dialog/settings/application.cc src/gui/dialog/settings/library.cc src/gui/dialog/settings/services.cc src/gui/widgets/drop_list_widget.cc src/main.cc src/services/anilist.cc src/services/kitsu.cc src/services/services.cc src/track/media.cc |
diffstat | 25 files changed, 676 insertions(+), 98 deletions(-) [+] |
line wrap: on
line diff
--- a/Makefile.am Tue Jun 11 15:11:09 2024 -0400 +++ b/Makefile.am Wed Jun 12 04:07:10 2024 -0400 @@ -143,6 +143,7 @@ include/gui/widgets/anime_button.h \ include/gui/widgets/anime_info.h \ include/gui/widgets/clickable_label.h \ + include/gui/widgets/drop_list_widget.h \ include/gui/widgets/graph.h \ include/gui/widgets/optional_date.h \ include/gui/widgets/poster.h \ @@ -168,6 +169,7 @@ include/core/torrent.h \ include/library/library.h \ include/services/anilist.h \ + include/services/kitsu.h \ include/services/services.h \ include/sys/glib/dark_theme.h \ include/sys/osx/dark_theme.h \ @@ -220,6 +222,7 @@ src/gui/widgets/anime_button.cc \ src/gui/widgets/anime_info.cc \ src/gui/widgets/clickable_label.cc \ + src/gui/widgets/drop_list_widget.cc \ src/gui/widgets/elided_label.cc \ src/gui/widgets/optional_date.cc \ src/gui/widgets/poster.cc \ @@ -230,6 +233,7 @@ src/gui/window.cc \ src/library/library.cc \ src/services/anilist.cc \ + src/services/kitsu.cc \ src/services/services.cc \ src/track/media.cc \ src/main.cc \
--- a/include/core/anime.h Tue Jun 11 15:11:09 2024 -0400 +++ b/include/core/anime.h Wed Jun 12 04:07:10 2024 -0400 @@ -111,7 +111,7 @@ ScoreFormat::Point10, ScoreFormat::Point5, ScoreFormat::Point3}; struct ListInformation { - int id = 0; + std::string id; int progress = 0; int score = 0; // this will ALWAYS be in POINT_100 format internally ListStatus status = ListStatus::NotInList; @@ -135,7 +135,7 @@ std::vector<std::string> genres; std::vector<std::string> producers; SeriesFormat format = SeriesFormat::Unknown; - int audience_score = 0; + double audience_score = 0; std::string synopsis; int duration = 0; std::string poster_url; @@ -144,6 +144,7 @@ class Anime { public: /* User list data */ + std::string GetUserId() const; ListStatus GetUserStatus() const; int GetUserProgress() const; int GetUserScore() const; @@ -156,6 +157,7 @@ uint64_t GetUserTimeUpdated() const; std::string GetUserNotes() const; + void SetUserId(const std::string& id); void SetUserStatus(ListStatus status); void SetUserScore(int score); void SetUserProgress(int progress); @@ -179,7 +181,7 @@ std::vector<std::string> GetProducers() const; SeriesFormat GetFormat() const; SeriesSeason GetSeason() const; - int GetAudienceScore() const; + double GetAudienceScore() const; std::string GetSynopsis() const; int GetDuration() const; std::string GetPosterUrl() const; @@ -196,7 +198,7 @@ void SetGenres(std::vector<std::string> const& genres); void SetProducers(std::vector<std::string> const& producers); void SetFormat(SeriesFormat format); - void SetAudienceScore(int audience_score); + void SetAudienceScore(double audience_score); void SetSynopsis(std::string synopsis); void SetDuration(int duration); void SetPosterUrl(std::string poster);
--- a/include/core/anime_db.h Tue Jun 11 15:11:09 2024 -0400 +++ b/include/core/anime_db.h Wed Jun 12 04:07:10 2024 -0400 @@ -25,6 +25,12 @@ bool ParseDatabaseJSON(const nlohmann::json& json); bool LoadDatabaseFromDisk(); + + /* These are here to make sure that our service IDs don't collide + * and make the whole thing go boom. */ + int GetUnusedId(); + int LookupServiceId(Service service, const std::string& id_to_find); + int LookupServiceIdOrUnused(Service service, const std::string& id_to_find); }; extern Database db;
--- a/include/core/config.h Tue Jun 11 15:11:09 2024 -0400 +++ b/include/core/config.h Wed Jun 12 04:07:10 2024 -0400 @@ -2,6 +2,7 @@ #define MINORI_CORE_CONFIG_H_ #include "core/anime.h" +#include "core/time.h" #include "gui/locale.h" #include "gui/theme.h" @@ -40,6 +41,12 @@ std::string auth_token; int user_id; } anilist; + struct { + std::string access_token; + Time::Timestamp access_token_expiration; /* Unix time */ + std::string refresh_token; + std::string user_id; + } kitsu; } auth; struct {
--- a/include/core/date.h Tue Jun 11 15:11:09 2024 -0400 +++ b/include/core/date.h Wed Jun 12 04:07:10 2024 -0400 @@ -4,9 +4,14 @@ #include "json/json_fwd.hpp" #include <optional> +#include <string> class QDate; +/* TODO: refactor constructors, as they aren't meant + * to be used in this way and may cause problems down + * the line */ + class Date { public: using Year = unsigned int; @@ -29,6 +34,7 @@ Date(); Date(Year y); Date(Year y, Month m, Day d); + Date(const std::string& str); Date(const QDate& date); Date(const nlohmann::json& json); bool IsValid() const;
--- a/include/core/http.h Tue Jun 11 15:11:09 2024 -0400 +++ b/include/core/http.h Wed Jun 12 04:07:10 2024 -0400 @@ -10,6 +10,12 @@ namespace HTTP { +/* calls libcurl to encode/decode */ +std::string UrlEncode(const std::string& data); +std::string UrlDecode(const std::string& data); + +std::string EncodeParamsList(std::string base, const std::map<std::string, std::string>& params); + enum class Type { Get, Post
--- a/include/core/json.h Tue Jun 11 15:11:09 2024 -0400 +++ b/include/core/json.h Wed Jun 12 04:07:10 2024 -0400 @@ -24,6 +24,7 @@ namespace JSON { +/* TODO: refactor these to return a std::optional... */ template<typename T = std::string> T GetString(const nlohmann::json& json, const nlohmann::json::json_pointer& ptr, T def) { if (json.contains(ptr) && json[ptr].is_string())
--- a/include/gui/dialog/settings.h Tue Jun 11 15:11:09 2024 -0400 +++ b/include/gui/dialog/settings.h Wed Jun 12 04:07:10 2024 -0400 @@ -15,22 +15,6 @@ #include <QVBoxLayout> #include <QWidget> -/* !!! MOVE THIS ELSEWHERE! */ -class DroppableListWidget : public QListWidget { - Q_OBJECT - -public: - explicit DroppableListWidget(QWidget* parent); - -signals: - void FilesDropped(QStringList list); - -protected: - void dragEnterEvent(QDragEnterEvent* event) override; - void dragMoveEvent(QDragMoveEvent* event) override; - void dropEvent(QDropEvent* event) override; -}; - class SettingsPage : public QWidget { Q_OBJECT @@ -56,6 +40,7 @@ private: QWidget* CreateMainPage(); QWidget* CreateAniListPage(); + QWidget* CreateKitsuPage(); decltype(session.config.service) service; };
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/include/gui/widgets/drop_list_widget.h Wed Jun 12 04:07:10 2024 -0400 @@ -0,0 +1,27 @@ +#ifndef MINORI_GUI_WIDGETS_DROP_LIST_WIDGET_H_ +#define MINORI_GUI_WIDGETS_DROP_LIST_WIDGET_H_ + +#include <QListWidget> +#include <QString> +#include <QWidget> + +class QDragEnterEvent; +class QDragMoveEvent; +class QDropEvent; + +class DroppableListWidget : public QListWidget { + Q_OBJECT + +public: + explicit DroppableListWidget(QWidget* parent); + +signals: + void FilesDropped(QStringList list); + +protected: + void dragEnterEvent(QDragEnterEvent* event) override; + void dragMoveEvent(QDragMoveEvent* event) override; + void dropEvent(QDropEvent* event) override; +}; + +#endif // MINORI_GUI_WIDGETS_DROP_LIST_WIDGET_H_ \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/include/services/kitsu.h Wed Jun 12 04:07:10 2024 -0400 @@ -0,0 +1,24 @@ +#ifndef MINORI_SERVICES_KITSU_H_ +#define MINORI_SERVICES_KITSU_H_ + +#include "core/anime.h" +#include "core/date.h" + +#include <string> +#include <vector> + +namespace Services { +namespace Kitsu { + +/* neither of these are stored in the config and only held temporarily */ +bool AuthorizeUser(const std::string& email, const std::string& password); + +int GetAnimeList(); +std::vector<int> Search(const std::string& search); +std::vector<int> GetSeason(Anime::SeriesSeason season, Date::Year year); +int UpdateAnimeEntry(int id); + +} // namespace Kitsu +} // namespace Services + +#endif // MINORI_SERVICES_KITSU_H_
--- a/include/services/services.h Tue Jun 11 15:11:09 2024 -0400 +++ b/include/services/services.h Wed Jun 12 04:07:10 2024 -0400 @@ -13,7 +13,6 @@ std::vector<int> Search(const std::string& search); std::vector<int> GetSeason(Anime::SeriesSeason season, Date::Year year); void UpdateAnimeEntry(int id); -bool Authorize(); }; // namespace Services
--- a/src/core/anime.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/core/anime.cc Wed Jun 12 04:07:10 2024 -0400 @@ -33,6 +33,11 @@ list_info_.reset(); } +std::string Anime::GetUserId() const { + assert(list_info_.has_value()); + return list_info_->id; +} + ListStatus Anime::GetUserStatus() const { assert(list_info_.has_value()); return list_info_->status; @@ -117,6 +122,11 @@ return list_info_->notes; } +void Anime::SetUserId(const std::string& id) { + assert(list_info_.has_value()); + list_info_->id = id; +} + void Anime::SetUserStatus(ListStatus status) { assert(list_info_.has_value()); list_info_->status = status; @@ -235,7 +245,7 @@ return (month.has_value() ? GetSeasonForMonth(month.value()) : SeriesSeason::Unknown); } -int Anime::GetAudienceScore() const { +double Anime::GetAudienceScore() const { return info_.audience_score; } @@ -311,7 +321,7 @@ info_.format = format; } -void Anime::SetAudienceScore(int audience_score) { +void Anime::SetAudienceScore(double audience_score) { info_.audience_score = audience_score; }
--- a/src/core/anime_db.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/core/anime_db.cc Wed Jun 12 04:07:10 2024 -0400 @@ -12,7 +12,9 @@ #include <fstream> #include <exception> +#include <cstdlib> #include <iostream> +#include <random> namespace Anime { @@ -300,6 +302,42 @@ return true; } +int Database::GetUnusedId() { + /* TODO: move these out of here */ + + std::random_device rd; + std::mt19937 gen(rd()); + std::uniform_int_distribution<int> distrib(1, INT_MAX); + int res; + + do { + res = distrib(gen); + } while (items.count(res)); + + return res; +} + +int Database::LookupServiceId(Service service, const std::string& id_to_find) { + for (const auto& [id, anime] : items) { + std::optional<std::string> service_id = anime.GetServiceId(service); + if (!service_id) + continue; + + if (service_id == id_to_find) + return id; + } + + return 0; +} + +int Database::LookupServiceIdOrUnused(Service service, const std::string& id_to_find) { + int id = LookupServiceId(service, id_to_find); + if (id) + return id; + + return GetUnusedId(); +} + Database db; } // namespace Anime
--- a/src/core/config.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/core/config.cc Wed Jun 12 04:07:10 2024 -0400 @@ -60,6 +60,11 @@ auth.anilist.auth_token = INI::GetIniString(ini, "Authentication/AniList", "Auth Token", ""); auth.anilist.user_id = INI::GetIniInteger<int>(ini, "Authentication/AniList", "User ID", 0); + auth.kitsu.access_token = INI::GetIniString(ini, "Authentication/Kitsu", "Access Token", ""); + auth.kitsu.access_token_expiration = INI::GetIniInteger<Time::Timestamp>(ini, "Authentication/Kitsu", "Access Token Expiration", 0); + auth.kitsu.refresh_token = INI::GetIniString(ini, "Authentication/Kitsu", "Refresh Token", ""); + auth.kitsu.user_id = INI::GetIniString(ini, "Authentication/Kitsu", "User ID", ""); + torrents.feed_link = INI::GetIniString(ini, "Torrents", "RSS feed", "https://www.tokyotosho.info/rss.php?filter=1,11&zwnj=0"); @@ -135,6 +140,11 @@ ini["Authentication/AniList"]["Auth Token"] = auth.anilist.auth_token; ini["Authentication/AniList"]["User ID"] = Strings::ToUtf8String(auth.anilist.user_id); + ini["Authentication/Kitsu"]["Access Token"] = auth.kitsu.access_token; + ini["Authentication/Kitsu"]["Access Token Expiration"] = Strings::ToUtf8String(auth.kitsu.access_token_expiration); + ini["Authentication/Kitsu"]["Refresh Token"] = auth.kitsu.refresh_token; + ini["Authentication/Kitsu"]["User ID"] = auth.kitsu.user_id; + ini["Appearance"]["Theme"] = Translate::ToString(theme.GetTheme()); ini["Torrents"]["RSS feed"] = torrents.feed_link;
--- a/src/core/date.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/core/date.cc Wed Jun 12 04:07:10 2024 -0400 @@ -5,6 +5,7 @@ #include <QDebug> #include <algorithm> +#include <cstdio> /* An implementation of AniList's "fuzzy date" */ @@ -21,6 +22,22 @@ SetDay(d); } +Date::Date(const std::string& str) { + unsigned int y, m, d; + + /* I don't like this that much, but it works... */ + int amt = std::sscanf(str.c_str(), "%4u-%2u-%2u", &y, &m, &d); + + if (amt > 0) + SetYear(y); + + if (amt > 1) + SetMonth(static_cast<Date::Month>(m - 1)); + + if (amt > 2) + SetDay(d); +} + Date::Date(const QDate& date) { SetYear(date.year()); auto m = date.month();
--- a/src/core/http.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/core/http.cc Wed Jun 12 04:07:10 2024 -0400 @@ -8,6 +8,60 @@ namespace HTTP { +std::string UrlEncode(const std::string& data) { + /* why do I need to init curl just for this? wtf? */ + CURL *curl = curl_easy_init(); + if (!curl) + return ""; /* no way! */ + + char* output = curl_easy_escape(curl, data.data(), data.size()); + if (!output) { + curl_easy_cleanup(curl); + return ""; + } + + std::string str(output); + + curl_free(output); + curl_easy_cleanup(curl); + + return str; +} + +std::string UrlDecode(const std::string& data) { + CURL *curl = curl_easy_init(); + if (!curl) + return ""; + + int outlength; + char* output = curl_easy_unescape(curl, data.data(), data.size(), &outlength); + if (!output) { + curl_easy_cleanup(curl); + return ""; + } + + std::string str(output, outlength); + + curl_free(output); + curl_easy_cleanup(curl); + + return str; +} + +std::string EncodeParamsList(std::string base, const std::map<std::string, std::string>& params) { + std::size_t count = 0; + for (const auto& param : params) { + base += (!count ? "?" : "&"); + base += UrlEncode(param.first); + base += "="; + base += UrlEncode(param.second); + + count++; + } + + return base; +} + static size_t WriteCallback(void* contents, size_t size, size_t nmemb, void* userdata) { reinterpret_cast<QByteArray*>(userdata)->append(reinterpret_cast<char*>(contents), size * nmemb); return size * nmemb; @@ -35,7 +89,7 @@ session.IncrementRequests(); curl_easy_cleanup(curl); if (res != CURLE_OK) - std::cerr << "curl_easy_perform(curl) failed!: " << curl_easy_strerror(res) << std::endl; + session.SetStatusBar(std::string("curl_easy_perform(curl) failed!: ") + curl_easy_strerror(res)); } return userdata; } @@ -122,7 +176,7 @@ callback_data_mutex_.lock(); if (res != CURLE_OK && !(res == CURLE_WRITE_ERROR && cancelled_)) - std::cerr << "curl_easy_perform(curl) failed!: " << curl_easy_strerror(res) << std::endl; + session.SetStatusBar(std::string("curl_easy_perform(curl) failed!: ") + curl_easy_strerror(res)); callback_data_mutex_.unlock(); }
--- a/src/gui/dialog/settings/application.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/gui/dialog/settings/application.cc Wed Jun 12 04:07:10 2024 -0400 @@ -111,7 +111,7 @@ } { - /* Hopefully I made this easy to parse... */ + /* Hopefully I made this easy to read... */ QCheckBox* hl_above_anime_box = new QCheckBox(tr("Display highlighted anime above others"), appearance_group_box); hl_above_anime_box->setCheckState(highlighted_anime_above_others ? Qt::Checked : Qt::Unchecked);
--- a/src/gui/dialog/settings/library.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/gui/dialog/settings/library.cc Wed Jun 12 04:07:10 2024 -0400 @@ -1,63 +1,19 @@ #include "core/session.h" #include "core/strings.h" #include "gui/dialog/settings.h" +#include "gui/widgets/drop_list_widget.h" #include <QCheckBox> #include <QDir> -#include <QDropEvent> #include <QFileDialog> -#include <QFileInfo> #include <QGroupBox> #include <QLabel> #include <QListWidget> #include <QListWidgetItem> -#include <QMimeData> #include <QPushButton> #include <QSizePolicy> #include <QVBoxLayout> -#include <algorithm> -#include <iostream> - -DroppableListWidget::DroppableListWidget(QWidget* parent) : QListWidget(parent) { - setAcceptDrops(true); -} - -void DroppableListWidget::dragMoveEvent(QDragMoveEvent* event) { - if (event->mimeData()->hasUrls()) - event->acceptProposedAction(); -} - -void DroppableListWidget::dragEnterEvent(QDragEnterEvent* event) { - if (event->mimeData()->hasUrls()) - event->acceptProposedAction(); -} - -void DroppableListWidget::dropEvent(QDropEvent* event) { - const QMimeData* mime_data = event->mimeData(); - - if (!mime_data->hasUrls()) - return; - - QStringList path_list; - QList<QUrl> url_list = mime_data->urls(); - - for (const auto& url : url_list) { - if (!url.isLocalFile()) - continue; - - const QString file = url.toLocalFile(); - const QFileInfo fileinfo(file); - if (fileinfo.exists() && fileinfo.isDir()) - path_list.append(file); - } - - if (!path_list.isEmpty()) - emit FilesDropped(path_list); - - event->acceptProposedAction(); -} - QWidget* SettingsPageLibrary::CreateFoldersWidget() { QWidget* result = new QWidget(this); result->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Maximum);
--- a/src/gui/dialog/settings/services.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/gui/dialog/settings/services.cc Wed Jun 12 04:07:10 2024 -0400 @@ -4,6 +4,7 @@ #include "gui/dialog/settings.h" #include "gui/translate/anime.h" #include "services/anilist.h" +#include "services/kitsu.h" #include <QComboBox> #include <QGroupBox> #include <QLabel> @@ -59,6 +60,71 @@ return result; } +QWidget* SettingsPageServices::CreateKitsuPage() { + QWidget* result = new QWidget(this); + result->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum); + + QVBoxLayout* full_layout = new QVBoxLayout(result); + + { + /* Account */ + QGroupBox* group_box = new QGroupBox(tr("Account"), result); + group_box->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Maximum); + + QVBoxLayout* group_box_layout = new QVBoxLayout(group_box); + + { + QWidget* credentials_grid = new QWidget(group_box); + QGridLayout* credentials_grid_layout = new QGridLayout(credentials_grid); + + /* E-mail */ + QLabel* email_label = new QLabel(tr("&E-mail"), credentials_grid); + QLineEdit* email = new QLineEdit(credentials_grid); + email_label->setBuddy(email); + credentials_grid_layout->addWidget(email_label, 0, 0); + credentials_grid_layout->addWidget(email, 1, 0); + + QLabel* password_label = new QLabel(tr("&Password:"), credentials_grid); + QLineEdit* password = new QLineEdit(credentials_grid); + password->setEchoMode(QLineEdit::Password); + password_label->setBuddy(password); + credentials_grid_layout->addWidget(password_label, 0, 1); + credentials_grid_layout->addWidget(password, 1, 1); + + { + QPushButton* auth_button = new QPushButton(credentials_grid); + connect(auth_button, &QPushButton::clicked, this, [email, password] { + Services::Kitsu::AuthorizeUser(Strings::ToUtf8String(email->text()), Strings::ToUtf8String(password->text())); + }); + auth_button->setText(session.config.auth.kitsu.access_token.empty() ? tr("Authorize...") + : tr("Re-authorize...")); + credentials_grid_layout->addWidget(auth_button, 1, 2); + } + + credentials_grid_layout->setContentsMargins(0, 0, 0, 0); + + group_box_layout->addWidget(credentials_grid); + } + + { + /* Note on password storing */ + QLabel* note_label = new QLabel(tr("Your e-mail and password are never stored by Minori and will only be used to authorize with Kitsu.\nFor more information, see <a href=\"https://kitsu.docs.apiary.io/#introduction/authentication\">Kitsu's API documentation</a>"), group_box); + note_label->setTextFormat(Qt::RichText); + note_label->setWordWrap(true); + note_label->setTextInteractionFlags(Qt::TextBrowserInteraction); + note_label->setOpenExternalLinks(true); + group_box_layout->addWidget(note_label); + } + + full_layout->addWidget(group_box); + } + + full_layout->setSpacing(10); + full_layout->addStretch(); + + return result; +} + QWidget* SettingsPageServices::CreateAniListPage() { QWidget* result = new QWidget(this); result->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum); @@ -93,6 +159,7 @@ QLabel* note_label = new QLabel(tr("<a href=\"http://anilist.co/\">Create a new AniList account</a>"), group_box); note_label->setTextFormat(Qt::RichText); + note_label->setWordWrap(true); note_label->setTextInteractionFlags(Qt::TextBrowserInteraction); note_label->setOpenExternalLinks(true); layout->addWidget(note_label); @@ -114,5 +181,6 @@ SettingsPageServices::SettingsPageServices(QWidget* parent) : SettingsPage(parent, tr("Services")) { service = session.config.service; AddTab(CreateMainPage(), tr("Main")); + AddTab(CreateKitsuPage(), tr("Kitsu")); AddTab(CreateAniListPage(), tr("AniList")); }
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/gui/widgets/drop_list_widget.cc Wed Jun 12 04:07:10 2024 -0400 @@ -0,0 +1,47 @@ +#include "gui/widgets/drop_list_widget.h" + +#include <QDragMoveEvent> +#include <QDropEvent> +#include <QMimeData> +#include <QFileInfo> + +/* currently this only sends local paths that are folders */ + +DroppableListWidget::DroppableListWidget(QWidget* parent) : QListWidget(parent) { + setAcceptDrops(true); +} + +void DroppableListWidget::dragMoveEvent(QDragMoveEvent* event) { + if (event->mimeData()->hasUrls()) + event->acceptProposedAction(); +} + +void DroppableListWidget::dragEnterEvent(QDragEnterEvent* event) { + if (event->mimeData()->hasUrls()) + event->acceptProposedAction(); +} + +void DroppableListWidget::dropEvent(QDropEvent* event) { + const QMimeData* mime_data = event->mimeData(); + + if (!mime_data->hasUrls()) + return; + + QStringList path_list; + QList<QUrl> url_list = mime_data->urls(); + + for (const auto& url : url_list) { + if (!url.isLocalFile()) + continue; + + const QString file = url.toLocalFile(); + const QFileInfo fileinfo(file); + if (fileinfo.exists() && fileinfo.isDir()) + path_list.append(file); + } + + if (!path_list.isEmpty()) + emit FilesDropped(path_list); + + event->acceptProposedAction(); +} \ No newline at end of file
--- a/src/main.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/main.cc Wed Jun 12 04:07:10 2024 -0400 @@ -9,7 +9,7 @@ #include <QStyleFactory> #include <QTranslator> -#include <iostream> +#include <cstdlib> int main(int argc, char** argv) { QApplication app(argc, argv);
--- a/src/services/anilist.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/services/anilist.cc Wed Jun 12 04:07:10 2024 -0400 @@ -23,6 +23,9 @@ #include <iostream> +/* This file really sucks because it was made when I was first + * really "learning" C++ */ + using namespace nlohmann::literals::json_literals; namespace Services { @@ -54,19 +57,17 @@ "synonyms\n" \ "description(asHtml: false)\n" -class Account { -public: +/* FIXME: why is this here */ + +static struct { int UserId() const { return session.config.auth.anilist.user_id; } void SetUserId(const int id) { session.config.auth.anilist.user_id = id; } std::string AuthToken() const { return session.config.auth.anilist.auth_token; } void SetAuthToken(const std::string& auth_token) { session.config.auth.anilist.auth_token = auth_token; } - bool Authenticated() const { return !AuthToken().empty(); } - bool IsValid() const { return UserId() && Authenticated(); } -}; - -static Account account; + bool IsValid() const { return UserId() && !AuthToken().empty(); } +} account; static std::string SendRequest(const std::string& data) { std::vector<std::string> headers = {"Authorization: Bearer " + account.AuthToken(), "Accept: application/json", @@ -89,7 +90,7 @@ if (out.contains("/errors"_json_pointer) && out.at("/errors"_json_pointer).is_array()) { for (const auto& error : out.at("/errors"_json_pointer)) - std::cerr << "[AniList] Received an error in response: " + std::cerr << "AniList: Received an error in response: " << JSON::GetString<std::string>(error, "/message"_json_pointer, "") << std::endl; session.SetStatusBar("AniList: Received an error in response!"); @@ -149,14 +150,19 @@ } static int ParseMediaJson(const nlohmann::json& json) { - int id = JSON::GetNumber(json, "/id"_json_pointer); - if (!id) + if (!json.contains("/id"_json_pointer) || !json["/id"_json_pointer].is_number()) return 0; + std::string service_id = Strings::ToUtf8String(json["/id"_json_pointer].get<int>()); + + int id = Anime::db.LookupServiceIdOrUnused(Anime::Service::AniList, service_id); + Anime::Anime& anime = Anime::db.items[id]; anime.SetId(id); - anime.SetServiceId(Anime::Service::AniList, Strings::ToUtf8String(id)); - anime.SetServiceId(Anime::Service::MyAnimeList, Strings::ToUtf8String(JSON::GetNumber(json, "/id_mal"_json_pointer))); + anime.SetServiceId(Anime::Service::AniList, service_id); + + if (json.contains("/id_mal"_json_pointer)) + anime.SetServiceId(Anime::Service::MyAnimeList, json["/id_mal"_json_pointer].get<std::string>()); ParseTitle(json.at("/title"_json_pointer), anime); @@ -185,7 +191,9 @@ } static int ParseListItem(const nlohmann::json& json) { - int id = ParseMediaJson(json["media"]); + int id = ParseMediaJson(json); + if (!id) + return 0; Anime::Anime& anime = Anime::db.items[id]; @@ -205,15 +213,15 @@ } static int ParseList(const nlohmann::json& json) { - for (const auto& entry : json["entries"].items()) { + for (const auto& entry : json["entries"].items()) ParseListItem(entry.value()); - } + return 1; } int GetAnimeList() { if (!account.IsValid()) { - session.SetStatusBar("AniList: Account isn't valid!"); + session.SetStatusBar("AniList: Account isn't valid! (unauthorized?)"); return 0; }
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/services/kitsu.cc Wed Jun 12 04:07:10 2024 -0400 @@ -0,0 +1,308 @@ +#include "services/anilist.h" +#include "core/anime.h" +#include "core/anime_db.h" +#include "core/date.h" +#include "core/config.h" +#include "core/http.h" +#include "core/json.h" +#include "core/session.h" +#include "core/strings.h" +#include "core/time.h" +#include "gui/translate/anilist.h" + +#include <QByteArray> +#include <QDate> +#include <QDesktopServices> +#include <QInputDialog> +#include <QLineEdit> +#include <QMessageBox> +#include <QUrl> + +#include <chrono> +#include <exception> +#include <string_view> + +#include <iostream> + +using namespace nlohmann::literals::json_literals; + +static constexpr std::string_view CLIENT_ID = "dd031b32d2f56c990b1425efe6c42ad847e7fe3ab46bf1299f05ecd856bdb7dd"; +static constexpr std::string_view CLIENT_SECRET = "54d7307928f63414defd96399fc31ba847961ceaecef3a5fd93144e960c0e151"; + +static constexpr std::string_view BASE_API_PATH = "https://kitsu.io/api/edge"; +static constexpr std::string_view OAUTH_PATH = "https://kitsu.io/api/oauth/token"; + +namespace Services { +namespace Kitsu { + +/* This nifty little function basically handles authentication AND reauthentication. */ +static bool SendAuthRequest(const nlohmann::json& data) { + static const std::vector<std::string> headers = { + {"Content-Type: application/json"} + }; + + const std::string ret = Strings::ToUtf8String(HTTP::Request(std::string(OAUTH_PATH), headers, data.dump(), HTTP::Type::Post)); + if (ret.empty()) { + session.SetStatusBar("Kitsu: Request returned empty data!"); + return false; + } + + nlohmann::json result; + try { + result = nlohmann::json::parse(ret, nullptr, false); + } catch (const std::exception& ex) { + session.SetStatusBar(std::string("Kitsu: Failed to parse authorization data with error \"") + ex.what() + "\"!"); + return false; + } + + if (result.contains("/error"_json_pointer)) { + std::string status = "Kitsu: Failed with error \""; + status += result["/error"_json_pointer].get<std::string>(); + + if (result.contains("/error_description"_json_pointer)) { + status += "\" and description \""; + status += result["/error_description"_json_pointer].get<std::string>(); + } + + status += "\"!"; + + session.SetStatusBar(status); + return false; + } + + const std::vector<nlohmann::json::json_pointer> required = { + "/access_token"_json_pointer, + "/created_at"_json_pointer, + "/expires_in"_json_pointer, + "/refresh_token"_json_pointer, + "/scope"_json_pointer, + "/token_type"_json_pointer + }; + + for (const auto& ptr : required) { + if (!result.contains(ptr)) { + session.SetStatusBar("Kitsu: Authorization request returned bad data!"); + return false; + } + } + + session.config.auth.kitsu.access_token = result["/access_token"_json_pointer].get<std::string>(); + session.config.auth.kitsu.access_token_expiration + = result["/created_at"_json_pointer].get<Time::Timestamp>(); + + result["/expires_in"_json_pointer].get<Time::Timestamp>(); + session.config.auth.kitsu.refresh_token = result["/refresh_token"_json_pointer].get<std::string>(); + + /* the next two are not that important */ + + return true; +} + +static bool RefreshAccessToken(std::string& access_token, const std::string& refresh_token) { + const nlohmann::json request = { + {"grant_type", "refresh_token"}, + {"refresh_token", refresh_token} + }; + + if (!SendAuthRequest(request)) + return false; + + return true; +} + +/* ----------------------------------------------------------------------------- */ + +static std::optional<std::string> AccountAccessToken() { + auto& auth = session.config.auth.kitsu; + + if (Time::GetSystemTime() >= session.config.auth.kitsu.access_token_expiration) + if (!RefreshAccessToken(auth.access_token, auth.refresh_token)) + return std::nullopt; + + return auth.access_token; +} + +/* ----------------------------------------------------------------------------- */ + +static std::optional<std::string> SendRequest(const std::string& path, const std::map<std::string, std::string>& params) { + std::optional<std::string> token = AccountAccessToken(); + if (!token) + return std::nullopt; + + const std::vector<std::string> headers = { + "Accept: application/vnd.api+json", + "Authorization: Bearer " + token.value(), + "Content-Type: application/vnd.api+json" + }; + + const std::string url = HTTP::EncodeParamsList(std::string(BASE_API_PATH) + path, params); + + return Strings::ToUtf8String(HTTP::Request(url, headers, "", HTTP::Type::Get)); +} + +static void ParseTitleJson(Anime::Anime& anime, const nlohmann::json& json) { + static const std::map<std::string, Anime::TitleLanguage> lookup = { + {"en", Anime::TitleLanguage::English}, + {"en_jp", Anime::TitleLanguage::Romaji}, + {"ja_jp", Anime::TitleLanguage::Native} + }; + + for (const auto& [string, title] : lookup) + if (json.contains(string)) + anime.SetTitle(title, json[string].get<std::string>()); +} + +static void ParseSubtype(Anime::Anime& anime, const std::string& str) { + static const std::map<std::string, Anime::SeriesFormat> lookup = { + {"ONA", Anime::SeriesFormat::Ona}, + {"OVA", Anime::SeriesFormat::Ova}, + {"TV", Anime::SeriesFormat::Tv}, + {"movie", Anime::SeriesFormat::Movie}, + {"music", Anime::SeriesFormat::Music}, + {"special", Anime::SeriesFormat::Special} + }; + + if (lookup.find(str) == lookup.end()) + return; + + anime.SetFormat(lookup.at(str)); +} + +static const std::string FAILED_TO_PARSE = "Kitsu: Failed to parse anime object!"; + +static int ParseAnimeJson(const nlohmann::json& json) { + const std::string service_id = json["/id"_json_pointer].get<std::string>(); + if (service_id.empty()) { + session.SetStatusBar(FAILED_TO_PARSE); + return 0; + } + + if (!json.contains("/attributes"_json_pointer)) { + session.SetStatusBar(FAILED_TO_PARSE); + return 0; + } + + const auto& attributes = json["/attributes"_json_pointer]; + + int id = Anime::db.LookupServiceIdOrUnused(Anime::Service::Kitsu, service_id); + if (!id) { + session.SetStatusBar(FAILED_TO_PARSE); + return 0; + } + + Anime::Anime& anime = Anime::db.items[id]; + + anime.SetId(id); + anime.SetServiceId(Anime::Service::Kitsu, service_id); + anime.SetSynopsis(attributes["/synopsis"_json_pointer].get<std::string>()); + ParseTitleJson(anime, attributes["/titles"_json_pointer]); + + // FIXME: parse abbreviatedTitles for synonyms?? + + anime.SetAudienceScore(JSON::GetNumber<double>(attributes, "/averageRating"_json_pointer)); + + if (attributes.contains("/startDate"_json_pointer)) + anime.SetAirDate(attributes["/startDate"_json_pointer].get<std::string>()); + + // TODO: endDate + + if (attributes.contains("/subtype"_json_pointer)) + ParseSubtype(anime, attributes["/subtype"_json_pointer].get<std::string>()); + + anime.SetPosterUrl(attributes["/posterImage/original"_json_pointer].get<std::string>()); + anime.SetEpisodes(attributes["/episodeCount"_json_pointer].get<int>()); + anime.SetDuration(attributes["/episodeLength"_json_pointer].get<int>()); + + return id; +} + +static int ParseLibraryJson(const nlohmann::json& json) { + if (!json.contains("/relationships/anime/data"_json_pointer) + || !json.contains("/attributes"_json_pointer) + || !json.contains("/id"_json_pointer)) { + session.SetStatusBar("Kitsu: Failed to parse library object!"); + return 0; + } + + int id = ParseAnimeJson(json["/relationships/anime/data"_json_pointer]); + if (!id) + return 0; + + const auto& attributes = json["/attributes"_json_pointer]; + + const std::string library_id = json["/id"_json_pointer].get<std::string>(); + + Anime::Anime& anime = Anime::db.items[id]; + + anime.AddToUserList(); + + anime.SetUserId(library_id); + anime.SetUserDateStarted(Date(attributes["/startedAt"_json_pointer].get<std::string>())); + anime.SetUserDateCompleted(Date(attributes["/finishedAt"_json_pointer].get<std::string>())); + anime.SetUserNotes(attributes["/notes"_json_pointer].get<std::string>()); + anime.SetUserProgress(attributes["/progress"_json_pointer].get<int>()); + anime.SetUserScore(attributes["/ratingTwenty"_json_pointer].get<int>() * 5); + anime.SetUserIsPrivate(attributes["/private"_json_pointer].get<bool>()); + anime.SetUserRewatchedTimes(attributes["/reconsumeCount"_json_pointer].get<int>()); + anime.SetUserIsRewatching(attributes["/reconsuming"_json_pointer].get<bool>()); /* "reconsuming". really? */ + // anime.SetUserStatus(); + // anime.SetUserLastUpdated(); + + return id; +} + +int GetAnimeList() { + return 0; +} + +/* unimplemented for now */ +std::vector<int> Search(const std::string& search) { + return {}; +} + +std::vector<int> GetSeason(Anime::SeriesSeason season, Date::Year year) { + return {}; +} + +int UpdateAnimeEntry(int id) { + return 0; +} + +bool AuthorizeUser(const std::string& email, const std::string& password) { + const nlohmann::json body = { + {"grant_type", "password"}, + {"username", email}, + {"password", HTTP::UrlEncode(password)} + }; + + if (!SendAuthRequest(body)) + return false; + + static const std::map<std::string, std::string> params = { + {"filter[self]", "true"} + }; + + std::optional<std::string> response = SendRequest("/users", params); + if (!response) + return false; // whuh? + + nlohmann::json json; + try { + json = nlohmann::json::parse(response.value()); + } catch (const std::exception& ex) { + session.SetStatusBar(std::string("Kitsu: Failed to parse user data with error \"") + ex.what() + "\"!"); + return false; + } + + if (!json.contains("/data/0/id"_json_pointer)) { + session.SetStatusBar("Kitsu: Failed to retrieve user ID!"); + return false; + } + + session.SetStatusBar("Kitsu: Successfully retrieved user data!"); + session.config.auth.kitsu.user_id = json["/data/0/id"_json_pointer].get<std::string>(); + + return true; +} + +} // namespace Kitsu +} // namespace Services
--- a/src/services/services.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/services/services.cc Wed Jun 12 04:07:10 2024 -0400 @@ -1,13 +1,14 @@ #include "services/services.h" #include "core/session.h" -#include "gui/dialog/settings.h" #include "services/anilist.h" +#include "services/kitsu.h" namespace Services { void Synchronize() { switch (session.config.service) { case Anime::Service::AniList: AniList::GetAnimeList(); break; + case Anime::Service::Kitsu: Kitsu::GetAnimeList(); break; default: break; } } @@ -15,6 +16,7 @@ std::vector<int> Search(const std::string& search) { switch (session.config.service) { case Anime::Service::AniList: return AniList::Search(search); + case Anime::Service::Kitsu: return Kitsu::Search(search); default: return {}; } } @@ -22,6 +24,7 @@ std::vector<int> GetSeason(Anime::SeriesSeason season, Date::Year year) { switch (session.config.service) { case Anime::Service::AniList: return AniList::GetSeason(season, year); + case Anime::Service::Kitsu: return Kitsu::GetSeason(season, year); default: return {}; } } @@ -29,15 +32,9 @@ void UpdateAnimeEntry(int id) { switch (session.config.service) { case Anime::Service::AniList: AniList::UpdateAnimeEntry(id); break; + case Anime::Service::Kitsu: Kitsu::UpdateAnimeEntry(id); break; default: break; } } -bool Authorize() { - switch (session.config.service) { - case Anime::Service::AniList: return AniList::AuthorizeUser(); - default: return true; - } -} - }; // namespace Services
--- a/src/track/media.cc Tue Jun 11 15:11:09 2024 -0400 +++ b/src/track/media.cc Wed Jun 12 04:07:10 2024 -0400 @@ -41,8 +41,6 @@ for (const auto& result : results) { for (const auto& media : result.media) { for (const auto& info : media.information) { - std::cout << info.value << std::endl; - switch (info.type) { case animone::MediaInfoType::File: vec.push_back(std::filesystem::path(info.value).filename().u8string());