From 43f73b5d71eb5fc8bec88239f220782135070c36 Mon Sep 17 00:00:00 2001 From: "J. King" Date: Sun, 30 Apr 2017 18:14:11 -0400 Subject: [PATCH] Make better use of prepared statements when updating feeds --- lib/Database.php | 130 ++++++++++++++++++++--------------------------- 1 file changed, 56 insertions(+), 74 deletions(-) diff --git a/lib/Database.php b/lib/Database.php index 63982d19..62fc794c 100644 --- a/lib/Database.php +++ b/lib/Database.php @@ -458,15 +458,65 @@ class Database { $this->db->rollback(); throw $e; } - // finally actually perform updates - foreach($feed->newItems as $item) { - $this->articleAdd($feedID, $item); + //prepare the necessary statements to perform the update + if(sizeof($feed->newItems) || sizeof($feed->changedItems)) { + $qInsertCategory = $this->db->prepare('INSERT INTO arsse_categories(article,name) values(?,?)', 'int', 'str'); + $qInsertEdition = $this->db->prepare('INSERT INTO arse_editions(article) values(?)', 'int'); } - foreach($feed->changedItems as $id => $item) { - $this->articleAdd($feedID, $item, $id); + if(sizeof($feed->newItems)) { + $qInsertArticle = $this->db->prepare( + 'INSERT INTO arsse_articles(url,title,author,published,edited,guid,content,url_title_hash,url_content_hash,title_content_hash,feed) values(?,?,?,?,?,?,?,?,?,?,?)', + 'str', 'str', 'str', 'datetime', 'datetime', 'str', 'str', 'str', 'str', 'str', 'int' + ); + } + if(sizeof($feed->changedItems)) { + $qDeleteCategories = $this->db->prepare('DELETE FROM arsse_categories WHERE article is ?', 'int'); + $qUpdateArticle = $this->db->prepare( + 'UPDATE arsse_articles SET url = ?, title = ?, author = ?, published = ?, edited = ?, modified = CURRENT_TIMESTAMP, guid = ?, content = ?, url_title_hash = ?, url_content_hash = ?, title_content_hash = ? WHERE id is ?', + 'str', 'str', 'str', 'datetime', 'datetime', 'str', 'str', 'str', 'str', 'str', 'int' + ); + } + // actually perform updates + foreach($feed->newItems as $article) { + $articleID = $qInsertArticle->run( + $article->url, + $article->title, + $article->author, + $article->publishedDate, + $article->updatedDate, + $article->id, + $article->content, + $article->urlTitleHash, + $article->urlContentHash, + $article->titleContentHash, + $feedID + )->lastId(); + foreach($article->getTag('category') as $c) { + $qInsertCategories->run($articleID, $c); + } + $qInsertEdition->run($articleID); + } + foreach($feed->changedItems as $articleID => $article) { + $qUpdateArticle->run( + $article->url, + $article->title, + $article->author, + $article->publishedDate, + $article->updatedDate, + $article->id, + $article->content, + $article->urlTitleHash, + $article->urlContentHash, + $article->titleContentHash, + $articleID + ); + $qDeleteCategories->run($articleID); + foreach($article->getTag('category') as $c) { + $qInsertCategories->run($articleID, $c); + } + $qInsertEdition->run($articleID); } // lastly update the feed database itself with updated information. - $next = $this->feedNextFetch($feedID, $feed); $this->db->prepare( 'UPDATE arsse_feeds SET url = ?, title = ?, favicon = ?, source = ?, updated = CURRENT_TIMESTAMP, modified = ?, etag = ?, err_count = 0, err_msg = "", next_fetch = ? WHERE id is ?', 'str', 'str', 'str', 'str', 'datetime', 'str', 'datetime', 'int' @@ -508,72 +558,4 @@ class Database { 'int', $tId, $tHashUT, $tHashUC, $tHashTC )->run($feedID, $ids, $hashesUT, $hashesUC, $hashesTC); } - - public function articleAdd(int $feedID, \PicoFeed\Parser\Item $article, int $articleID = null): int { - $this->db->begin(); - try { - if(is_null($articleID)) { - $articleID = $this->db->prepare( - 'INSERT INTO arsse_articles(url,title,author,published,edited,guid,content,url_title_hash,url_content_hash,title_content_hash,feed) values(?,?,?,?,?,?,?,?,?,?,?)', - 'str', 'str', 'str', 'datetime', 'datetime', 'str', 'str', 'str', 'str', 'str', 'int' - )->run( - $article->url, - $article->title, - $article->author, - $article->publishedDate, - $article->updatedDate, - $article->id, - $article->content, - $article->urlTitleHash, - $article->urlContentHash, - $article->titleContentHash, - $feedID - )->lastId(); - } else { - $this->db->prepare( - 'UPDATE arsse_articles SET url = ?, title = ?, author = ?, published = ?, edited = ?, modified = CURRENT_TIMESTAMP, guid = ?, content = ?, url_title_hash = ?, url_content_hash = ?, title_content_hash = ? WHERE id is ?', - 'str', 'str', 'str', 'datetime', 'datetime', 'str', 'str', 'str', 'str', 'str', 'int' - )->run( - $article->url, - $article->title, - $article->author, - $article->publishedDate, - $article->updatedDate, - $article->id, - $article->content, - $article->urlTitleHash, - $article->urlContentHash, - $article->titleContentHash, - $articleID - ); - } - // If the article has categories add them into the categories database. - $this->db->prepare('DELETE FROM arsse_categories WHERE article is ?', 'int')->run($articleID); - $this->categoriesAdd($articleID, $article); - // add a new article edition ID - $this->db->prepare('INSERT INTO arse_editions(article) values(?)', 'int')->run($articleID); - } catch(\Throwable $e) { - $this->db->rollback(); - throw $e; - } - $this->db->commit(); - return $articleID; - } - - public function categoriesAdd(int $articleID, \PicoFeed\Parser\Item $article): int { - $categories = $article->getTag('category'); - $this->db->begin(); - try { - if(count($categories) > 0) { - foreach($categories as $c) { - $this->db->prepare('INSERT INTO arsse_categories(article,name) values(?,?)', 'int', 'str')->run($articleID, $c); - } - } - } catch(\Throwable $e) { - $this->db->rollback(); - throw $e; - } - $this->db->commit(); - return count($categories); - } } \ No newline at end of file