<?php declare(strict_types = 1); // {{{ License // This file is part of GNU social - https://www.gnu.org/software/social // // GNU social is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // GNU social is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with GNU social. If not, see <http://www.gnu.org/licenses/>. // }}} /** * OEmbed and OpenGraph implementation for GNU social * * @package GNUsocial * * @author Mikael Nordfeldth * @author Stephen Paul Weber * @author hannes * @author Mikael Nordfeldth * @author Miguel Dantas * @author Hugo Sales <hugo@hsal.es> * @author Diogo Peralta Cordeiro <mail@diogo.site> * @copyright 2014-2021 Free Software Foundation, Inc http://www.fsf.org * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ namespace Plugin\Embed; use App\Core\Cache; use App\Core\DB\DB; use App\Core\Event; use App\Core\GSFile; use App\Core\HTTPClient; use function App\Core\I18n\_m; use App\Core\Log; use App\Core\Modules\Plugin; use App\Core\Router\RouteLoader; use App\Core\Router\Router; use App\Entity\Note; use App\Util\Common; use App\Util\Exception\ClientException; use App\Util\Exception\DuplicateFoundException; use App\Util\Exception\NotFoundException; use App\Util\Exception\ServerException; use App\Util\Formatting; use App\Util\TemporaryFile; use Component\Attachment\Entity\Attachment; use Component\Link\Entity\Link; use Embed\Embed as LibEmbed; use Exception; use Symfony\Component\HttpFoundation\Request; use Symfony\Contracts\HttpClient\Exception\ClientExceptionInterface; use Symfony\Contracts\HttpClient\Exception\RedirectionExceptionInterface; use Symfony\Contracts\HttpClient\Exception\ServerExceptionInterface; use Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface; /** * Base class for the Embed plugin that does most of the heavy lifting to get * and display representations for remote content. * * @copyright 2014-2021 Free Software Foundation, Inc http://www.fsf.org * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ class Embed extends Plugin { public function version(): string { return '3.0.1'; } /** * Settings which can be set in social.local.yaml * WARNING, these are _regexps_ (slashes added later). Always escape your dots and end ('$') your strings */ public bool $check_whitelist = false; public bool $check_blacklist = false; public array $domain_whitelist = [ // hostname '.*', // Default to allowing any host ]; public array $domain_blacklist = []; // Whether to maintain a copy of the original media or only a thumbnail of it public bool $store_image = true; public ?int $thumbnail_width; public ?int $thumbnail_height; public ?int $max_size; public ?bool $smart_crop; // TODO: storeThumbs setting private function getMaxFileSize(): int { return min(Common::config('plugin_embed', 'max_file_size'), Common::config('attachments', 'file_quota')); } /** * This code executes when GNU social creates the page routing, and we hook * on this event to add our action handler for Embed. * * @param RouteLoader $m the router that was initialized * * @throws Exception */ public function onAddRoute(RouteLoader $m): bool { $m->connect('oembed', 'main/oembed', Controller\OEmbed::class); return Event::next; } /** * Insert oembed and opengraph tags in all HTML head elements */ public function onShowHeadElements(Request $request, array &$result): bool { $matches = []; preg_match(',/?([^/]+)/?(.*),', $request->getPathInfo(), $matches); $url = match ($matches[1]) { 'attachment' => "{$matches[1]}/{$matches[2]}", default => null, }; if (\is_null($url)) { foreach (['xml', 'json'] as $format) { $result[] = [ 'link' => [ 'rel' => 'alternate', 'type' => "application/{$format}+oembed", 'href' => Router::url('oembed', ['format' => $format, 'url' => $url]), 'title' => 'oEmbed', ], ]; } } return Event::next; } /** * Show this attachment enhanced with the corresponding Embed data, if available */ public function onViewLink(array $vars, array &$res): bool { $link = $vars['link']; try { $embed = Cache::get( 'attachment-embed-' . $link->getId(), fn () => DB::findOneBy('attachment_embed', ['link_id' => $link->getId()]), ); } catch (DuplicateFoundException $e) { Log::warning($e->getMessage()); return Event::next; } catch (NotFoundException) { Log::debug("Embed doesn't have a representation for the link id={$link->getId()}. Must have been stored before the plugin was enabled."); return Event::next; } $attributes = $embed->getImageHTMLAttributes(); $res[] = Formatting::twigRenderFile( 'embed/embedView.html.twig', ['embed' => $embed, 'attributes' => $attributes, 'link' => $link, 'note' => $vars['note']], ); return Event::stop; } /** * Save embedding information for an Attachment, if applicable. * * @throws DuplicateFoundException */ public function onNewLinkFromNote(Link $link, Note $note): bool { // Only handle text mime $mimetype = $link->getMimetype(); if (\is_null($mimetype) || !(Formatting::startsWith($mimetype, 'text/html') || Formatting::startsWith($mimetype, 'application/xhtml+xml'))) { return Event::next; } // Ignore if already handled $attachment_embed = DB::find('attachment_embed', ['link_id' => $link->getId()]); if (!\is_null($attachment_embed)) { return Event::next; } // If an attachment already exist, do not create an Embed for it. Some other plugin must have done things $attachment_to_link = DB::find('attachment_to_link', ['link_id' => $link->getId()]); if (!\is_null($attachment_to_link)) { $attachment_id = $attachment_to_link->getAttachmentId(); try { $attachment = DB::findOneBy('attachment', ['id' => $attachment_id]); $attachment->livesIncrementAndGet(); return Event::next; } catch (DuplicateFoundException|NotFoundException $e) { Log::error($e->getMessage(), context: [$e]); } } // Create an Embed representation for this URL $embed_data = $this->getEmbedLibMetadata($link->getUrl()); $embed_data['link_id'] = $link->getId(); $img_data = $this->downloadThumbnail($embed_data['thumbnail_url']); switch ($img_data) { case null: // URL isn't usable $embed_data['thumbnail_url'] = null; // no break case false: // Thumbnail isn't acceptable DB::persist($attachment = Attachment::create(['mimetype' => $link->getMimetype()])); Event::handle('AttachmentStoreNew', [&$attachment]); break; default: // String is valid image data $temp_file = new TemporaryFile(); $temp_file->write($img_data); try { $attachment = GSFile::storeFileAsAttachment($temp_file); $embed_data['attachment_id'] = $attachment->getId(); } catch (ClientException) { DB::persist($attachment = Attachment::create(['mimetype' => $link->getMimetype()])); Event::handle('AttachmentStoreNew', [&$attachment]); } } $embed_data['attachment_id'] = $attachment->getId(); DB::persist(Entity\AttachmentEmbed::create($embed_data)); DB::flush(); return Event::stop; } /** * @return bool true if allowed by the lists, false otherwise */ private function allowedLink(string $url): bool { $passed_whitelist = !$this->check_whitelist; $passed_blacklist = !$this->check_blacklist; if ($this->check_whitelist) { $passed_whitelist = false; // don't trust be default $host = parse_url($url, \PHP_URL_HOST); foreach ($this->domain_whitelist as $regex => $provider) { if (preg_match("/{$regex}/", $host)) { $passed_whitelist = true; // we trust this source } } } if ($this->check_blacklist) { // assume it passed by default $host = parse_url($url, \PHP_URL_HOST); foreach ($this->domain_blacklist as $regex => $provider) { if (preg_match("/{$regex}/", $host)) { $passed_blacklist = false; // we blocked this source } } } return $passed_whitelist && $passed_blacklist; } /** * Perform an oEmbed or OpenGraph lookup for the given $url. * * Some known hosts are whitelisted with API endpoints where we * know they exist but autodiscovery data isn't available. * * Throws exceptions on failure. */ private function getEmbedLibMetadata(string $url): array { Log::info("Trying to find Embed data for {$url} with 'oscarotero/Embed'"); $embed = new LibEmbed(); $info = $embed->get($url); $metadata['title'] = $info->title; $metadata['description'] = $info->description; $metadata['author_name'] = $info->authorName; $root_url = parse_url($url); $root_url = "{$root_url['scheme']}://{$root_url['host']}"; $metadata['author_url'] = $info->authorUrl ? (string) $info->authorUrl : $root_url; $metadata['provider_name'] = $info->providerName; $metadata['provider_url'] = (string) ($info->providerUrl ?? $metadata['author_name']); if (!\is_null($info->image)) { $thumbnail_url = (string) $info->image; } else { $thumbnail_url = (string) $info->favicon; } // Check thumbnail URL validity $metadata['thumbnail_url'] = $thumbnail_url; return self::normalizeEmbedLibMetadata($metadata); } /** * Normalize fetched info. */ private static function normalizeEmbedLibMetadata(array $metadata): array { if (isset($metadata['thumbnail_url'])) { // sometimes sites serve the path, not the full URL, for images // let's "be liberal in what you accept from others"! // add protocol and host if the thumbnail_url starts with / if ($metadata['thumbnail_url'][0] == '/') { $metadata['thumbnail_url'] = "{$metadata['provider_url']}{$metadata['thumbnail_url']}"; } // Some wordpress opengraph implementations sometimes return a white blank image // no need for us to save that! if ($metadata['thumbnail_url'] == 'https://s0.wp.com/i/blank.jpg') { $metadata['thumbnail_url'] = null; } } return $metadata; } /** * Private helper that: * - checks if given URL is valid and is in fact an image (basic test), returns null if not; * - checks if respects file quota and whitelist/blacklist, returns false if not; * - downloads the thumbnail, returns a string if successful. * * @param string $url URL to the remote thumbnail */ private function downloadThumbnail(string $url): bool|string|null { // Is this a valid URL? if (!Common::isValidHttpUrl($url)) { Log::debug("Invalid URL ({$url}) in Embed->downloadThumbnail."); return null; } // Is this URL trusted? if (!$this->allowedLink($url)) { Log::info("Blocked URL ({$url}) in Embed->downloadThumbnail."); return false; } // Validate if the URL really does point to a remote image $head = HTTPClient::head($url); try { $headers = $head->getHeaders(); } catch (ClientExceptionInterface|RedirectionExceptionInterface|ServerExceptionInterface|TransportExceptionInterface $e) { Log::debug('Embed->downloadThumbnail@HTTPHead->getHeaders: ' . $e->getMessage(), [$e]); return null; } if (empty($headers['content-type']) || GSFile::mimetypeMajor($headers['content-type'][0]) !== 'image') { Log::debug("URL ({$url}) doesn't point to an image (content-type: " . (!empty($headers['content-type'][0]) ? $headers['content-type'][0] : 'not available') . ') in Embed->downloadThumbnail.'); return null; } // Does it respect the file quota? $file_size = $headers['content-length'][0] ?? null; $max_size = $this->getMaxFileSize(); if (\is_null($file_size) || $file_size > $max_size) { Log::debug("Went to download remote thumbnail of size {$file_size} but the plugin's filesize limit is {$max_size} so we aborted in Embed->downloadThumbnail."); return false; } // Download and return the file Log::debug("Downloading remote thumbnail from URL: {$url} in Embed->downloadThumbnail."); return HTTPClient::get($url)->getContent(); } public function onAttachmentGetBestTitle(Attachment $attachment, Note $note, ?string &$title) { try { $embed = DB::findOneBy('attachment_embed', ['attachment_id' => $attachment->getId()]); $title = $embed->getTitle(); return Event::stop; } catch (NotFoundException) { } return Event::next; } /** * Event raised when GNU social polls the plugin for information about it. * Adds this plugin's version information to $versions array * * @param array $versions inherited from parent * * @throws ServerException * * @return bool true hook value */ public function onPluginVersion(array &$versions): bool { $versions[] = [ 'name' => 'Embed', 'version' => $this->version(), 'author' => 'Mikael Nordfeldth, Hugo Sales, Diogo Peralta Cordeiro', 'homepage' => GNUSOCIAL_PROJECT_URL, 'description', // TRANS: Plugin description. => _m('Plugin for using and representing oEmbed, OpenGraph and other data.'), ]; return Event::next; } }