This page lists files in the current directory. You can view content, get download/execute commands for Wget, Curl, or PowerShell, or filter the list using wildcards (e.g., `*.sh`).
wget 'https://sme10.lists2.roe3.org/FreshRSS/app/.htaccess'
# Apache 2.2
<IfModule !mod_authz_core.c>
Order Allow,Deny
Deny from all
Satisfy all
</IfModule>
# Apache 2.4
<IfModule mod_authz_core.c>
Require all denied
</IfModule>
wget 'https://sme10.lists2.roe3.org/FreshRSS/app/FreshRSS.php'
<?php
declare(strict_types=1);
class FreshRSS extends Minz_FrontController {
/**
* Initialize the different FreshRSS / Minz components.
*
* PLEASE DON’T CHANGE THE ORDER OF INITIALIZATIONS UNLESS YOU KNOW WHAT YOU DO!!
*
* Here is the list of components:
* - Create a configuration setter and register it to system conf
* - Init extension manager and enable system extensions (has to be done asap)
* - Init authentication system
* - Init user configuration (need auth system)
* - Init FreshRSS context (need user conf)
* - Init i18n (need context)
* - Init sharing system (need user conf and i18n)
* - Init generic styles and scripts (need user conf)
* - Enable user extensions (need all the other initializations)
*/
public function init(): void {
if (!isset($_SESSION)) {
Minz_Session::init('FreshRSS');
}
FreshRSS_Context::initSystem();
if (!FreshRSS_Context::hasSystemConf()) {
$message = 'Error during context system init!';
Minz_Error::error(500, $message, false);
die($message);
}
if (FreshRSS_Context::systemConf()->logo_html != '') {
// Relax Content Security Policy to allow external images if a custom logo HTML is used
Minz_ActionController::_defaultCsp([
'default-src' => "'self'",
'img-src' => '* data:',
]);
}
// Load list of extensions and enable the "system" ones.
Minz_ExtensionManager::init();
// Auth has to be initialized before using currentUser session parameter
// because it’s this part which create this parameter.
self::initAuth();
if (!FreshRSS_Context::hasUserConf()) {
FreshRSS_Context::initUser();
}
if (!FreshRSS_Context::hasUserConf()) {
$message = 'Error during context user init!';
Minz_Error::error(500, $message, false);
die($message);
}
// Complete initialization of the other FreshRSS / Minz components.
self::initI18n();
// Enable extensions for the current (logged) user.
if (FreshRSS_Auth::hasAccess() || FreshRSS_Context::systemConf()->allow_anonymous) {
$ext_list = FreshRSS_Context::userConf()->extensions_enabled;
Minz_ExtensionManager::enableByList($ext_list, 'user');
}
if (FreshRSS_Context::systemConf()->force_email_validation && !FreshRSS_Auth::hasAccess('admin')) {
self::checkEmailValidated();
}
Minz_ExtensionManager::callHookVoid('freshrss_init');
}
private static function initAuth(): void {
FreshRSS_Auth::init();
if (Minz_Request::isPost()) {
if (!FreshRSS_Context::hasSystemConf() || !(FreshRSS_Auth::isCsrfOk() ||
(Minz_Request::controllerName() === 'auth' && Minz_Request::actionName() === 'login') ||
(Minz_Request::controllerName() === 'user' && Minz_Request::actionName() === 'create' && !FreshRSS_Auth::hasAccess('admin')) ||
(Minz_Request::controllerName() === 'feed' && Minz_Request::actionName() === 'actualize' &&
FreshRSS_Context::systemConf()->allow_anonymous_refresh) ||
(Minz_Request::controllerName() === 'javascript' && Minz_Request::actionName() === 'actualize' &&
FreshRSS_Context::systemConf()->allow_anonymous)
)) {
// Token-based protection against XSRF attacks, except for the login or self-create user forms
self::initI18n();
Minz_Error::error(403, ['error' => [_t('feedback.access.denied'), ' [CSRF]']]);
}
}
}
private static function initI18n(): void {
$userLanguage = FreshRSS_Context::hasUserConf() ? FreshRSS_Context::userConf()->language : null;
$systemLanguage = FreshRSS_Context::hasSystemConf() ? FreshRSS_Context::systemConf()->language : null;
$language = Minz_Translate::getLanguage($userLanguage, Minz_Request::getPreferredLanguages(), $systemLanguage);
Minz_Session::_param('language', $language);
Minz_Translate::init($language);
$timezone = FreshRSS_Context::hasUserConf() ? FreshRSS_Context::userConf()->timezone : '';
if ($timezone == '') {
$timezone = FreshRSS_Context::defaultTimeZone();
}
date_default_timezone_set($timezone);
}
private static function getThemeFileUrl(string $theme_id, string $filename): string {
$filetime = @filemtime(PUBLIC_PATH . '/themes/' . $theme_id . '/' . $filename);
return '/themes/' . $theme_id . '/' . $filename . '?' . $filetime;
}
public static function loadStylesAndScripts(): void {
if (!FreshRSS_Context::hasUserConf()) {
return;
}
$theme = FreshRSS_Themes::load(FreshRSS_Context::userConf()->theme);
if ($theme) {
foreach (array_reverse($theme['files']) as $file) {
switch (substr($file, -3)) {
case '.js':
$theme_id = $theme['id'];
$filename = $file;
FreshRSS_View::prependScript(Minz_Url::display(FreshRSS::getThemeFileUrl($theme_id, $filename)));
break;
case '.css':
default:
if ($file[0] === '_') {
$theme_id = 'base-theme';
$filename = substr($file, 1);
} else {
$theme_id = $theme['id'];
$filename = $file;
}
if (_t('gen.dir') === 'rtl') {
$filename = substr($filename, 0, -4);
$filename = $filename . '.rtl.css';
}
FreshRSS_View::prependStyle(Minz_Url::display(FreshRSS::getThemeFileUrl($theme_id, $filename)));
}
}
if (!empty($theme['theme-color'])) {
FreshRSS_View::appendThemeColors($theme['theme-color']);
}
}
//Use prepend to insert before extensions. Added in reverse order.
if (!in_array(Minz_Request::controllerName(), ['index', ''], true)) {
FreshRSS_View::prependScript(Minz_Url::display('/scripts/extra.js?' . @filemtime(PUBLIC_PATH . '/scripts/extra.js')));
}
FreshRSS_View::prependScript(Minz_Url::display('/scripts/main.js?' . @filemtime(PUBLIC_PATH . '/scripts/main.js')));
}
public static function preLayout(): void {
header("X-Content-Type-Options: nosniff");
FreshRSS_Share::load(join_path(APP_PATH, 'shares.php'));
self::loadStylesAndScripts();
}
private static function checkEmailValidated(): void {
$email_not_verified = FreshRSS_Auth::hasAccess() &&
FreshRSS_Context::hasUserConf() && FreshRSS_Context::userConf()->email_validation_token !== '';
$action_is_allowed = (
Minz_Request::is('user', 'validateEmail') ||
Minz_Request::is('user', 'sendValidationEmail') ||
Minz_Request::is('user', 'profile') ||
Minz_Request::is('user', 'delete') ||
Minz_Request::is('auth', 'logout') ||
Minz_Request::is('feed', 'actualize') ||
Minz_Request::is('javascript', 'nonce')
);
if ($email_not_verified && !$action_is_allowed) {
Minz_Request::forward([
'c' => 'user',
'a' => 'validateEmail',
], true);
}
}
}
wget 'https://sme10.lists2.roe3.org/FreshRSS/app/actualize_script.php'
#!/usr/bin/env php
<?php
// declare(strict_types=1); // Need to wait for PHP 8+ due to https://php.net/ob-implicit-flush
require(__DIR__ . '/../cli/_cli.php');
session_cache_limiter('');
ob_implicit_flush(false);
ob_start();
$begin_date = date_create('now');
// Set the header params ($_GET) to call the FRSS application.
$_GET['c'] = 'feed';
$_GET['a'] = 'actualize';
$_GET['ajax'] = 1;
$_GET['maxFeeds'] = PHP_INT_MAX;
$_SERVER['HTTP_HOST'] = '';
$app = new FreshRSS();
FreshRSS_Context::initSystem();
FreshRSS_Context::systemConf()->auth_type = 'none'; // avoid necessity to be logged in (not saved!)
define('SIMPLEPIE_SYSLOG_ENABLED', FreshRSS_Context::systemConf()->simplepie_syslog_enabled);
/**
* Writes to FreshRSS admin log, and if it is not already done by default,
* writes to syslog (only if simplepie_syslog_enabled in FreshRSS configuration) and to STDOUT
*/
function notice(string $message): void {
Minz_Log::notice($message, ADMIN_LOG);
if (!COPY_LOG_TO_SYSLOG && SIMPLEPIE_SYSLOG_ENABLED) {
syslog(LOG_NOTICE, $message);
}
if (defined('STDOUT') && !COPY_SYSLOG_TO_STDERR) {
fwrite(STDOUT, $message . "\n"); //Unbuffered
}
}
// <Mutex>
// Avoid having multiple actualization processes at the same time
$mutexFile = TMP_PATH . '/actualize.freshrss.lock';
$mutexTtl = 900; // seconds (refreshed before each new feed)
if (file_exists($mutexFile) && ((time() - (@filemtime($mutexFile) ?: 0)) > $mutexTtl)) {
unlink($mutexFile);
}
if (($handle = @fopen($mutexFile, 'x')) === false) {
notice('FreshRSS feeds actualization was already running, so aborting new run at ' . $begin_date->format('c'));
die();
}
fclose($handle);
register_shutdown_function(static function () use ($mutexFile) {
unlink($mutexFile);
});
// </Mutex>
notice('FreshRSS starting feeds actualization at ' . $begin_date->format('c'));
// make sure the PHP setup of the CLI environment is compatible with FreshRSS as well
echo 'Failed requirements!', "\n";
performRequirementCheck(FreshRSS_Context::systemConf()->db['type'] ?? '');
ob_clean();
echo 'Results: ', "\n"; //Buffered
// Create the list of users to actualize.
// Users are processed in a random order but always start with default user
$users = listUsers();
shuffle($users);
if (FreshRSS_Context::systemConf()->default_user !== '') {
array_unshift($users, FreshRSS_Context::systemConf()->default_user);
$users = array_unique($users);
}
$limits = FreshRSS_Context::systemConf()->limits;
$min_last_activity = time() - $limits['max_inactivity'];
foreach ($users as $user) {
FreshRSS_Context::initUser($user);
if (!FreshRSS_Context::hasUserConf()) {
notice('Invalid user ' . $user);
continue;
}
if (!FreshRSS_Context::userConf()->enabled) {
notice('FreshRSS skip disabled user ' . $user);
continue;
}
if (($user !== FreshRSS_Context::systemConf()->default_user) &&
(FreshRSS_UserDAO::mtime($user) < $min_last_activity)) {
notice('FreshRSS skip inactive user ' . $user);
continue;
}
FreshRSS_Auth::giveAccess();
// NB: Extensions and hooks are reinitialised there
$app->init();
Minz_ExtensionManager::addHook('feed_before_actualize', static function (FreshRSS_Feed $feed) use ($mutexFile) {
touch($mutexFile);
return $feed;
});
notice('FreshRSS actualize ' . $user . '…');
echo $user, ' '; //Buffered
$app->run();
if (!invalidateHttpCache()) {
Minz_Log::warning('FreshRSS write access problem in ' . join_path(USERS_PATH, $user, LOG_FILENAME), ADMIN_LOG);
if (defined('STDERR')) {
fwrite(STDERR, 'FreshRSS write access problem in ' . join_path(USERS_PATH, $user, LOG_FILENAME) . "\n");
}
}
gc_collect_cycles();
}
$end_date = date_create('now');
$duration = date_diff($end_date, $begin_date);
notice('FreshRSS actualization done for ' . count($users) .
' users, using ' . format_bytes(memory_get_peak_usage(true)) . ' of memory, in ' .
$duration->format('%a day(s), %h hour(s), %i minute(s) and %s seconds.'));
echo 'End.', "\n";
ob_end_flush();
wget 'https://sme10.lists2.roe3.org/FreshRSS/app/index.html'
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en-GB" lang="en-GB">
<head>
<meta charset="UTF-8" />
<meta http-equiv="Refresh" content="0; url=/" />
<title>Redirection</title>
<meta name="robots" content="noindex" />
</head>
<body>
<p><a href="/">Redirection</a></p>
</body>
</html>
wget 'https://sme10.lists2.roe3.org/FreshRSS/app/install.php'
<?php
declare(strict_types=1);
if (function_exists('opcache_reset')) {
opcache_reset();
}
header("Content-Security-Policy: default-src 'self'");
require(LIB_PATH . '/lib_install.php');
Minz_Session::init('FreshRSS');
if (isset($_GET['step'])) {
define('STEP', (int)$_GET['step']);
} else {
define('STEP', 0);
}
if (STEP === 2 && isset($_POST['type'])) {
Minz_Session::_param('bd_type', $_POST['type']);
}
function param(string $key, string $default = ''): string {
return isset($_POST[$key]) && is_string($_POST[$key]) ? trim($_POST[$key]) : $default;
}
// gestion internationalisation
function initTranslate(): void {
Minz_Translate::init();
$available_languages = Minz_Translate::availableLanguages();
if (Minz_Session::paramString('language') == '') {
Minz_Session::_param('language', get_best_language());
}
if (!in_array(Minz_Session::paramString('language'), $available_languages, true)) {
Minz_Session::_param('language', 'en');
}
Minz_Translate::reset(Minz_Session::paramString('language'));
}
function get_best_language(): string {
$accept = empty($_SERVER['HTTP_ACCEPT_LANGUAGE']) ? '' : $_SERVER['HTTP_ACCEPT_LANGUAGE'];
return strtolower(substr($accept, 0, 2));
}
/*** SAUVEGARDES ***/
function saveLanguage(): bool {
if (!empty($_POST)) {
if (!isset($_POST['language'])) {
return false;
}
Minz_Session::_param('language', $_POST['language']);
Minz_Session::_param('sessionWorking', 'ok');
header('Location: index.php?step=1');
}
return true;
}
function saveStep1(): void {
if (isset($_POST['freshrss-keep-install']) &&
$_POST['freshrss-keep-install'] === '1') {
// We want to keep our previous installation of FreshRSS
// so we need to make next steps valid by setting $_SESSION vars
// with values from the previous installation
// First, we try to get previous configurations
FreshRSS_Context::initSystem();
FreshRSS_Context::initUser(FreshRSS_Context::systemConf()->default_user, false);
// Then, we set $_SESSION vars
Minz_Session::_params([
'title' => FreshRSS_Context::systemConf()->title,
'auth_type' => FreshRSS_Context::systemConf()->auth_type,
'default_user' => Minz_User::name() ?? '',
'passwordHash' => FreshRSS_Context::userConf()->passwordHash,
'bd_type' => FreshRSS_Context::systemConf()->db['type'] ?? '',
'bd_host' => FreshRSS_Context::systemConf()->db['host'] ?? '',
'bd_user' => FreshRSS_Context::systemConf()->db['user'] ?? '',
'bd_password' => FreshRSS_Context::systemConf()->db['password'] ?? '',
'bd_base' => FreshRSS_Context::systemConf()->db['base'] ?? '',
'bd_prefix' => FreshRSS_Context::systemConf()->db['prefix'] ?? '',
'bd_error' => false,
]);
header('Location: index.php?step=4');
}
}
function saveStep2(): void {
if (!empty($_POST)) {
if (Minz_Session::paramString('bd_type') === 'sqlite') {
Minz_Session::_params([
'bd_base' => false,
'bd_host' => false,
'bd_user' => false,
'bd_password' => false,
'bd_prefix' => false,
]);
} else {
if (empty($_POST['type']) ||
empty($_POST['host']) ||
empty($_POST['user']) ||
empty($_POST['base'])) {
Minz_Session::_param('bd_error', 'Missing parameters!');
}
Minz_Session::_params([
'bd_base' => substr($_POST['base'], 0, 64),
'bd_host' => $_POST['host'],
'bd_user' => $_POST['user'],
'bd_password' => $_POST['pass'],
'bd_prefix' => substr($_POST['prefix'], 0, 16),
]);
}
// We use dirname to remove the /i part
$base_url = dirname(Minz_Request::guessBaseUrl());
$config_array = [
'salt' => generateSalt(),
'base_url' => $base_url,
'default_user' => '_',
'db' => [
'type' => Minz_Session::paramString('bd_type'),
'host' => Minz_Session::paramString('bd_host'),
'user' => Minz_Session::paramString('bd_user'),
'password' => Minz_Session::paramString('bd_password'),
'base' => Minz_Session::paramString('bd_base'),
'prefix' => Minz_Session::paramString('bd_prefix'),
'pdo_options' => [],
],
'pubsubhubbub_enabled' => Minz_Request::serverIsPublic($base_url),
];
if (Minz_Session::paramString('title') != '') {
$config_array['title'] = Minz_Session::paramString('title');
}
$customConfigPath = DATA_PATH . '/config.custom.php';
if (file_exists($customConfigPath)) {
$customConfig = include($customConfigPath);
if (is_array($customConfig)) {
$config_array = array_merge($customConfig, $config_array);
}
}
@unlink(DATA_PATH . '/config.php'); //To avoid access-rights problems
file_put_contents(DATA_PATH . '/config.php', "<?php\n return " . var_export($config_array, true) . ";\n");
if (function_exists('opcache_reset')) {
opcache_reset();
}
FreshRSS_Context::initSystem(true);
$ok = false;
try {
Minz_User::change($config_array['default_user']);
$error = initDb();
Minz_User::change();
if ($error != '') {
Minz_Session::_param('bd_error', $error);
} else {
$ok = true;
}
} catch (Exception $ex) {
Minz_Session::_param('bd_error', $ex->getMessage());
$ok = false;
}
if (!$ok) {
@unlink(join_path(DATA_PATH, 'config.php'));
}
if ($ok) {
Minz_Session::_param('bd_error');
header('Location: index.php?step=3');
} elseif (Minz_Session::paramString('bd_error') == '') {
Minz_Session::_param('bd_error', 'Unknown error!');
}
}
invalidateHttpCache();
}
function saveStep3(): bool {
FreshRSS_Context::initSystem();
Minz_Translate::init(Minz_Session::paramString('language'));
if (!empty($_POST)) {
$auth_type = param('auth_type', 'form');
if (in_array($auth_type, ['form', 'http_auth', 'none'], true)) {
FreshRSS_Context::systemConf()->auth_type = $auth_type;
Minz_Session::_param('auth_type', FreshRSS_Context::systemConf()->auth_type);
} else {
return false;
}
$password_plain = param('passwordPlain', '');
if (FreshRSS_Context::systemConf()->auth_type === 'form' && $password_plain == '') {
return false;
}
if (FreshRSS_user_Controller::checkUsername(param('default_user', ''))) {
FreshRSS_Context::systemConf()->default_user = param('default_user', '');
Minz_Session::_param('default_user', FreshRSS_Context::systemConf()->default_user);
} else {
return false;
}
if (FreshRSS_Context::systemConf()->auth_type === 'http_auth' &&
connectionRemoteAddress() !== '' &&
empty($_SERVER['REMOTE_USER']) && empty($_SERVER['REDIRECT_REMOTE_USER']) && // No safe authentication HTTP headers
(!empty($_SERVER['HTTP_REMOTE_USER']) || !empty($_SERVER['HTTP_X_WEBAUTH_USER'])) // but has unsafe authentication HTTP headers
) {
// Trust by default the remote IP address (e.g. last proxy) used during install to provide remote user name via unsafe HTTP header
FreshRSS_Context::systemConf()->trusted_sources[] = connectionRemoteAddress();
FreshRSS_Context::systemConf()->trusted_sources = array_unique(FreshRSS_Context::systemConf()->trusted_sources);
}
// Create default user files but first, we delete previous data to
// avoid access right problems.
recursive_unlink(USERS_PATH . '/' . Minz_Session::paramString('default_user'));
$ok = false;
try {
$ok = FreshRSS_user_Controller::createUser(
Minz_Session::paramString('default_user'),
'', //TODO: Add e-mail
$password_plain,
[
'language' => Minz_Session::paramString('language'),
'is_admin' => true,
'enabled' => true,
]
);
} catch (Exception $e) {
Minz_Session::_param('bd_error', $e->getMessage());
$ok = false;
}
if (!$ok) {
return false;
}
FreshRSS_Context::systemConf()->save();
header('Location: index.php?step=4');
}
return true;
}
/*** VÉRIFICATIONS ***/
function checkStep(): void {
$s0 = checkStep0();
$s1 = checkRequirements();
$s2 = checkStep2();
$s3 = checkStep3();
if (STEP > 0 && $s0['all'] !== 'ok') {
header('Location: index.php?step=0');
} elseif (STEP > 1 && $s1['all'] !== 'ok') {
header('Location: index.php?step=1');
} elseif (STEP > 2 && $s2['all'] !== 'ok') {
header('Location: index.php?step=2');
} elseif (STEP > 3 && $s3['all'] !== 'ok') {
header('Location: index.php?step=3');
}
Minz_Session::_param('actualize_feeds', true);
}
/** @return array<string,string> */
function checkStep0(): array {
$languages = Minz_Translate::availableLanguages();
$language = Minz_Session::paramString('language') != '' && in_array(Minz_Session::paramString('language'), $languages, true);
$sessionWorking = Minz_Session::paramString('sessionWorking') === 'ok';
return [
'language' => $language ? 'ok' : 'ko',
'sessionWorking' => $sessionWorking ? 'ok' : 'ko',
'all' => $language && $sessionWorking ? 'ok' : 'ko',
];
}
function freshrss_already_installed(): bool {
$conf_path = join_path(DATA_PATH, 'config.php');
if (!file_exists($conf_path)) {
return false;
}
// A configuration file already exists, we try to load it.
$system_conf = null;
try {
$system_conf = FreshRSS_SystemConfiguration::init($conf_path);
} catch (Minz_FileNotExistException $e) {
return false;
}
// ok, the global conf exists… but what about default user conf?
$current_user = $system_conf->default_user;
try {
FreshRSS_UserConfiguration::init(USERS_PATH . '/' . $current_user . '/config.php');
} catch (Minz_FileNotExistException $e) {
return false;
}
// ok, ok, default user exists too!
return true;
}
/** @return array<string,string> */
function checkStep2(): array {
$conf = is_writable(join_path(DATA_PATH, 'config.php'));
$bd = Minz_Session::paramString('bd_type') != '';
$conn = Minz_Session::paramString('bd_error') == '';
return [
'bd' => $bd ? 'ok' : 'ko',
'conn' => $conn ? 'ok' : 'ko',
'conf' => $conf ? 'ok' : 'ko',
'all' => $bd && $conn && $conf ? 'ok' : 'ko',
];
}
/** @return array<string,string> */
function checkStep3(): array {
$conf = Minz_Session::paramString('default_user') != '';
$form = Minz_Session::paramString('auth_type') != '';
$defaultUser = empty($_POST['default_user']) ? null : $_POST['default_user'];
if ($defaultUser === null) {
$defaultUser = Minz_Session::paramString('default_user') == '' ? '' : Minz_Session::paramString('default_user');
}
$data = is_writable(join_path(USERS_PATH, $defaultUser, 'config.php'));
return [
'conf' => $conf ? 'ok' : 'ko',
'form' => $form ? 'ok' : 'ko',
'data' => $data ? 'ok' : 'ko',
'all' => $conf && $form && $data ? 'ok' : 'ko',
];
}
/* select language */
function printStep0(): void {
$actual = Minz_Translate::language();
$languages = Minz_Translate::availableLanguages();
$s0 = checkStep0();
?>
<?php if ($s0['all'] === 'ok') { ?>
<p class="alert alert-success"><span class="alert-head"><?= _t('gen.short.ok') ?></span> <?= _t('install.language.defined') ?></p>
<?php } elseif (!empty($_POST) && $s0['sessionWorking'] !== 'ok') { ?>
<p class="alert alert-error"><span class="alert-head"><?= _t('gen.short.damn') ?></span> <?= _t('install.session.nok') ?></p>
<?php } ?>
<div class="form-group">
<label class="group-name"><?= _t('index.about') ?></label>
<div class="group-controls">
<?= _t('index.about.freshrss_description') ?>
</div>
</div>
<div class="form-group">
<label class="group-name"><?= _t('index.about.project_website') ?></label>
<div class="group-controls">
<a href="<?= FRESHRSS_WEBSITE ?>" target="_blank"><?= FRESHRSS_WEBSITE ?></a>
</div>
</div>
<div class="form-group">
<label class="group-name"><?= _t('index.about.documentation') ?></label>
<div class="group-controls">
<a href="<?= FRESHRSS_WIKI ?>" target="_blank"><?= FRESHRSS_WIKI ?></a>
</div>
</div>
<div class="form-group">
<label class="group-name"><?= _t('index.about.version') ?></label>
<div class="group-controls">
<?= FRESHRSS_VERSION ?>
</div>
</div>
<h2><?= _t('install.language.choose') ?></h2>
<form action="index.php?step=0" method="post">
<div class="form-group">
<label class="group-name" for="language"><?= _t('install.language') ?></label>
<div class="group-controls">
<select name="language" id="language" tabindex="1" >
<?php foreach ($languages as $lang) { ?>
<option value="<?= $lang ?>"<?= $actual == $lang ? ' selected="selected"' : '' ?>>
<?= _t('gen.lang.' . $lang) ?>
</option>
<?php } ?>
</select>
</div>
</div>
<div class="form-group form-actions">
<div class="group-controls">
<button type="submit" class="btn btn-important" tabindex="2" ><?= _t('gen.action.submit') ?></button>
<?php if ($s0['all'] == 'ok') { ?>
<a class="next-step" href="?step=1" tabindex="4" ><?= _t('install.action.next_step') ?></a>
<?php } ?>
</div>
</div>
</form>
<?php
}
/**
* Alert box template
* @param array<string> $messageParams
* */
function printStep1Template(string $key, string $value, array $messageParams = []): void {
if ('ok' === $value) {
$message = _t("install.check.{$key}.ok", ...$messageParams);
?><p class="alert alert-success"><span class="alert-head"><?= _t('gen.short.ok') ?></span> <?= $message ?></p><?php
} else {
$message = _t("install.check.{$key}.nok", ...$messageParams);
?><p class="alert alert-error"><span class="alert-head"><?= _t('gen.short.damn') ?></span> <?= $message ?></p><?php
}
}
function getProcessUsername(): string {
if (function_exists('posix_getpwuid') && function_exists('posix_geteuid')) {
$processUser = posix_getpwuid(posix_geteuid()) ?: [];
if (!empty($processUser['name'])) {
return $processUser['name'];
}
}
if (function_exists('exec')) {
exec('whoami', $output);
if (!empty($output[0])) {
return $output[0];
}
}
return _t('install.check.unknown_process_username');
}
// @todo refactor this view with the check_install action
/* check system environment */
function printStep1(): void {
$res = checkRequirements();
$processUsername = getProcessUsername();
?>
<h2><?= _t('admin.check_install.php') ?></h2>
<noscript><p class="alert alert-warn"><span class="alert-head"><?= _t('gen.short.attention') ?></span> <?= _t('install.javascript_is_better') ?></p></noscript>
<?php
$version = function_exists('curl_version') ? curl_version() : [];
printStep1Template('php', $res['php'], [PHP_VERSION, FRESHRSS_MIN_PHP_VERSION]);
printStep1Template('pdo', $res['pdo']);
printStep1Template('curl', $res['curl'], [$version['version'] ?? '']);
printStep1Template('json', $res['json']);
printStep1Template('pcre', $res['pcre']);
printStep1Template('ctype', $res['ctype']);
printStep1Template('dom', $res['dom']);
printStep1Template('xml', $res['xml']);
printStep1Template('mbstring', $res['mbstring']);
printStep1Template('fileinfo', $res['fileinfo']);
?>
<h2><?= _t('admin.check_install.files') ?></h2>
<?php
printStep1Template('data', $res['data'], [DATA_PATH, $processUsername]);
printStep1Template('cache', $res['cache'], [CACHE_PATH, $processUsername]);
printStep1Template('tmp', $res['tmp'], [TMP_PATH, $processUsername]);
printStep1Template('users', $res['users'], [USERS_PATH, $processUsername]);
printStep1Template('favicons', $res['favicons'], [DATA_PATH . '/favicons', $processUsername]);
?>
<?php if (freshrss_already_installed() && $res['all'] == 'ok') { ?>
<p class="alert alert-warn"><span class="alert-head"><?= _t('gen.short.attention') ?></span> <?= _t('install.check.already_installed') ?></p>
<div class="form-group form-actions">
<div class="group-controls">
<form action="index.php?step=1" method="post">
<input type="hidden" name="freshrss-keep-install" value="1" />
<button type="submit" class="btn btn-important" tabindex="1"><?= _t('install.action.keep_install') ?></button>
<a class="btn btn-attention confirm" data-str-confirm="<?= _t('install.js.confirm_reinstall') ?>"
href="?step=2" tabindex="2"><?= _t('install.action.reinstall') ?></a>
</form>
</div>
</div>
<?php } elseif ($res['all'] == 'ok') { ?>
<div class="form-group form-actions">
<div class="group-controls">
<a class="btn btn-important" href="?step=2" tabindex="1"><?= _t('install.action.next_step') ?></a>
</div>
</div>
<?php } else { ?>
<p class="alert alert-error"><?= _t('install.action.fix_errors_before') ?></p>
<div class="form-group form-actions">
<div class="group-controls">
<a id="actualize" class="btn" href="./index.php?step=1" title="<?= _t('install.check.reload') ?>" tabindex="1">
<img class="icon" src="../themes/icons/refresh.svg" alt="🔃" loading="lazy" />
</a>
</div>
</div>
<?php } ?>
<?php
}
/**
* Select database & configuration
* @throws Minz_ConfigurationNamespaceException
*/
function printStep2(): void {
$system_default_config = FreshRSS_SystemConfiguration::get('default_system');
$s2 = checkStep2();
if ($s2['all'] == 'ok') { ?>
<p class="alert alert-success"><span class="alert-head"><?= _t('gen.short.ok') ?></span> <?= _t('install.bdd.conf.ok') ?></p>
<?php } elseif ($s2['conn'] == 'ko') { ?>
<p class="alert alert-error"><span class="alert-head"><?= _t('gen.short.damn') ?></span> <?= _t('install.bdd.conf.ko'),
(empty($_SESSION['bd_error']) ? '' : ' : ' . $_SESSION['bd_error']) ?></p>
<?php } ?>
<h2><?= _t('install.bdd.conf') ?></h2>
<form action="index.php?step=2" method="post" autocomplete="off">
<div class="form-group">
<label class="group-name" for="type"><?= _t('install.bdd.type') ?></label>
<div class="group-controls">
<select name="type" id="type" tabindex="1">
<?php if (extension_loaded('pdo_sqlite')) {?>
<option value="sqlite"
<?= isset($_SESSION['bd_type']) && $_SESSION['bd_type'] === 'sqlite' ? 'selected="selected"' : '' ?>>
SQLite
</option>
<?php }?>
<?php if (extension_loaded('pdo_mysql')) {?>
<option value="mysql"
<?= isset($_SESSION['bd_type']) && $_SESSION['bd_type'] === 'mysql' ? 'selected="selected"' : '' ?>>
MySQL / MariaDB
</option>
<?php }?>
<?php if (extension_loaded('pdo_pgsql')) {?>
<option value="pgsql"
<?= isset($_SESSION['bd_type']) && $_SESSION['bd_type'] === 'pgsql' ? 'selected="selected"' : '' ?>>
PostgreSQL
</option>
<?php }?>
</select>
</div>
</div>
<div id="mysql">
<div class="form-group">
<label class="group-name" for="host"><?= _t('install.bdd.host') ?></label>
<div class="group-controls">
<input type="text" id="host" name="host" pattern="[0-9A-Z/a-z_.\-]{1,64}(:[0-9]{2,5})?" value="<?=
$_SESSION['bd_host'] ?? $system_default_config->db['host'] ?? '' ?>" tabindex="2" />
</div>
</div>
<div class="form-group">
<label class="group-name" for="user"><?= _t('install.bdd.username') ?></label>
<div class="group-controls">
<input type="text" id="user" name="user" maxlength="64" pattern="[0-9A-Za-z@_.\-]{1,64}" value="<?=
$_SESSION['bd_user'] ?? '' ?>" tabindex="3" />
</div>
</div>
<div class="form-group">
<label class="group-name" for="pass"><?= _t('install.bdd.password') ?></label>
<div class="group-controls">
<div class="stick">
<input type="password" id="pass" name="pass" value="<?=
$_SESSION['bd_password'] ?? '' ?>" tabindex="4" autocomplete="off" />
<a class="btn toggle-password" data-toggle="pass" tabindex="5"><?= FreshRSS_Themes::icon('key') ?></a>
</div>
</div>
</div>
<div class="form-group">
<label class="group-name" for="base"><?= _t('install.bdd') ?></label>
<div class="group-controls">
<input type="text" id="base" name="base" maxlength="64" pattern="[0-9A-Za-z_\-]{1,64}" value="<?=
$_SESSION['bd_base'] ?? '' ?>" tabindex="6" />
</div>
</div>
<div class="form-group">
<label class="group-name" for="prefix"><?= _t('install.bdd.prefix') ?></label>
<div class="group-controls">
<input type="text" id="prefix" name="prefix" maxlength="16" pattern="[0-9A-Za-z_]{1,16}" value="<?=
$_SESSION['bd_prefix'] ?? $system_default_config->db['prefix'] ?? '' ?>" tabindex="7" />
</div>
</div>
</div>
<div class="form-group form-actions">
<div class="group-controls">
<button type="submit" class="btn btn-important" tabindex="8" ><?= _t('gen.action.submit') ?></button>
<button type="reset" class="btn" tabindex="9" ><?= _t('gen.action.cancel') ?></button>
<?php if ($s2['all'] == 'ok') { ?>
<a class="next-step" href="?step=3" tabindex="10" ><?= _t('install.action.next_step') ?></a>
<?php } ?>
</div>
</div>
</form>
<?php
}
function no_auth(string $auth_type): bool {
return !in_array($auth_type, ['form', 'http_auth', 'none'], true);
}
/* Create default user */
function printStep3(): void {
$auth_type = $_SESSION['auth_type'] ?? '';
$s3 = checkStep3();
if ($s3['all'] == 'ok') { ?>
<p class="alert alert-success"><span class="alert-head"><?= _t('gen.short.ok') ?></span> <?= _t('install.conf.ok') ?></p>
<?php } elseif (!empty($_POST)) { ?>
<p class="alert alert-error"><?= _t('install.fix_errors_before') ?></p>
<?php } ?>
<h2><?= _t('install.conf') ?></h2>
<form action="index.php?step=3" method="post">
<div class="form-group">
<label class="group-name" for="default_user"><?= _t('install.default_user') ?></label>
<div class="group-controls">
<input type="text" id="default_user" name="default_user" autocomplete="username" required="required" size="16"
pattern="<?= FreshRSS_user_Controller::USERNAME_PATTERN ?>" value="<?= $_SESSION['default_user'] ?? '' ?>"
placeholder="<?= httpAuthUser(false) == '' ? 'alice' : httpAuthUser(false) ?>" tabindex="1" />
<p class="help"><?= _i('help') ?> <?= _t('install.default_user.max_char') ?></p>
</div>
</div>
<div class="form-group">
<label class="group-name" for="auth_type"><?= _t('install.auth.type') ?></label>
<div class="group-controls">
<select id="auth_type" name="auth_type" required="required" tabindex="2">
<option value="form"<?= $auth_type === 'form' || (no_auth($auth_type) && cryptAvailable()) ? ' selected="selected"' : '',
cryptAvailable() ? '' : ' disabled="disabled"' ?>><?= _t('install.auth.form') ?></option>
<option value="http_auth"<?= $auth_type === 'http_auth' ? ' selected="selected"' : '',
httpAuthUser(false) == '' ? ' disabled="disabled"' : '' ?>>
<?= _t('install.auth.http') ?> (REMOTE_USER = '<?= httpAuthUser(false) ?>')</option>
<option value="none"<?= $auth_type === 'none' || (no_auth($auth_type) && !cryptAvailable()) ? ' selected="selected"' : ''
?>><?= _t('install.auth.none') ?></option>
</select>
</div>
</div>
<div class="form-group">
<label class="group-name" for="passwordPlain"><?= _t('install.auth.password_form') ?></label>
<div class="group-controls">
<div class="stick">
<input type="password" id="passwordPlain" name="passwordPlain" pattern=".{7,}"
autocomplete="off" <?= $auth_type === 'form' ? ' required="required"' : '' ?> tabindex="3" />
<button type="button" class="btn toggle-password" data-toggle="passwordPlain" tabindex="4"><?= FreshRSS_Themes::icon('key') ?></button>
</div>
<p class="help"><?= _i('help') ?> <?= _t('install.auth.password_format') ?></p>
<noscript><b><?= _t('gen.js.should_be_activated') ?></b></noscript>
</div>
</div>
<div class="form-group form-actions">
<div class="group-controls">
<button type="submit" class="btn btn-important" tabindex="5" ><?= _t('gen.action.submit') ?></button>
<button type="reset" class="btn" tabindex="6" ><?= _t('gen.action.cancel') ?></button>
<?php if ($s3['all'] == 'ok') { ?>
<a class="next-step" href="?step=4" tabindex="7" ><?= _t('install.action.next_step') ?></a>
<?php } ?>
</div>
</div>
</form>
<?php
}
/* congrats. Installation successful completed */
function printStep4(): void {
?>
<p class="alert alert-success"><span class="alert-head"><?= _t('install.congratulations') ?></span> <?= _t('install.ok') ?></p>
<div class="form-group form-actions">
<div class="group-controls">
<a class="btn btn-important" href="?step=5" tabindex="1"><?= _t('install.action.finish') ?></a>
</div>
</div>
<?php
}
/* failed */
function printStep5(): void {
?>
<p class="alert alert-error">
<span class="alert-head"><?= _t('gen.short.damn') ?></span>
<?= _t('install.missing_applied_migrations', DATA_PATH . '/applied_migrations.txt') ?>
</p>
<?php
}
initTranslate();
checkStep();
switch (STEP) {
case 0:
default:
saveLanguage();
break;
case 1:
saveStep1();
break;
case 2:
saveStep2();
break;
case 3:
saveStep3();
break;
case 4:
break;
case 5:
if (setupMigrations()) {
header('Location: index.php');
}
break;
}
?>
<!DOCTYPE html>
<html <?php
if (_t('gen.dir') === 'rtl') {
echo ' dir="rtl" class="rtl"';
}
?>>
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="initial-scale=1.0" />
<script id="jsonVars" type="application/json">{}</script>
<title><?= _t('install.title') ?>: <?= _t('install.step', STEP + 1) ?></title>
<link rel="stylesheet" href="../themes/base-theme/frss.css?<?= @filemtime(PUBLIC_PATH . '/themes/base-theme/frss.css') ?>" />
<link rel="stylesheet" href="../themes/Origine/origine.css?<?= @filemtime(PUBLIC_PATH . '/themes/Origine/origine.css') ?>" />
<link rel="shortcut icon" id="favicon" type="image/x-icon" sizes="16x16 64x64" href="../favicon.ico" />
<link rel="icon msapplication-TileImage apple-touch-icon" type="image/png" sizes="256x256" href="../themes/icons/favicon-256.png" />
<link rel="apple-touch-icon" href="../themes/icons/apple-touch-icon.png" />
<meta name="apple-mobile-web-app-capable" content="yes" />
<meta name="apple-mobile-web-app-status-bar-style" content="black" />
<meta name="apple-mobile-web-app-title" content="FreshRSS">
<meta name="robots" content="noindex,nofollow" />
</head>
<body>
<header class="header">
<div class="item title">
<div id="logo-wrapper">
<a href="./">
<img class="logo" src="../themes/icons/FreshRSS-logo.svg" alt="" loading="lazy">
</a>
</div>
</div>
<div class="item"></div>
<div class="item configure">
<a class="btn only-mobile" href="#aside"><?= _i('view-normal') ?></a>
</div>
</header>
<div id="global">
<nav class="nav nav-list aside" id="aside">
<a class="toggle_aside" href="#close"><img class="icon" src="../themes/icons/close.svg" loading="lazy" alt="❌"></a>
<ul>
<li class="item nav-section">
<div class="nav-header"><?= _t('install.steps') ?></div>
<ol>
<li class="item<?= STEP == 0 ? ' active' : '' ?>">
<a href="?step=0" title="<?= _t('install.step', 0) ?>: <?= _t('install.language') ?>"><?= _t('install.language') ?></a>
</li>
<li class="item<?= STEP == 1 ? ' active' : '' ?>">
<?php if (STEP > 0) {?>
<a href="?step=1" title="<?= _t('install.step', 1) ?>: <?= _t('install.check') ?>"><?= _t('install.check') ?></a>
<?php } else { ?>
<span><?= _t('install.check') ?></span>
<?php } ?>
</li>
<li class="item<?= STEP == 2 ? ' active' : '' ?>">
<?php if (STEP > 1) {?>
<a href="?step=2" title="<?= _t('install.step', 2) ?>: <?= _t('install.bdd.conf') ?>"><?= _t('install.bdd.conf') ?></a>
<?php } else { ?>
<span><?= _t('install.bdd.conf') ?></span>
<?php } ?>
</li>
<li class="item<?= STEP == 3 ? ' active' : '' ?>">
<?php if (STEP > 2) {?>
<a href="?step=3" title="<?= _t('install.step', 3) ?>: <?= _t('install.conf') ?>"><?= _t('install.conf') ?></a>
<?php } else { ?>
<span><?= _t('install.conf') ?></span>
<?php } ?>
</li>
<li class="item<?= STEP == 4 ? ' active' : '' ?>">
<?php if (STEP > 3) {?>
<a href="?step=4" title="<?= _t('install.step', 4) ?>: <?= _t('install.this_is_the_end') ?>"><?= _t('install.this_is_the_end') ?></a>
<?php } else { ?>
<span><?= _t('install.this_is_the_end') ?></span>
<?php } ?>
</li>
</ol>
</li>
</ul>
</nav>
<a class="close-aside" href="#close">❌</a>
<main class="post">
<h1><?= _t('install.title') ?>: <?= _t('install.step', STEP + 1) ?></h1>
<?php
switch (STEP) {
case 0:
default:
printStep0();
break;
case 1:
printStep1();
break;
case 2:
printStep2();
break;
case 3:
printStep3();
break;
case 4:
printStep4();
break;
case 5:
printStep5();
break;
}
?>
</main>
</div>
<script src="../scripts/install.js?<?= @filemtime(PUBLIC_PATH . '/scripts/install.js') ?>"></script>
</body>
</html>
wget 'https://sme10.lists2.roe3.org/FreshRSS/app/shares.php'
<?php
declare(strict_types=1);
/*
* This is a configuration file. You shouldn’t modify it unless you know what
* you are doing. If you want to add a share type, this is where you need to do
* it.
*
* For each share there is different configuration options. Here is the description
* of those options:
* - 'deprecated' (optional) is a boolean. Default: 'false'.
* 'true', if the sharing service is planned to remove in the future.
* Add more information into the documentation center.
* - 'HTMLtag' (optional). If it is 'button' then an HTML <button> is used,
* else an <a href=""> is used. Add a click event in main.js additionally.
* - 'url' is a mandatory option. It is a string representing the share URL. It
* supports 4 different placeholders for custom data. The ~URL~ placeholder
* represents the URL of the system used to share, it is configured by the
* user. The ~LINK~ placeholder represents the link of the shared article.
* The ~TITLE~ placeholder represents the title of the shared article. The
* ~ID~ placeholder represents the id of the shared article (only useful
* for internal use)
* - 'transform' is an array of transformation to apply on links and titles
* - 'help' is a URL to a help page (mandatory for form = 'advanced')
* - 'form' is the type of form to display during configuration. It’s either
* 'simple' or 'advanced'. 'simple' is used when only the name is configurable,
* 'advanced' is used when the name and the location are configurable.
* - 'method' is the HTTP method (POST or GET) used to share a link.
*/
return [
'archiveORG' => [
'url' => 'https://web.archive.org/save/~LINK~',
'transform' => [],
'help' => 'https://web.archive.org',
'form' => 'simple',
'method' => 'GET',
],
'archiveIS' => [
'url' => 'https://archive.is/submit/?url=~LINK~',
'transform' => [],
'help' => 'https://archive.is/',
'form' => 'simple',
'method' => 'GET',
],
'archivePH' => [
'url' => 'https://archive.ph/submit/?url=~LINK~',
'transform' => [],
'help' => 'https://archive.ph/',
'form' => 'simple',
'method' => 'GET',
],
'buffer' => [
'url' => 'https://publish.buffer.com/compose?url=~LINK~&text=~TITLE~',
'transform' => ['rawurlencode'],
'help' => 'https://support.buffer.com/hc/en-us/articles/360035587394-Scheduling-posts',
'form' => 'simple',
'method' => 'GET',
],
'clipboard' => [
'HTMLtag' => 'button',
'url' => '~LINK~',
'transform' => [],
'form' => 'simple',
'method' => 'GET',
],
'diaspora' => [
'url' => '~URL~/bookmarklet?url=~LINK~&title=~TITLE~',
'transform' => ['rawurlencode'],
'help' => 'https://diasporafoundation.org/',
'form' => 'advanced',
'method' => 'GET',
],
'email' => [
'url' => 'mailto:?subject=~TITLE~&body=~LINK~',
'transform' => ['rawurlencode'],
'form' => 'simple',
'method' => 'GET',
],
'email-webmail-firefox-fix' => [ // see https://github.com/FreshRSS/FreshRSS/issues/2666
'url' => 'mailto:?subject=~TITLE~&body=~LINK~',
'transform' => ['rawurlencode'],
'form' => 'simple',
'method' => 'GET',
],
'facebook' => [
'url' => 'https://www.facebook.com/sharer.php?u=~LINK~&t=~TITLE~',
'transform' => ['rawurlencode'],
'form' => 'simple',
'method' => 'GET',
],
'gnusocial' => [
'url' => '~URL~/notice/new?content=~TITLE~%20~LINK~',
'transform' => ['urlencode'],
'help' => 'https://gnu.io/social/',
'form' => 'advanced',
'method' => 'GET',
],
'jdh' => [
'url' => 'https://www.journalduhacker.net/stories/new?url=~LINK~&title=~TITLE~',
'transform' => ['rawurlencode'],
'form' => 'simple',
'method' => 'GET',
],
'Known' => [
'url' => '~URL~/share?share_url=~LINK~&share_title=~TITLE~',
'transform' => ['rawurlencode'],
'help' => 'https://withknown.com/',
'form' => 'advanced',
'method' => 'GET',
],
'lemmy' => [
'url' => '~URL~/create_post?url=~LINK~&title=~TITLE~',
'transform' => ['rawurlencode'],
'help' => 'https://join-lemmy.org/',
'form' => 'advanced',
'method' => 'GET',
],
'linkding' => [
'url' => '~URL~/bookmarks/new?url=~LINK~&title=~TITLE~&auto_close',
'transform' => ['rawurlencode'],
'help' => 'https://github.com/sissbruecker/linkding/blob/master/docs/how-to.md',
'form' => 'advanced',
'method' => 'GET',
],
'linkedin' => [
'url' => 'https://www.linkedin.com/shareArticle?url=~LINK~&title=~TITLE~&source=FreshRSS',
'transform' => ['rawurlencode'],
'form' => 'simple',
'method' => 'GET',
],
'mastodon' => [
'url' => '~URL~/share?title=~TITLE~&url=~LINK~',
'transform' => ['rawurlencode'],
'help' => 'https://joinmastodon.org/',
'form' => 'advanced',
'method' => 'GET',
],
'movim' => [
'url' => '~URL~/?share/~LINK~',
'transform' => ['urlencode'],
'help' => 'https://movim.eu/',
'form' => 'advanced',
'method' => 'GET',
],
'omnivore' => [
'url' => '~URL~/api/save?url=~LINK~',
'transform' => ['urlencode'],
'help' => 'https://omnivore.app/',
'form' => 'advanced',
'method' => 'GET',
],
'pinboard' => [
'url' => 'https://pinboard.in/add?next=same&url=~LINK~&title=~TITLE~',
'transform' => ['urlencode'],
'help' => 'https://pinboard.in/api/',
'form' => 'simple',
'method' => 'GET',
],
'pinterest' => [
'url' => 'https://pinterest.com/pin/create/button/?url=~LINK~',
'transform' => ['rawurlencode'],
'help' => 'https://pinterest.com/',
'form' => 'simple',
'method' => 'GET',
],
'pocket' => [
'url' => 'https://getpocket.com/save?url=~LINK~&title=~TITLE~',
'transform' => ['rawurlencode'],
'form' => 'simple',
'method' => 'GET',
],
'print' => [
'HTMLtag' => 'button',
'url' => '#',
'transform' => [],
'form' => 'simple',
'method' => 'GET',
],
'raindrop' => [
'url' => 'https://app.raindrop.io/add?link=~LINK~&title=~TITLE~',
'transform' => ['rawurlencode'],
'form' => 'simple',
'method' => 'GET',
],
'reddit' => [
'url' => 'https://www.reddit.com/submit?url=~LINK~',
'transform' => ['rawurlencode'],
'help' => 'https://www.reddit.com/wiki/submitting?v=c2ae883a-04b9-11e4-a68c-12313b01a1fc',
'form' => 'simple',
'method' => 'GET',
],
'shaarli' => [
'url' => '~URL~?post=~LINK~&title=~TITLE~&source=FreshRSS',
'transform' => ['rawurlencode'],
'help' => 'http://sebsauvage.net/wiki/doku.php?id=php:shaarli',
'form' => 'advanced',
'method' => 'GET',
],
'twitter' => [
'url' => 'https://twitter.com/share?url=~LINK~&text=~TITLE~',
'transform' => ['rawurlencode'],
'form' => 'simple',
'method' => 'GET',
],
'wallabag' => [
'url' => '~URL~?action=add&url=~LINK~',
'transform' => ['rawurlencode'],
'help' => 'http://www.wallabag.org/',
'form' => 'advanced',
'method' => 'GET',
],
'wallabagv2' => [
'url' => '~URL~/bookmarklet?url=~LINK~',
'transform' => ['rawurlencode'],
'help' => 'http://www.wallabag.org/',
'form' => 'advanced',
'method' => 'GET',
],
'web-sharing-api' => [
'HTMLtag' => 'button',
'url' => '~LINK~',
'transform' => [],
'form' => 'simple',
'method' => 'GET',
],
'whatsapp' => [
'url' => 'https://wa.me/?text=~TITLE~ | ~LINK~',
'transform' => ['rawurlencode'],
'help' => 'https://faq.whatsapp.com/iphone/how-to-link-to-whatsapp-from-a-different-app/?lang=en',
'form' => 'simple',
'method' => 'GET',
],
'xing' => [
'url' => 'https://www.xing.com/spi/shares/new?url=~LINK~',
'transform' => ['rawurlencode'],
'help' => 'https://dev.xing.com/plugins/share_button/docs',
'form' => 'simple',
'method' => 'GET',
],
];