forked from GNUsocial/gnu-social
		
	
		
			
				
	
	
		
			181 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			181 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| // {{{ 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/>.
 | |
| // }}}
 | |
| 
 | |
| namespace Plugin\StoreRemoteMedia;
 | |
| 
 | |
| use App\Core\DB\DB;
 | |
| use App\Core\Event;
 | |
| use App\Core\GSFile;
 | |
| use App\Core\HTTPClient;
 | |
| use App\Core\Modules\Plugin;
 | |
| use App\Entity\AttachmentThumbnail;
 | |
| use App\Entity\AttachmentToNote;
 | |
| use App\Entity\Note;
 | |
| use App\Entity\RemoteURL;
 | |
| use App\Entity\RemoteURLToAttachment;
 | |
| use App\Util\Common;
 | |
| use App\Util\TemporaryFile;
 | |
| 
 | |
| /**
 | |
|  * The StoreRemoteMedia plugin downloads remotely attached files to local server.
 | |
|  *
 | |
|  * @package   GNUsocial
 | |
|  *
 | |
|  * @author    Mikael Nordfeldth
 | |
|  * @author    Stephen Paul Weber
 | |
|  * @author    Mikael Nordfeldth
 | |
|  * @author    Miguel Dantas
 | |
|  * @author    Diogo Peralta Cordeiro
 | |
|  * @copyright 2015-2016, 2019-2021 Free Software Foundation, Inc http://www.fsf.org
 | |
|  * @license   https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
 | |
|  */
 | |
| class StoreRemoteMedia extends Plugin
 | |
| {
 | |
|     public function version(): string
 | |
|     {
 | |
|         return '3.0.0';
 | |
|     }
 | |
| 
 | |
|     public bool $store_original = false; // Whether to maintain a copy of the original media or only a thumbnail of it
 | |
|     public ?int $thumbnail_width;
 | |
|     public ?int $thumbnail_height;
 | |
|     public ?int $max_size;
 | |
|     public ?bool $smart_crop;
 | |
| 
 | |
|     private function getStoreOriginal(): bool
 | |
|     {
 | |
|         return $this->store_original;
 | |
|     }
 | |
|     private function getThumbnailWidth(): int
 | |
|     {
 | |
|         return $this->thumbnail_width ?? Common::config('thumbnail', 'width');
 | |
|     }
 | |
| 
 | |
|     private function getThumbnailHeight(): int
 | |
|     {
 | |
|         return $this->thumbnail_height ?? Common::config('thumbnail', 'height');
 | |
|     }
 | |
| 
 | |
|     private function getMaxSize(): int
 | |
|     {
 | |
|         return $this->max_size ?? Common::config('attachments', 'file_quota');
 | |
|     }
 | |
| 
 | |
|     private function getSmartCrop(): bool
 | |
|     {
 | |
|         return $this->smart_crop ?? Common::config('thumbnail', 'smart_crop');
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param RemoteURL $remote_url
 | |
|      *
 | |
|      * @return bool
 | |
|      */
 | |
|     public function onNewRemoteURLFromNote(RemoteURL $remote_url, Note $note): bool
 | |
|     {
 | |
|         // Embed is the plugin to handle these
 | |
|         if ($remote_url->getMimetypeMajor() === 'text') {
 | |
|             return Event::next;
 | |
|         }
 | |
| 
 | |
|         // Have we handled it already?
 | |
|         $remoteurl_to_attachment = DB::find('remoteurl_to_attachment',
 | |
|             ['remoteurl_id' => $remote_url->getId()]);
 | |
| 
 | |
|         // If it was handled already
 | |
|         if (!is_null($remoteurl_to_attachment)) {
 | |
|             // Relate the note with the existing attachment
 | |
|             DB::persist(AttachmentToNote::create([
 | |
|                 'attachment_id' => $remoteurl_to_attachment->getAttachmentId(),
 | |
|                 'note_id'       => $note->getId(),
 | |
|             ]));
 | |
|             DB::flush();
 | |
|             return Event::stop;
 | |
|         } else {
 | |
|             // Retrieve media
 | |
|             $get_response = HTTPClient::get($remote_url->getRemoteUrl());
 | |
|             $media        = $get_response->getContent();
 | |
|             $mimetype     = $get_response->getHeaders()['content-type'][0];
 | |
|             unset($get_response);
 | |
| 
 | |
|             // Ensure we still want to handle it
 | |
|             if ($mimetype != $remote_url->getMimetype()) {
 | |
|                 $remote_url->setMimetype($mimetype);
 | |
|                 DB::persist($remote_url);
 | |
|                 DB::flush();
 | |
|                 if ($remote_url->getMimetypeMajor() === 'text') {
 | |
|                     return Event::next;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // Create an attachment for this
 | |
|             $temp_file = new TemporaryFile();
 | |
|             $temp_file->write($media);
 | |
|             $attachment = GSFile::sanitizeAndStoreFileAsAttachment($temp_file);
 | |
| 
 | |
|             // Relate the remoteurl with the attachment
 | |
|             DB::persist(RemoteURLToAttachment::create([
 | |
|                 'remoteurl_id'  => $remote_url->getId(),
 | |
|                 'attachment_id' => $attachment->getId(),
 | |
|             ]));
 | |
| 
 | |
|             // Relate the note with the attachment
 | |
|             DB::persist(AttachmentToNote::create([
 | |
|                 'attachment_id' => $attachment->getId(),
 | |
|                 'note_id'       => $note->getId(),
 | |
|             ]));
 | |
| 
 | |
|             DB::flush();
 | |
| 
 | |
|             // Should we create a thumb and delete the original file?
 | |
|             if (!$this->getStoreOriginal()) {
 | |
|                 $thumbnail = AttachmentThumbnail::getOrCreate(
 | |
|                     attachment: $attachment,
 | |
|                     width: $this->getThumbnailWidth(),
 | |
|                     height: $this->getThumbnailHeight(),
 | |
|                     crop: $this->getSmartCrop()
 | |
|                 );
 | |
|                 $attachment->deleteStorage();
 | |
|             }
 | |
| 
 | |
|             return Event::stop;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Event raised when GNU social polls the plugin for information about it.
 | |
|      * Adds this plugin's version information to $versions array
 | |
|      *
 | |
|      * @param &$versions array inherited from parent
 | |
|      *
 | |
|      * @return bool true hook value
 | |
|      */
 | |
|     public function onPluginVersion(array &$versions): bool
 | |
|     {
 | |
|         $versions[] = [
 | |
|             'name'        => 'StoreRemoteMedia',
 | |
|             'version'     => $this->version(),
 | |
|             'author'      => 'Mikael Nordfeldth, Diogo Peralta Cordeiro',
 | |
|             'homepage'    => GNUSOCIAL_PROJECT_URL,
 | |
|             'description' => // TRANS: Plugin description.
 | |
|                 _m('Plugin for downloading remotely attached files to local server.'),
 | |
|         ];
 | |
|         return Event::next;
 | |
|     }
 | |
| }
 |