// This file is part of RSS Guard. // // Copyright (C) 2011-2015 by Martin Rotter // // RSS Guard is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // RSS Guard is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with RSS Guard. If not, see . #include "services/tt-rss/ttrssfeed.h" #include "definitions/definitions.h" #include "miscellaneous/application.h" #include "miscellaneous/databasefactory.h" #include "miscellaneous/iconfactory.h" #include "miscellaneous/textfactory.h" #include "services/tt-rss/definitions.h" #include "services/tt-rss/ttrssserviceroot.h" #include "services/tt-rss/network/ttrssnetworkfactory.h" #include TtRssFeed::TtRssFeed(RootItem *parent) : Feed(parent), m_customId(NO_PARENT_CATEGORY), m_totalCount(0), m_unreadCount(0) { } TtRssFeed::TtRssFeed(const QSqlRecord &record) : Feed(NULL), m_totalCount(0), m_unreadCount(0) { setTitle(record.value(FDS_DB_TITLE_INDEX).toString()); setId(record.value(FDS_DB_ID_INDEX).toInt()); setIcon(qApp->icons()->fromByteArray(record.value(FDS_DB_ICON_INDEX).toByteArray())); setAutoUpdateType(static_cast(record.value(FDS_DB_UPDATE_TYPE_INDEX).toInt())); setAutoUpdateInitialInterval(record.value(FDS_DB_UPDATE_INTERVAL_INDEX).toInt()); setCustomId(record.value(FDS_DB_CUSTOM_ID_INDEX).toInt()); } TtRssFeed::~TtRssFeed() { } TtRssServiceRoot *TtRssFeed::serviceRoot() { return qobject_cast(getParentServiceRoot()); } void TtRssFeed::updateCounts(bool including_total_count) { QSqlDatabase database = qApp->database()->connection(metaObject()->className(), DatabaseFactory::FromSettings); QSqlQuery query_all(database); query_all.setForwardOnly(true); if (including_total_count) { if (query_all.exec(QString("SELECT count(*) FROM Messages WHERE feed = '%1' AND is_deleted = 0 AND account_id = %2;").arg(QString::number(customId()), QString::number(serviceRoot()->accountId()))) && query_all.next()) { m_totalCount = query_all.value(0).toInt(); } } // Obtain count of unread messages. if (query_all.exec(QString("SELECT count(*) FROM Messages WHERE feed = '%1' AND is_deleted = 0 AND is_read = 0 AND account_id = %2;").arg(QString::number(customId()), QString::number(serviceRoot()->accountId()))) && query_all.next()) { int new_unread_count = query_all.value(0).toInt(); if (status() == NewMessages && new_unread_count < m_unreadCount) { setStatus(Normal); } m_unreadCount = new_unread_count; } } int TtRssFeed::countOfAllMessages() const { return m_totalCount; } int TtRssFeed::countOfUnreadMessages() const { return m_unreadCount; } int TtRssFeed::update() { QNetworkReply::NetworkError error; QList messages; int newly_added_messages = 0; int limit = MAX_MESSAGES; int skip = 0; do { TtRssGetHeadlinesResponse headlines = serviceRoot()->network()->getHeadlines(customId(), true, limit, skip, true, true, false, error); if (error != QNetworkReply::NoError) { setStatus(Feed::NetworkError); return 0; } else { QList new_messages = headlines.messages(); messages.append(new_messages); newly_added_messages = new_messages.size(); skip += newly_added_messages; } } while (newly_added_messages > 0); return updateMessages(messages); } QList TtRssFeed::undeletedMessages() const { QList messages; int account_id = const_cast(this)->serviceRoot()->accountId(); QSqlDatabase database = qApp->database()->connection(metaObject()->className(), DatabaseFactory::FromSettings); QSqlQuery query_read_msg(database); query_read_msg.setForwardOnly(true); query_read_msg.prepare("SELECT title, url, author, date_created, contents, enclosures, custom_id " "FROM Messages " "WHERE is_deleted = 0 AND feed = :feed AND account_id = :account_id;"); query_read_msg.bindValue(QSL(":feed"), customId()); query_read_msg.bindValue(QSL(":account_id"), account_id); // FIXME: Fix those const functions, this is fucking ugly. if (query_read_msg.exec()) { while (query_read_msg.next()) { Message message; message.m_feedId = QString::number(customId()); message.m_title = query_read_msg.value(0).toString(); message.m_url = query_read_msg.value(1).toString(); message.m_author = query_read_msg.value(2).toString(); message.m_created = TextFactory::parseDateTime(query_read_msg.value(3).value()); message.m_contents = query_read_msg.value(4).toString(); message.m_enclosures = Enclosures::decodeEnclosuresFromString(query_read_msg.value(5).toString()); message.m_accountId = account_id; message.m_customId = query_read_msg.value(6).toString(); messages.append(message); } } return messages; } int TtRssFeed::customId() const { return m_customId; } void TtRssFeed::setCustomId(int custom_id) { m_customId = custom_id; } int TtRssFeed::updateMessages(const QList &messages) { if (messages.isEmpty()) { return 0; } int feed_id = customId(); int updated_messages = 0; QSqlDatabase database = qApp->database()->connection(metaObject()->className(), DatabaseFactory::FromSettings); int account_id = serviceRoot()->accountId(); // Prepare queries. QSqlQuery query_insert(database); QSqlQuery query_select(database); QSqlQuery query_update(database); query_update.setForwardOnly(true); query_update.prepare("UPDATE Messages " "SET title = :title, is_read = :is_read, is_important = :is_important, url = :url, author = :author, date_created = :date_created, contents = :contents, enclosures = :enclosures " "WHERE id = :id;"); query_select.setForwardOnly(true); query_select.prepare("SELECT id, date_created, is_read, is_important FROM Messages " "WHERE account_id = :account_id AND custom_id = :custom_id;"); // Used to insert new messages. query_insert.setForwardOnly(true); query_insert.prepare("INSERT INTO Messages " "(feed, title, is_read, is_important, url, author, date_created, contents, enclosures, custom_id, account_id) " "VALUES (:feed, :title, :is_read, :is_important, :url, :author, :date_created, :contents, :enclosures, :custom_id, :account_id);"); if (!database.transaction()) { database.rollback(); qDebug("Transaction start for message downloader failed."); return updated_messages; } foreach (Message message, messages) { query_select.bindValue(QSL(":account_id"), account_id); query_select.bindValue(QSL(":custom_id"), message.m_customId); query_select.exec(); int id_existing_message = -1; qint64 date_existing_message; bool is_read_existing_message; bool is_important_existing_message; if (query_select.next()) { id_existing_message = query_select.value(0).toInt(); date_existing_message = query_select.value(1).value(); is_read_existing_message = query_select.value(2).toBool(); is_important_existing_message = query_select.value(3).toBool(); } query_select.finish(); // Now, check if this message is already in the DB. if (id_existing_message >= 0) { // Message is already in the DB. if (message.m_created.toMSecsSinceEpoch() != date_existing_message || message.m_isRead != is_read_existing_message || message.m_isImportant != is_important_existing_message) { // Message exists, it is changed, update it. query_update.bindValue(QSL(":title"), message.m_title); query_update.bindValue(QSL(":is_read"), (int) message.m_isRead); query_update.bindValue(QSL(":is_important"), (int) message.m_isImportant); query_update.bindValue(QSL(":url"), message.m_url); query_update.bindValue(QSL(":author"), message.m_author); query_update.bindValue(QSL(":date_created"), message.m_created.toMSecsSinceEpoch()); query_update.bindValue(QSL(":contents"), message.m_contents); query_update.bindValue(QSL(":enclosures"), Enclosures::encodeEnclosuresToString(message.m_enclosures)); query_update.bindValue(QSL(":id"), id_existing_message); if (query_update.exec()) { updated_messages++; } query_update.finish(); qDebug("Updating message '%s' in DB.", qPrintable(message.m_title)); } } else { // Message with this URL is not fetched in this feed yet. query_insert.bindValue(QSL(":feed"), feed_id); query_insert.bindValue(QSL(":title"), message.m_title); query_insert.bindValue(QSL(":is_read"), (int) message.m_isRead); query_insert.bindValue(QSL(":is_important"), (int) message.m_isImportant); query_insert.bindValue(QSL(":url"), message.m_url); query_insert.bindValue(QSL(":author"), message.m_author); query_insert.bindValue(QSL(":date_created"), message.m_created.toMSecsSinceEpoch()); query_insert.bindValue(QSL(":contents"), message.m_contents); query_insert.bindValue(QSL(":enclosures"), Enclosures::encodeEnclosuresToString(message.m_enclosures)); query_insert.bindValue(QSL(":custom_id"), message.m_customId); query_insert.bindValue(QSL(":account_id"), account_id); if (query_insert.exec() && query_insert.numRowsAffected() == 1) { updated_messages++; } query_insert.finish(); qDebug("Adding new message '%s' to DB.", qPrintable(message.m_title)); } } if (!database.commit()) { database.rollback(); qDebug("Transaction commit for message downloader failed."); } else { setStatus(NewMessages); updateCounts(true); serviceRoot()->itemChanged(QList() << this); } return updated_messages; }