3 * @copyright Copyright (C) 2010-2023, the Friendica project
5 * @license GNU AGPL version 3 or any later version
7 * This program is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU Affero General Public License as
9 * published by the Free Software Foundation, either version 3 of the
10 * License, or (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU Affero General Public License for more details.
17 * You should have received a copy of the GNU Affero General Public License
18 * along with this program. If not, see <https://www.gnu.org/licenses/>.
22 namespace Friendica\Model\Post;
24 use Friendica\Core\Logger;
25 use Friendica\Core\System;
26 use Friendica\Database\Database;
27 use Friendica\Database\DBA;
29 use Friendica\Network\HTTPClient\Client\HttpClientAccept;
30 use Friendica\Network\HTTPClient\Client\HttpClientOptions;
31 use Friendica\Util\HTTPSignature;
32 use Friendica\Util\Images;
33 use Friendica\Util\Proxy;
34 use Friendica\Object\Image;
39 * This Model class handles post related external links
44 * Check if the link is stored
47 * @param string $url URL
48 * @return bool Whether record has been found
50 public static function exists(int $uriId, string $url): bool
52 return DBA::exists('post-link', ['uri-id' => $uriId, 'url' => $url]);
56 * Returns URL by URI id and other URL
61 * @return string Found link URL + id on success, $url on failure
63 public static function getByLink(int $uriId, string $url, string $size = ''): string
65 if (empty($uriId) || empty($url) || Proxy::isLocalImage($url)) {
69 if (!in_array(parse_url($url, PHP_URL_SCHEME), ['http', 'https'])) {
70 Logger::info('Bad URL, quitting', ['uri-id' => $uriId, 'url' => $url, 'callstack' => System::callstack(20)]);
74 $link = DBA::selectFirst('post-link', ['id'], ['uri-id' => $uriId, 'url' => $url]);
75 if (!empty($link['id'])) {
77 Logger::info('Found', ['id' => $id, 'uri-id' => $uriId, 'url' => $url]);
79 $fields = self::fetchMimeType($url);
80 $fields['uri-id'] = $uriId;
81 $fields['url'] = $url;
83 DBA::insert('post-link', $fields, Database::INSERT_IGNORE);
84 $id = DBA::lastInsertId();
85 Logger::info('Inserted', $fields);
92 $url = DI::baseUrl() . '/photo/link/';
94 case Proxy::SIZE_MICRO:
95 $url .= Proxy::PIXEL_MICRO . '/';
98 case Proxy::SIZE_THUMB:
99 $url .= Proxy::PIXEL_THUMB . '/';
102 case Proxy::SIZE_SMALL:
103 $url .= Proxy::PIXEL_SMALL . '/';
106 case Proxy::SIZE_MEDIUM:
107 $url .= Proxy::PIXEL_MEDIUM . '/';
110 case Proxy::SIZE_LARGE:
111 $url .= Proxy::PIXEL_LARGE . '/';
118 * Fetches MIME type by URL and Accept: header
120 * @param string $url URL to fetch
121 * @param string $accept Comma-separated list of expected response MIME type(s)
122 * @return array Discovered MIME type and blurhash or empty array on failure
124 private static function fetchMimeType(string $url, string $accept = HttpClientAccept::DEFAULT): array
126 $timeout = DI::config()->get('system', 'xrd_timeout');
129 $curlResult = HTTPSignature::fetchRaw($url, 0, [HttpClientOptions::TIMEOUT => $timeout, HttpClientOptions::ACCEPT_CONTENT => $accept]);
130 if (empty($curlResult) || !$curlResult->isSuccess()) {
133 } catch (\Exception $exception) {
134 Logger::notice('Error fetching url', ['url' => $url, 'exception' => $exception]);
137 $fields = ['mimetype' => $curlResult->getHeader('Content-Type')[0]];
139 $img_str = $curlResult->getBody();
140 $image = new Image($img_str, Images::getMimeTypeByData($img_str));
141 if ($image->isValid()) {
142 $fields['mimetype'] = $image->getType();
143 $fields['width'] = $image->getWidth();
144 $fields['height'] = $image->getHeight();
145 $fields['blurhash'] = $image->getBlurHash();
152 * Add external links and replace them in the body
154 * @param integer $uriId
155 * @param string $body Item body formatted with BBCodes
156 * @return string Body with replaced links
158 public static function insertFromBody(int $uriId, string $body): string
160 if (preg_match_all("/\[img\=([0-9]*)x([0-9]*)\](http.*?)\[\/img\]/ism", $body, $pictures, PREG_SET_ORDER)) {
161 foreach ($pictures as $picture) {
162 $body = str_replace($picture[3], self::getByLink($uriId, $picture[3]), $body);
166 if (preg_match_all("/\[img=(http[^\[\]]*)\]([^\[\]]*)\[\/img\]/Usi", $body, $pictures, PREG_SET_ORDER)) {
167 foreach ($pictures as $picture) {
168 $body = str_replace($picture[1], self::getByLink($uriId, $picture[1]), $body);
172 if (preg_match_all("/\[img\](http[^\[\]]*)\[\/img\]/ism", $body, $pictures, PREG_SET_ORDER)) {
173 foreach ($pictures as $picture) {
174 $body = str_replace($picture[1], self::getByLink($uriId, $picture[1]), $body);