2020-03-29 19:33:16 +01:00
< ? php
2020-03-29 20:56:35 +01:00
// {{{ License
2021-04-15 23:28:28 +01:00
2020-05-20 17:53:53 +01:00
// This file is part of GNU social - https://www.gnu.org/software/social
2020-03-29 20:56:35 +01:00
//
// GNU social is free software: you can redistribute it and/or modify
2020-05-10 21:43:15 +01:00
// it under the terms of the GNU Affero General Public License as published by
2020-03-29 20:56:35 +01:00
// 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.
//
2020-05-10 21:43:15 +01:00
// You should have received a copy of the GNU Affero General Public License
2020-03-29 20:56:35 +01:00
// along with GNU social. If not, see <http://www.gnu.org/licenses/>.
2021-04-15 23:28:28 +01:00
2020-03-29 20:56:35 +01:00
// }}}
2020-03-29 19:33:16 +01:00
namespace App\Entity ;
2021-04-16 17:11:34 +01:00
use App\Core\Cache ;
2021-04-16 11:46:53 +01:00
use App\Core\DB\DB ;
2020-08-15 07:18:23 +01:00
use App\Core\Entity ;
2021-04-16 11:46:53 +01:00
use App\Core\Event ;
2021-04-18 02:17:57 +01:00
use App\Core\GSFile ;
2021-07-22 20:56:29 +01:00
use function App\Core\I18n\_m ;
2021-04-16 11:46:53 +01:00
use App\Core\Log ;
2021-08-18 17:30:54 +01:00
use App\Core\Router\Router ;
2021-04-16 16:57:25 +01:00
use App\Util\Common ;
2021-07-22 20:56:29 +01:00
use App\Util\Exception\ClientException ;
2021-04-16 11:46:53 +01:00
use App\Util\Exception\NotFoundException ;
2021-08-18 17:30:54 +01:00
use App\Util\Exception\NotStoredLocallyException ;
2021-04-16 11:46:53 +01:00
use App\Util\Exception\ServerException ;
2021-09-06 23:47:28 +01:00
use App\Util\TemporaryFile ;
2020-05-10 21:43:15 +01:00
use DateTimeInterface ;
2021-07-22 20:56:29 +01:00
use Symfony\Component\Mime\MimeTypes ;
2020-05-10 21:43:15 +01:00
2020-03-29 19:33:16 +01:00
/**
2021-04-15 23:28:28 +01:00
* Entity for Attachment thumbnails
2020-03-29 19:33:16 +01:00
*
* @ category DB
* @ package GNUsocial
*
* @ author Zach Copley < zach @ status . net >
* @ copyright 2010 StatusNet Inc .
* @ author Mikael Nordfeldth < mmn @ hethane . se >
* @ copyright 2009 - 2014 Free Software Foundation , Inc http :// www . fsf . org
2021-02-19 23:29:43 +00:00
* @ author Hugo Sales < hugo @ hsal . es >
2021-07-20 21:17:53 +01:00
* @ author Diogo Peralta Cordeiro < mail @ diogo . site >
2021-02-19 23:29:43 +00:00
* @ copyright 2020 - 2021 Free Software Foundation , Inc http :// www . fsf . org
2020-03-29 19:33:16 +01:00
* @ license https :// www . gnu . org / licenses / agpl . html GNU AGPL v3 or later
*/
2021-04-16 20:27:33 +01:00
class AttachmentThumbnail extends Entity
2020-03-29 19:33:16 +01:00
{
2020-03-30 15:00:13 +01:00
// {{{ Autocode
2021-05-05 17:03:03 +01:00
// @codeCoverageIgnoreStart
2021-04-15 23:28:28 +01:00
private int $attachment_id ;
2021-07-22 20:56:29 +01:00
private ? string $mimetype ;
2020-03-30 16:13:51 +01:00
private int $width ;
private int $height ;
2021-05-01 22:48:44 +01:00
private string $filename ;
2021-04-27 22:24:48 +01:00
private \DateTimeInterface $modified ;
2020-03-30 16:13:51 +01:00
2021-04-15 23:28:28 +01:00
public function setAttachmentId ( int $attachment_id ) : self
2020-03-30 16:13:51 +01:00
{
2021-04-15 23:28:28 +01:00
$this -> attachment_id = $attachment_id ;
2020-03-30 16:13:51 +01:00
return $this ;
}
2020-08-08 17:11:18 +01:00
2021-04-15 23:28:28 +01:00
public function getAttachmentId () : int
2020-03-30 16:13:51 +01:00
{
2021-04-15 23:28:28 +01:00
return $this -> attachment_id ;
2020-03-30 16:13:51 +01:00
}
2021-07-22 20:56:29 +01:00
public function setMimetype ( ? string $mimetype ) : self
{
$this -> mimetype = $mimetype ;
return $this ;
}
public function getMimetype () : ? string
{
return $this -> mimetype ;
}
2020-03-30 16:13:51 +01:00
public function setWidth ( int $width ) : self
{
$this -> width = $width ;
return $this ;
}
2020-08-08 17:11:18 +01:00
2020-03-30 16:13:51 +01:00
public function getWidth () : int
{
return $this -> width ;
}
public function setHeight ( int $height ) : self
{
$this -> height = $height ;
return $this ;
}
2020-08-08 17:11:18 +01:00
2020-03-30 16:13:51 +01:00
public function getHeight () : int
{
return $this -> height ;
}
2021-05-01 22:48:44 +01:00
public function setFilename ( string $filename ) : self
{
$this -> filename = $filename ;
return $this ;
}
public function getFilename () : string
{
return $this -> filename ;
}
2021-05-02 16:02:26 +01:00
public function setModified ( DateTimeInterface $modified ) : self
2020-03-30 16:13:51 +01:00
{
$this -> modified = $modified ;
return $this ;
}
2020-08-08 17:11:18 +01:00
2021-05-02 16:02:26 +01:00
public function getModified () : DateTimeInterface
2020-03-30 16:13:51 +01:00
{
return $this -> modified ;
}
2021-05-05 17:03:03 +01:00
// @codeCoverageIgnoreEnd
2020-03-30 15:00:13 +01:00
// }}} Autocode
2020-03-29 19:33:16 +01:00
2021-08-18 17:30:54 +01:00
private ? Attachment $attachment = null ;
2021-04-16 16:57:25 +01:00
2021-08-18 17:30:54 +01:00
public function setAttachment ( ? Attachment $attachment )
2021-04-16 16:57:25 +01:00
{
$this -> attachment = $attachment ;
}
public function getAttachment ()
{
2021-08-18 17:30:54 +01:00
if ( isset ( $this -> attachment ) && ! is_null ( $this -> attachment )) {
2021-04-16 16:57:25 +01:00
return $this -> attachment ;
} else {
return $this -> attachment = DB :: findOneBy ( 'attachment' , [ 'id' => $this -> attachment_id ]);
}
}
2021-07-20 21:17:53 +01:00
/**
* @ param Attachment $attachment
* @ param int $width
* @ param int $height
* @ param bool $crop
*
2021-08-18 22:10:19 +01:00
* @ throws ClientException
* @ throws NotFoundException
2021-07-20 21:17:53 +01:00
* @ throws ServerException
*
* @ return mixed
*/
2021-05-01 22:48:44 +01:00
public static function getOrCreate ( Attachment $attachment , int $width , int $height , bool $crop )
2021-04-16 11:46:53 +01:00
{
2021-07-20 23:31:53 +01:00
// We need to keep these in mind for DB indexing
$predicted_width = null ;
$predicted_height = null ;
2021-04-16 11:46:53 +01:00
try {
2021-08-14 15:07:17 +01:00
if ( is_null ( $attachment -> getWidth ()) || is_null ( $attachment -> getHeight ())) {
2021-08-18 17:30:54 +01:00
// @codeCoverageIgnoreStart
2021-08-14 15:07:17 +01:00
// TODO: check if we can generate from an existing thumbnail
throw new ClientException ( _m ( 'Invalid dimensions requested for thumbnail.' ));
2021-08-18 17:30:54 +01:00
// @codeCoverageIgnoreEnd
2021-08-14 15:07:17 +01:00
}
2021-04-16 17:11:34 +01:00
return Cache :: get ( 'thumb-' . $attachment -> getId () . " - { $width } x { $height } " ,
2021-07-20 23:31:53 +01:00
function () use ( $crop , $attachment , $width , $height , & $predicted_width , & $predicted_height ) {
2021-07-22 20:56:29 +01:00
[ $predicted_width , $predicted_height ] = self :: predictScalingValues ( $attachment -> getWidth (), $attachment -> getHeight (), $width , $height , $crop );
2021-05-01 22:48:44 +01:00
return DB :: findOneBy ( 'attachment_thumbnail' , [ 'attachment_id' => $attachment -> getId (), 'width' => $predicted_width , 'height' => $predicted_height ]);
});
2021-04-16 11:46:53 +01:00
} catch ( NotFoundException $e ) {
2021-08-18 17:30:54 +01:00
if ( ! file_exists ( $attachment -> getPath ())) {
throw new NotStoredLocallyException ();
2021-08-14 15:07:17 +01:00
}
2021-08-18 22:10:19 +01:00
$thumbnail = self :: create ([ 'attachment_id' => $attachment -> getId ()]);
$mimetype = $attachment -> getMimetype ();
$event_map [ $mimetype ] = [];
$major_mime = GSFile :: mimetypeMajor ( $mimetype );
$event_map [ $major_mime ] = [];
Event :: handle ( 'FileResizerAvailable' , [ & $event_map , $mimetype ]);
// Always prefer specific encoders
2021-09-06 23:47:28 +01:00
/** @var callable[] function(string $source, ?TemporaryFile &$destination, int &$width, int &$height, bool $smart_crop, ?string &$mimetype): bool */
2021-08-18 22:10:19 +01:00
$encoders = array_merge ( $event_map [ $mimetype ], $event_map [ $major_mime ]);
foreach ( $encoders as $encoder ) {
2021-09-06 23:47:28 +01:00
/** @var ?TemporaryFile */
2021-07-22 20:56:29 +01:00
$temp = null ; // Let the EncoderPlugin create a temporary file for us
2021-08-18 22:10:19 +01:00
if ( $encoder ( $attachment -> getPath (), $temp , $width , $height , $crop , $mimetype )) {
2021-08-18 17:30:54 +01:00
$thumbnail -> setAttachment ( $attachment );
2021-07-22 20:56:29 +01:00
$thumbnail -> setWidth ( $predicted_width );
$thumbnail -> setHeight ( $predicted_height );
2021-09-06 23:47:28 +01:00
$mimetype = $temp -> getMimeType ();
$ext = '.' . MimeTypes :: getDefault () -> getExtensions ( $mimetype )[ 0 ];
2021-08-14 16:47:45 +01:00
$filename = " { $predicted_width } x { $predicted_height } { $ext } - " . $attachment -> getFilehash ();
2021-07-22 20:56:29 +01:00
$thumbnail -> setFilename ( $filename );
2021-07-22 21:17:23 +01:00
$thumbnail -> setMimetype ( $mimetype );
2021-07-22 20:56:29 +01:00
DB :: persist ( $thumbnail );
DB :: flush ();
$temp -> move ( Common :: config ( 'thumbnail' , 'dir' ), $filename );
return $thumbnail ;
}
2021-04-16 11:46:53 +01:00
}
2021-08-18 22:10:19 +01:00
throw new ClientException ( _m ( 'Can not generate thumbnail for attachment with id={id}' , [ 'id' => $attachment -> getId ()]));
2021-04-16 11:46:53 +01:00
}
}
2021-04-16 16:57:25 +01:00
public function getPath ()
{
2021-08-19 01:45:11 +01:00
return Common :: config ( 'thumbnail' , 'dir' ) . DIRECTORY_SEPARATOR . $this -> getFilename ();
2021-04-16 16:57:25 +01:00
}
2021-04-25 22:26:53 +01:00
public function getUrl ()
{
return Router :: url ( 'attachment_thumbnail' , [ 'id' => $this -> getAttachmentId (), 'w' => $this -> getWidth (), 'h' => $this -> getHeight ()]);
}
/**
* Get the HTML attributes for this thumbnail
*/
public function getHTMLAttributes ( array $orig = [], bool $overwrite = true )
{
$attrs = [
'height' => $this -> getHeight (),
2021-05-02 16:02:26 +01:00
'width' => $this -> getWidth (),
'src' => $this -> getUrl (),
2021-04-25 22:26:53 +01:00
];
return $overwrite ? array_merge ( $orig , $attrs ) : array_merge ( $attrs , $orig );
}
2020-08-07 03:03:55 +01:00
/**
2021-04-29 19:12:32 +01:00
* Delete an attachment thumbnail
2020-08-07 03:03:55 +01:00
*/
2021-04-29 19:12:32 +01:00
public function delete ( bool $flush = true ) : void
2020-08-07 03:03:55 +01:00
{
2021-04-29 19:12:32 +01:00
$filepath = $this -> getPath ();
if ( file_exists ( $filepath )) {
if ( @ unlink ( $filepath ) === false ) {
2021-08-18 17:30:54 +01:00
// @codeCoverageIgnoreStart
2021-04-29 19:12:32 +01:00
Log :: warning ( " Failed deleting file for attachment thumbnail with id= { $this -> attachment_id } , width= { $this -> width } , height= { $this -> height } at { $filepath } " );
2021-08-18 17:30:54 +01:00
// @codeCoverageIgnoreEnd
2021-04-29 19:12:32 +01:00
}
}
DB :: remove ( $this );
if ( $flush ) {
DB :: flush ();
}
2020-08-07 03:03:55 +01:00
}
2021-05-01 22:48:44 +01:00
/**
* Gets scaling values for images of various types . Cropping can be enabled .
*
* Values will scale _up_ to fit max values if cropping is enabled !
* With cropping disabled , the max value of each axis will be respected .
*
2021-09-06 23:47:28 +01:00
* @ param int $existing_width Original width
* @ param int $existing_height Original height
* @ param int $requested_width Resulting max width
* @ param int $requested_height Resulting max height
* @ param bool $crop Crop to the size ( not preserving aspect ratio )
2021-05-01 22:48:44 +01:00
*
* @ return array [ predicted width , predicted height ]
*/
public static function predictScalingValues (
2021-08-18 17:30:54 +01:00
int $existing_width ,
int $existing_height ,
int $requested_width ,
int $requested_height ,
2021-05-01 22:48:44 +01:00
bool $crop
2021-05-02 16:02:26 +01:00
) : array {
2021-05-01 22:48:44 +01:00
if ( $crop ) {
2021-08-18 17:30:54 +01:00
$rw = min ( $existing_width , $requested_width );
$rh = min ( $existing_height , $requested_height );
2021-05-01 22:48:44 +01:00
} else {
2021-08-18 17:30:54 +01:00
if ( $existing_width > $existing_height ) {
$rw = min ( $existing_width , $requested_width );
$rh = ceil ( $existing_height * $rw / $existing_width );
} else {
$rh = min ( $existing_height , $requested_height );
$rw = ceil ( $existing_width * $rh / $existing_height );
2021-05-01 22:48:44 +01:00
}
}
2021-08-18 17:30:54 +01:00
2021-05-02 16:02:26 +01:00
return [( int ) $rw , ( int ) $rh ];
2021-05-01 22:48:44 +01:00
}
2020-03-29 19:33:16 +01:00
public static function schemaDef () : array
{
return [
2021-05-02 16:02:26 +01:00
'name' => 'attachment_thumbnail' ,
2020-03-29 19:33:16 +01:00
'fields' => [
2021-04-15 23:28:28 +01:00
'attachment_id' => [ 'type' => 'int' , 'foreign key' => true , 'target' => 'Attachment.id' , 'multiplicity' => 'one to one' , 'not null' => true , 'description' => 'thumbnail for what attachment' ],
2021-08-18 22:10:19 +01:00
'mimetype' => [ 'type' => 'varchar' , 'length' => 129 , 'description' => 'resource mime type 64+1+64, images hardly will show up with long mimetypes, this is probably safe considering rfc6838#section-4.2' ],
2021-07-22 21:17:23 +01:00
'width' => [ 'type' => 'int' , 'not null' => true , 'description' => 'width of thumbnail' ],
'height' => [ 'type' => 'int' , 'not null' => true , 'description' => 'height of thumbnail' ],
'filename' => [ 'type' => 'varchar' , 'length' => 191 , 'not null' => true , 'description' => 'thumbnail filename' ],
'modified' => [ 'type' => 'timestamp' , 'not null' => true , 'default' => 'CURRENT_TIMESTAMP' , 'description' => 'date this record was modified' ],
2020-03-29 19:33:16 +01:00
],
2021-04-15 23:28:28 +01:00
'primary key' => [ 'attachment_id' , 'width' , 'height' ],
2021-05-02 16:02:26 +01:00
'indexes' => [
2021-04-15 23:28:28 +01:00
'attachment_thumbnail_attachment_id_idx' => [ 'attachment_id' ],
2020-06-30 17:26:40 +01:00
],
2020-03-29 19:33:16 +01:00
];
}
2020-06-30 17:26:40 +01:00
}