#!/usr/bin/env php <?php // 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/>. /** * EmbedPlugin implementation for GNU social * * @package GNUsocial * @author Mikael Nordfeldth * @author Diogo Cordeiro <diogo@fc.up.pt> * @copyright 2019 Free Software Foundation, Inc http://www.fsf.org * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ define('INSTALLDIR', dirname(__DIR__, 3)); define('PUBLICDIR', INSTALLDIR . DIRECTORY_SEPARATOR . 'public'); $longoptions = ['dry-run', 'h-bug', 'broken-oembed', 'limit=']; $helptext = <<<END_OF_USERROLE_HELP fixup_files.php [options] Patches attachments with broken oembed. --dry-run look but don't touch --h-bug Patches up file entries with corrupted types and titles (the "h bug") --broken-oembed Attempts refecting info for broken attachments --limit [date] Only affect files from after this date. This is a timestamp, format is: yyyy-mm-dd (optional time hh:mm:ss may be provided) END_OF_USERROLE_HELP; require_once INSTALLDIR.'/scripts/commandline.inc'; $dry = have_option('dry-run'); $h_bug = have_option('h-bug'); $broken = have_option('broken-oembed'); $limit = get_option_value('limit'); if (!($broken ^ $h_bug)) { echo "Exactly one of --h-bug and --broken-oembed are required\n"; die(); } $query = " SELECT DISTINCT file_to_post.file_id FROM file_to_post INNER JOIN file ON file.id = file_to_post.file_id INNER JOIN notice ON notice.id = file_to_post.post_id WHERE"; $f = new File(); if ($h_bug) { $query .= " file.title = 'h' AND file.mimetype = 'h' AND file.size = 0 AND file.protected = 0"; } elseif ($broken) { $query .= " file.filename is NULL"; } $query .= empty($limit) ? "" : " AND notice.modified >= '{$limit}' ORDER BY notice.modified ASC"; // echo $query; $fn = new DB_DataObject(); $fn->query($query); if ($h_bug) { echo "Found {$fn->N} bad items:\n"; } else { echo "Found {$fn->N} files.\n"; } while ($fn->fetch()) { $f = File::getByID($fn->file_id); try { $data = File_embed::getByFile($f); } catch (Exception $e) { // Carry on } if ($broken && $data instanceof File_embed) { try { $thumb = File_thumbnail::byFile($f, true /* not null url */); $thumb->getPath(); // Check we have the file } catch (Exception $e) { $no_thumb = true; // Doesn't exist, no problem } } if ($h_bug) { echo "ID: {$f->id}, URL {$f->url}"; if ($dry) { if ($data instanceof File_embed) { echo " (unchanged)\n"; } else { echo " (unchanged, but embedding lookup failed)\n"; } } elseif (!$dry) { $sql = "UPDATE file " . "SET mimetype=null, title=null,size=null,protected=null " . "WHERE id={$f->id}"; $f->query($sql); $f->decache(); if ($data instanceof File_embed) { $fetch = true; echo " (ok)\n"; } else { echo " (ok, but embedding lookup failed)\n"; } } } elseif ($broken && (!$data instanceof File_embed || empty($data->title) || empty($f->title) || ($thumb instanceof File_thumbnail && empty($thumb->filename)) )) { // print_r($thumb); if (!$dry) { echo "Will refetch for file with "; } else { echo "Found broken file with "; } echo "ID: {$f->id}, URL {$f->url}\n"; if (!$dry) { $fetch = true; $sql = "UPDATE file SET title=null, size=null, protected=null " . "WHERE id={$f->id}"; $f->query($sql); $f->decache(); if ($data instanceof File_embed) { $data->delete(); $data->decache(); } if ($thumb instanceof File_thumbnail) { // Delete all thumbnails, not just this one $f->query("DELETE from file_thumbnail WHERE file_id = {$f->id}"); $thumb->decache(); } } } if (isset($fetch) && $fetch === true && !$dry) { $fetch = false; echo "Attempting to fetch Embed data\n"; Event::handle('EndFileSaveNew', array($f)); } } echo "Done.\n";