Performance: Consolidated multiple preg_replace calls into a single, efficient regex pass. Support: Added handling for YouTube Shorts, Music, and Embed links. Features: Enhanced link detection to preserve video timestamps (?t=...). Robustness: Improved URL sanitization to prevent broken links (double slashes) and added rel="noopener" for security. UI: Cleaned up the footer note by displaying a clickable hostname instead of the full URL.
127 lines
5.1 KiB
PHP
127 lines
5.1 KiB
PHP
<?php
|
|
/*
|
|
* Name: invidious
|
|
* Description: Replaces links to youtube.com to an invidious instance in all displays of postings on a node.
|
|
* Version: 0.7
|
|
* Author: Matthias Ebers <https://loma.ml/profile/feb>
|
|
* Author: Michael Vogel <https://pirati.ca/profile/heluecht>
|
|
* Status:
|
|
*/
|
|
|
|
use Friendica\Core\Hook;
|
|
use Friendica\Core\Renderer;
|
|
use Friendica\DI;
|
|
|
|
CONST INVIDIOUS_DEFAULT = 'https://invidio.us';
|
|
|
|
function invidious_install()
|
|
{
|
|
Hook::register('prepare_body_final', __FILE__, 'invidious_render');
|
|
Hook::register('addon_settings', __FILE__, 'invidious_settings');
|
|
Hook::register('addon_settings_post', __FILE__, 'invidious_settings_post');
|
|
}
|
|
|
|
/* Handle the send data from the admin settings
|
|
*/
|
|
function invidious_addon_admin_post()
|
|
{
|
|
// Sanitize and validate the input as a valid URL
|
|
$url = filter_var(trim($_POST['invidiousserver'], " \n\r\t\v\x00/"), FILTER_VALIDATE_URL);
|
|
if ($url !== false) {
|
|
DI::config()->set('invidious', 'server', $url);
|
|
}
|
|
}
|
|
|
|
/* Hook into the admin settings to let the admin choose an
|
|
* invidious server to use for the replacement.
|
|
*/
|
|
function invidious_addon_admin(string &$o)
|
|
{
|
|
$invidiousserver = DI::config()->get('invidious', 'server', INVIDIOUS_DEFAULT);
|
|
$t = Renderer::getMarkupTemplate('admin.tpl', 'addon/invidious/');
|
|
$o = Renderer::replaceMacros($t, [
|
|
'$settingdescription' => DI::l10n()->t('Which Invidious server shall be used for the replacements in the post bodies? Use the URL with servername and protocol. See %s for a list of available public Invidious servers.', 'https://redirect.invidious.io'),
|
|
'$invidiousserver' => ['invidiousserver', DI::l10n()->t('Invidious server'), $invidiousserver, DI::l10n()->t('See %s for a list of available Invidious servers.', '<a href="https://api.invidious.io/">https://api.invidious.io/</a>')],
|
|
'$submit' => DI::l10n()->t('Save Settings'),
|
|
]);
|
|
}
|
|
|
|
function invidious_settings(array &$data)
|
|
{
|
|
if (!DI::userSession()->getLocalUserId()) {
|
|
return;
|
|
}
|
|
|
|
$enabled = DI::pConfig()->get(DI::userSession()->getLocalUserId(), 'invidious', 'enabled');
|
|
$server = DI::pConfig()->get(DI::userSession()->getLocalUserId(), 'invidious', 'server', DI::config()->get('invidious', 'server', INVIDIOUS_DEFAULT));
|
|
|
|
$t = Renderer::getMarkupTemplate('settings.tpl', 'addon/invidious/');
|
|
$html = Renderer::replaceMacros($t, [
|
|
'$enabled' => ['enabled', DI::l10n()->t('Replace Youtube links with links to an Invidious server'), $enabled, DI::l10n()->t('If enabled, Youtube links are replaced with the links to the specified Invidious server.')],
|
|
'$server' => ['server', DI::l10n()->t('Invidious server'), $server, DI::l10n()->t('See %s for a list of available Invidious servers.', '<a href="https://api.invidious.io/">https://api.invidious.io/</a>')],
|
|
]);
|
|
|
|
$data = [
|
|
'addon' => 'invidious',
|
|
'title' => DI::l10n()->t('Invidious Settings'),
|
|
'html' => $html,
|
|
];
|
|
}
|
|
|
|
function invidious_settings_post(array &$b)
|
|
{
|
|
if (!DI::userSession()->getLocalUserId() || empty($_POST['invidious-submit'])) {
|
|
return;
|
|
}
|
|
|
|
DI::pConfig()->set(DI::userSession()->getLocalUserId(), 'invidious', 'enabled', (bool)$_POST['enabled']);
|
|
|
|
$server = trim($_POST['server'], " \n\r\t\v\x00/");
|
|
// Sanitize and validate the server URL before saving
|
|
$validatedServer = filter_var($server, FILTER_VALIDATE_URL);
|
|
if ($validatedServer !== false && $validatedServer != DI::config()->get('invidious', 'server', INVIDIOUS_DEFAULT)) {
|
|
DI::pConfig()->set(DI::userSession()->getLocalUserId(), 'invidious', 'server', $validatedServer);
|
|
} else {
|
|
DI::pConfig()->delete(DI::userSession()->getLocalUserId(), 'invidious', 'server');
|
|
}
|
|
}
|
|
|
|
/*
|
|
* replace "youtube.com" with the chosen Invidious instance
|
|
*/
|
|
function invidious_render(array &$b)
|
|
{
|
|
if (!DI::userSession()->getLocalUserId() || !DI::pConfig()->get(DI::userSession()->getLocalUserId(), 'invidious', 'enabled')) {
|
|
return;
|
|
}
|
|
|
|
$original = $b['html'];
|
|
|
|
$server = DI::pConfig()->get(DI::userSession()->getLocalUserId(), 'invidious', 'server', DI::config()->get('invidious', 'server', INVIDIOUS_DEFAULT));
|
|
$server = rtrim($server, '/');
|
|
|
|
$pattern = "~https?://(?:(?:www\.|m\.|music\.)?youtube\.com/(?:watch\?v=|embed/|shorts/)|youtu\.be/)([a-zA-Z0-9_-]{11})([^ \n\r\t\v\x00\"<>]*[?&][tT]=[^ \n\r\t\v\x00\"<>]*)?~ism";
|
|
|
|
$b['html'] = preg_replace_callback($pattern, function($matches) use ($server) {
|
|
$videoId = $matches[1];
|
|
$params = $matches[2] ?? '';
|
|
|
|
if (!empty($params)) {
|
|
$params = str_replace('&', '&', $params);
|
|
if (strpos($params, '?') === false && strpos($params, '&') === 0) {
|
|
$params = '?' . ltrim($params, '&');
|
|
}
|
|
}
|
|
|
|
return $server . '/watch?v=' . $videoId . $params;
|
|
}, $b['html']);
|
|
|
|
if ($original != $b['html']) {
|
|
$displayHost = parse_url($server, PHP_URL_HOST);
|
|
$serverLink = '<a href="' . $server . '" target="_blank" rel="noopener noreferrer">' . $displayHost . '</a>';
|
|
|
|
$b['html'] .= '<hr><p><small class="invidious-note">' .
|
|
DI::l10n()->t('(Invidious addon enabled: YouTube links via %s)', $serverLink) .
|
|
'</small></p>';
|
|
}
|
|
}
|