Work on TT-RSS network.

This commit is contained in:
Martin Rotter 2015-12-07 09:42:46 +01:00
parent 8fbb434a56
commit 32a8fc81cb
19 changed files with 217 additions and 58 deletions

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.9 KiB

View File

@ -22,7 +22,7 @@ CREATE TABLE IF NOT EXISTS Accounts (
INSERT INTO Accounts (type) VALUES ('std-rss'); INSERT INTO Accounts (type) VALUES ('std-rss');
-- ! -- !
CREATE TABLE IF NOT EXISTS TtRssAccounts ( CREATE TABLE IF NOT EXISTS TtRssAccounts (
id INTEGER PRIMARY KEY, id INTEGER,
username TEXT NOT NULL, username TEXT NOT NULL,
password TEXT, password TEXT,
url TEXT NOT NULL, url TEXT NOT NULL,

View File

@ -16,7 +16,7 @@ CREATE TABLE IF NOT EXISTS Accounts (
INSERT INTO Accounts (type) VALUES ('std-rss'); INSERT INTO Accounts (type) VALUES ('std-rss');
-- ! -- !
CREATE TABLE IF NOT EXISTS TtRssAccounts ( CREATE TABLE IF NOT EXISTS TtRssAccounts (
id INTEGER PRIMARY KEY, id INTEGER,
username TEXT NOT NULL, username TEXT NOT NULL,
password TEXT, password TEXT,
url TEXT NOT NULL, url TEXT NOT NULL,

View File

@ -8,7 +8,7 @@ INSERT INTO Accounts (type) VALUES ('std-rss');
DROP TABLE IF EXISTS FeedsData; DROP TABLE IF EXISTS FeedsData;
-- ! -- !
CREATE TABLE IF NOT EXISTS TtRssAccounts ( CREATE TABLE IF NOT EXISTS TtRssAccounts (
id INTEGER PRIMARY KEY, id INTEGER,
username TEXT NOT NULL, username TEXT NOT NULL,
password TEXT, password TEXT,
url TEXT NOT NULL, url TEXT NOT NULL,

View File

@ -8,7 +8,7 @@ INSERT INTO Accounts (type) VALUES ('std-rss');
DROP TABLE IF EXISTS FeedsData; DROP TABLE IF EXISTS FeedsData;
-- ! -- !
CREATE TABLE IF NOT EXISTS TtRssAccounts ( CREATE TABLE IF NOT EXISTS TtRssAccounts (
id INTEGER PRIMARY KEY, id INTEGER,
username TEXT NOT NULL, username TEXT NOT NULL,
password TEXT, password TEXT,
url TEXT NOT NULL, url TEXT NOT NULL,

View File

@ -42,7 +42,7 @@ RootItem::~RootItem() {
qDeleteAll(m_childItems); qDeleteAll(m_childItems);
} }
QList<QAction*> RootItem::contextMenuActions() { QList<QAction*> RootItem::contextMenu() {
return QList<QAction*>(); return QList<QAction*>();
} }

View File

@ -76,7 +76,7 @@ class RootItem : public QObject {
// Returns list of specific actions which can be done with the item. // Returns list of specific actions which can be done with the item.
// Do not include general actions here like actions: Mark as read, Update, ... // Do not include general actions here like actions: Mark as read, Update, ...
// NOTE: Ownership of returned actions is not switched to caller, free them when needed. // NOTE: Ownership of returned actions is not switched to caller, free them when needed.
virtual QList<QAction*> contextMenuActions(); virtual QList<QAction*> contextMenu();
// Can properties of this item be edited? // Can properties of this item be edited?
virtual bool canBeEdited(); virtual bool canBeEdited();

View File

@ -303,7 +303,7 @@ QMenu *FeedsView::initializeContextMenuCategories(RootItem *clicked_item) {
m_contextMenuCategories->clear(); m_contextMenuCategories->clear();
} }
QList<QAction*> specific_actions = clicked_item->contextMenuActions(); QList<QAction*> specific_actions = clicked_item->contextMenu();
m_contextMenuCategories->addActions(QList<QAction*>() << m_contextMenuCategories->addActions(QList<QAction*>() <<
qApp->mainForm()->m_ui->m_actionUpdateSelectedItems << qApp->mainForm()->m_ui->m_actionUpdateSelectedItems <<
@ -329,7 +329,7 @@ QMenu *FeedsView::initializeContextMenuFeeds(RootItem *clicked_item) {
m_contextMenuFeeds->clear(); m_contextMenuFeeds->clear();
} }
QList<QAction*> specific_actions = clicked_item->contextMenuActions(); QList<QAction*> specific_actions = clicked_item->contextMenu();
m_contextMenuFeeds->addActions(QList<QAction*>() << m_contextMenuFeeds->addActions(QList<QAction*>() <<
qApp->mainForm()->m_ui->m_actionUpdateSelectedItems << qApp->mainForm()->m_ui->m_actionUpdateSelectedItems <<
@ -365,7 +365,7 @@ QMenu *FeedsView::initializeContextMenuOtherItem(RootItem *clicked_item) {
m_contextMenuOtherItems->clear(); m_contextMenuOtherItems->clear();
} }
QList<QAction*> specific_actions = clicked_item->contextMenuActions(); QList<QAction*> specific_actions = clicked_item->contextMenu();
if (!specific_actions.isEmpty()) { if (!specific_actions.isEmpty()) {
m_contextMenuOtherItems->addSeparator(); m_contextMenuOtherItems->addSeparator();

View File

@ -18,6 +18,9 @@
#include "services/abstract/serviceroot.h" #include "services/abstract/serviceroot.h"
#include "core/feedsmodel.h" #include "core/feedsmodel.h"
#include "miscellaneous/application.h"
#include <QSqlQuery>
ServiceRoot::ServiceRoot(RootItem *parent) : RootItem(parent), m_accountId(NO_PARENT_CATEGORY) { ServiceRoot::ServiceRoot(RootItem *parent) : RootItem(parent), m_accountId(NO_PARENT_CATEGORY) {
@ -27,6 +30,34 @@ ServiceRoot::ServiceRoot(RootItem *parent) : RootItem(parent), m_accountId(NO_PA
ServiceRoot::~ServiceRoot() { ServiceRoot::~ServiceRoot() {
} }
bool ServiceRoot::deleteViaGui() {
QSqlDatabase connection = qApp->database()->connection(metaObject()->className(), DatabaseFactory::FromSettings);
// Remove all messages.
if (!QSqlQuery(connection).exec(QString("DELETE FROM Messages WHERE account_id = %1;").arg(accountId()))) {
return false;
}
// Remove all feeds.
if (!QSqlQuery(connection).exec(QString("DELETE FROM Feeds WHERE account_id = %1;").arg(accountId()))) {
return false;
}
// Remove all categories.
if (!QSqlQuery(connection).exec(QString("DELETE FROM Categories WHERE account_id = %1;").arg(accountId()))) {
return false;
}
// Switch "existence" flag.
bool data_removed = QSqlQuery(connection).exec(QString("DELETE FROM Accounts WHERE id = %1;").arg(accountId()));
if (data_removed) {
requestItemRemoval(this);
}
return data_removed;
}
void ServiceRoot::itemChanged(QList<RootItem*> items) { void ServiceRoot::itemChanged(QList<RootItem*> items) {
emit dataChanged(items); emit dataChanged(items);
} }

View File

@ -44,6 +44,8 @@ class ServiceRoot : public RootItem {
// /* Members to override. // /* Members to override.
///////////////////////////////////////// /////////////////////////////////////////
bool deleteViaGui();
// Returns list of specific actions for "Add new item" main window menu. // Returns list of specific actions for "Add new item" main window menu.
// So typical list of returned actions could look like: // So typical list of returned actions could look like:
// a) Add new feed // a) Add new feed

View File

@ -98,7 +98,7 @@ int StandardFeed::countOfUnreadMessages() const {
return m_unreadCount; return m_unreadCount;
} }
QList<QAction*> StandardFeed::contextMenuActions() { QList<QAction*> StandardFeed::contextMenu() {
return serviceRoot()->getContextMenuForFeed(this); return serviceRoot()->getContextMenuForFeed(this);
} }

View File

@ -61,7 +61,7 @@ class StandardFeed : public Feed {
int countOfAllMessages() const; int countOfAllMessages() const;
int countOfUnreadMessages() const; int countOfUnreadMessages() const;
QList<QAction*> contextMenuActions(); QList<QAction*> contextMenu();
bool canBeEdited() { bool canBeEdited() {
return true; return true;

View File

@ -109,31 +109,7 @@ bool StandardServiceRoot::canBeDeleted() {
} }
bool StandardServiceRoot::deleteViaGui() { bool StandardServiceRoot::deleteViaGui() {
QSqlDatabase connection = qApp->database()->connection(metaObject()->className(), DatabaseFactory::FromSettings); return ServiceRoot::deleteViaGui();
// Remove all messages.
if (!QSqlQuery(connection).exec(QString("DELETE FROM Messages WHERE account_id = %1;").arg(accountId()))) {
return false;
}
// Remove all feeds.
if (!QSqlQuery(connection).exec(QString("DELETE FROM Feeds WHERE account_id = %1;").arg(accountId()))) {
return false;
}
// Remove all categories.
if (!QSqlQuery(connection).exec(QString("DELETE FROM Categories WHERE account_id = %1;").arg(accountId()))) {
return false;
}
// Switch "existence" flag.
bool data_removed = QSqlQuery(connection).exec(QString("DELETE FROM Accounts WHERE id = %1;").arg(accountId()));
if (data_removed) {
requestItemRemoval(this);
}
return data_removed;
} }
QVariant StandardServiceRoot::data(int column, int role) const { QVariant StandardServiceRoot::data(int column, int role) const {
@ -622,7 +598,7 @@ QList<QAction*> StandardServiceRoot::serviceMenu() {
return m_serviceMenu; return m_serviceMenu;
} }
QList<QAction*> StandardServiceRoot::contextMenuActions() { QList<QAction*> StandardServiceRoot::contextMenu() {
return serviceMenu(); return serviceMenu();
} }

View File

@ -64,7 +64,7 @@ class StandardServiceRoot : public ServiceRoot {
// Return menu to be shown in "Services -> service" menu. // Return menu to be shown in "Services -> service" menu.
QList<QAction*> serviceMenu(); QList<QAction*> serviceMenu();
QList<QAction*> contextMenuActions(); QList<QAction*> contextMenu();
// Message stuff. // Message stuff.
bool loadMessagesForItem(RootItem *item, QSqlTableModel *model); bool loadMessagesForItem(RootItem *item, QSqlTableModel *model);

View File

@ -23,4 +23,7 @@
// Logout. // Logout.
#define LOGOUT_OK "OK" #define LOGOUT_OK "OK"
// Get feed tree.
#define GFT_TYPE_CATEGORY "category"
#endif // DEFINITIONS_H #endif // DEFINITIONS_H

View File

@ -18,12 +18,13 @@
#include "services/tt-rss/network/ttrssnetworkfactory.h" #include "services/tt-rss/network/ttrssnetworkfactory.h"
#include "definitions/definitions.h" #include "definitions/definitions.h"
#include "core/rootitem.h"
#include "services/tt-rss/definitions.h" #include "services/tt-rss/definitions.h"
#include "network-web/networkfactory.h" #include "network-web/networkfactory.h"
TtRssNetworkFactory::TtRssNetworkFactory() TtRssNetworkFactory::TtRssNetworkFactory()
: m_url(QString()), m_username(QString()), m_password(QString()), m_session_Id(QString()) { : m_url(QString()), m_username(QString()), m_password(QString()), m_sessionId(QString()) {
} }
TtRssNetworkFactory::~TtRssNetworkFactory() { TtRssNetworkFactory::~TtRssNetworkFactory() {
@ -63,22 +64,56 @@ void TtRssNetworkFactory::setPassword(const QString &password) {
LoginResult TtRssNetworkFactory::login() { LoginResult TtRssNetworkFactory::login() {
if (!m_sessionId.isEmpty()) {
logout();
}
QtJson::JsonObject json; QtJson::JsonObject json;
json["op"] = "login"; json["op"] = "login";
json["user"] = m_username; json["user"] = m_username;
json["password"] = m_password; json["password"] = m_password;
QByteArray result_raw; QByteArray result_raw;
NetworkResult res = NetworkFactory::uploadData(m_url, DOWNLOAD_TIMEOUT, QtJson::serialize(json), CONTENT_TYPE, result_raw); NetworkResult network_reply = NetworkFactory::uploadData(m_url, DOWNLOAD_TIMEOUT, QtJson::serialize(json), CONTENT_TYPE, result_raw);
LoginResult result(network_reply.first, TtRssLoginResponse(QString::fromUtf8(result_raw)));
if (res.first != QNetworkReply::NoError) { if (network_reply.first == QNetworkReply::NoError) {
return LoginResult(res.first, TtRssLoginResponse()); m_sessionId = result.second.sessionId();
} }
else {
LoginResult result(res.first, TtRssLoginResponse(QString::fromUtf8(result_raw)));
m_session_Id = result.second.sessionId();
return result; return result;
} }
LogoutResult TtRssNetworkFactory::logout() {
QtJson::JsonObject json;
json["op"] = "logout";
json["sid"] = m_sessionId;
QByteArray result_raw;
NetworkResult network_reply = NetworkFactory::uploadData(m_url, DOWNLOAD_TIMEOUT, QtJson::serialize(json), CONTENT_TYPE, result_raw);
return LogoutResult(network_reply.first, TtRssResponse(QString::fromUtf8(result_raw)));
}
GetFeedTreeResult TtRssNetworkFactory::getFeedTree() {
QtJson::JsonObject json;
json["op"] = "getFeedTree";
json["sid"] = m_sessionId;
json["include_empty"] = true;
QByteArray result_raw;
NetworkResult network_reply = NetworkFactory::uploadData(m_url, DOWNLOAD_TIMEOUT, QtJson::serialize(json), CONTENT_TYPE, result_raw);
GetFeedTreeResult result(network_reply.first, TtRssGetFeedTreeResponse(QString::fromUtf8(result_raw)));
if (result.second.isNotLoggedIn()) {
// We are not logged in.
login();
network_reply = NetworkFactory::uploadData(m_url, DOWNLOAD_TIMEOUT, QtJson::serialize(json), CONTENT_TYPE, result_raw);
result = GetFeedTreeResult(network_reply.first, TtRssGetFeedTreeResponse(QString::fromUtf8(result_raw)));
}
return result;
} }
TtRssResponse::TtRssResponse(const QString &raw_content) { TtRssResponse::TtRssResponse(const QString &raw_content) {
@ -110,6 +145,10 @@ int TtRssResponse::status() const {
} }
} }
bool TtRssResponse::isNotLoggedIn() const {
return status() == API_STATUS_ERR && hasError() && error() == NOT_LOGGED_IN;
}
TtRssLoginResponse::TtRssLoginResponse(const QString &raw_content) : TtRssResponse(raw_content) { TtRssLoginResponse::TtRssLoginResponse(const QString &raw_content) : TtRssResponse(raw_content) {
} }
@ -135,7 +174,7 @@ QString TtRssLoginResponse::sessionId() const {
} }
} }
QString TtRssLoginResponse::error() const { QString TtRssResponse::error() const {
if (!isLoaded()) { if (!isLoaded()) {
return QString(); return QString();
} }
@ -144,7 +183,7 @@ QString TtRssLoginResponse::error() const {
} }
} }
bool TtRssLoginResponse::hasError() const { bool TtRssResponse::hasError() const {
if (!isLoaded()) { if (!isLoaded()) {
return false; return false;
} }
@ -152,3 +191,58 @@ bool TtRssLoginResponse::hasError() const {
return m_rawContent["content"].toMap().contains("error"); return m_rawContent["content"].toMap().contains("error");
} }
} }
TtRssGetFeedTreeResponse::TtRssGetFeedTreeResponse(const QString &raw_content) : TtRssResponse(raw_content) {
}
TtRssGetFeedTreeResponse::~TtRssGetFeedTreeResponse() {
}
QList<RootItem*> TtRssGetFeedTreeResponse::getTree() {
QList<RootItem*> items;
if (status() == API_STATUS_OK) {
// We have data, construct object tree according to data.
QList<QVariant> items_to_process = m_rawContent["content"].toMap()["categories"].toMap()["items"].toList();
processSubtree(true, items, NULL, items_to_process);
}
return items;
}
void TtRssGetFeedTreeResponse::processSubtree(bool is_top_level, QList<RootItem *> &top_level_items,
RootItem *parent, const QList<QVariant> &items) {
foreach (QVariant item, items) {
QMap<QString,QVariant> map_item = item.toMap();
if (map_item.contains("type") && map_item["type"].toString() == GFT_TYPE_CATEGORY) {
// TODO: pokračovat tady
// We have category, create it, add it to "parent".
// Then process all its children.
//
// TtRssCategory *new_category = new TtRssCategory();
// naplnit informace.....
// parent->appendChild(new_category);
// if (is_top_level) {
// top_level_items.append(new_category);
// }
// else {
// parent->appendChild(new_category);
// }
// processSubtree(false, top_level_items, new_category, map_item["items"].toList());
}
else {
// We have feed, add it.
// TtRssFeed *new_feed = new TtRssFeed();
// naplnit informace.....
// parent->appendChild(new_feed);
// if (is_top_level) {
// top_level_items.append(new_feed);
// }
}
}
}

View File

@ -25,6 +25,8 @@
#include <QNetworkReply> #include <QNetworkReply>
class RootItem;
class TtRssResponse { class TtRssResponse {
public: public:
explicit TtRssResponse(const QString &raw_content = QString()); explicit TtRssResponse(const QString &raw_content = QString());
@ -34,6 +36,9 @@ class TtRssResponse {
int seq() const; int seq() const;
int status() const; int status() const;
QString error() const;
bool hasError() const;
bool isNotLoggedIn() const;
protected: protected:
QtJson::JsonObject m_rawContent; QtJson::JsonObject m_rawContent;
@ -46,11 +51,24 @@ class TtRssLoginResponse : public TtRssResponse {
int apiLevel() const; int apiLevel() const;
QString sessionId() const; QString sessionId() const;
QString error() const;
bool hasError() const;
}; };
typedef QPair<QNetworkReply::NetworkError,TtRssLoginResponse> LoginResult; typedef QPair<QNetworkReply::NetworkError,TtRssLoginResponse> LoginResult;
typedef QPair<QNetworkReply::NetworkError,TtRssResponse> LogoutResult;
class TtRssGetFeedTreeResponse : public TtRssResponse {
public:
explicit TtRssGetFeedTreeResponse(const QString &raw_content = QString());
virtual ~TtRssGetFeedTreeResponse();
QList<RootItem*> getTree();
private:
void processSubtree(bool is_top_level, QList<RootItem*> &top_level_items,
RootItem *parent, const QList<QVariant> &items);
};
typedef QPair<QNetworkReply::NetworkError,TtRssGetFeedTreeResponse> GetFeedTreeResult;
class TtRssNetworkFactory { class TtRssNetworkFactory {
public: public:
@ -67,13 +85,21 @@ class TtRssNetworkFactory {
void setPassword(const QString &password); void setPassword(const QString &password);
// Operations. // Operations.
// Logs user in.
LoginResult login(); LoginResult login();
// Logs user out.
LogoutResult logout();
// Gets tree from feeds/categories obtained from the server.
GetFeedTreeResult getFeedTree();
private: private:
QString m_url; QString m_url;
QString m_username; QString m_username;
QString m_password; QString m_password;
QString m_session_Id; QString m_sessionId;
}; };
#endif // TTRSSNETWORKFACTORY_H #endif // TTRSSNETWORKFACTORY_H

View File

@ -30,7 +30,7 @@
TtRssServiceRoot::TtRssServiceRoot(RootItem *parent) TtRssServiceRoot::TtRssServiceRoot(RootItem *parent)
: ServiceRoot(parent), m_network(new TtRssNetworkFactory) { : ServiceRoot(parent), m_network(new TtRssNetworkFactory), m_actionSyncIn(NULL), m_serviceMenu(QList<QAction*>()) {
setIcon(TtRssServiceEntryPoint().icon()); setIcon(TtRssServiceEntryPoint().icon());
setCreationDate(QDateTime::currentDateTime()); setCreationDate(QDateTime::currentDateTime());
} }
@ -63,8 +63,17 @@ bool TtRssServiceRoot::editViaGui() {
} }
bool TtRssServiceRoot::deleteViaGui() { bool TtRssServiceRoot::deleteViaGui() {
QSqlDatabase connection = qApp->database()->connection(metaObject()->className(), DatabaseFactory::FromSettings);
// Remove extra entry in "Tiny Tiny RSS accounts list" and then delete
// all the categories/feeds and messages.
if (!QSqlQuery(connection).exec(QString("DELETE FROM TtRssAccounts WHERE id = %1;").arg(accountId()))) {
return false; return false;
} }
else {
return ServiceRoot::deleteViaGui();
}
}
bool TtRssServiceRoot::canBeEdited() { bool TtRssServiceRoot::canBeEdited() {
return true; return true;
@ -105,7 +114,19 @@ bool TtRssServiceRoot::loadMessagesForItem(RootItem *item, QSqlTableModel *model
} }
QList<QAction*> TtRssServiceRoot::serviceMenu() { QList<QAction*> TtRssServiceRoot::serviceMenu() {
return QList<QAction*>(); if (m_serviceMenu.isEmpty()) {
m_actionSyncIn = new QAction(qApp->icons()->fromTheme(QSL("item-sync")), tr("Sync in"), this);
connect(m_actionSyncIn, SIGNAL(triggered()), this, SLOT(syncIn()));
m_serviceMenu.append(m_actionSyncIn);
}
return m_serviceMenu;
}
QList<QAction*> TtRssServiceRoot::contextMenu() {
return serviceMenu();
} }
bool TtRssServiceRoot::onBeforeSetMessagesRead(RootItem *selected_item, QList<int> message_db_ids, RootItem::ReadStatus read) { bool TtRssServiceRoot::onBeforeSetMessagesRead(RootItem *selected_item, QList<int> message_db_ids, RootItem::ReadStatus read) {
@ -207,4 +228,5 @@ void TtRssServiceRoot::syncIn() {
// TODO: provede stažení kanálů/kategorií // TODO: provede stažení kanálů/kategorií
// ze serveru, a sloučení s aktuálními // ze serveru, a sloučení s aktuálními
// neprovádí aktualizace kanálů ani stažení počtu nepřečtených zpráv // neprovádí aktualizace kanálů ani stažení počtu nepřečtených zpráv
m_network->getFeedTree();
} }

View File

@ -46,6 +46,7 @@ class TtRssServiceRoot : public ServiceRoot {
QList<QAction*> addItemMenu(); QList<QAction*> addItemMenu();
QList<QAction*> serviceMenu(); QList<QAction*> serviceMenu();
QList<QAction*> contextMenu();
RecycleBin *recycleBin(); RecycleBin *recycleBin();
@ -69,9 +70,13 @@ class TtRssServiceRoot : public ServiceRoot {
void loadFromDatabase(); void loadFromDatabase();
void updateTitle(); void updateTitle();
private: private slots:
void syncIn(); void syncIn();
private:
QAction *m_actionSyncIn;
QList<QAction*> m_serviceMenu;
TtRssNetworkFactory *m_network; TtRssNetworkFactory *m_network;
}; };