2014-10-29 21:28:23 -04:00
|
|
|
<?php
|
|
|
|
|
2016-08-18 21:02:49 -04:00
|
|
|
require __DIR__.'/../app/common.php';
|
2014-10-29 21:28:23 -04:00
|
|
|
|
2016-08-24 21:25:24 -04:00
|
|
|
use Miniflux\Handler;
|
|
|
|
use Miniflux\Model;
|
2017-01-03 20:57:33 -05:00
|
|
|
use Miniflux\Session\SessionStorage;
|
2016-12-26 09:44:53 -05:00
|
|
|
|
|
|
|
register_shutdown_function(function () {
|
|
|
|
Miniflux\Helper\write_debug_file();
|
|
|
|
});
|
2014-10-29 21:28:23 -04:00
|
|
|
|
|
|
|
// Route handler
|
|
|
|
function route($name, Closure $callback = null)
|
|
|
|
{
|
|
|
|
static $routes = array();
|
|
|
|
|
|
|
|
if ($callback !== null) {
|
|
|
|
$routes[$name] = $callback;
|
2016-04-17 19:44:45 -04:00
|
|
|
} elseif (isset($routes[$name])) {
|
2014-10-29 21:28:23 -04:00
|
|
|
$routes[$name]();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Serialize the payload in Json (XML is not supported)
|
|
|
|
function response(array $response)
|
|
|
|
{
|
|
|
|
header('Content-Type: application/json');
|
|
|
|
echo json_encode($response);
|
|
|
|
exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fever authentication
|
|
|
|
function auth()
|
|
|
|
{
|
2016-12-26 09:44:53 -05:00
|
|
|
$api_key = isset($_POST['api_key']) && ctype_alnum($_POST['api_key']) ? $_POST['api_key'] : null;
|
|
|
|
$user = Model\User\get_user_by_token('fever_api_key', $api_key);
|
|
|
|
$authenticated = $user !== null;
|
2014-10-29 21:28:23 -04:00
|
|
|
|
2017-01-03 20:57:33 -05:00
|
|
|
if ($authenticated) {
|
|
|
|
SessionStorage::getInstance()->setUser($user);
|
|
|
|
}
|
|
|
|
|
2014-10-29 21:28:23 -04:00
|
|
|
$response = array(
|
|
|
|
'api_version' => 3,
|
2016-12-26 09:44:53 -05:00
|
|
|
'auth' => (int) $authenticated,
|
2014-10-29 21:28:23 -04:00
|
|
|
'last_refreshed_on_time' => time(),
|
|
|
|
);
|
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
return array($user, $authenticated, $response);
|
2014-10-29 21:28:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Call: ?api&groups
|
2016-04-17 19:44:45 -04:00
|
|
|
route('groups', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list($user, $authenticated, $response) = auth();
|
2014-10-29 21:28:23 -04:00
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
if ($authenticated) {
|
2016-12-29 18:04:56 -05:00
|
|
|
$response['groups'] = array();
|
2015-08-05 01:01:21 +02:00
|
|
|
$response['feeds_groups'] = array();
|
2015-08-05 01:01:21 +02:00
|
|
|
|
2016-12-29 18:04:56 -05:00
|
|
|
$groups = Model\Group\get_all($user['id']);
|
|
|
|
$feed_groups = Model\Group\get_groups_feed_ids($user['id']);
|
|
|
|
|
|
|
|
foreach ($groups as $group) {
|
|
|
|
$response['groups'][] = array(
|
|
|
|
'id' => $group['id'],
|
|
|
|
'title' => $group['title'],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($feed_groups as $group_id => $feed_ids) {
|
2015-08-05 01:01:21 +02:00
|
|
|
$response['feeds_groups'][] = array(
|
|
|
|
'group_id' => $group_id,
|
|
|
|
'feed_ids' => implode(',', $feed_ids)
|
|
|
|
);
|
|
|
|
}
|
2014-10-29 21:28:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Call: ?api&feeds
|
2016-04-17 19:44:45 -04:00
|
|
|
route('feeds', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list($user, $authenticated, $response) = auth();
|
2014-10-29 21:28:23 -04:00
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
if ($authenticated) {
|
2014-10-29 21:28:23 -04:00
|
|
|
$response['feeds'] = array();
|
2015-08-05 01:01:21 +02:00
|
|
|
$response['feeds_groups'] = array();
|
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
$feeds = Model\Feed\get_feeds($user['id']);
|
2014-10-29 21:28:23 -04:00
|
|
|
|
|
|
|
foreach ($feeds as $feed) {
|
|
|
|
$response['feeds'][] = array(
|
|
|
|
'id' => (int) $feed['id'],
|
2015-03-23 15:01:54 +07:00
|
|
|
'favicon_id' => (int) $feed['id'],
|
2014-10-29 21:28:23 -04:00
|
|
|
'title' => $feed['title'],
|
|
|
|
'url' => $feed['feed_url'],
|
|
|
|
'site_url' => $feed['site_url'],
|
|
|
|
'is_spark' => 0,
|
|
|
|
'last_updated_on_time' => $feed['last_checked'] ?: time(),
|
|
|
|
);
|
|
|
|
}
|
2015-08-05 01:01:21 +02:00
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
$group_map = Model\Group\get_groups_feed_ids($user['id']);
|
2015-08-05 01:01:21 +02:00
|
|
|
foreach ($group_map as $group_id => $feed_ids) {
|
|
|
|
$response['feeds_groups'][] = array(
|
|
|
|
'group_id' => $group_id,
|
|
|
|
'feed_ids' => implode(',', $feed_ids)
|
|
|
|
);
|
|
|
|
}
|
2014-10-29 21:28:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Call: ?api&favicons
|
2016-04-17 19:44:45 -04:00
|
|
|
route('favicons', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list($user, $authenticated, $response) = auth();
|
|
|
|
if ($authenticated) {
|
|
|
|
$favicons = Model\Favicon\get_favicons_with_data_url($user['id']);
|
2014-10-29 21:28:23 -04:00
|
|
|
$response['favicons'] = array();
|
2016-12-26 09:44:53 -05:00
|
|
|
|
2015-03-23 15:01:54 +07:00
|
|
|
foreach ($favicons as $favicon) {
|
|
|
|
$response['favicons'][] = array(
|
|
|
|
'id' => (int) $favicon['feed_id'],
|
2016-12-29 18:04:56 -05:00
|
|
|
'data' => $favicon['data_url'],
|
2015-03-23 15:01:54 +07:00
|
|
|
);
|
|
|
|
}
|
2014-10-29 21:28:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Call: ?api&items
|
2016-04-17 19:44:45 -04:00
|
|
|
route('items', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list($user, $authenticated, $response) = auth();
|
2014-10-29 21:28:23 -04:00
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
if ($authenticated) {
|
|
|
|
$since_id = isset($_GET['since_id']) && ctype_digit($_GET['since_id']) ? $_GET['since_id'] : null;
|
|
|
|
$item_ids = ! empty($_GET['with_ids']) ? explode(',', $_GET['with_ids']) : array();
|
|
|
|
$items = Model\Item\get_items($user['id'], $since_id, $item_ids);
|
2014-10-29 21:28:23 -04:00
|
|
|
$response['items'] = array();
|
|
|
|
|
|
|
|
foreach ($items as $item) {
|
|
|
|
$response['items'][] = array(
|
2016-12-26 09:44:53 -05:00
|
|
|
'id' => (int) $item['id'],
|
2014-10-29 21:28:23 -04:00
|
|
|
'feed_id' => (int) $item['feed_id'],
|
|
|
|
'title' => $item['title'],
|
|
|
|
'author' => $item['author'],
|
|
|
|
'html' => $item['content'],
|
|
|
|
'url' => $item['url'],
|
|
|
|
'is_saved' => (int) $item['bookmark'],
|
|
|
|
'is_read' => $item['status'] == 'read' ? 1 : 0,
|
|
|
|
'created_on_time' => $item['updated'],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
$response['total_items'] = Model\Item\count_by_status(
|
|
|
|
$user['id'],
|
|
|
|
array(Model\Item\STATUS_READ, Model\Item\STATUS_UNREAD)
|
|
|
|
);
|
2014-10-29 21:28:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Call: ?api&links
|
2016-04-17 19:44:45 -04:00
|
|
|
route('links', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list(, $authenticated, $response) = auth();
|
2014-10-29 21:28:23 -04:00
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
if ($authenticated) {
|
2014-10-29 21:28:23 -04:00
|
|
|
$response['links'] = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Call: ?api&unread_item_ids
|
2016-04-17 19:44:45 -04:00
|
|
|
route('unread_item_ids', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list($user, $authenticated, $response) = auth();
|
2014-10-29 21:28:23 -04:00
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
if ($authenticated) {
|
|
|
|
$item_ids = Model\Item\get_item_ids_by_status($user['id'], Model\Item\STATUS_UNREAD);
|
2014-10-29 21:28:23 -04:00
|
|
|
$response['unread_item_ids'] = implode(',', $item_ids);
|
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Call: ?api&saved_item_ids
|
2016-04-17 19:44:45 -04:00
|
|
|
route('saved_item_ids', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list($user, $authenticated, $response) = auth();
|
2014-10-29 21:28:23 -04:00
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
if ($authenticated) {
|
|
|
|
$item_ids = Model\Bookmark\get_bookmarked_item_ids($user['id']);
|
2014-10-29 21:28:23 -04:00
|
|
|
$response['saved_item_ids'] = implode(',', $item_ids);
|
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// handle write items
|
2016-04-17 19:44:45 -04:00
|
|
|
route('write_items', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list($user, $authenticated, $response) = auth();
|
2014-10-29 21:28:23 -04:00
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
if ($authenticated && ctype_digit($_POST['id'])) {
|
|
|
|
$item_id = $_POST['id'];
|
2014-10-29 21:28:23 -04:00
|
|
|
|
|
|
|
if ($_POST['as'] === 'saved') {
|
2016-12-26 09:44:53 -05:00
|
|
|
Model\Bookmark\set_flag($user['id'], $item_id, 1);
|
|
|
|
Handler\Service\sync($user['id'], $item_id);
|
2016-04-17 19:44:45 -04:00
|
|
|
} elseif ($_POST['as'] === 'unsaved') {
|
2016-12-26 09:44:53 -05:00
|
|
|
Model\Bookmark\set_flag($user['id'], $item_id, 0);
|
2016-04-17 19:44:45 -04:00
|
|
|
} elseif ($_POST['as'] === 'read') {
|
2016-12-26 09:44:53 -05:00
|
|
|
Model\Item\change_item_status($user['id'], $item_id, Model\Item\STATUS_READ);
|
2016-04-17 19:44:45 -04:00
|
|
|
} elseif ($_POST['as'] === 'unread') {
|
2016-12-26 09:44:53 -05:00
|
|
|
Model\Item\change_item_status($user['id'], $item_id, Model\Item\STATUS_UNREAD);
|
2014-10-29 21:28:23 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// handle write feeds
|
2016-04-17 19:44:45 -04:00
|
|
|
route('write_feeds', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list($user, $authenticated, $response) = auth();
|
|
|
|
|
|
|
|
if ($authenticated && ctype_digit($_POST['id']) && ctype_digit($_POST['before'])) {
|
|
|
|
Model\ItemFeed\change_items_status(
|
|
|
|
$user['id'],
|
|
|
|
$_POST['id'],
|
|
|
|
Model\Item\STATUS_UNREAD,
|
|
|
|
Model\Item\STATUS_READ,
|
|
|
|
$_POST['before']
|
|
|
|
);
|
2014-10-29 21:28:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// handle write groups
|
2016-04-17 19:44:45 -04:00
|
|
|
route('write_groups', function () {
|
2016-12-26 09:44:53 -05:00
|
|
|
list($user, $authenticated, $response) = auth();
|
|
|
|
|
|
|
|
if ($authenticated && ctype_digit($_POST['id']) && ctype_digit($_POST['before'])) {
|
|
|
|
Model\ItemGroup\change_items_status(
|
|
|
|
$user['id'],
|
|
|
|
$_POST['id'],
|
|
|
|
Model\Item\STATUS_UNREAD,
|
|
|
|
Model\Item\STATUS_READ,
|
|
|
|
$_POST['before']
|
|
|
|
);
|
2014-10-29 21:28:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
response($response);
|
|
|
|
});
|
|
|
|
|
|
|
|
foreach (array_keys($_GET) as $action) {
|
|
|
|
route($action);
|
|
|
|
}
|
|
|
|
|
2014-11-06 19:54:12 +01:00
|
|
|
if (! empty($_POST['mark']) && ! empty($_POST['as'])
|
2016-05-03 17:45:07 +09:00
|
|
|
&& filter_input(INPUT_POST, 'id', FILTER_VALIDATE_INT, array('options' => array('default' => null, 'min_range' => -1))) !== null) {
|
2014-10-29 21:28:23 -04:00
|
|
|
if ($_POST['mark'] === 'item') {
|
|
|
|
route('write_items');
|
2016-04-17 19:44:45 -04:00
|
|
|
} elseif ($_POST['mark'] === 'feed' && ! empty($_POST['before'])) {
|
2014-10-29 21:28:23 -04:00
|
|
|
route('write_feeds');
|
2016-04-17 19:44:45 -04:00
|
|
|
} elseif ($_POST['mark'] === 'group' && ! empty($_POST['before'])) {
|
2014-10-29 21:28:23 -04:00
|
|
|
route('write_groups');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-26 09:44:53 -05:00
|
|
|
list(, , $response) = auth();
|
|
|
|
response($response);
|