2019-06-12 22:54:06 +00:00
< ? php
/*
* Name : Transcode Image
* Author : Matthew Barbour < matthew @ darkholme . net >
* Description : Allows admins to automatically and manually transcode images .
* License : MIT
* Version : 1.0
* Documentation :
* Can transcode on - demand and automatically on upload . Config screen allows choosing an output format for each of the supported input formats .
2019-06-14 12:47:50 +00:00
* Supports GD and ImageMagick . Both support bmp , gif , jpg , png , and webp as inputs , and jpg , png , and lossy webp as outputs .
2019-06-12 22:54:06 +00:00
* ImageMagick additionally supports tiff and psd inputs , and webp lossless output .
* If and image is uanble to be transcoded for any reason , the upload will continue unaffected .
*/
/*
* This is used by the image transcoding code when there is an error while transcoding
*/
2019-06-14 12:47:50 +00:00
class ImageTranscodeException extends SCoreException
{
}
2019-06-12 22:54:06 +00:00
class TranscodeImage extends Extension
{
const CONVERSION_ENGINES = [
" GD " => " gd " ,
" ImageMagick " => " convert " ,
];
const ENGINE_INPUT_SUPPORT = [
" gd " => [
" bmp " ,
" gif " ,
" jpg " ,
" png " ,
" webp " ,
],
" convert " => [
" bmp " ,
" gif " ,
" jpg " ,
" png " ,
" psd " ,
" tiff " ,
" webp " ,
]
];
const ENGINE_OUTPUT_SUPPORT = [
" gd " => [
" jpg " ,
" png " ,
" webp-lossy " ,
],
" convert " => [
" jpg " ,
" png " ,
" webp-lossy " ,
" webp-lossless " ,
]
];
const LOSSLESS_FORMATS = [
" webp-lossless " ,
" png " ,
];
const INPUT_FORMATS = [
" BMP " => " bmp " ,
" GIF " => " gif " ,
" JPG " => " jpg " ,
" PNG " => " png " ,
" PSD " => " psd " ,
" TIFF " => " tiff " ,
" WEBP " => " webp " ,
];
const FORMAT_ALIASES = [
" tif " => " tiff " ,
" jpeg " => " jpg " ,
];
const OUTPUT_FORMATS = [
" " => " " ,
" JPEG (lossy) " => " jpg " ,
" PNG (lossless) " => " png " ,
" WEBP (lossy) " => " webp-lossy " ,
" WEBP (lossless) " => " webp-lossless " ,
];
/**
2019-06-13 16:45:34 +00:00
* Needs to be after upload , but before the processing extensions
2019-06-12 22:54:06 +00:00
*/
public function get_priority () : int
{
return 45 ;
}
public function onInitExt ( InitExtEvent $event )
{
global $config ;
$config -> set_default_bool ( 'transcode_enabled' , true );
$config -> set_default_bool ( 'transcode_upload' , false );
$config -> set_default_string ( 'transcode_engine' , " gd " );
$config -> set_default_int ( 'transcode_quality' , 80 );
2019-06-14 12:47:50 +00:00
foreach ( array_values ( self :: INPUT_FORMATS ) as $format ) {
2019-06-12 22:54:06 +00:00
$config -> set_default_string ( 'transcode_upload_' . $format , " " );
}
}
public function onImageAdminBlockBuilding ( ImageAdminBlockBuildingEvent $event )
{
global $user , $config ;
if ( $user -> is_admin () && $config -> get_bool ( " resize_enabled " )) {
$engine = $config -> get_string ( " transcode_engine " );
2019-06-14 12:47:50 +00:00
if ( $this -> can_convert_format ( $engine , $event -> image -> ext )) {
2019-06-12 22:54:06 +00:00
$options = $this -> get_supported_output_formats ( $engine , $event -> image -> ext );
$event -> add_part ( $this -> theme -> get_transcode_html ( $event -> image , $options ));
}
}
}
public function onSetupBuilding ( SetupBuildingEvent $event )
{
global $config ;
$engine = $config -> get_string ( " transcode_engine " );
$sb = new SetupBlock ( " Image Transcode " );
$sb -> add_bool_option ( " transcode_enabled " , " Allow transcoding images: " );
$sb -> add_bool_option ( " transcode_upload " , " <br>Transcode on upload: " );
2019-06-14 12:47:50 +00:00
$sb -> add_choice_option ( 'transcode_engine' , self :: CONVERSION_ENGINES , " <br />Transcode engine: " );
foreach ( self :: INPUT_FORMATS as $display => $format ) {
if ( in_array ( $format , self :: ENGINE_INPUT_SUPPORT [ $engine ])) {
2019-06-12 22:54:06 +00:00
$outputs = $this -> get_supported_output_formats ( $engine , $format );
2019-06-14 12:47:50 +00:00
$sb -> add_choice_option ( 'transcode_upload_' . $format , $outputs , " <br /> $display to: " );
}
2019-06-12 22:54:06 +00:00
}
$sb -> add_int_option ( " transcode_quality " , " <br/>Lossy format quality: " );
$event -> panel -> add_block ( $sb );
}
public function onDataUpload ( DataUploadEvent $event )
{
global $config , $page ;
2019-06-14 12:47:50 +00:00
if ( $config -> get_bool ( " transcode_upload " ) == true ) {
2019-06-12 22:54:06 +00:00
$ext = strtolower ( $event -> type );
$ext = $this -> clean_format ( $ext );
2019-06-14 12:47:50 +00:00
if ( $event -> type == " gif " && is_animated_gif ( $event -> tmpname )) {
2019-06-12 22:54:06 +00:00
return ;
}
2019-06-14 12:47:50 +00:00
if ( in_array ( $ext , array_values ( self :: INPUT_FORMATS ))) {
2019-06-12 22:54:06 +00:00
$target_format = $config -> get_string ( " transcode_upload_ " . $ext );
2019-06-14 12:47:50 +00:00
if ( empty ( $target_format )) {
2019-06-12 22:54:06 +00:00
return ;
}
try {
$new_image = $this -> transcode_image ( $event -> tmpname , $ext , $target_format );
$event -> set_type ( $this -> determine_ext ( $target_format ));
$event -> set_tmpname ( $new_image );
2019-06-14 12:47:50 +00:00
} catch ( Exception $e ) {
log_error ( " transcode " , " Error while performing upload transcode: " . $e -> getMessage ());
// We don't want to interfere with the upload process,
2019-06-12 22:54:06 +00:00
// so if something goes wrong the untranscoded image jsut continues
}
}
2019-06-14 12:47:50 +00:00
}
2019-06-12 22:54:06 +00:00
}
public function onPageRequest ( PageRequestEvent $event )
{
global $page , $user ;
if ( $event -> page_matches ( " transcode " ) && $user -> is_admin ()) {
2019-06-14 12:47:50 +00:00
$image_id = int_escape ( $event -> get_arg ( 0 ));
if ( empty ( $image_id )) {
$image_id = isset ( $_POST [ 'image_id' ]) ? int_escape ( $_POST [ 'image_id' ]) : null ;
}
// Try to get the image ID
if ( empty ( $image_id )) {
throw new ImageTranscodeException ( " Can not resize Image: No valid Image ID given. " );
}
$image_obj = Image :: by_id ( $image_id );
if ( is_null ( $image_obj )) {
$this -> theme -> display_error ( 404 , " Image not found " , " No image in the database has the ID # $image_id " );
} else {
if ( isset ( $_POST [ 'transcode_format' ])) {
try {
2019-06-12 22:54:06 +00:00
$this -> transcode_and_replace_image ( $image_obj , $_POST [ 'transcode_format' ]);
$page -> set_mode ( " redirect " );
$page -> set_redirect ( make_link ( " post/view/ " . $image_id ));
2019-06-14 12:47:50 +00:00
} catch ( ImageTranscodeException $e ) {
$this -> theme -> display_transcode_error ( $page , " Error Transcoding " , $e -> getMessage ());
}
}
}
}
2019-06-12 22:54:06 +00:00
}
public function onBulkActionBlockBuilding ( BulkActionBlockBuildingEvent $event )
{
global $user , $config ;
$engine = $config -> get_string ( " transcode_engine " );
if ( $user -> is_admin ()) {
2019-06-14 12:47:50 +00:00
$event -> add_action ( " bulk_transcode " , " Transcode " , " " , $this -> theme -> get_transcode_picker_html ( $this -> get_supported_output_formats ( $engine )));
2019-06-12 22:54:06 +00:00
}
}
public function onBulkAction ( BulkActionEvent $event )
{
global $user , $database ;
2019-06-14 12:47:50 +00:00
switch ( $event -> action ) {
2019-06-12 22:54:06 +00:00
case " bulk_transcode " :
if ( ! isset ( $_POST [ 'transcode_format' ])) {
return ;
}
if ( $user -> is_admin ()) {
$format = $_POST [ 'transcode_format' ];
$total = 0 ;
foreach ( $event -> items as $id ) {
try {
$database -> beginTransaction ();
$image = Image :: by_id ( $id );
2019-06-14 12:47:50 +00:00
if ( $image == null ) {
2019-06-12 22:54:06 +00:00
continue ;
}
2019-06-13 16:45:34 +00:00
2019-06-12 22:54:06 +00:00
$this -> transcode_and_replace_image ( $image , $format );
// If a subsequent transcode fails, the database need to have everything about the previous transcodes recorded already,
// otherwise the image entries will be stuck pointing to missing image files
$database -> commit ();
$total ++ ;
2019-06-14 12:47:50 +00:00
} catch ( Exception $e ) {
2019-06-12 22:54:06 +00:00
log_error ( " transcode " , " Error while bulk transcode on item $id to $format : " . $e -> getMessage ());
try {
$database -> rollback ();
2019-06-14 12:47:50 +00:00
} catch ( Exception $e ) {
}
2019-06-12 22:54:06 +00:00
}
}
flash_message ( " Transcoded $total items " );
}
break ;
}
}
2019-06-14 12:47:50 +00:00
private function clean_format ( $format ) : ? string
{
if ( array_key_exists ( $format , self :: FORMAT_ALIASES )) {
2019-06-12 22:54:06 +00:00
return self :: FORMAT_ALIASES [ $format ];
}
return $format ;
}
2019-06-14 12:47:50 +00:00
private function can_convert_format ( $engine , $format ) : bool
2019-06-12 22:54:06 +00:00
{
$format = $this -> clean_format ( $format );
2019-06-14 12:47:50 +00:00
if ( ! in_array ( $format , self :: ENGINE_INPUT_SUPPORT [ $engine ])) {
2019-06-12 22:54:06 +00:00
return false ;
}
return true ;
}
2019-06-14 12:47:50 +00:00
private function get_supported_output_formats ( $engine , ? String $omit_format = null ) : array
2019-06-12 22:54:06 +00:00
{
$omit_format = $this -> clean_format ( $omit_format );
$output = [];
2019-06-14 12:47:50 +00:00
foreach ( self :: OUTPUT_FORMATS as $key => $value ) {
if ( $value == " " ) {
2019-06-12 22:54:06 +00:00
$output [ $key ] = $value ;
continue ;
}
2019-06-14 12:47:50 +00:00
if ( in_array ( $value , self :: ENGINE_OUTPUT_SUPPORT [ $engine ])
2019-06-12 22:54:06 +00:00
&& ( empty ( $omit_format ) || $omit_format != $this -> determine_ext ( $value ))) {
$output [ $key ] = $value ;
2019-06-14 12:47:50 +00:00
}
2019-06-12 22:54:06 +00:00
}
return $output ;
}
2019-06-14 12:47:50 +00:00
private function determine_ext ( String $format ) : String
2019-06-12 22:54:06 +00:00
{
2019-06-14 12:47:50 +00:00
switch ( $format ) {
2019-06-12 22:54:06 +00:00
case " webp-lossless " :
case " webp-lossy " :
return " webp " ;
default :
return $format ;
}
}
private function transcode_and_replace_image ( Image $image_obj , String $target_format )
{
$target_format = $this -> clean_format ( $target_format );
$original_file = warehouse_path ( " images " , $image_obj -> hash );
$tmp_filename = $this -> transcode_image ( $original_file , $image_obj -> ext , $target_format );
$new_image = new Image ();
$new_image -> hash = md5_file ( $tmp_filename );
$new_image -> filesize = filesize ( $tmp_filename );
$new_image -> filename = $image_obj -> filename ;
$new_image -> width = $image_obj -> width ;
$new_image -> height = $image_obj -> height ;
$new_image -> ext = $this -> determine_ext ( $target_format );
/* Move the new image into the main storage location */
$target = warehouse_path ( " images " , $new_image -> hash );
if ( !@ copy ( $tmp_filename , $target )) {
throw new ImageTranscodeException ( " Failed to copy new image file from temporary location ( { $tmp_filename } ) to archive ( $target ) " );
}
/* Remove temporary file */
@ unlink ( $tmp_filename );
send_event ( new ImageReplaceEvent ( $image_obj -> id , $new_image ));
2019-06-14 12:47:50 +00:00
}
2019-06-12 22:54:06 +00:00
private function transcode_image ( String $source_name , String $source_format , string $target_format ) : string
{
global $config ;
2019-06-14 12:47:50 +00:00
if ( $source_format == $this -> determine_ext ( $target_format )) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " Source and target formats are the same: " . $source_format );
}
$engine = $config -> get_string ( " transcode_engine " );
2019-06-14 12:47:50 +00:00
if ( ! $this -> can_convert_format ( $engine , $source_format )) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " Engine $engine does not support input format $source_format " );
}
2019-06-14 12:47:50 +00:00
if ( ! in_array ( $target_format , self :: ENGINE_OUTPUT_SUPPORT [ $engine ])) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " Engine $engine does not support output format $target_format " );
}
2019-06-14 12:47:50 +00:00
switch ( $engine ) {
2019-06-12 22:54:06 +00:00
case " gd " :
return $this -> transcode_image_gd ( $source_name , $source_format , $target_format );
case " convert " :
return $this -> transcode_image_convert ( $source_name , $source_format , $target_format );
}
}
private function transcode_image_gd ( String $source_name , String $source_format , string $target_format ) : string
{
global $config ;
$q = $config -> get_int ( " transcode_quality " );
$tmp_name = tempnam ( " /tmp " , " shimmie_transcode " );
$image = imagecreatefromstring ( file_get_contents ( $source_name ));
try {
$result = false ;
2019-06-14 12:47:50 +00:00
switch ( $target_format ) {
2019-06-12 22:54:06 +00:00
case " webp-lossy " :
$result = imagewebp ( $image , $tmp_name , $q );
break ;
case " png " :
$result = imagepng ( $image , $tmp_name , 9 );
break ;
case " jpg " :
// In case of alpha channels
$width = imagesx ( $image );
$height = imagesy ( $image );
$new_image = imagecreatetruecolor ( $width , $height );
2019-06-14 12:47:50 +00:00
if ( $new_image === false ) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " Could not create image with dimensions $width x $height " );
}
2019-06-14 12:47:50 +00:00
try {
$black = imagecolorallocate ( $new_image , 0 , 0 , 0 );
if ( $black === false ) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " Could not allocate background color " );
}
2019-06-14 12:47:50 +00:00
if ( imagefilledrectangle ( $new_image , 0 , 0 , $width , $height , $black ) === false ) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " Could not fill background color " );
}
2019-06-14 12:47:50 +00:00
if ( imagecopy ( $new_image , $image , 0 , 0 , 0 , 0 , $width , $height ) === false ) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " Could not copy source image to new image " );
}
$result = imagejpeg ( $new_image , $tmp_name , $q );
} finally {
imagedestroy ( $new_image );
}
break ;
}
2019-06-14 12:47:50 +00:00
if ( $result === false ) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " Error while transcoding " . $source_name . " to " . $target_format );
}
return $tmp_name ;
} finally {
imagedestroy ( $image );
}
}
private function transcode_image_convert ( String $source_name , String $source_format , string $target_format ) : string
{
global $config ;
$q = $config -> get_int ( " transcode_quality " );
$convert = $config -> get_string ( " thumb_convert_path " );
2019-06-14 12:47:50 +00:00
if ( $convert == null || $convert == " " ) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " ImageMagick path not configured " );
}
$ext = $this -> determine_ext ( $target_format );
2019-06-13 16:45:34 +00:00
$args = " -flatten " ;
2019-06-12 22:54:06 +00:00
$bg = " none " ;
2019-06-14 12:47:50 +00:00
switch ( $target_format ) {
2019-06-12 22:54:06 +00:00
case " webp-lossless " :
2019-06-13 16:45:34 +00:00
$args .= '-define webp:lossless=true' ;
2019-06-12 22:54:06 +00:00
break ;
case " webp-lossy " :
2019-06-13 16:45:34 +00:00
$args .= '' ;
2019-06-12 22:54:06 +00:00
break ;
case " png " :
2019-06-13 16:45:34 +00:00
$args .= '-define png:compression-level=9' ;
2019-06-12 22:54:06 +00:00
break ;
default :
2019-06-13 16:45:34 +00:00
$bg = " black " ;
2019-06-12 22:54:06 +00:00
break ;
}
$tmp_name = tempnam ( " /tmp " , " shimmie_transcode " );
$format = '"%s" %s -quality %u -background %s "%s" %s:"%s"' ;
$cmd = sprintf ( $format , $convert , $args , $q , $bg , $source_name , $ext , $tmp_name );
$cmd = str_replace ( " \" convert \" " , " convert " , $cmd ); // quotes are only needed if the path to convert contains a space; some other times, quotes break things, see github bug #27
exec ( $cmd , $output , $ret );
log_debug ( 'transcode' , " Transcoding with command ` $cmd `, returns $ret " );
2019-06-14 12:47:50 +00:00
if ( $ret !== 0 ) {
2019-06-12 22:54:06 +00:00
throw new ImageTranscodeException ( " Transcoding failed with command " . $cmd );
}
return $tmp_name ;
}
}