function css() { $theme = \Model\Config\get('theme'); if ($theme !== 'original') { $css_file = THEME_DIRECTORY . '/' . $theme . '/css/app.css'; if (file_exists($css_file)) { return $css_file . '?version=' . filemtime($css_file); } } return 'assets/css/app.css?version=' . filemtime('assets/css/app.css'); }
} else { Session\flash_error(t('Unable to update your preferences.')); } Response\redirect('?action=config'); } Response\html(Template\layout('config', array('errors' => $errors, 'values' => Model\Config\get_all() + array('csrf' => Model\Config\generate_csrf()), 'languages' => Model\Config\get_languages(), 'timezones' => Model\Config\get_timezones(), 'autoflush_read_options' => Model\Config\get_autoflush_read_options(), 'autoflush_unread_options' => Model\Config\get_autoflush_unread_options(), 'paging_options' => Model\Config\get_paging_options(), 'theme_options' => Model\Config\get_themes(), 'sorting_options' => Model\Config\get_sorting_directions(), 'redirect_nothing_to_read_options' => Model\Config\get_nothing_to_read_redirections(), 'display_mode' => Model\Config\get_display_mode(), 'nb_unread_items' => Model\Item\count_by_status('unread'), 'menu' => 'config', 'title' => t('Preferences')))); }); // Get configuration parameters (AJAX request) Router\post_action('get-config', function () { $return = array(); $options = Request\values(); if (empty($options)) { $return = Model\Config\get_all(); } else { foreach ($options as $name) { $return[$name] = Model\Config\get($name); } } Response\json($return); }); // Display help page Router\get_action('help', function () { Response\html(Template\layout('help', array('config' => Model\Config\get_all(), 'nb_unread_items' => Model\Item\count_by_status('unread'), 'menu' => 'config', 'title' => t('Preferences')))); }); // Display about page Router\get_action('about', function () { Response\html(Template\layout('about', array('csrf' => Model\Config\generate_csrf(), 'config' => Model\Config\get_all(), 'nb_unread_items' => Model\Item\count_by_status('unread'), 'menu' => 'config', 'title' => t('Preferences')))); }); // Display database page Router\get_action('database', function () { Response\html(Template\layout('database', array('csrf' => Model\Config\generate_csrf(), 'config' => Model\Config\get_all(), 'db_size' => filesize(\Model\Database\get_path()), 'nb_unread_items' => Model\Item\count_by_status('unread'), 'menu' => 'config', 'title' => t('Preferences'))));
<?php // Display history page Router\get_action('history', function () { $offset = Request\int_param('offset', 0); $nb_items = Model\Item\count_by_status('read'); $items = Model\Item\get_all_by_status('read', array(), $offset, Model\Config\get('items_per_page'), 'updated', Model\Config\get('items_sorting_direction')); Response\html(Template\layout('history', array('favicons' => Model\Favicon\get_item_favicons($items), 'original_marks_read' => Model\Config\get('original_marks_read'), 'items' => $items, 'order' => '', 'direction' => '', 'display_mode' => Model\Config\get('items_display_mode'), 'nb_items' => $nb_items, 'nb_unread_items' => Model\Item\count_by_status('unread'), 'offset' => $offset, 'items_per_page' => Model\Config\get('items_per_page'), 'nothing_to_read' => Request\int_param('nothing_to_read'), 'menu' => 'history', 'title' => t('History') . ' (' . $nb_items . ')'))); }); // Confirmation box to flush history Router\get_action('confirm-flush-history', function () { Response\html(Template\layout('confirm_flush_items', array('nb_unread_items' => Model\Item\count_by_status('unread'), 'menu' => 'history', 'title' => t('Confirmation')))); }); // Flush history Router\get_action('flush-history', function () { Model\Item\mark_all_as_removed(); Response\redirect('?action=history'); });
$feed_id = Request\int_param('feed_id', 0); $offset = Request\int_param('offset', 0); $nb_items = Model\Item\count_by_feed($feed_id); $feed = Model\Feed\get($feed_id); $order = Request\param('order', 'updated'); $direction = Request\param('direction', Model\Config\get('items_sorting_direction')); $items = Model\Item\get_all_by_feed($feed_id, $offset, Model\Config\get('items_per_page'), $order, $direction); Response\html(Template\layout('feed_items', array('favicons' => Model\Feed\get_favicons(array($feed['id'])), 'original_marks_read' => Model\Config\get('original_marks_read'), 'order' => $order, 'direction' => $direction, 'display_mode' => Model\Config\get('items_display_mode'), 'feed' => $feed, 'items' => $items, 'nb_items' => $nb_items, 'nb_unread_items' => Model\Item\count_by_status('unread'), 'offset' => $offset, 'items_per_page' => Model\Config\get('items_per_page'), 'menu' => 'feed-items', 'title' => '(' . $nb_items . ') ' . $feed['title']))); }); // Ajax call to download an item (fetch the full content from the original website) Router\post_action('download-item', function () { $id = Request\param('id'); $item = Model\Item\get($id); $feed = Model\Feed\get($item['feed_id']); $download = Model\Item\download_content_id($id); $download['content'] = Model\Proxy\addProxyToTags($download['content'], $item['url'], Model\Config\get('image_proxy'), $feed['cloak_referrer']); Response\json($download); }); // Ajax call change item status Router\post_action('change-item-status', function () { $id = Request\param('id'); Response\json(array('item_id' => $id, 'status' => Model\Item\switch_status($id))); }); // Ajax call to mark item read Router\post_action('mark-item-read', function () { Model\Item\set_read(Request\param('id')); Response\json(array('Ok')); }); // Ajax call to mark item as removed Router\post_action('mark-item-removed', function () { Model\Item\set_removed(Request\param('id'));
if ($source === 'show') { Response\Redirect('?action=show&menu=' . $menu . '&id=' . $id); } Response\Redirect('?action=' . $menu . '&offset=' . $offset . '&feed_id=' . $feed_id . '#item-' . $id); }); // Display bookmarks page Router\get_action('bookmarks', function () { $offset = Request\int_param('offset', 0); $nb_items = Model\Item\count_bookmarks(); $items = Model\Item\get_bookmarks($offset, Model\Config\get('items_per_page')); Response\html(Template\layout('bookmarks', array('favicons' => Model\Feed\get_item_favicons($items), 'original_marks_read' => Model\Config\get('original_marks_read'), 'order' => '', 'direction' => '', 'display_mode' => Model\Config\get('items_display_mode'), 'items' => $items, 'nb_items' => $nb_items, 'offset' => $offset, 'items_per_page' => Model\Config\get('items_per_page'), 'nothing_to_read' => Request\int_param('nothing_to_read'), 'nb_unread_items' => Model\Item\count_by_status('unread'), 'menu' => 'bookmarks', 'title' => t('Bookmarks') . ' (' . $nb_items . ')'))); }); // Display bookmark feeds Router\get_action('bookmark-feed', function () { // Check token $feed_token = Model\Config\get('feed_token'); $request_token = Request\param('token'); if ($feed_token !== $request_token) { Response\text('Access Forbidden', 403); } // Build Feed $writer = new Atom(); $writer->title = t('Bookmarks') . ' - Miniflux'; $writer->site_url = Helper\get_current_base_url(); $writer->feed_url = $writer->site_url . '?action=bookmark-feed&token=' . urlencode($feed_token); $bookmarks = Model\Item\get_bookmarks(); foreach ($bookmarks as $bookmark) { $article = Model\Item\get($bookmark['id']); $writer->items[] = array('id' => $article['id'], 'title' => $article['title'], 'updated' => $article['updated'], 'url' => $article['url'], 'content' => $article['content']); } Response\xml($writer->execute());
} // These actions are considered to be safe even for unauthenticated users $safe_actions = array('login', 'bookmark-feed', 'select-db', 'logout', 'notfound'); if (!Model\User\is_loggedin() && !in_array($action, $safe_actions)) { if (!Model\RememberMe\authenticate()) { Model\User\logout(); Response\redirect('?action=login'); } } elseif (Model\RememberMe\has_cookie()) { Model\RememberMe\refresh(); } // Load translations $language = Model\Config\get('language') ?: 'en_US'; Translator\load($language); // Set timezone date_default_timezone_set(Model\Config\get('timezone') ?: 'UTC'); // HTTP secure headers Response\csp(array('media-src' => '*', 'img-src' => '*', 'frame-src' => Model\Config\get_iframe_whitelist(), 'referrer' => 'no-referrer')); Response\xframe(); Response\xss(); Response\nosniff(); if (ENABLE_HSTS && Helper\is_secure_connection()) { Response\hsts(); } }); // Show help Router\get_action('show-help', function () { Response\html(Template\load('show_help')); }); // Show the menu for the mobile view Router\get_action('more', function () {
$feed_id = Request\int_param('feed_id', 0); $offset = Request\int_param('offset', 0); $nb_items = Model\Item\count_by_feed($feed_id); $feed = Model\Feed\get($feed_id); $order = Request\param('order', 'updated'); $direction = Request\param('direction', Model\Config\get('items_sorting_direction')); $items = Model\Item\get_all_by_feed($feed_id, $offset, Model\Config\get('items_per_page'), $order, $direction); Response\html(Template\layout('feed_items', array('favicons' => Model\Feed\get_favicons(array($feed['id'])), 'original_marks_read' => Model\Config\get('original_marks_read'), 'order' => $order, 'direction' => $direction, 'display_mode' => Model\Config\get('items_display_mode'), 'feed' => $feed, 'items' => $items, 'nb_items' => $nb_items, 'nb_unread_items' => Model\Item\count_by_status('unread'), 'offset' => $offset, 'items_per_page' => Model\Config\get('items_per_page'), 'menu' => 'feed-items', 'title' => '(' . $nb_items . ') ' . $feed['title']))); }); // Ajax call to download an item (fetch the full content from the original website) Router\post_action('download-item', function () { $id = Request\param('id'); $item = Model\Item\get($id); $feed = Model\Feed\get($item['feed_id']); $download = Model\Item\download_content_id($id); $download['content'] = Model\Proxy\rewrite_html($download['content'], $item['url'], Model\Config\get('image_proxy'), $feed['cloak_referrer']); Response\json($download); }); // Ajax call to mark item read Router\post_action('mark-item-read', function () { Model\Item\set_read(Request\param('id')); Response\json(array('Ok')); }); // Ajax call to mark item as removed Router\post_action('mark-item-removed', function () { Model\Item\set_removed(Request\param('id')); Response\json(array('Ok')); }); // Ajax call to mark item unread Router\post_action('mark-item-unread', function () { Model\Item\set_unread(Request\param('id'));
// Display form to add one feed Router\get_action('add', function () { $values = array('download_content' => 0, 'rtl' => 0, 'cloak_referrer' => 0, 'create_group' => '', 'feed_group_ids' => array()); Response\html(Template\layout('add', array('values' => $values + array('csrf' => Model\Config\generate_csrf()), 'errors' => array(), 'nb_unread_items' => Model\Item\count_by_status('unread'), 'groups' => Model\Group\get_all(), 'menu' => 'feeds', 'title' => t('New subscription')))); }); // Add a feed with the form or directly from the url, it can be used by a bookmarklet by example Router\action('subscribe', function () { if (Request\is_post()) { $values = Request\values(); Model\Config\check_csrf_values($values); $url = isset($values['url']) ? $values['url'] : ''; } else { $values = array(); $url = Request\param('url'); $token = Request\param('token'); if ($token !== Model\Config\get('bookmarklet_token')) { Response\text('Access Forbidden', 403); } } $values += array('url' => trim($url), 'download_content' => 0, 'rtl' => 0, 'cloak_referrer' => 0, 'create_group' => '', 'feed_group_ids' => array()); try { $feed_id = Model\Feed\create($values['url'], $values['download_content'], $values['rtl'], $values['cloak_referrer'], $values['feed_group_ids'], $values['create_group']); } catch (UnexpectedValueException $e) { $error_message = t('This subscription already exists.'); } catch (PicoFeed\Client\InvalidCertificateException $e) { $error_message = t('Invalid SSL certificate.'); } catch (PicoFeed\Client\InvalidUrlException $e) { // picoFeed uses this exception for multiple reasons, but doesn't // provide an exception code to distinguish what exactly happend here $error_message = $e->getMessage(); } catch (PicoFeed\Client\MaxRedirectException $e) {
<?php require __DIR__ . '/common.php'; use JsonRPC\Server; use PicoFeed\PicoFeedException; $server = new Server(); $server->authentication(array(\Model\Config\get('username') => \Model\Config\get('api_token'))); // Get version $server->register('app.version', function () { return array('version' => APP_VERSION); }); // Get all feeds $server->register('feed.list', function () { return Model\Feed\get_all(); }); // Get one feed $server->register('feed.info', function ($feed_id) { return Model\Feed\get($feed_id); }); // Add a new feed $server->register('feed.create', function ($url) { try { $result = Model\Feed\create($url); } catch (PicoFeedException $e) { $result = false; } catch (UnexpectedValueException $e) { $result = false; } Model\Config\write_debug(); return $result; });