2013-12-22 20:55:53 -05:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Model\Item;
|
|
|
|
|
|
|
|
require_once 'vendor/Readability/Readability.php';
|
|
|
|
require_once 'vendor/PicoFeed/Grabber.php';
|
|
|
|
require_once 'vendor/PicoFeed/Filter.php';
|
|
|
|
|
|
|
|
// Get all items
|
|
|
|
function get_all($status, $offset = null, $limit = null, $order_column = 'updated', $order_direction = 'desc')
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->columns(
|
|
|
|
'items.id',
|
|
|
|
'items.title',
|
|
|
|
'items.updated',
|
|
|
|
'items.url',
|
|
|
|
'items.bookmark',
|
|
|
|
'items.feed_id',
|
|
|
|
'items.status',
|
|
|
|
'items.content',
|
|
|
|
'feeds.site_url',
|
|
|
|
'feeds.title AS feed_title'
|
|
|
|
)
|
|
|
|
->join('feeds', 'id', 'feed_id')
|
|
|
|
->eq('status', $status)
|
|
|
|
->orderBy($order_column, $order_direction)
|
|
|
|
->offset($offset)
|
|
|
|
->limit($limit)
|
|
|
|
->findAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the number of items per status
|
|
|
|
function count_by_status($status)
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('status', $status)
|
|
|
|
->count();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the number of bookmarks
|
|
|
|
function count_bookmarks()
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('bookmark', 1)
|
|
|
|
->in('status', array('read', 'unread'))
|
|
|
|
->count();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get all bookmarks
|
|
|
|
function get_bookmarks($offset = null, $limit = null)
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->columns(
|
|
|
|
'items.id',
|
|
|
|
'items.title',
|
|
|
|
'items.updated',
|
|
|
|
'items.url',
|
|
|
|
'items.bookmark',
|
|
|
|
'items.status',
|
|
|
|
'items.content',
|
|
|
|
'items.feed_id',
|
|
|
|
'feeds.site_url',
|
|
|
|
'feeds.title AS feed_title'
|
|
|
|
)
|
|
|
|
->join('feeds', 'id', 'feed_id')
|
|
|
|
->in('status', array('read', 'unread'))
|
|
|
|
->eq('bookmark', 1)
|
|
|
|
->orderBy('updated', \Model\Config\get('items_sorting_direction'))
|
|
|
|
->offset($offset)
|
|
|
|
->limit($limit)
|
|
|
|
->findAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the number of items per feed
|
|
|
|
function count_by_feed($feed_id)
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('feed_id', $feed_id)
|
|
|
|
->in('status', array('unread', 'read'))
|
|
|
|
->count();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get all items per feed
|
|
|
|
function get_all_by_feed($feed_id, $offset = null, $limit = null, $order_column = 'updated', $order_direction = 'desc')
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->columns(
|
|
|
|
'items.id',
|
|
|
|
'items.title',
|
|
|
|
'items.updated',
|
|
|
|
'items.url',
|
|
|
|
'items.feed_id',
|
|
|
|
'items.status',
|
|
|
|
'items.content',
|
|
|
|
'items.bookmark',
|
|
|
|
'feeds.site_url'
|
|
|
|
)
|
|
|
|
->join('feeds', 'id', 'feed_id')
|
|
|
|
->in('status', array('unread', 'read'))
|
|
|
|
->eq('feed_id', $feed_id)
|
|
|
|
->orderBy($order_column, $order_direction)
|
|
|
|
->offset($offset)
|
|
|
|
->limit($limit)
|
|
|
|
->findAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get one item by id
|
|
|
|
function get($id)
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('id', $id)
|
|
|
|
->findOne();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get item naviguation (next/prev items)
|
|
|
|
function get_nav($item, $status = array('unread'), $bookmark = array(1, 0), $feed_id = null)
|
|
|
|
{
|
|
|
|
$query = \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->columns('id', 'status', 'title', 'bookmark')
|
|
|
|
->neq('status', 'removed')
|
|
|
|
->orderBy('updated', \Model\Config\get('items_sorting_direction'));
|
|
|
|
|
|
|
|
if ($feed_id) $query->eq('feed_id', $feed_id);
|
|
|
|
|
|
|
|
$items = $query->findAll();
|
|
|
|
|
|
|
|
$next_item = null;
|
|
|
|
$previous_item = null;
|
|
|
|
|
|
|
|
for ($i = 0, $ilen = count($items); $i < $ilen; $i++) {
|
|
|
|
|
|
|
|
if ($items[$i]['id'] == $item['id']) {
|
|
|
|
|
|
|
|
if ($i > 0) {
|
|
|
|
|
|
|
|
$j = $i - 1;
|
|
|
|
|
|
|
|
while ($j >= 0) {
|
|
|
|
|
|
|
|
if (in_array($items[$j]['status'], $status) && in_array($items[$j]['bookmark'], $bookmark)) {
|
|
|
|
$previous_item = $items[$j];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
$j--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($i < ($ilen - 1)) {
|
|
|
|
|
|
|
|
$j = $i + 1;
|
|
|
|
|
|
|
|
while ($j < $ilen) {
|
|
|
|
|
|
|
|
if (in_array($items[$j]['status'], $status) && in_array($items[$j]['bookmark'], $bookmark)) {
|
|
|
|
$next_item = $items[$j];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
$j++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array(
|
|
|
|
'next' => $next_item,
|
|
|
|
'previous' => $previous_item
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change item status to removed and clear content
|
|
|
|
function set_removed($id)
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('id', $id)
|
|
|
|
->save(array('status' => 'removed', 'content' => ''));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change item status to read
|
|
|
|
function set_read($id)
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('id', $id)
|
|
|
|
->save(array('status' => 'read'));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change item status to unread
|
|
|
|
function set_unread($id)
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('id', $id)
|
|
|
|
->save(array('status' => 'unread'));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change item status to "read", "unread" or "removed"
|
|
|
|
function set_status($status, array $items)
|
|
|
|
{
|
|
|
|
if (! in_array($status, array('read', 'unread', 'removed'))) return false;
|
|
|
|
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->in('id', $items)
|
|
|
|
->save(array('status' => $status));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enable/disable bookmark flag
|
|
|
|
function set_bookmark_value($id, $value)
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('id', $id)
|
|
|
|
->save(array('bookmark' => $value));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Swap item status read <-> unread
|
|
|
|
function switch_status($id)
|
|
|
|
{
|
|
|
|
$item = \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->columns('status')
|
|
|
|
->eq('id', $id)
|
|
|
|
->findOne();
|
|
|
|
|
|
|
|
if ($item['status'] == 'unread') {
|
|
|
|
|
|
|
|
\PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('id', $id)
|
|
|
|
->save(array('status' => 'read'));
|
|
|
|
|
|
|
|
return 'read';
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
|
|
|
|
\PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('id', $id)
|
|
|
|
->save(array('status' => 'unread'));
|
|
|
|
|
|
|
|
return 'unread';
|
|
|
|
}
|
|
|
|
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark all unread items as read
|
|
|
|
function mark_all_as_read()
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('status', 'unread')
|
|
|
|
->save(array('status' => 'read'));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark all read items to removed
|
|
|
|
function mark_all_as_removed()
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('status', 'read')
|
|
|
|
->eq('bookmark', 0)
|
|
|
|
->save(array('status' => 'removed', 'content' => ''));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark only specified items as read
|
|
|
|
function mark_items_as_read(array $items_id)
|
|
|
|
{
|
|
|
|
\PicoTools\singleton('db')->startTransaction();
|
|
|
|
|
|
|
|
foreach ($items_id as $id) {
|
|
|
|
set_read($id);
|
|
|
|
}
|
|
|
|
|
|
|
|
\PicoTools\singleton('db')->closeTransaction();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark all items of a feed as read
|
|
|
|
function mark_feed_as_read($feed_id)
|
|
|
|
{
|
|
|
|
return \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->columns('items.id')
|
|
|
|
->eq('status', 'unread')
|
|
|
|
->eq('feed_id', $feed_id)
|
|
|
|
->update(array('status' => 'read'));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark all read items to removed after X days
|
|
|
|
function autoflush()
|
|
|
|
{
|
2013-12-22 21:25:54 -05:00
|
|
|
$autoflush = (int) \Model\Config\get('autoflush');
|
2013-12-22 20:55:53 -05:00
|
|
|
|
2013-12-22 21:25:54 -05:00
|
|
|
if ($autoflush > 0) {
|
2013-12-22 20:55:53 -05:00
|
|
|
|
2013-12-22 21:25:54 -05:00
|
|
|
// Mark read items removed after X days
|
2013-12-22 20:55:53 -05:00
|
|
|
\PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('bookmark', 0)
|
|
|
|
->eq('status', 'read')
|
|
|
|
->lt('updated', strtotime('-'.$autoflush.'day'))
|
2013-12-22 21:25:54 -05:00
|
|
|
->save(array('status' => 'removed', 'content' => ''));
|
|
|
|
}
|
|
|
|
else if ($autoflush === -1) {
|
|
|
|
|
|
|
|
// Mark read items removed immediately
|
|
|
|
\PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('bookmark', 0)
|
|
|
|
->eq('status', 'read')
|
2013-12-22 20:55:53 -05:00
|
|
|
->save(array('status' => 'removed', 'content' => ''));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update all items
|
|
|
|
function update_all($feed_id, array $items, $grabber = false)
|
|
|
|
{
|
|
|
|
$nocontent = (bool) \Model\Config\get('nocontent');
|
|
|
|
|
|
|
|
$items_in_feed = array();
|
|
|
|
$db = \PicoTools\singleton('db');
|
|
|
|
|
|
|
|
$db->startTransaction();
|
|
|
|
|
|
|
|
foreach ($items as $item) {
|
|
|
|
|
|
|
|
// Item parsed correctly?
|
|
|
|
if ($item->id) {
|
|
|
|
|
|
|
|
// Insert only new item
|
|
|
|
if ($db->table('items')->eq('id', $item->id)->count() !== 1) {
|
|
|
|
|
|
|
|
if (! $item->content && ! $nocontent && $grabber) {
|
2013-12-22 22:30:49 -05:00
|
|
|
$item->content = download_content_url($item->url);
|
2013-12-22 20:55:53 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
$db->table('items')->save(array(
|
|
|
|
'id' => $item->id,
|
|
|
|
'title' => $item->title,
|
|
|
|
'url' => $item->url,
|
|
|
|
'updated' => $item->updated,
|
|
|
|
'author' => $item->author,
|
|
|
|
'content' => $nocontent ? '' : $item->content,
|
|
|
|
'status' => 'unread',
|
|
|
|
'feed_id' => $feed_id
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Items inside this feed
|
|
|
|
$items_in_feed[] = $item->id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove from the database items marked as "removed"
|
|
|
|
// and not present inside the feed
|
|
|
|
if (! empty($items_in_feed)) {
|
|
|
|
|
|
|
|
$removed_items = \PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->columns('id')
|
|
|
|
->notin('id', $items_in_feed)
|
|
|
|
->eq('status', 'removed')
|
|
|
|
->eq('feed_id', $feed_id)
|
|
|
|
->desc('updated')
|
|
|
|
->findAllByColumn('id');
|
|
|
|
|
|
|
|
// Keep a buffer of 2 items
|
|
|
|
// It's workaround for buggy feeds (cache issue with some Wordpress plugins)
|
|
|
|
if (is_array($removed_items)) {
|
|
|
|
|
|
|
|
$items_to_remove = array_slice($removed_items, 2);
|
|
|
|
|
|
|
|
if (! empty($items_to_remove)) {
|
|
|
|
|
|
|
|
\PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->in('id', $items_to_remove)
|
|
|
|
->eq('status', 'removed')
|
|
|
|
->eq('feed_id', $feed_id)
|
|
|
|
->remove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$db->closeTransaction();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Download content from an URL
|
|
|
|
function download_content_url($url)
|
|
|
|
{
|
|
|
|
$client = \PicoFeed\Client::create();
|
|
|
|
$client->url = $url;
|
|
|
|
$client->timeout = HTTP_TIMEOUT;
|
|
|
|
$client->user_agent = \Model\Config\HTTP_FAKE_USERAGENT;
|
|
|
|
$client->execute();
|
|
|
|
|
|
|
|
$html = $client->getContent();
|
|
|
|
|
|
|
|
if (! empty($html)) {
|
|
|
|
|
|
|
|
// Try first with PicoFeed grabber and with Readability after
|
|
|
|
$grabber = new \PicoFeed\Grabber($url, $html, $client->getEncoding());
|
|
|
|
$content = '';
|
|
|
|
|
|
|
|
if ($grabber->parse()) {
|
|
|
|
$content = $grabber->content;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($content)) {
|
|
|
|
$content = download_content_readability($grabber->html, $url);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Filter content
|
|
|
|
$filter = new \PicoFeed\Filter($content, $url);
|
|
|
|
return $filter->execute();
|
|
|
|
}
|
|
|
|
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
// Download content from item ID
|
|
|
|
function download_content_id($item_id)
|
|
|
|
{
|
|
|
|
$item = get($item_id);
|
|
|
|
$content = download_content_url($item['url']);
|
|
|
|
|
|
|
|
if (! empty($content)) {
|
|
|
|
|
|
|
|
if (! \Model\Config\get('nocontent')) {
|
|
|
|
|
|
|
|
// Save content
|
|
|
|
\PicoTools\singleton('db')
|
|
|
|
->table('items')
|
|
|
|
->eq('id', $item['id'])
|
|
|
|
->save(array('content' => $content));
|
|
|
|
}
|
|
|
|
|
|
|
|
\Model\Config\write_debug();
|
|
|
|
|
|
|
|
return array(
|
|
|
|
'result' => true,
|
|
|
|
'content' => $content
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
\Model\Config\write_debug();
|
|
|
|
|
|
|
|
return array(
|
|
|
|
'result' => false,
|
|
|
|
'content' => ''
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Download content with Readability PHP port
|
|
|
|
function download_content_readability($content, $url)
|
|
|
|
{
|
|
|
|
if (! empty($content)) {
|
|
|
|
|
|
|
|
$readability = new \Readability($content, $url);
|
|
|
|
|
|
|
|
if ($readability->init()) {
|
|
|
|
return $readability->getContent()->innerHTML;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return '';
|
|
|
|
}
|