1<?php 2////////////////////////////////////////////////////////////// 3// phpThumb() by James Heinrich <info@silisoftware.com> // 4// available at http://phpthumb.sourceforge.net // 5// and/or https://github.com/JamesHeinrich/phpThumb // 6////////////////////////////////////////////////////////////// 7/// // 8// See: phpthumb.readme.txt for usage instructions // 9// /// 10////////////////////////////////////////////////////////////// 11 12if (!class_exists('phpthumb_functions', false)) { 13 ob_start(); 14 if (!include_once __DIR__ . '/phpthumb.functions.php') { 15 ob_end_flush(); 16 die('failed to include_once("' . __DIR__ . '/phpthumb.functions.php")'); 17 } 18 ob_end_clean(); 19} 20 21// make sure all image type constants are defined, even in older PHP versions that don't natively support them 22$predefined_IMG_constants = array( 23 'IMG_GIF' => 1, 24 'IMG_JPG' => 2, // not a typo, both IMG_JPG and IMG_JPEG have a value of "2" 25 'IMG_JPEG' => 2, // not a typo, both IMG_JPG and IMG_JPEG have a value of "2" 26 'IMG_PNG' => 4, 27 'IMG_WBMP' => 8, 28 'IMG_XPM' => 16, 29 'IMG_WEBP' => 32, // PHP 7.0.10 30 'IMG_BMP' => 64, // PHP 7.2.0 31 'IMG_WEBP_LOSSLESS' => 101, // PHP 8.1.0 32 'IMG_AVIF' => 256, // PHP 8.1.0 33); 34$predefined_IMAGETYPE_constants = array( 35 'IMAGETYPE_GIF' => 1, 36 'IMAGETYPE_JPEG' => 2, 37 'IMAGETYPE_PNG' => 3, 38 'IMAGETYPE_SWF' => 4, 39 'IMAGETYPE_PSD' => 5, 40 'IMAGETYPE_BMP' => 6, 41 'IMAGETYPE_TIFF_II' => 7, 42 'IMAGETYPE_TIFF_MM' => 8, 43 'IMAGETYPE_JPC' => 9, 44 'IMAGETYPE_JP2' => 10, 45 'IMAGETYPE_JPX' => 11, 46 'IMAGETYPE_JB2' => 12, 47 'IMAGETYPE_SWC' => 13, 48 'IMAGETYPE_IFF' => 14, 49 'IMAGETYPE_WBMP' => 15, 50 'IMAGETYPE_XBM' => 16, 51 'IMAGETYPE_ICO' => 17, 52 'IMAGETYPE_WEBP' => 18, // PHP 7.0.10 53 'IMAGETYPE_AVIF' => 19, // PHP 8.1.0 54); 55foreach ($predefined_IMG_constants as $PHP_constant_name => $PHP_constant_value) { 56 if (!defined($PHP_constant_name)) { 57 define($PHP_constant_name, $PHP_constant_value); 58 } 59} 60foreach ($predefined_IMAGETYPE_constants as $PHP_constant_name => $PHP_constant_value) { 61 if (!defined($PHP_constant_name)) { 62 define($PHP_constant_name, $PHP_constant_value); 63 } 64} 65unset($predefined_IMG_constants, $predefined_IMAGETYPE_constants, $PHP_constant_name, $PHP_constant_value); 66 67 68class phpthumb { 69 70 // public: 71 // START PARAMETERS (for object mode and phpThumb.php) 72 // See phpthumb.readme.txt for descriptions of what each of these values are 73 public $src = null; // SouRCe filename 74 public $new = null; // NEW image (phpThumb.php only) 75 public $w = null; // Width 76 public $h = null; // Height 77 public $wp = null; // Width (Portrait Images Only) 78 public $hp = null; // Height (Portrait Images Only) 79 public $wl = null; // Width (Landscape Images Only) 80 public $hl = null; // Height (Landscape Images Only) 81 public $ws = null; // Width (Square Images Only) 82 public $hs = null; // Height (Square Images Only) 83 public $f = null; // output image Format 84 public $q = 75; // jpeg output Quality 85 public $sx = null; // Source crop top-left X position 86 public $sy = null; // Source crop top-left Y position 87 public $sw = null; // Source crop Width 88 public $sh = null; // Source crop Height 89 public $zc = null; // Zoom Crop 90 public $ica = null; // Image Crop Auto 91 public $bc = null; // Border Color 92 public $bg = null; // BackGround color 93 public $fltr = array(); // FiLTeRs 94 public $goto = null; // GO TO url after processing 95 public $err = null; // default ERRor image filename 96 public $xto = null; // extract eXif Thumbnail Only 97 public $ra = null; // Rotate by Angle 98 public $ar = null; // Auto Rotate 99 public $aoe = null; // Allow Output Enlargement 100 public $far = null; // Fixed Aspect Ratio 101 public $iar = null; // Ignore Aspect Ratio 102 public $maxb = null; // MAXimum Bytes 103 public $down = null; // DOWNload thumbnail filename 104 public $md5s = null; // MD5 hash of Source image 105 public $sfn = 0; // Source Frame Number 106 public $dpi = 150; // Dots Per Inch for vector source formats 107 public $sia = null; // Save Image As filename 108 109 public $file = null; // >>>deprecated, DO NOT USE, will be removed in future versions<<< 110 111 public $phpThumbDebug = null; 112 // END PARAMETERS 113 114 115 // public: 116 // START CONFIGURATION OPTIONS (for object mode only) 117 // See phpThumb.config.php for descriptions of what each of these settings do 118 119 // * Directory Configuration 120 public $config_cache_directory = null; 121 public $config_cache_directory_depth = 0; 122 public $config_cache_disable_warning = true; 123 public $config_cache_source_enabled = false; 124 public $config_cache_source_directory = null; 125 public $config_temp_directory = null; 126 public $config_document_root = null; 127 128 // * Default output configuration: 129 public $config_output_format = 'jpeg'; 130 public $config_output_maxwidth = 0; 131 public $config_output_maxheight = 0; 132 public $config_output_interlace = true; 133 134 // * Error message configuration 135 public $config_error_image_width = 400; 136 public $config_error_image_height = 100; 137 public $config_error_message_image_default = ''; 138 public $config_error_bgcolor = 'CCCCFF'; 139 public $config_error_textcolor = 'FF0000'; 140 public $config_error_fontsize = 1; 141 public $config_error_die_on_error = false; 142 public $config_error_silent_die_on_error = false; 143 public $config_error_die_on_source_failure = true; 144 145 // * Anti-Hotlink Configuration: 146 public $config_nohotlink_enabled = true; 147 public $config_nohotlink_valid_domains = array(); 148 public $config_nohotlink_erase_image = true; 149 public $config_nohotlink_text_message = 'Off-server thumbnailing is not allowed'; 150 // * Off-server Linking Configuration: 151 public $config_nooffsitelink_enabled = false; 152 public $config_nooffsitelink_valid_domains = array(); 153 public $config_nooffsitelink_require_refer = false; 154 public $config_nooffsitelink_erase_image = true; 155 public $config_nooffsitelink_watermark_src = ''; 156 public $config_nooffsitelink_text_message = 'Off-server linking is not allowed'; 157 158 // * Border & Background default colors 159 public $config_border_hexcolor = '000000'; 160 public $config_background_hexcolor = 'FFFFFF'; 161 162 // * TrueType Fonts 163 public $config_ttf_directory = './fonts'; 164 165 public $config_max_source_pixels = null; 166 public $config_use_exif_thumbnail_for_speed = false; 167 public $config_allow_local_http_src = false; 168 169 public $config_imagemagick_path = null; 170 public $config_prefer_imagemagick = true; 171 public $config_imagemagick_use_thumbnail = true; 172 173 public $config_cache_maxage = null; 174 public $config_cache_maxsize = null; 175 public $config_cache_maxfiles = null; 176 public $config_cache_source_filemtime_ignore_local = false; 177 public $config_cache_source_filemtime_ignore_remote = true; 178 public $config_cache_default_only_suffix = false; 179 public $config_cache_force_passthru = true; 180 public $config_cache_prefix = ''; // default value set in the constructor below 181 182 // * MySQL 183 public $config_mysql_extension = null; 184 public $config_mysql_query = null; 185 public $config_mysql_hostname = null; 186 public $config_mysql_username = null; 187 public $config_mysql_password = null; 188 public $config_mysql_database = null; 189 190 // * Security 191 public $config_high_security_enabled = true; 192 public $config_high_security_password = null; 193 public $config_high_security_url_separator = '&'; 194 public $config_disable_debug = true; 195 public $config_allow_src_above_docroot = false; 196 public $config_allow_src_above_phpthumb = true; 197 public $config_auto_allow_symlinks = true; // allow symlink target directories without explicitly whitelisting them 198 public $config_additional_allowed_dirs = array(); // additional directories to allow source images to be read from 199 public $config_file_create_mask = 0755; 200 public $config_dir_create_mask = 0755; 201 202 // * HTTP fopen 203 public $config_http_fopen_timeout = 10; 204 public $config_http_follow_redirect = true; 205 206 // * Compatability 207 public $config_disable_pathinfo_parsing = false; 208 public $config_disable_imagecopyresampled = false; 209 public $config_disable_onlycreateable_passthru = false; 210 public $config_disable_realpath = false; 211 212 public $config_http_user_agent = 'Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.7.12) Gecko/20050915 Firefox/1.0.7'; 213 214 // END CONFIGURATION OPTIONS 215 216 217 // public: error messages (read-only; persistant) 218 public $debugmessages = array(); 219 public $debugtiming = array(); 220 public $fatalerror = null; 221 222 223 // private: (should not be modified directly) 224 public $thumbnailQuality = 75; 225 public $thumbnailFormat = null; 226 227 public $sourceFilename = null; 228 public $rawImageData = null; 229 public $IMresizedData = null; 230 public $outputImageData = null; 231 232 public $useRawIMoutput = false; 233 234 public $gdimg_output = null; 235 public $gdimg_source = null; 236 237 public $getimagesizeinfo = null; 238 239 public $source_width = null; 240 public $source_height = null; 241 242 public $thumbnailCropX = null; 243 public $thumbnailCropY = null; 244 public $thumbnailCropW = null; 245 public $thumbnailCropH = null; 246 247 public $exif_thumbnail_width = null; 248 public $exif_thumbnail_height = null; 249 public $exif_thumbnail_type = null; 250 public $exif_thumbnail_data = null; 251 public $exif_raw_data = null; 252 253 public $thumbnail_width = null; 254 public $thumbnail_height = null; 255 public $thumbnail_image_width = null; 256 public $thumbnail_image_height = null; 257 258 public $tempFilesToDelete = array(); 259 public $cache_filename = null; 260 261 public $AlphaCapableFormats = array( 'png', 'ico', 'gif', 'webp', 'avif'); 262 public $is_alpha = false; 263 264 public $iswindows = null; 265 public $issafemode = null; 266 public $php_memory_limit = null; 267 268 public $phpthumb_version = '1.7.19-202210110924'; 269 270 ////////////////////////////////////////////////////////////////////// 271 272 // public: constructor 273 public function __construct() { 274 $this->phpThumb(); 275 } 276 277 public function phpThumb() { 278 $this->DebugTimingMessage('phpThumb() constructor', __FILE__, __LINE__); 279 $this->DebugMessage('phpThumb() v'.$this->phpthumb_version, __FILE__, __LINE__); 280 281 foreach (array(ini_get('memory_limit'), get_cfg_var('memory_limit')) as $php_config_memory_limit) { 282 if (!empty($php_config_memory_limit)) { 283 if (strtoupper($php_config_memory_limit[ strlen($php_config_memory_limit) - 1 ]) == 'G') { // PHP memory limit expressed in Gigabytes 284 $php_config_memory_limit = (int) substr($php_config_memory_limit, 0, -1) * 1073741824; 285 } elseif (strtoupper($php_config_memory_limit[ strlen($php_config_memory_limit) - 1 ]) == 'M') { // PHP memory limit expressed in Megabytes 286 $php_config_memory_limit = (int) substr($php_config_memory_limit, 0, -1) * 1048576; 287 } 288 $this->php_memory_limit = max($this->php_memory_limit, $php_config_memory_limit); 289 } 290 } 291 if ($this->php_memory_limit > 0) { // could be "-1" for "no limit" 292 $this->config_max_source_pixels = round($this->php_memory_limit * 0.20); // 20% of memory_limit 293 } 294 295 $this->iswindows = (bool) (strtoupper(substr(PHP_OS, 0, 3)) == 'WIN'); 296 $this->issafemode = (bool) preg_match('#(1|ON)#i', ini_get('safe_mode')); 297 $this->config_document_root = (!empty($_SERVER['DOCUMENT_ROOT']) ? $_SERVER['DOCUMENT_ROOT'] : $this->config_document_root); 298 $this->config_cache_prefix = ( isset($_SERVER['SERVER_NAME']) ? $_SERVER['SERVER_NAME'].'_' : ''); 299 300 $this->purgeTempFiles(); // purge existing temp files if re-initializing object 301 302 $php_sapi_name = strtolower(function_exists('php_sapi_name') ? PHP_SAPI : ''); 303 if ($php_sapi_name == 'cli') { 304 $this->config_allow_src_above_docroot = true; 305 } 306 307 if (!$this->config_disable_debug) { 308 // if debug mode is enabled, force phpThumbDebug output, do not allow normal thumbnails to be generated 309 $this->phpThumbDebug = (null === $this->phpThumbDebug ? 9 : max(1, (int) $this->phpThumbDebug)); 310 } 311 } 312 313 public function __destruct() { 314 $this->purgeTempFiles(); 315 } 316 317 // public: 318 public function purgeTempFiles() { 319 foreach ($this->tempFilesToDelete as $tempFileToDelete) { 320 if (file_exists($tempFileToDelete)) { 321 $this->DebugMessage('Deleting temp file "'.$tempFileToDelete.'"', __FILE__, __LINE__); 322 @unlink($tempFileToDelete); 323 } 324 } 325 $this->tempFilesToDelete = array(); 326 return true; 327 } 328 329 // public: 330 public function setSourceFilename($sourceFilename) { 331 //$this->resetObject(); 332 //$this->rawImageData = null; 333 $this->sourceFilename = $sourceFilename; 334 $this->src = $sourceFilename; 335 if (null === $this->config_output_format) { 336 $sourceFileExtension = strtolower(substr(strrchr($sourceFilename, '.'), 1)); 337 if (preg_match('#^[a-z]{3,4}$#', $sourceFileExtension)) { 338 $this->config_output_format = $sourceFileExtension; 339 $this->DebugMessage('setSourceFilename('.$sourceFilename.') set $this->config_output_format to "'.$sourceFileExtension.'"', __FILE__, __LINE__); 340 } else { 341 $this->DebugMessage('setSourceFilename('.$sourceFilename.') did NOT set $this->config_output_format to "'.$sourceFileExtension.'" because it did not seem like an appropriate image format', __FILE__, __LINE__); 342 } 343 } 344 $this->DebugMessage('setSourceFilename('.$sourceFilename.') set $this->sourceFilename to "'.$this->sourceFilename.'"', __FILE__, __LINE__); 345 return true; 346 } 347 348 // public: 349 public function setSourceData($rawImageData, $sourceFilename='') { 350 //$this->resetObject(); 351 //$this->sourceFilename = null; 352 $this->rawImageData = $rawImageData; 353 $this->DebugMessage('setSourceData() setting $this->rawImageData ('.strlen($this->rawImageData).' bytes; magic="'.substr($this->rawImageData, 0, 4).'" ('.phpthumb_functions::HexCharDisplay(substr($this->rawImageData, 0, 4)).'))', __FILE__, __LINE__); 354 if ($this->config_cache_source_enabled) { 355 $sourceFilename = ($sourceFilename ? $sourceFilename : md5($rawImageData)); 356 if (!is_dir($this->config_cache_source_directory)) { 357 $this->ErrorImage('$this->config_cache_source_directory ('.$this->config_cache_source_directory.') is not a directory'); 358 } elseif (!@is_writable($this->config_cache_source_directory)) { 359 $this->ErrorImage('$this->config_cache_source_directory ('.$this->config_cache_source_directory.') is not writable'); 360 } 361 $this->DebugMessage('setSourceData() attempting to save source image to "'.$this->config_cache_source_directory.DIRECTORY_SEPARATOR.urlencode($sourceFilename).'"', __FILE__, __LINE__); 362 if ($fp = @fopen($this->config_cache_source_directory.DIRECTORY_SEPARATOR.urlencode($sourceFilename), 'wb')) { 363 fwrite($fp, $rawImageData); 364 fclose($fp); 365 } elseif (!$this->phpThumbDebug) { 366 $this->ErrorImage('setSourceData() failed to write to source cache ('.$this->config_cache_source_directory.DIRECTORY_SEPARATOR.urlencode($sourceFilename).')'); 367 } 368 } 369 return true; 370 } 371 372 // public: 373 public function setSourceImageResource($gdimg) { 374 //$this->resetObject(); 375 $this->gdimg_source = $gdimg; 376 return true; 377 } 378 379 // public: 380 public function setParameter($param, $value) { 381 if ($param == 'src') { 382 $this->setSourceFilename($this->ResolveFilenameToAbsolute($value)); 383 } elseif (@is_array($this->$param)) { 384 if (is_array($value)) { 385 foreach ($value as $arraykey => $arrayvalue) { 386 array_push($this->$param, $arrayvalue); 387 } 388 } else { 389 array_push($this->$param, $value); 390 } 391 } else { 392 $this->$param = $value; 393 } 394 return true; 395 } 396 397 // public: 398 public function getParameter($param) { 399 //if (property_exists('phpThumb', $param)) { 400 return $this->$param; 401 //} 402 //$this->DebugMessage('setParameter() attempting to get non-existant parameter "'.$param.'"', __FILE__, __LINE__); 403 //return false; 404 } 405 406 407 // public: 408 public function GenerateThumbnail() { 409 410 $this->setOutputFormat(); 411 $this->phpThumbDebug('8a'); 412 $this->ResolveSource(); 413 $this->phpThumbDebug('8b'); 414 $this->SetCacheFilename(); 415 $this->phpThumbDebug('8c'); 416 $this->ExtractEXIFgetImageSize(); 417 $this->phpThumbDebug('8d'); 418 if ($this->useRawIMoutput) { 419 $this->DebugMessage('Skipping rest of GenerateThumbnail() because ($this->useRawIMoutput == true)', __FILE__, __LINE__); 420 return true; 421 } 422 $this->phpThumbDebug('8e'); 423 if (!$this->SourceImageToGD()) { 424 $this->DebugMessage('SourceImageToGD() failed', __FILE__, __LINE__); 425 return false; 426 } 427 $this->phpThumbDebug('8f'); 428 $this->ImageCropAuto(); 429 $this->phpThumbDebug('8h'); 430 $this->Rotate(); 431 $this->phpThumbDebug('8h'); 432 $this->CreateGDoutput(); 433 $this->phpThumbDebug('8i'); 434 435 // default values, also applicable for far="C" 436 $destination_offset_x = round(($this->thumbnail_width - $this->thumbnail_image_width) / 2); 437 $destination_offset_y = round(($this->thumbnail_height - $this->thumbnail_image_height) / 2); 438 if (($this->far == 'L') || ($this->far == 'TL') || ($this->far == 'BL')) { 439 $destination_offset_x = 0; 440 } 441 if (($this->far == 'R') || ($this->far == 'TR') || ($this->far == 'BR')) { 442 $destination_offset_x = round($this->thumbnail_width - $this->thumbnail_image_width); 443 } 444 if (($this->far == 'T') || ($this->far == 'TL') || ($this->far == 'TR')) { 445 $destination_offset_y = 0; 446 } 447 if (($this->far == 'B') || ($this->far == 'BL') || ($this->far == 'BR')) { 448 $destination_offset_y = round($this->thumbnail_height - $this->thumbnail_image_height); 449 } 450 451// // copy/resize image to appropriate dimensions 452// $borderThickness = 0; 453// if (!empty($this->fltr)) { 454// foreach ($this->fltr as $key => $value) { 455// if (preg_match('#^bord\|([0-9]+)#', $value, $matches)) { 456// $borderThickness = $matches[1]; 457// break; 458// } 459// } 460// } 461// if ($borderThickness > 0) { 462// //$this->DebugMessage('Skipping ImageResizeFunction() because BorderThickness="'.$borderThickness.'"', __FILE__, __LINE__); 463// $this->thumbnail_image_height /= 2; 464// } 465 $this->ImageResizeFunction( 466 $this->gdimg_output, 467 $this->gdimg_source, 468 $destination_offset_x, 469 $destination_offset_y, 470 $this->thumbnailCropX, 471 $this->thumbnailCropY, 472 $this->thumbnail_image_width, 473 $this->thumbnail_image_height, 474 $this->thumbnailCropW, 475 $this->thumbnailCropH 476 ); 477 478 $this->DebugMessage('memory_get_usage() after copy-resize = '.(function_exists('memory_get_usage') ? @memory_get_usage() : 'n/a'), __FILE__, __LINE__); 479 imagedestroy($this->gdimg_source); 480 $this->DebugMessage('memory_get_usage() after imagedestroy = '.(function_exists('memory_get_usage') ? @memory_get_usage() : 'n/a'), __FILE__, __LINE__); 481 482 $this->phpThumbDebug('8i'); 483 $this->AntiOffsiteLinking(); 484 $this->phpThumbDebug('8j'); 485 $this->ApplyFilters(); 486 $this->phpThumbDebug('8k'); 487 $this->AlphaChannelFlatten(); 488 $this->phpThumbDebug('8l'); 489 $this->MaxFileSize(); 490 $this->phpThumbDebug('8m'); 491 492 $this->DebugMessage('GenerateThumbnail() completed successfully', __FILE__, __LINE__); 493 return true; 494 } 495 496 497 // public: 498 public function RenderOutput() { 499 if (!$this->useRawIMoutput && !(is_resource($this->gdimg_output) || (is_object($this->gdimg_source) && $this->gdimg_source instanceOf \GdImage))) { 500 $this->DebugMessage('RenderOutput() failed because !is_resource($this->gdimg_output)', __FILE__, __LINE__); 501 return false; 502 } 503 if (!$this->thumbnailFormat) { 504 $this->DebugMessage('RenderOutput() failed because $this->thumbnailFormat is empty', __FILE__, __LINE__); 505 return false; 506 } 507 if ($this->useRawIMoutput) { 508 $this->DebugMessage('RenderOutput copying $this->IMresizedData ('.strlen($this->IMresizedData).' bytes) to $this->outputImage', __FILE__, __LINE__); 509 $this->outputImageData = $this->IMresizedData; 510 return true; 511 } 512 513 $builtin_formats = array(); 514 if (function_exists('imagetypes')) { 515 $imagetypes = imagetypes(); 516 $builtin_formats['wbmp'] = (bool) ($imagetypes & IMG_WBMP); 517 $builtin_formats['jpg'] = (bool) ($imagetypes & IMG_JPG); 518 $builtin_formats['gif'] = (bool) ($imagetypes & IMG_GIF); 519 $builtin_formats['png'] = (bool) ($imagetypes & IMG_PNG); 520 $builtin_formats['webp'] = (bool) ($imagetypes & IMG_WEBP); // PHP 7.0.10 521 $builtin_formats['bmp'] = (bool) ($imagetypes & IMG_BMP); // PHP 7.2.0 522 $builtin_formats['avif'] = (bool) ($imagetypes & IMG_AVIF); // PHP 8.1.0 523 } 524 525 $this->DebugMessage('imageinterlace($this->gdimg_output, '. (int) $this->config_output_interlace .')', __FILE__, __LINE__); 526 imageinterlace($this->gdimg_output, (int) $this->config_output_interlace); 527 528 $this->DebugMessage('RenderOutput() attempting image'.strtolower(@$this->thumbnailFormat).'($this->gdimg_output)', __FILE__, __LINE__); 529 ob_start(); 530 switch ($this->thumbnailFormat) { 531 case 'wbmp': 532 if (empty($builtin_formats['wbmp'])) { 533 $this->DebugMessage('GD does not have required built-in support for WBMP output', __FILE__, __LINE__); 534 ob_end_clean(); 535 return false; 536 } 537 imagewbmp($this->gdimg_output, null, $this->thumbnailQuality); 538 $this->outputImageData = ob_get_contents(); 539 break; 540 541 case 'jpeg': 542 case 'jpg': // should be "jpeg" not "jpg" but just in case... 543 if (empty($builtin_formats['jpg'])) { 544 $this->DebugMessage('GD does not have required built-in support for JPEG output', __FILE__, __LINE__); 545 ob_end_clean(); 546 return false; 547 } 548 imagejpeg($this->gdimg_output, null, $this->thumbnailQuality); 549 $this->outputImageData = ob_get_contents(); 550 break; 551 552 case 'png': 553 if (empty($builtin_formats['png'])) { 554 $this->DebugMessage('GD does not have required built-in support for PNG output', __FILE__, __LINE__); 555 ob_end_clean(); 556 return false; 557 } 558 if (phpthumb_functions::version_compare_replacement(PHP_VERSION, '5.1.2', '>=')) { 559 // https://github.com/JamesHeinrich/phpThumb/issues/24 560 561 /* http://php.net/manual/en/function.imagepng.php: 562 from php source (gd.h): 563 2.0.12: Compression level: 0-9 or -1, where 0 is NO COMPRESSION at all, 564 :: 1 is FASTEST but produces larger files, 9 provides the best 565 :: compression (smallest files) but takes a long time to compress, and 566 :: -1 selects the default compiled into the zlib library. 567 Conclusion: Based on the Zlib manual (http://www.zlib.net/manual.html) the default compression level is set to 6. 568 */ 569 if (($this->thumbnailQuality >= -1) && ($this->thumbnailQuality <= 9)) { 570 $PNGquality = $this->thumbnailQuality; 571 } else { 572 $this->DebugMessage('Specified thumbnailQuality "'.$this->thumbnailQuality.'" is outside the accepted range (0-9, or -1). Using 6 as default value.', __FILE__, __LINE__); 573 $PNGquality = 6; 574 } 575 imagepng($this->gdimg_output, null, $PNGquality); 576 } else { 577 imagepng($this->gdimg_output); 578 } 579 $this->outputImageData = ob_get_contents(); 580 break; 581 582 case 'gif': 583 if (empty($builtin_formats['gif'])) { 584 $this->DebugMessage('GD does not have required built-in support for GIF output', __FILE__, __LINE__); 585 ob_end_clean(); 586 return false; 587 } 588 imagegif($this->gdimg_output); 589 $this->outputImageData = ob_get_contents(); 590 break; 591 592 case 'webp': 593 if (empty($builtin_formats['webp'])) { 594 $this->DebugMessage('GD does not have required built-in support for WebP output', __FILE__, __LINE__); 595 ob_end_clean(); 596 return false; 597 } 598 imagewebp($this->gdimg_output, null, $this->thumbnailQuality); 599 $this->outputImageData = ob_get_contents(); 600 break; 601 602 case 'avif': 603 if (empty($builtin_formats['avif'])) { 604 $this->DebugMessage('GD does not have required built-in support for AVIF output', __FILE__, __LINE__); 605 ob_end_clean(); 606 return false; 607 } 608 imageavif($this->gdimg_output, null, $this->thumbnailQuality); 609 $this->outputImageData = ob_get_contents(); 610 break; 611 612 case 'bmp': 613 if (!empty($builtin_formats['bmp'])) { 614 imagebmp($this->gdimg_output); 615 $this->outputImageData = ob_get_contents(); 616 break; 617 } 618 $this->DebugMessage('GD does not have required built-in support for BMP output', __FILE__, __LINE__); 619 if (!@include_once __DIR__ .'/phpthumb.bmp.php' ) { 620 $this->DebugMessage('Error including "'. __DIR__ .'/phpthumb.bmp.php" which is required for BMP format output', __FILE__, __LINE__); 621 ob_end_clean(); 622 return false; 623 } 624 $phpthumb_bmp = new phpthumb_bmp(); 625 $this->outputImageData = $phpthumb_bmp->GD2BMPstring($this->gdimg_output); 626 unset($phpthumb_bmp); 627 break; 628 629 case 'ico': 630 if (!@include_once __DIR__ .'/phpthumb.ico.php' ) { 631 $this->DebugMessage('Error including "'. __DIR__ .'/phpthumb.ico.php" which is required for ICO format output', __FILE__, __LINE__); 632 ob_end_clean(); 633 return false; 634 } 635 $phpthumb_ico = new phpthumb_ico(); 636 $arrayOfOutputImages = array($this->gdimg_output); 637 $this->outputImageData = $phpthumb_ico->GD2ICOstring($arrayOfOutputImages); 638 unset($phpthumb_ico); 639 break; 640 641 default: 642 $this->DebugMessage('RenderOutput failed because $this->thumbnailFormat "'.$this->thumbnailFormat.'" is not valid', __FILE__, __LINE__); 643 ob_end_clean(); 644 return false; 645 } 646 ob_end_clean(); 647 if (!$this->outputImageData) { 648 $this->DebugMessage('RenderOutput() for "'.$this->thumbnailFormat.'" failed', __FILE__, __LINE__); 649 ob_end_clean(); 650 return false; 651 } 652 $this->DebugMessage('RenderOutput() completing with $this->outputImageData = '.strlen($this->outputImageData).' bytes', __FILE__, __LINE__); 653 return true; 654 } 655 656 657 // public: 658 public function RenderToFile($filename) { 659 if (preg_match('#^[a-z0-9]+://#i', $filename)) { 660 $this->DebugMessage('RenderToFile() failed because $filename ('.$filename.') is a URL', __FILE__, __LINE__); 661 return false; 662 } 663 // render thumbnail to this file only, do not cache, do not output to browser 664 //$renderfilename = $this->ResolveFilenameToAbsolute(dirname($filename)).DIRECTORY_SEPARATOR.basename($filename); 665 $renderfilename = $filename; 666 if (($filename[0] != '/') && ($filename[0] != '\\') && ($filename[1] != ':')) { 667 $renderfilename = $this->ResolveFilenameToAbsolute($renderfilename); 668 } 669 if (!@is_writable(dirname($renderfilename))) { 670 $this->DebugMessage('RenderToFile() failed because "'.dirname($renderfilename).'/" is not writable', __FILE__, __LINE__); 671 return false; 672 } 673 if (@is_file($renderfilename) && !@is_writable($renderfilename)) { 674 $this->DebugMessage('RenderToFile() failed because "'.$renderfilename.'" is not writable', __FILE__, __LINE__); 675 return false; 676 } 677 678 if ($this->RenderOutput()) { 679 if (file_put_contents($renderfilename, $this->outputImageData)) { 680 @chmod($renderfilename, $this->getParameter('config_file_create_mask')); 681 $this->DebugMessage('RenderToFile('.$renderfilename.') succeeded', __FILE__, __LINE__); 682 return true; 683 } 684 if (!@file_exists($renderfilename)) { 685 $this->DebugMessage('RenderOutput ['.$this->thumbnailFormat.'('.$renderfilename.')] did not appear to fail, but the output image does not exist either...', __FILE__, __LINE__); 686 } 687 } else { 688 $this->DebugMessage('RenderOutput ['.$this->thumbnailFormat.'('.$renderfilename.')] failed', __FILE__, __LINE__); 689 } 690 return false; 691 } 692 693 694 // public: 695 public function OutputThumbnail() { 696 $this->purgeTempFiles(); 697 698 if (!$this->useRawIMoutput && !(is_resource($this->gdimg_output) || (is_object($this->gdimg_source) && $this->gdimg_source instanceOf \GdImage))) { 699 $this->DebugMessage('OutputThumbnail() failed because !is_resource($this->gdimg_output)', __FILE__, __LINE__); 700 return false; 701 } 702 if (headers_sent()) { 703 return $this->ErrorImage('OutputThumbnail() failed - headers already sent'); 704 } 705 706 $downloadfilename = phpthumb_functions::SanitizeFilename(is_string($this->sia) ? $this->sia : ($this->down ? $this->down : 'phpThumb_generated_thumbnail'.'.'.$this->thumbnailFormat)); 707 $this->DebugMessage('Content-Disposition header filename set to "'.$downloadfilename.'"', __FILE__, __LINE__); 708 if ($downloadfilename) { 709 header('Content-Disposition: '.($this->down ? 'attachment' : 'inline').'; filename="'.$downloadfilename.'"'); 710 } else { 711 $this->DebugMessage('failed to send Content-Disposition header because $downloadfilename is empty', __FILE__, __LINE__); 712 } 713 714 if ($this->useRawIMoutput) { 715 716 header('Content-Type: '.phpthumb_functions::ImageTypeToMIMEtype($this->thumbnailFormat)); 717 echo $this->IMresizedData; 718 719 } else { 720 721 $this->DebugMessage('imageinterlace($this->gdimg_output, '. (int) $this->config_output_interlace .')', __FILE__, __LINE__); 722 imageinterlace($this->gdimg_output, (int) $this->config_output_interlace); 723 switch ($this->thumbnailFormat) { 724 case 'gif': 725 case 'jpeg': 726 case 'png': 727 case 'webp': 728 case 'avif': 729 $ImageOutFunction = 'image'.$this->thumbnailFormat; 730 if (!function_exists($ImageOutFunction)) { 731 $this->DebugMessage($ImageOutFunction.' is not available', __FILE__, __LINE__); 732 return false; 733 } 734 header('Content-Type: '.phpthumb_functions::ImageTypeToMIMEtype($this->thumbnailFormat)); 735 if ($this->thumbnailFormat == 'gif') { 736 @$ImageOutFunction($this->gdimg_output); 737 } else { 738 @$ImageOutFunction($this->gdimg_output, null, $this->thumbnailQuality); 739 } 740 break; 741 742 case 'bmp': 743 if (function_exists('imagebmp')) { 744 header('Content-Type: '.phpthumb_functions::ImageTypeToMIMEtype($this->thumbnailFormat)); 745 imagebmp($this->gdimg_output); 746 break; 747 } 748 if (!@include_once __DIR__ .'/phpthumb.bmp.php' ) { 749 $this->DebugMessage('Error including "'. __DIR__ .'/phpthumb.bmp.php" which is required for BMP format output', __FILE__, __LINE__); 750 return false; 751 } 752 $phpthumb_bmp = new phpthumb_bmp(); 753 if (is_object($phpthumb_bmp)) { 754 $bmp_data = $phpthumb_bmp->GD2BMPstring($this->gdimg_output); 755 unset($phpthumb_bmp); 756 if (!$bmp_data) { 757 $this->DebugMessage('$phpthumb_bmp->GD2BMPstring() failed', __FILE__, __LINE__); 758 return false; 759 } 760 header('Content-Type: '.phpthumb_functions::ImageTypeToMIMEtype($this->thumbnailFormat)); 761 echo $bmp_data; 762 } else { 763 $this->DebugMessage('new phpthumb_bmp() failed', __FILE__, __LINE__); 764 return false; 765 } 766 break; 767 768 case 'ico': 769 if (!@include_once __DIR__ .'/phpthumb.ico.php' ) { 770 $this->DebugMessage('Error including "'. __DIR__ .'/phpthumb.ico.php" which is required for ICO format output', __FILE__, __LINE__); 771 return false; 772 } 773 $phpthumb_ico = new phpthumb_ico(); 774 if (is_object($phpthumb_ico)) { 775 $arrayOfOutputImages = array($this->gdimg_output); 776 $ico_data = $phpthumb_ico->GD2ICOstring($arrayOfOutputImages); 777 unset($phpthumb_ico); 778 if (!$ico_data) { 779 $this->DebugMessage('$phpthumb_ico->GD2ICOstring() failed', __FILE__, __LINE__); 780 return false; 781 } 782 header('Content-Type: '.phpthumb_functions::ImageTypeToMIMEtype($this->thumbnailFormat)); 783 echo $ico_data; 784 } else { 785 $this->DebugMessage('new phpthumb_ico() failed', __FILE__, __LINE__); 786 return false; 787 } 788 break; 789 790 default: 791 $this->DebugMessage('OutputThumbnail failed because $this->thumbnailFormat "'.$this->thumbnailFormat.'" is not valid', __FILE__, __LINE__); 792 return false; 793 break; 794 } 795 796 } 797 return true; 798 } 799 800 801 // public: 802 public function CleanUpCacheDirectory() { 803 $this->DebugMessage('CleanUpCacheDirectory() set to purge ('.(null === $this->config_cache_maxage ? 'NULL' : number_format($this->config_cache_maxage / 86400, 1)).' days; '.(null === $this->config_cache_maxsize ? 'NULL' : number_format($this->config_cache_maxsize / 1048576, 2)).' MB; '.(null === $this->config_cache_maxfiles ? 'NULL' : number_format($this->config_cache_maxfiles)).' files)', __FILE__, __LINE__); 804 805 if (!is_writable($this->config_cache_directory)) { 806 $this->DebugMessage('CleanUpCacheDirectory() skipped because "'.$this->config_cache_directory.'" is not writable', __FILE__, __LINE__); 807 return true; 808 } 809 810 // cache status of cache directory for 1 hour to avoid hammering the filesystem functions 811 $phpThumbCacheStats_filename = $this->config_cache_directory.DIRECTORY_SEPARATOR.'phpThumbCacheStats.txt'; 812 if (file_exists($phpThumbCacheStats_filename) && is_readable($phpThumbCacheStats_filename) && (filemtime($phpThumbCacheStats_filename) >= (time() - 3600))) { 813 $this->DebugMessage('CleanUpCacheDirectory() skipped because "'.$phpThumbCacheStats_filename.'" is recently modified', __FILE__, __LINE__); 814 return true; 815 } 816 if (!@touch($phpThumbCacheStats_filename)) { 817 $this->DebugMessage('touch('.$phpThumbCacheStats_filename.') failed', __FILE__, __LINE__); 818 } 819 820 $DeletedKeys = array(); 821 $AllFilesInCacheDirectory = array(); 822 if (($this->config_cache_maxage > 0) || ($this->config_cache_maxsize > 0) || ($this->config_cache_maxfiles > 0)) { 823 $CacheDirOldFilesAge = array(); 824 $CacheDirOldFilesSize = array(); 825 $AllFilesInCacheDirectory = phpthumb_functions::GetAllFilesInSubfolders($this->config_cache_directory); 826 foreach ($AllFilesInCacheDirectory as $fullfilename) { 827 if (preg_match('#'.preg_quote($this->config_cache_prefix).'#i', $fullfilename) && file_exists($fullfilename)) { 828 $CacheDirOldFilesAge[$fullfilename] = @fileatime($fullfilename); 829 if ($CacheDirOldFilesAge[$fullfilename] == 0) { 830 $CacheDirOldFilesAge[$fullfilename] = @filemtime($fullfilename); 831 } 832 $CacheDirOldFilesSize[$fullfilename] = @filesize($fullfilename); 833 } 834 } 835 if (empty($CacheDirOldFilesSize)) { 836 $this->DebugMessage('CleanUpCacheDirectory() skipped because $CacheDirOldFilesSize is empty (phpthumb_functions::GetAllFilesInSubfolders('.$this->config_cache_directory.') found no files)', __FILE__, __LINE__); 837 return true; 838 } 839 $DeletedKeys['zerobyte'] = array(); 840 foreach ($CacheDirOldFilesSize as $fullfilename => $filesize) { 841 // purge all zero-size files more than an hour old (to prevent trying to delete just-created and/or in-use files) 842 $cutofftime = time() - 3600; 843 if (($filesize == 0) && ($CacheDirOldFilesAge[$fullfilename] < $cutofftime)) { 844 $this->DebugMessage('deleting "'.$fullfilename.'"', __FILE__, __LINE__); 845 if (@unlink($fullfilename)) { 846 $DeletedKeys['zerobyte'][] = $fullfilename; 847 unset($CacheDirOldFilesSize[$fullfilename]); 848 unset($CacheDirOldFilesAge[$fullfilename]); 849 } 850 } 851 } 852 $this->DebugMessage('CleanUpCacheDirectory() purged '.count($DeletedKeys['zerobyte']).' zero-byte files', __FILE__, __LINE__); 853 asort($CacheDirOldFilesAge); 854 855 if ($this->config_cache_maxfiles > 0) { 856 $TotalCachedFiles = count($CacheDirOldFilesAge); 857 $DeletedKeys['maxfiles'] = array(); 858 foreach ($CacheDirOldFilesAge as $fullfilename => $filedate) { 859 if ($TotalCachedFiles > $this->config_cache_maxfiles) { 860 $this->DebugMessage('deleting "'.$fullfilename.'"', __FILE__, __LINE__); 861 if (@unlink($fullfilename)) { 862 $TotalCachedFiles--; 863 $DeletedKeys['maxfiles'][] = $fullfilename; 864 } 865 } else { 866 // there are few enough files to keep the rest 867 break; 868 } 869 } 870 $this->DebugMessage('CleanUpCacheDirectory() purged '.count($DeletedKeys['maxfiles']).' files based on (config_cache_maxfiles='.$this->config_cache_maxfiles.')', __FILE__, __LINE__); 871 foreach ($DeletedKeys['maxfiles'] as $fullfilename) { 872 unset($CacheDirOldFilesAge[$fullfilename]); 873 unset($CacheDirOldFilesSize[$fullfilename]); 874 } 875 } 876 877 if ($this->config_cache_maxage > 0) { 878 $mindate = time() - $this->config_cache_maxage; 879 $DeletedKeys['maxage'] = array(); 880 foreach ($CacheDirOldFilesAge as $fullfilename => $filedate) { 881 if ($filedate > 0) { 882 if ($filedate < $mindate) { 883 $this->DebugMessage('deleting "'.$fullfilename.'"', __FILE__, __LINE__); 884 if (@unlink($fullfilename)) { 885 $DeletedKeys['maxage'][] = $fullfilename; 886 } 887 } else { 888 // the rest of the files are new enough to keep 889 break; 890 } 891 } 892 } 893 $this->DebugMessage('CleanUpCacheDirectory() purged '.count($DeletedKeys['maxage']).' files based on (config_cache_maxage='.$this->config_cache_maxage.')', __FILE__, __LINE__); 894 foreach ($DeletedKeys['maxage'] as $fullfilename) { 895 unset($CacheDirOldFilesAge[$fullfilename]); 896 unset($CacheDirOldFilesSize[$fullfilename]); 897 } 898 } 899 900 if ($this->config_cache_maxsize > 0) { 901 $TotalCachedFileSize = array_sum($CacheDirOldFilesSize); 902 $DeletedKeys['maxsize'] = array(); 903 foreach ($CacheDirOldFilesAge as $fullfilename => $filedate) { 904 if ($TotalCachedFileSize > $this->config_cache_maxsize) { 905 $this->DebugMessage('deleting "'.$fullfilename.'"', __FILE__, __LINE__); 906 if (@unlink($fullfilename)) { 907 $TotalCachedFileSize -= $CacheDirOldFilesSize[$fullfilename]; 908 $DeletedKeys['maxsize'][] = $fullfilename; 909 } 910 } else { 911 // the total filesizes are small enough to keep the rest of the files 912 break; 913 } 914 } 915 $this->DebugMessage('CleanUpCacheDirectory() purged '.count($DeletedKeys['maxsize']).' files based on (config_cache_maxsize='.$this->config_cache_maxsize.')', __FILE__, __LINE__); 916 foreach ($DeletedKeys['maxsize'] as $fullfilename) { 917 unset($CacheDirOldFilesAge[$fullfilename]); 918 unset($CacheDirOldFilesSize[$fullfilename]); 919 } 920 } 921 922 } else { 923 $this->DebugMessage('skipping CleanUpCacheDirectory() because config set to not use it', __FILE__, __LINE__); 924 } 925 $totalpurged = 0; 926 foreach ($DeletedKeys as $key => $value) { 927 $totalpurged += count($value); 928 } 929 $this->DebugMessage('CleanUpCacheDirectory() purged '.$totalpurged.' files (from '.count($AllFilesInCacheDirectory).') based on config settings', __FILE__, __LINE__); 930 if ($totalpurged > 0) { 931 $empty_dirs = array(); 932 foreach ($AllFilesInCacheDirectory as $fullfilename) { 933 if (is_dir($fullfilename)) { 934 $empty_dirs[$this->realPathSafe($fullfilename)] = 1; 935 } else { 936 unset($empty_dirs[$this->realPathSafe(dirname($fullfilename))]); 937 } 938 } 939 krsort($empty_dirs); 940 $totalpurgeddirs = 0; 941 foreach ($empty_dirs as $empty_dir => $dummy) { 942 if ($empty_dir == $this->config_cache_directory) { 943 // shouldn't happen, but just in case, don't let it delete actual cache directory 944 continue; 945 } elseif (@rmdir($empty_dir)) { 946 $totalpurgeddirs++; 947 } else { 948 $this->DebugMessage('failed to rmdir('.$empty_dir.')', __FILE__, __LINE__); 949 } 950 } 951 $this->DebugMessage('purged '.$totalpurgeddirs.' empty directories', __FILE__, __LINE__); 952 } 953 return true; 954 } 955 956 ////////////////////////////////////////////////////////////////////// 957 958 // private: re-initializator (call between rendering multiple images with one object) 959 public function resetObject() { 960 $class_vars = get_class_vars(get_class($this)); 961 foreach ($class_vars as $key => $value) { 962 // do not clobber debug or config info 963 if (!preg_match('#^(config_|debug|fatalerror)#i', $key)) { 964 $this->$key = $value; 965 } 966 } 967 $this->phpThumb(); // re-initialize some class variables 968 return true; 969 } 970 971 ////////////////////////////////////////////////////////////////////// 972 973 public function ResolveSource() { 974 if (is_resource($this->gdimg_source) || (is_object($this->gdimg_source) && $this->gdimg_source instanceOf \GdImage)) { 975 $this->DebugMessage('ResolveSource() exiting because is_resource($this->gdimg_source)', __FILE__, __LINE__); 976 return true; 977 } 978 if ($this->rawImageData) { 979 $this->sourceFilename = null; 980 $this->DebugMessage('ResolveSource() exiting because $this->rawImageData is set ('.number_format(strlen($this->rawImageData)).' bytes)', __FILE__, __LINE__); 981 return true; 982 } 983 if ($this->sourceFilename) { 984 $this->sourceFilename = $this->ResolveFilenameToAbsolute($this->sourceFilename); 985 $this->DebugMessage('$this->sourceFilename set to "'.$this->sourceFilename.'"', __FILE__, __LINE__); 986 } elseif ($this->src) { 987 $this->sourceFilename = $this->ResolveFilenameToAbsolute($this->src); 988 $this->DebugMessage('$this->sourceFilename set to "'.$this->sourceFilename.'" from $this->src ('.$this->src.')', __FILE__, __LINE__); 989 } else { 990 return $this->ErrorImage('$this->sourceFilename and $this->src are both empty'); 991 } 992 if ($this->iswindows && ((substr($this->sourceFilename, 0, 2) == '//') || (substr($this->sourceFilename, 0, 2) == '\\\\'))) { 993 // Windows \\share\filename.ext 994 } elseif (preg_match('#^[a-z0-9]+://#i', $this->sourceFilename, $protocol_matches)) { 995 if (preg_match('#^(f|ht)tps?\://#i', $this->sourceFilename)) { 996 // URL 997 if ($this->config_http_user_agent) { 998 ini_set('user_agent', $this->config_http_user_agent); 999 } 1000 } else { 1001 return $this->ErrorImage('only FTP and HTTP/HTTPS protocols are allowed, "'.$protocol_matches[1].'" is not'); 1002 } 1003 } elseif (!@file_exists($this->sourceFilename)) { 1004 return $this->ErrorImage('"'.$this->sourceFilename.'" does not exist'); 1005 } elseif (!@is_file($this->sourceFilename)) { 1006 return $this->ErrorImage('"'.$this->sourceFilename.'" is not a file'); 1007 } 1008 return true; 1009 } 1010 1011 1012 public function setOutputFormat() { 1013 static $alreadyCalled = false; 1014 if ($this->thumbnailFormat && $alreadyCalled) { 1015 return true; 1016 } 1017 $alreadyCalled = true; 1018 1019 $AvailableImageOutputFormats = array(); 1020 $AvailableImageOutputFormats[] = 'text'; 1021 if (@is_readable( __DIR__ .'/phpthumb.ico.php')) { 1022 $AvailableImageOutputFormats[] = 'ico'; 1023 } 1024 if (@is_readable( __DIR__ .'/phpthumb.bmp.php')) { 1025 $AvailableImageOutputFormats[] = 'bmp'; 1026 } 1027 1028 $this->thumbnailFormat = 'ico'; 1029 1030 // Set default output format based on what image types are available 1031 if (function_exists('imagetypes')) { 1032 $imagetypes = imagetypes(); 1033 if ($imagetypes & IMG_WBMP) { 1034 $this->thumbnailFormat = 'wbmp'; 1035 $AvailableImageOutputFormats[] = 'wbmp'; 1036 } 1037 if ($imagetypes & IMG_GIF) { 1038 $this->thumbnailFormat = 'gif'; 1039 $AvailableImageOutputFormats[] = 'gif'; 1040 } 1041 if ($imagetypes & IMG_AVIF) { 1042 $this->thumbnailFormat = 'avif'; 1043 $AvailableImageOutputFormats[] = 'avif'; 1044 } 1045 if ($imagetypes & IMG_WEBP) { 1046 $this->thumbnailFormat = 'webp'; 1047 $AvailableImageOutputFormats[] = 'webp'; 1048 } 1049 if ($imagetypes & IMG_PNG) { 1050 $this->thumbnailFormat = 'png'; 1051 $AvailableImageOutputFormats[] = 'png'; 1052 } 1053 if ($imagetypes & IMG_JPG) { 1054 $this->thumbnailFormat = 'jpeg'; 1055 $AvailableImageOutputFormats[] = 'jpeg'; 1056 } 1057 } else { 1058 $this->DebugMessage('imagetypes() does not exist - GD support might not be enabled?', __FILE__, __LINE__); 1059 } 1060 if ($this->ImageMagickVersion()) { 1061 $IMformats = array('jpeg', 'png', 'gif', 'bmp', 'ico', 'wbmp', 'webp', 'avif'); 1062 $this->DebugMessage('Addding ImageMagick formats to $AvailableImageOutputFormats ('.implode(';', $AvailableImageOutputFormats).')', __FILE__, __LINE__); 1063 foreach ($IMformats as $key => $format) { 1064 $AvailableImageOutputFormats[] = $format; 1065 } 1066 } 1067 $AvailableImageOutputFormats = array_unique($AvailableImageOutputFormats); 1068 $this->DebugMessage('$AvailableImageOutputFormats = array('.implode(';', $AvailableImageOutputFormats).')', __FILE__, __LINE__); 1069 1070 $this->f = (!empty($this->f) ? $this->f : ''); 1071 $this->f = preg_replace('#[^a-z]#', '', strtolower($this->f)); 1072 if (strtolower($this->config_output_format) == 'jpg') { 1073 $this->config_output_format = 'jpeg'; 1074 } 1075 if (strtolower($this->f) == 'jpg') { 1076 $this->f = 'jpeg'; 1077 } 1078 if (phpthumb_functions::CaseInsensitiveInArray($this->config_output_format, $AvailableImageOutputFormats)) { 1079 // set output format to config default if that format is available 1080 $this->DebugMessage('$this->thumbnailFormat set to $this->config_output_format "'.strtolower($this->config_output_format).'"', __FILE__, __LINE__); 1081 $this->thumbnailFormat = strtolower($this->config_output_format); 1082 } elseif ($this->config_output_format) { 1083 $this->DebugMessage('$this->thumbnailFormat staying as "'.$this->thumbnailFormat.'" because $this->config_output_format ('.strtolower($this->config_output_format).') is not in $AvailableImageOutputFormats', __FILE__, __LINE__); 1084 } 1085 if ($this->f && phpthumb_functions::CaseInsensitiveInArray($this->f, $AvailableImageOutputFormats) ) { 1086 // override output format if $this->f is set and that format is available 1087 $this->DebugMessage('$this->thumbnailFormat set to $this->f "'.strtolower($this->f).'"', __FILE__, __LINE__); 1088 $this->thumbnailFormat = strtolower($this->f); 1089 } elseif ($this->f) { 1090 $this->DebugMessage('$this->thumbnailFormat staying as "'.$this->thumbnailFormat.'" because $this->f ('.strtolower($this->f).') is not in $AvailableImageOutputFormats', __FILE__, __LINE__); 1091 } 1092 1093 // for JPEG images, quality 1 (worst) to 99 (best) 1094 // quality < 25 is nasty, with not much size savings - not recommended 1095 // problems with 100 - invalid JPEG? 1096 $this->thumbnailQuality = max(1, min(99, ($this->q ? (int) $this->q : 75))); 1097 $this->DebugMessage('$this->thumbnailQuality set to "'.$this->thumbnailQuality.'"', __FILE__, __LINE__); 1098 1099 return true; 1100 } 1101 1102 1103 public function setCacheDirectory() { 1104 // resolve cache directory to absolute pathname 1105 $this->DebugMessage('setCacheDirectory() starting with config_cache_directory = "'.$this->config_cache_directory.'"', __FILE__, __LINE__); 1106 if ($this->config_cache_directory && ($this->config_cache_directory[0] == '.')) { 1107 if (preg_match('#^(f|ht)tps?\://#i', $this->src)) { 1108 if (!$this->config_cache_disable_warning) { 1109 $this->ErrorImage('$this->config_cache_directory ('.$this->config_cache_directory.') cannot be used for remote images. Adjust "cache_directory" or "cache_disable_warning" in phpThumb.config.php'); 1110 } 1111 } elseif ($this->src) { 1112 // resolve relative cache directory to source image 1113 $this->config_cache_directory = dirname($this->ResolveFilenameToAbsolute($this->src)).DIRECTORY_SEPARATOR.$this->config_cache_directory; 1114 } else { 1115 // $this->new is probably set 1116 } 1117 } 1118 if (substr($this->config_cache_directory, -1) == '/') { 1119 $this->config_cache_directory = substr($this->config_cache_directory, 0, -1); 1120 } 1121 if ($this->iswindows) { 1122 $this->config_cache_directory = str_replace('/', DIRECTORY_SEPARATOR, $this->config_cache_directory); 1123 } 1124 if ($this->config_cache_directory) { 1125 $real_cache_path = $this->realPathSafe($this->config_cache_directory); 1126 if (!$real_cache_path) { 1127 $this->DebugMessage('$this->realPathSafe($this->config_cache_directory) failed for "'.$this->config_cache_directory.'"', __FILE__, __LINE__); 1128 if (!is_dir($this->config_cache_directory)) { 1129 $this->DebugMessage('!is_dir('.$this->config_cache_directory.')', __FILE__, __LINE__); 1130 } 1131 } 1132 if ($real_cache_path) { 1133 $this->DebugMessage('setting config_cache_directory to $this->realPathSafe('.$this->config_cache_directory.') = "'.$real_cache_path.'"', __FILE__, __LINE__); 1134 $this->config_cache_directory = $real_cache_path; 1135 } 1136 } 1137 if (!is_dir($this->config_cache_directory)) { 1138 if (!$this->config_cache_disable_warning) { 1139 $this->ErrorImage('$this->config_cache_directory ('.$this->config_cache_directory.') does not exist. Adjust "cache_directory" or "cache_disable_warning" in phpThumb.config.php'); 1140 } 1141 $this->DebugMessage('$this->config_cache_directory ('.$this->config_cache_directory.') is not a directory', __FILE__, __LINE__); 1142 $this->config_cache_directory = null; 1143 } elseif (!@is_writable($this->config_cache_directory)) { 1144 $this->DebugMessage('$this->config_cache_directory is not writable ('.$this->config_cache_directory.')', __FILE__, __LINE__); 1145 } 1146 1147 $this->InitializeTempDirSetting(); 1148 if (!@is_dir($this->config_temp_directory) && !@is_writable($this->config_temp_directory) && @is_dir($this->config_cache_directory) && @is_writable($this->config_cache_directory)) { 1149 $this->DebugMessage('setting $this->config_temp_directory = $this->config_cache_directory ('.$this->config_cache_directory.')', __FILE__, __LINE__); 1150 $this->config_temp_directory = $this->config_cache_directory; 1151 } 1152 return true; 1153 } 1154 1155 /* Takes the array of path segments up to now, and the next segment (maybe a modifier: empty, . or ..) 1156 Applies it, adding or removing from $segments as a result. Returns nothing. */ 1157 // http://support.silisoftware.com/phpBB3/viewtopic.php?t=961 1158 public function applyPathSegment(&$segments, $segment) { 1159 if ($segment == '.') { 1160 return; // always remove 1161 } 1162 if ($segment == '') { 1163 $test = array_pop($segments); 1164 if (null === $test) { 1165 $segments[] = $segment; // keep the first empty block 1166 } elseif ($test == '') { 1167 $test = array_pop($segments); 1168 if (null === $test) { 1169 $segments[] = $test; 1170 $segments[] = $segment; // keep the second one too 1171 } else { // put both back and ignore segment 1172 $segments[] = $test; 1173 $segments[] = $test; 1174 } 1175 } else { 1176 $segments[] = $test; // ignore empty blocks 1177 } 1178 } else { 1179 if ($segment == '..') { 1180 $test = array_pop($segments); 1181 if (null === $test) { 1182 $segments[] = $segment; 1183 } elseif ($test == '..') { 1184 $segments[] = $test; 1185 $segments[] = $segment; 1186 } else { 1187 if ($test == '') { 1188 $segments[] = $test; 1189 } // else nothing, remove both 1190 } 1191 } else { 1192 $segments[] = $segment; 1193 } 1194 } 1195 } 1196 1197 /* Takes array of path components, normalizes it: removes empty slots and '.', collapses '..' and folder names. Returns array. */ 1198 // http://support.silisoftware.com/phpBB3/viewtopic.php?t=961 1199 public function normalizePath($segments) { 1200 $parts = array(); 1201 foreach ($segments as $segment) { 1202 $this->applyPathSegment($parts, $segment); 1203 } 1204 return $parts; 1205 } 1206 1207 /* True if the provided path points (without resolving symbolic links) into one of the allowed directories. */ 1208 // http://support.silisoftware.com/phpBB3/viewtopic.php?t=961 1209 public function matchPath($path, $allowed_dirs) { 1210 if (!empty($allowed_dirs)) { 1211 foreach ($allowed_dirs as $one_dir) { 1212 if (preg_match('#^'.preg_quote(str_replace(DIRECTORY_SEPARATOR, '/', $this->realPathSafe($one_dir))).'#', $path)) { 1213 return true; 1214 } 1215 } 1216 } 1217 return false; 1218 } 1219 1220 /* True if the provided path points inside one of open_basedirs (or if open_basedirs are disabled) */ 1221 // http://support.silisoftware.com/phpBB3/viewtopic.php?t=961 1222 public function isInOpenBasedir($path) { 1223 static $open_basedirs = null; 1224 if (null === $open_basedirs) { 1225 $ini_text = ini_get('open_basedir'); 1226 $this->DebugMessage('open_basedir: "'.$ini_text.'"', __FILE__, __LINE__); 1227 $open_basedirs = array(); 1228 if (strlen($ini_text) > 0) { 1229 foreach (preg_split('#[;:]#', $ini_text) as $key => $value) { 1230 $open_basedirs[$key] = $this->realPathSafe($value); 1231 } 1232 } 1233 } 1234 return (empty($open_basedirs) || $this->matchPath($path, $open_basedirs)); 1235 } 1236 1237 /* Resolves all symlinks in $path, checking that each continuous part ends in an allowed zone. Returns null, if any component leads outside of allowed zone. */ 1238 // http://support.silisoftware.com/phpBB3/viewtopic.php?t=961 1239 public function resolvePath($path, $allowed_dirs) { 1240 $this->DebugMessage('resolvePath: '.$path.' (allowed_dirs: '.print_r($allowed_dirs, true).')', __FILE__, __LINE__); 1241 1242 // add base path to the top of the list 1243 if (!$this->config_allow_src_above_docroot) { 1244 array_unshift($allowed_dirs, $this->realPathSafe($this->config_document_root)); 1245 } else { 1246 if (!$this->config_allow_src_above_phpthumb) { 1247 array_unshift($allowed_dirs, $this->realPathSafe( __DIR__ )); 1248 } else { 1249 // no checks are needed, offload the work to realpath and forget about it 1250 $this->DebugMessage('resolvePath: checks disabled, returning '.$this->realPathSafe($path), __FILE__, __LINE__); 1251 return $this->realPathSafe($path); 1252 } 1253 } 1254 if ($path == '') { 1255 return null; // save us trouble 1256 } 1257 1258 do { 1259 $this->DebugMessage('resolvePath: iteration, path='.$path.', base path = '.$allowed_dirs[0], __FILE__, __LINE__); 1260 1261 $parts = array(); 1262 // do not use "cleaner" foreach version of this loop as later code relies on both $segments and $i 1263 // http://support.silisoftware.com/phpBB3/viewtopic.php?t=964 1264 $segments = explode(DIRECTORY_SEPARATOR, $path); 1265 for ($i = 0, $iMax = count($segments); $i < $iMax; $i++) { 1266 $this->applyPathSegment($parts, $segments[$i]); 1267 $thispart = implode(DIRECTORY_SEPARATOR, $parts); 1268 if ($this->isInOpenBasedir($thispart)) { 1269 if (is_link($thispart)) { 1270 break; 1271 } 1272 } 1273 } 1274 1275 $this->DebugMessage('resolvePath: stop at component '.$i, __FILE__, __LINE__); 1276 // test the part up to here 1277 $path = implode(DIRECTORY_SEPARATOR, $parts); 1278 $this->DebugMessage('resolvePath: stop at path='.$path, __FILE__, __LINE__); 1279 if (!$this->matchPath($path, $allowed_dirs)) { 1280 $this->DebugMessage('resolvePath: no match, returning null', __FILE__, __LINE__); 1281 return null; 1282 } 1283 if ($i >= count($segments)) { // reached end 1284 $this->DebugMessage('resolvePath: path parsed, over', __FILE__, __LINE__); 1285 break; 1286 } 1287 // else it's symlink, rewrite path 1288 $path = readlink($path); 1289 $this->DebugMessage('resolvePath: symlink matched, target='.$path, __FILE__, __LINE__); 1290 1291 /* 1292 Replace base path with symlink target. 1293 Assuming: 1294 /www/img/external -> /external 1295 This is allowed: 1296 GET /www/img/external/../external/test/pic.jpg 1297 This isn't: 1298 GET /www/img/external/../www/img/pic.jpg 1299 So there's only one base path which is the last symlink target, but any number of stable whitelisted paths. 1300 */ 1301 if ($this->config_auto_allow_symlinks) { 1302 $allowed_dirs[0] = $path; 1303 } 1304 $path = $path.DIRECTORY_SEPARATOR.implode(DIRECTORY_SEPARATOR, array_slice($segments,$i + 1)); 1305 } while (true); 1306 return $path; 1307 } 1308 1309 1310 public function realPathSafe($filename) { 1311 // http://php.net/manual/en/function.realpath.php -- "Note: The running script must have executable permissions on all directories in the hierarchy, otherwise realpath() will return FALSE" 1312 // realPathSafe() provides a reasonable facsimile of realpath() but does not resolve symbolic links, nor does it check that the file/path actually exists 1313 if (!$this->config_disable_realpath) { 1314 return realpath($filename); 1315 } 1316 1317 // http://stackoverflow.com/questions/21421569 1318 $newfilename = preg_replace('#[\\/]+#', DIRECTORY_SEPARATOR, $filename); 1319 1320 if (phpthumb_functions::is_windows()) { 1321 $isAlreadyAbsoluteFilename = preg_match('#^[A-Z]\\:#i', $newfilename); // C:\path\filename.ext 1322 } else { 1323 $isAlreadyAbsoluteFilename = ($newfilename[0] == DIRECTORY_SEPARATOR); // /path/filename.ext 1324 } 1325 if (!$isAlreadyAbsoluteFilename) { 1326 // not already an absolute filename, prepend current directory 1327 $newfilename = __DIR__ .DIRECTORY_SEPARATOR.$newfilename; 1328 } 1329 do { 1330 $beforeloop = $newfilename; 1331 1332 // Replace all sequences of more than one / with a single one [[ If you're working on a system that treats // at the start of a path as special, make sure you replace multiple / characters at the start with two of them. This is the only place where POSIX allows (but does not mandate) special handling for multiples, in all other cases, multiple / characters are equivalent to a single one.]] 1333 $newfilename = preg_replace('#'.preg_quote(DIRECTORY_SEPARATOR).'+#', DIRECTORY_SEPARATOR, $newfilename); 1334 1335 // Replace all occurrences of /./ with / 1336 $newfilename = preg_replace('#'.preg_quote(DIRECTORY_SEPARATOR).'\\.'.preg_quote(DIRECTORY_SEPARATOR).'#', DIRECTORY_SEPARATOR, $newfilename); 1337 1338 // Remove ./ if at the start 1339 $newfilename = preg_replace('#^\\.'.preg_quote(DIRECTORY_SEPARATOR).'#', '', $newfilename); 1340 1341 // Remove /. if at the end 1342 $newfilename = preg_replace('#'.preg_quote(DIRECTORY_SEPARATOR).'\\.$#', '', $newfilename); 1343 1344 // Replace /anything/../ with / 1345 $newfilename = preg_replace('#'.preg_quote(DIRECTORY_SEPARATOR).'[^'.preg_quote(DIRECTORY_SEPARATOR).']+'.preg_quote(DIRECTORY_SEPARATOR).'\\.\\.'.preg_quote(DIRECTORY_SEPARATOR).'#', DIRECTORY_SEPARATOR, $newfilename); 1346 1347 // Remove /anything/.. if at the end 1348 $newfilename = preg_replace('#'.preg_quote(DIRECTORY_SEPARATOR).'[^'.preg_quote(DIRECTORY_SEPARATOR).']+'.preg_quote(DIRECTORY_SEPARATOR).'\\.\\.$#', '', $newfilename); 1349 1350 } while ($newfilename != $beforeloop); 1351 return $newfilename; 1352 } 1353 1354 1355 public function ResolveFilenameToAbsolute($filename) { 1356 if (empty($filename)) { 1357 return false; 1358 } 1359 1360 if (preg_match('#^[a-z0-9]+\\:/{1,2}#i', $filename)) { 1361 // eg: http://host/path/file.jpg (HTTP URL) 1362 // eg: ftp://host/path/file.jpg (FTP URL) 1363 // eg: data1:/path/file.jpg (Netware path) 1364 1365 //$AbsoluteFilename = $filename; 1366 return $filename; 1367 1368 } elseif ($this->iswindows && isset($filename[1]) && ($filename[1] == ':')) { 1369 1370 // absolute pathname (Windows) 1371 $AbsoluteFilename = $filename; 1372 1373 } elseif ($this->iswindows && ((substr($filename, 0, 2) == '//') || (substr($filename, 0, 2) == '\\\\'))) { 1374 1375 // absolute pathname (Windows) 1376 $AbsoluteFilename = $filename; 1377 1378 } elseif ($filename[0] == '/') { 1379 1380 if (@is_readable($filename) && !@is_readable($this->config_document_root.$filename)) { 1381 1382 // absolute filename (*nix) 1383 $AbsoluteFilename = $filename; 1384 1385 } elseif (isset($filename[1]) && ($filename[1] == '~')) { 1386 1387 // /~user/path 1388 if ($ApacheLookupURIarray = phpthumb_functions::ApacheLookupURIarray($filename)) { 1389 $AbsoluteFilename = $ApacheLookupURIarray['filename']; 1390 } else { 1391 $AbsoluteFilename = $this->realPathSafe($filename); 1392 if (@is_readable($AbsoluteFilename)) { 1393 $this->DebugMessage('phpthumb_functions::ApacheLookupURIarray() failed for "'.$filename.'", but the correct filename ('.$AbsoluteFilename.') seems to have been resolved with $this->realPathSafe($filename)', __FILE__, __LINE__); 1394 } elseif (is_dir(dirname($AbsoluteFilename))) { 1395 $this->DebugMessage('phpthumb_functions::ApacheLookupURIarray() failed for "'.dirname($filename).'", but the correct directory ('.dirname($AbsoluteFilename).') seems to have been resolved with $this->realPathSafe(.)', __FILE__, __LINE__); 1396 } else { 1397 return $this->ErrorImage('phpthumb_functions::ApacheLookupURIarray() failed for "'.$filename.'". This has been known to fail on Apache2 - try using the absolute filename for the source image (ex: "/home/user/httpdocs/image.jpg" instead of "/~user/image.jpg")'); 1398 } 1399 } 1400 1401 } else { 1402 1403 // relative filename (any OS) 1404 if (preg_match('#^'.preg_quote($this->config_document_root).'#', $filename)) { 1405 $AbsoluteFilename = $filename; 1406 $this->DebugMessage('ResolveFilenameToAbsolute() NOT prepending $this->config_document_root ('.$this->config_document_root.') to $filename ('.$filename.') resulting in ($AbsoluteFilename = "'.$AbsoluteFilename.'")', __FILE__, __LINE__); 1407 } else { 1408 $AbsoluteFilename = $this->config_document_root.$filename; 1409 $this->DebugMessage('ResolveFilenameToAbsolute() prepending $this->config_document_root ('.$this->config_document_root.') to $filename ('.$filename.') resulting in ($AbsoluteFilename = "'.$AbsoluteFilename.'")', __FILE__, __LINE__); 1410 } 1411 1412 } 1413 1414 } else { 1415 1416 // relative to current directory (any OS) 1417 $AbsoluteFilename = __DIR__ .DIRECTORY_SEPARATOR.preg_replace('#[/\\\\]#', DIRECTORY_SEPARATOR, $filename); 1418 1419 if (substr(dirname(@$_SERVER['PHP_SELF']), 0, 2) == '/~') { 1420 if ($ApacheLookupURIarray = phpthumb_functions::ApacheLookupURIarray(dirname(@$_SERVER['PHP_SELF']))) { 1421 $AbsoluteFilename = $ApacheLookupURIarray['filename'].DIRECTORY_SEPARATOR.$filename; 1422 } else { 1423 $AbsoluteFilename = $this->realPathSafe('.').DIRECTORY_SEPARATOR.$filename; 1424 if (@is_readable($AbsoluteFilename)) { 1425 $this->DebugMessage('phpthumb_functions::ApacheLookupURIarray() failed for "'.dirname(@$_SERVER['PHP_SELF']).'", but the correct filename ('.$AbsoluteFilename.') seems to have been resolved with $this->realPathSafe(.)/$filename', __FILE__, __LINE__); 1426 } elseif (is_dir(dirname($AbsoluteFilename))) { 1427 $this->DebugMessage('phpthumb_functions::ApacheLookupURIarray() failed for "'.dirname(@$_SERVER['PHP_SELF']).'", but the correct directory ('.dirname($AbsoluteFilename).') seems to have been resolved with $this->realPathSafe(.)', __FILE__, __LINE__); 1428 } else { 1429 return $this->ErrorImage('phpthumb_functions::ApacheLookupURIarray() failed for "'.dirname(@$_SERVER['PHP_SELF']).'". This has been known to fail on Apache2 - try using the absolute filename for the source image'); 1430 } 1431 } 1432 } 1433 1434 } 1435 /* 1436 // removed 2014-May-30: http://support.silisoftware.com/phpBB3/viewtopic.php?t=961 1437 if (is_link($AbsoluteFilename)) { 1438 $this->DebugMessage('is_link()==true, changing "'.$AbsoluteFilename.'" to "'.readlink($AbsoluteFilename).'"', __FILE__, __LINE__); 1439 $AbsoluteFilename = readlink($AbsoluteFilename); 1440 } 1441 if ($this->realPathSafe($AbsoluteFilename)) { 1442 $AbsoluteFilename = $this->realPathSafe($AbsoluteFilename); 1443 } 1444 */ 1445 if ($this->iswindows) { 1446 $AbsoluteFilename = preg_replace('#^'.preg_quote($this->realPathSafe($this->config_document_root)).'#i', str_replace('\\', '\\\\', $this->realPathSafe($this->config_document_root)), $AbsoluteFilename); 1447 $AbsoluteFilename = str_replace(DIRECTORY_SEPARATOR, '/', $AbsoluteFilename); 1448 } 1449 $resolvedAbsoluteFilename = $this->resolvePath($AbsoluteFilename, $this->config_additional_allowed_dirs); 1450 if (!$this->config_allow_src_above_docroot && !preg_match('#^'.preg_quote(str_replace(DIRECTORY_SEPARATOR, '/', $this->realPathSafe($this->config_document_root))).'#', $resolvedAbsoluteFilename)) { 1451 $this->DebugMessage('!$this->config_allow_src_above_docroot therefore setting "'.$AbsoluteFilename.'" (outside "'.$this->realPathSafe($this->config_document_root).'") to null', __FILE__, __LINE__); 1452 return false; 1453 } 1454 if (!$this->config_allow_src_above_phpthumb && !preg_match('#^'.preg_quote(str_replace(DIRECTORY_SEPARATOR, '/', __DIR__ )).'#', $resolvedAbsoluteFilename)) { 1455 $this->DebugMessage('!$this->config_allow_src_above_phpthumb therefore setting "'.$AbsoluteFilename.'" (outside "'. __DIR__ .'") to null', __FILE__, __LINE__); 1456 return false; 1457 } 1458 return $resolvedAbsoluteFilename; 1459 } 1460 1461 1462 public function file_exists_ignoreopenbasedir($filename, $cached=true) { 1463 static $open_basedirs = null; 1464 static $file_exists_cache = array(); 1465 if (!$cached || !isset($file_exists_cache[$filename])) { 1466 if (is_null($open_basedirs)) { 1467 $open_basedirs = preg_split('#[;:]#', ini_get('open_basedir')); 1468 } 1469 if (is_null($filename)) { // shouldn't happen, but https://github.com/JamesHeinrich/phpThumb/issues/188 1470 $file_exists_cache[$filename] = false; 1471 } elseif (empty($open_basedirs) || in_array(dirname($filename), $open_basedirs)) { 1472 $file_exists_cache[$filename] = file_exists($filename); 1473 } elseif ($this->iswindows) { 1474 $ls_filename = trim(phpthumb_functions::SafeExec('dir /b '.phpthumb_functions::escapeshellarg_replacement($filename))); 1475 $file_exists_cache[$filename] = ($ls_filename == basename($filename)); // command dir /b return only filename without path 1476 } else { 1477 $ls_filename = trim(phpthumb_functions::SafeExec('ls '.phpthumb_functions::escapeshellarg_replacement($filename))); 1478 $file_exists_cache[$filename] = ($ls_filename == $filename); 1479 } 1480 } 1481 return $file_exists_cache[$filename]; 1482 } 1483 1484 1485 public function ImageMagickWhichConvert() { 1486 static $WhichConvert = null; 1487 if (null === $WhichConvert) { 1488 if ($this->iswindows) { 1489 $WhichConvert = false; 1490 } else { 1491 $IMwhichConvertCacheFilename = $this->config_cache_directory.DIRECTORY_SEPARATOR.'phpThumbCacheIMwhichConvert.txt'; 1492 if (($cachedwhichconvertstring = @file_get_contents($IMwhichConvertCacheFilename)) !== false) { 1493 $WhichConvert = $cachedwhichconvertstring; 1494 } else { 1495 $WhichConvert = trim(phpthumb_functions::SafeExec('which convert')); 1496 @file_put_contents($IMwhichConvertCacheFilename, $WhichConvert); 1497 @chmod($IMwhichConvertCacheFilename, $this->getParameter('config_file_create_mask')); 1498 } 1499 } 1500 } 1501 return $WhichConvert; 1502 } 1503 1504 1505 public function ImageMagickCommandlineBase() { 1506 static $commandline = null; 1507 if (null === $commandline) { 1508 if ($this->issafemode) { 1509 $commandline = ''; 1510 return $commandline; 1511 } 1512 1513 $IMcommandlineBaseCacheFilename = $this->config_cache_directory.DIRECTORY_SEPARATOR.'phpThumbCacheIMcommandlineBase.txt'; 1514 if (($commandline = @file_get_contents($IMcommandlineBaseCacheFilename)) !== false) { 1515 return $commandline; 1516 } 1517 1518 $commandline = (null !== $this->config_imagemagick_path ? $this->config_imagemagick_path : ''); 1519 1520 if ($this->config_imagemagick_path && ($this->config_imagemagick_path != $this->realPathSafe($this->config_imagemagick_path))) { 1521 if (@is_executable($this->realPathSafe($this->config_imagemagick_path))) { 1522 $this->DebugMessage('Changing $this->config_imagemagick_path ('.$this->config_imagemagick_path.') to $this->realPathSafe($this->config_imagemagick_path) ('.$this->realPathSafe($this->config_imagemagick_path).')', __FILE__, __LINE__); 1523 $this->config_imagemagick_path = $this->realPathSafe($this->config_imagemagick_path); 1524 } else { 1525 $this->DebugMessage('Leaving $this->config_imagemagick_path as ('.$this->config_imagemagick_path.') because !is_execuatable($this->realPathSafe($this->config_imagemagick_path)) ('.$this->realPathSafe($this->config_imagemagick_path).')', __FILE__, __LINE__); 1526 } 1527 } 1528 if (!empty($this->config_imagemagick_path)) { 1529 $this->DebugMessage(' file_exists('.$this->config_imagemagick_path.') = '. (int) (@file_exists($this->config_imagemagick_path)), __FILE__, __LINE__); 1530 $this->DebugMessage('file_exists_ignoreopenbasedir('.$this->config_imagemagick_path.') = '. (int) $this->file_exists_ignoreopenbasedir($this->config_imagemagick_path), __FILE__, __LINE__); 1531 $this->DebugMessage(' is_file('.$this->config_imagemagick_path.') = '. (int) (@is_file($this->config_imagemagick_path)), __FILE__, __LINE__); 1532 $this->DebugMessage(' is_executable('.$this->config_imagemagick_path.') = '. (int) (@is_executable($this->config_imagemagick_path)), __FILE__, __LINE__); 1533 } 1534 1535 if ($this->file_exists_ignoreopenbasedir($this->config_imagemagick_path)) { 1536 1537 $this->DebugMessage('using ImageMagick path from $this->config_imagemagick_path ('.$this->config_imagemagick_path.')', __FILE__, __LINE__); 1538 if ($this->iswindows) { 1539 $commandline = ''; 1540 $commandline .= substr($this->config_imagemagick_path, 0, 2); 1541 $commandline .= ' && cd '.phpthumb_functions::escapeshellarg_replacement(str_replace('/', DIRECTORY_SEPARATOR, substr(dirname($this->config_imagemagick_path), 2))); 1542 $commandline .= ' && '.phpthumb_functions::escapeshellarg_replacement(basename($this->config_imagemagick_path)); 1543 } else { 1544 $commandline = phpthumb_functions::escapeshellarg_replacement($this->config_imagemagick_path); 1545 } 1546 1547 } else { 1548 1549 $which_convert = $this->ImageMagickWhichConvert(); 1550 $IMversion = $this->ImageMagickVersion(); 1551 1552 if ($which_convert && ($which_convert[0] == '/') && $this->file_exists_ignoreopenbasedir($which_convert)) { 1553 1554 // `which convert` *should* return the path if "convert" exist, or nothing if it doesn't 1555 // other things *may* get returned, like "sh: convert: not found" or "no convert in /usr/local/bin /usr/sbin /usr/bin /usr/ccs/bin" 1556 // so only do this if the value returned exists as a file 1557 $this->DebugMessage('using ImageMagick path from `which convert` ('.$which_convert.')', __FILE__, __LINE__); 1558 $commandline = 'convert'; 1559 1560 } elseif ($IMversion) { 1561 1562 $this->DebugMessage('setting ImageMagick path to $this->config_imagemagick_path ('.$this->config_imagemagick_path.') ['.$IMversion.']', __FILE__, __LINE__); 1563 $commandline = $this->config_imagemagick_path; 1564 1565 } else { 1566 1567 $this->DebugMessage('ImageMagickThumbnailToGD() aborting because cannot find convert in $this->config_imagemagick_path ('.$this->config_imagemagick_path.'), and `which convert` returned ('.$which_convert.')', __FILE__, __LINE__); 1568 $commandline = ''; 1569 1570 } 1571 1572 } 1573 1574 @file_put_contents($IMcommandlineBaseCacheFilename, $commandline); 1575 @chmod($IMcommandlineBaseCacheFilename, $this->getParameter('config_file_create_mask')); 1576 } 1577 return $commandline; 1578 } 1579 1580 1581 public function ImageMagickVersion($returnRAW=false) { 1582 static $versionstring = null; 1583 if (null === $versionstring) { 1584 $versionstring = array(0=>false, 1=>false); 1585 1586 $IMversionCacheFilename = $this->config_cache_directory.DIRECTORY_SEPARATOR.'phpThumbCacheIMversion.txt'; 1587 if ($cachedversionstring = @file_get_contents($IMversionCacheFilename)) { 1588 1589 $versionstring = explode("\n", $cachedversionstring, 2); 1590 $versionstring[0] = ($versionstring[0] ? $versionstring[0] : false); // "false" is stored as an empty string in the cache file 1591 $versionstring[1] = ($versionstring[1] ? $versionstring[1] : false); // "false" is stored as an empty string in the cache file 1592 1593 } else { 1594 1595 $commandline = $this->ImageMagickCommandlineBase(); 1596 $commandline = (null !== $commandline ? $commandline : ''); 1597 if ($commandline) { 1598 $commandline .= ' --version'; 1599 $this->DebugMessage('ImageMagick version checked with "'.$commandline.'"', __FILE__, __LINE__); 1600 $versionstring[1] = trim(phpthumb_functions::SafeExec($commandline)); 1601 if (preg_match('#^Version: [^\d]*([ 0-9\\.\\:Q/\\-]+)#i', $versionstring[1], $matches)) { 1602 $versionstring[0] = trim($matches[1]); 1603 } else { 1604 $versionstring[0] = false; 1605 $this->DebugMessage('ImageMagick did not return recognized version string ('.$versionstring[1].')', __FILE__, __LINE__); 1606 } 1607 $this->DebugMessage('ImageMagick convert --version says "'.@$matches[0].'"', __FILE__, __LINE__); 1608 } 1609 1610 @file_put_contents($IMversionCacheFilename, $versionstring[0]."\n".$versionstring[1]); 1611 @chmod($IMversionCacheFilename, $this->getParameter('config_file_create_mask')); 1612 1613 } 1614 } 1615 return $versionstring[ (int) $returnRAW ]; 1616 } 1617 1618 1619 public function ImageMagickSwitchAvailable($switchname) { 1620 static $IMoptions = null; 1621 if (null === $IMoptions) { 1622 $IMoptions = array(); 1623 $commandline = $this->ImageMagickCommandlineBase(); 1624 if (null !== $commandline) { 1625 $commandline .= ' -help'; 1626 $IMhelp_lines = explode("\n", phpthumb_functions::SafeExec($commandline)); 1627 foreach ($IMhelp_lines as $line) { 1628 if (preg_match('#^[\\+\\-]([a-z\\-]+) #', trim($line), $matches)) { 1629 $IMoptions[$matches[1]] = true; 1630 } 1631 } 1632 } 1633 } 1634 if (is_array($switchname)) { 1635 $allOK = true; 1636 foreach ($switchname as $key => $value) { 1637 if (!isset($IMoptions[$value])) { 1638 $allOK = false; 1639 break; 1640 } 1641 } 1642 $this->DebugMessage('ImageMagickSwitchAvailable('.implode(';', $switchname).') = '. (int) $allOK .'', __FILE__, __LINE__); 1643 } else { 1644 $allOK = isset($IMoptions[$switchname]); 1645 $this->DebugMessage('ImageMagickSwitchAvailable('.$switchname.') = '. (int) $allOK .'', __FILE__, __LINE__); 1646 } 1647 return $allOK; 1648 } 1649 1650 1651 public function ImageMagickFormatsList() { 1652 static $IMformatsList = null; 1653 if (null === $IMformatsList) { 1654 $IMformatsList = ''; 1655 $commandline = $this->ImageMagickCommandlineBase(); 1656 if (!is_null($commandline)) { 1657 $commandline = dirname($commandline).DIRECTORY_SEPARATOR.str_replace('convert', 'identify', basename($commandline)); 1658 $commandline .= ' -list format'; 1659 $IMformatsList = phpthumb_functions::SafeExec($commandline); 1660 } 1661 } 1662 return $IMformatsList; 1663 } 1664 1665 1666 public function SourceDataToTempFile() { 1667 if ($IMtempSourceFilename = $this->phpThumb_tempnam()) { 1668 $IMtempSourceFilename = $this->realPathSafe($IMtempSourceFilename); 1669 ob_start(); 1670 $fp_tempfile = fopen($IMtempSourceFilename, 'wb'); 1671 $tempfile_open_error = ob_get_contents(); 1672 ob_end_clean(); 1673 if ($fp_tempfile) { 1674 fwrite($fp_tempfile, $this->rawImageData); 1675 fclose($fp_tempfile); 1676 @chmod($IMtempSourceFilename, $this->getParameter('config_file_create_mask')); 1677 $this->sourceFilename = $IMtempSourceFilename; 1678 $this->DebugMessage('ImageMagickThumbnailToGD() setting $this->sourceFilename to "'.$IMtempSourceFilename.'" from $this->rawImageData ('.strlen($this->rawImageData).' bytes)', __FILE__, __LINE__); 1679 } else { 1680 $this->DebugMessage('ImageMagickThumbnailToGD() FAILED setting $this->sourceFilename to "'.$IMtempSourceFilename.'" (failed to open for writing: "'.$tempfile_open_error.'")', __FILE__, __LINE__); 1681 } 1682 unset($tempfile_open_error, $IMtempSourceFilename); 1683 return true; 1684 } 1685 $this->DebugMessage('SourceDataToTempFile() FAILED because $this->phpThumb_tempnam() failed', __FILE__, __LINE__); 1686 return false; 1687 } 1688 1689 1690 public function ImageMagickThumbnailToGD() { 1691 // http://www.imagemagick.org/script/command-line-options.php 1692 1693 $this->useRawIMoutput = true; 1694 if (phpthumb_functions::gd_version()) { 1695 // if GD is not available, must use whatever ImageMagick can output 1696 1697 // $CannotMagickParameters contains options that cannot be used with ImageMagick 1698 $CannotMagickParameters = array('ica'); 1699 foreach ($CannotMagickParameters as $parameter) { 1700 if (isset($this->$parameter)) { 1701 $this->DebugMessage('cannot process with ImageMagick because "'.$parameter.'" is set', __FILE__, __LINE__); 1702 $this->useRawIMoutput = false; 1703 return false; 1704 } 1705 } 1706 1707 // $UnAllowedParameters contains options that can only be processed in GD, not ImageMagick 1708 // note: 'fltr' *may* need to be processed by GD, but we'll check that in more detail below 1709 $UnAllowedParameters = array('xto', 'ar', 'bg', 'bc'); 1710 // 'ra' may be part of this list, if not a multiple of 90 degrees 1711 foreach ($UnAllowedParameters as $parameter) { 1712 if (isset($this->$parameter)) { 1713 $this->DebugMessage('$this->useRawIMoutput=false because "'.$parameter.'" is set', __FILE__, __LINE__); 1714 $this->useRawIMoutput = false; 1715 break; 1716 } 1717 } 1718 } 1719 $this->DebugMessage('$this->useRawIMoutput='.($this->useRawIMoutput ? 'true' : 'false').' after checking $UnAllowedParameters', __FILE__, __LINE__); 1720 $ImageCreateFunction = ''; 1721 $outputFormat = $this->thumbnailFormat; 1722 if (phpthumb_functions::gd_version()) { 1723 if ($this->useRawIMoutput) { 1724 switch ($this->thumbnailFormat) { 1725 case 'gif': 1726 $ImageCreateFunction = 'imagecreatefromgif'; 1727 $this->is_alpha = true; 1728 break; 1729 case 'png': 1730 $ImageCreateFunction = 'imagecreatefrompng'; 1731 $this->is_alpha = true; 1732 break; 1733 case 'jpg': 1734 case 'jpeg': 1735 $ImageCreateFunction = 'imagecreatefromjpeg'; 1736 break; 1737 case 'webp': 1738 $ImageCreateFunction = 'imagecreatefromwebp'; 1739 $this->is_alpha = true; 1740 break; 1741 case 'avif': 1742 $ImageCreateFunction = 'imagecreatefromavif'; 1743 $this->is_alpha = true; 1744 break; 1745 default: 1746 $this->DebugMessage('Forcing output to PNG because $this->thumbnailFormat ('.$this->thumbnailFormat.' is not a GD-supported format)', __FILE__, __LINE__); 1747 $outputFormat = 'png'; 1748 $ImageCreateFunction = 'imagecreatefrompng'; 1749 $this->is_alpha = true; 1750 $this->useRawIMoutput = false; 1751 break; 1752 } 1753 if (!function_exists($ImageCreateFunction)) { 1754 // ImageMagickThumbnailToGD() depends on imagecreatefrompng/imagecreatefromgif 1755 //$this->DebugMessage('ImageMagickThumbnailToGD() aborting because '.@$ImageCreateFunction.'() is not available', __FILE__, __LINE__); 1756 $this->useRawIMoutput = true; 1757 //return false; 1758 } 1759 } else { 1760 $outputFormat = 'png'; 1761 $ImageCreateFunction = 'imagecreatefrompng'; 1762 $this->is_alpha = true; 1763 $this->useRawIMoutput = false; 1764 } 1765 } 1766 1767 // http://freealter.org/doc_distrib/ImageMagick-5.1.1/www/convert.html 1768 if (!$this->sourceFilename && $this->rawImageData) { 1769 $this->SourceDataToTempFile(); 1770 } 1771 if (!$this->sourceFilename) { 1772 $this->DebugMessage('ImageMagickThumbnailToGD() aborting because $this->sourceFilename is empty', __FILE__, __LINE__); 1773 $this->useRawIMoutput = false; 1774 return false; 1775 } 1776 if ($this->issafemode) { 1777 $this->DebugMessage('ImageMagickThumbnailToGD() aborting because safe_mode is enabled', __FILE__, __LINE__); 1778 $this->useRawIMoutput = false; 1779 return false; 1780 } 1781// TO BE FIXED 1782//if (true) { 1783// $this->DebugMessage('ImageMagickThumbnailToGD() aborting it is broken right now', __FILE__, __LINE__); 1784// $this->useRawIMoutput = false; 1785// return false; 1786//} 1787 1788 $commandline = $this->ImageMagickCommandlineBase(); 1789 if ($commandline) { 1790 $commandline .= ' '.phpthumb_functions::escapeshellarg_replacement(preg_replace('#[/\\\\]#', DIRECTORY_SEPARATOR, $this->sourceFilename).(($outputFormat == 'gif') ? '' : '['. (int) $this->sfn .']')); // [0] means first frame of (GIF) animation, can be ignored 1791 if ($IMtempfilename = $this->phpThumb_tempnam()) { 1792 $IMtempfilename = $this->realPathSafe($IMtempfilename); 1793 1794 $IMuseExplicitImageOutputDimensions = false; 1795 if ($this->ImageMagickSwitchAvailable('thumbnail') && $this->config_imagemagick_use_thumbnail) { 1796 $IMresizeParameter = 'thumbnail'; 1797 } else { 1798 $IMresizeParameter = 'resize'; 1799 1800 // some (older? around 2002) versions of IM won't accept "-resize 100x" but require "-resize 100x100" 1801 $commandline_test = $this->ImageMagickCommandlineBase().' logo: -resize 1x '.phpthumb_functions::escapeshellarg_replacement($IMtempfilename).' 2>&1'; 1802 $IMresult_test = phpthumb_functions::SafeExec($commandline_test); 1803 $IMuseExplicitImageOutputDimensions = preg_match('#image dimensions are zero#i', $IMresult_test); 1804 $this->DebugMessage('IMuseExplicitImageOutputDimensions = '. (int) $IMuseExplicitImageOutputDimensions, __FILE__, __LINE__); 1805 if ($fp_im_temp = @fopen($IMtempfilename, 'wb')) { 1806 // erase temp image so ImageMagick logo doesn't get output if other processing fails 1807 fclose($fp_im_temp); 1808 @chmod($IMtempfilename, $this->getParameter('config_file_create_mask')); 1809 } 1810 } 1811 1812 1813 ob_start(); 1814 $getimagesize = getimagesize($this->sourceFilename); 1815 $GetImageSizeError = ob_get_contents(); 1816 ob_end_clean(); 1817 if (is_array($getimagesize)) { 1818 $this->DebugMessage('getimagesize('.$this->sourceFilename.') SUCCEEDED: '.print_r($getimagesize, true), __FILE__, __LINE__); 1819 } else { 1820 $this->DebugMessage('getimagesize('.$this->sourceFilename.') FAILED with error "'.$GetImageSizeError.'"', __FILE__, __LINE__); 1821 } 1822 if (null !== $this->dpi && $this->ImageMagickSwitchAvailable('density')) { 1823 // for vector source formats only (WMF, PDF, etc) 1824 if (is_array($getimagesize) && isset($getimagesize[2]) && ($getimagesize[2] == IMAGETYPE_PNG)) { 1825 // explicitly exclude PNG from "-flatten" to make sure transparency is preserved 1826 // https://github.com/JamesHeinrich/phpThumb/issues/65 1827 } else { 1828 $commandline .= ' -flatten'; 1829 $commandline .= ' -density '.phpthumb_functions::escapeshellarg_replacement($this->dpi); 1830 } 1831 } 1832 if (is_array($getimagesize)) { 1833 $this->DebugMessage('getimagesize('.$this->sourceFilename.') returned [w='.$getimagesize[0].';h='.$getimagesize[1].';f='.$getimagesize[2].']', __FILE__, __LINE__); 1834 $this->source_width = $getimagesize[0]; 1835 $this->source_height = $getimagesize[1]; 1836 $this->DebugMessage('source dimensions set to '.$this->source_width.'x'.$this->source_height, __FILE__, __LINE__); 1837 $this->SetOrientationDependantWidthHeight(); 1838 1839 if (!preg_match('#('.implode('|', $this->AlphaCapableFormats).')#i', $outputFormat)) { 1840 // not a transparency-capable format 1841 $commandline .= ' -background '.phpthumb_functions::escapeshellarg_replacement('#'.($this->bg ? $this->bg : 'FFFFFF')); 1842 if (!stristr($commandline, ' -flatten')) { 1843 $commandline .= ' -flatten'; 1844 } 1845 } else { 1846 if ($getimagesize[2] == IMAGETYPE_PNG && !$this->bg) { 1847 $commandline .= ' -background none'; 1848 } 1849 } 1850 if ($getimagesize[2] == IMAGETYPE_GIF) { 1851 $commandline .= ' -coalesce'; // may be needed for animated GIFs 1852 } 1853 if ($this->source_width || $this->source_height) { 1854 if ($this->zc) { 1855 1856 $borderThickness = 0; 1857 if (!empty($this->fltr)) { 1858 foreach ($this->fltr as $key => $value) { 1859 if (preg_match('#^bord\|([\d]+)#', $value, $matches)) { 1860 $borderThickness = $matches[1]; 1861 break; 1862 } 1863 } 1864 } 1865 $wAll = (int) max($this->w, $this->wp, $this->wl, $this->ws) - (2 * $borderThickness); 1866 $hAll = (int) max($this->h, $this->hp, $this->hl, $this->hs) - (2 * $borderThickness); 1867 $imAR = $this->source_width / $this->source_height; 1868 $zcAR = (($wAll && $hAll) ? $wAll / $hAll : 1); 1869 $side = phpthumb_functions::nonempty_min($this->source_width, $this->source_height, max($wAll, $hAll)); 1870 $sideX = phpthumb_functions::nonempty_min($this->source_width, $wAll, round($hAll * $zcAR)); 1871 $sideY = phpthumb_functions::nonempty_min( $this->source_height, $hAll, round($wAll / $zcAR)); 1872 1873 $thumbnailH = round(max($sideY, ($sideY * $zcAR) / $imAR)); 1874 if ($this->aoe == 1) { 1875 $commandline .= ' -'.$IMresizeParameter.' "'.$wAll.'x'.$hAll.'^"'; 1876 } else { 1877 $commandline .= ' -'.$IMresizeParameter.' '.phpthumb_functions::escapeshellarg_replacement(($IMuseExplicitImageOutputDimensions ? $thumbnailH : '').'x'.$thumbnailH); 1878 } 1879 1880 switch (strtoupper($this->zc)) { 1881 case 'T': 1882 $commandline .= ' -gravity north'; 1883 break; 1884 case 'B': 1885 $commandline .= ' -gravity south'; 1886 break; 1887 case 'L': 1888 $commandline .= ' -gravity west'; 1889 break; 1890 case 'R': 1891 $commandline .= ' -gravity east'; 1892 break; 1893 case 'TL': 1894 $commandline .= ' -gravity northwest'; 1895 break; 1896 case 'TR': 1897 $commandline .= ' -gravity northeast'; 1898 break; 1899 case 'BL': 1900 $commandline .= ' -gravity southwest'; 1901 break; 1902 case 'BR': 1903 $commandline .= ' -gravity southeast'; 1904 break; 1905 case '1': 1906 case 'C': 1907 default: 1908 $commandline .= ' -gravity center'; 1909 break; 1910 } 1911 1912 if (($wAll > 0) && ($hAll > 0)) { 1913 $commandline .= ' -crop '.phpthumb_functions::escapeshellarg_replacement($wAll.'x'.$hAll.'+0+0'); 1914 } else { 1915 $commandline .= ' -crop '.phpthumb_functions::escapeshellarg_replacement($side.'x'.$side.'+0+0'); 1916 } 1917 if ($this->ImageMagickSwitchAvailable('repage')) { 1918 $commandline .= ' +repage'; 1919 } else { 1920 $this->DebugMessage('Skipping "+repage" because ImageMagick (v'.$this->ImageMagickVersion().') does not support it', __FILE__, __LINE__); 1921 } 1922 1923 } elseif ($this->sw || $this->sh || $this->sx || $this->sy) { 1924 1925 $crop_param = ''; 1926 $crop_param .= ($this->sw ? (($this->sw < 2) ? round($this->sw * $this->source_width) : $this->sw) : $this->source_width); 1927 $crop_param .= 'x'.($this->sh ? (($this->sh < 2) ? round($this->sh * $this->source_height) : $this->sh) : $this->source_height); 1928 $crop_param .= '+'.(($this->sx < 2) ? round($this->sx * $this->source_width) : $this->sx); 1929 $crop_param .= '+'.(($this->sy < 2) ? round($this->sy * $this->source_height) : $this->sy); 1930// TO BE FIXED 1931// makes 1x1 output 1932// http://trainspotted.com/phpThumb/phpThumb.php?src=/content/CNR/47/CNR-4728-LD-L-20110723-898.jpg&w=100&h=100&far=1&f=png&fltr[]=lvl&sx=0.05&sy=0.25&sw=0.92&sh=0.42 1933// '/usr/bin/convert' -density 150 -thumbnail 100x100 -contrast-stretch '0.1%' '/var/www/vhosts/trainspotted.com/httpdocs/content/CNR/47/CNR-4728-LD-L-20110723-898.jpg[0]' png:'/var/www/vhosts/trainspotted.com/httpdocs/phpThumb/_cache/pThumbIIUlvj' 1934 $commandline .= ' -crop '.phpthumb_functions::escapeshellarg_replacement($crop_param); 1935 1936 // this is broken for aoe=1, but unsure how to fix. Send advice to info@silisoftware.com 1937 if ($this->w || $this->h) { 1938 //if ($this->ImageMagickSwitchAvailable('repage')) { 1939if (false) { 1940// TO BE FIXED 1941// newer versions of ImageMagick require -repage <geometry> 1942 $commandline .= ' -repage'; 1943 } else { 1944 $this->DebugMessage('Skipping "-repage" because ImageMagick (v'.$this->ImageMagickVersion().') does not support it', __FILE__, __LINE__); 1945 } 1946 if ($IMuseExplicitImageOutputDimensions) { 1947 if ($this->w && !$this->h) { 1948 $this->h = ceil($this->w / ($this->source_width / $this->source_height)); 1949 } elseif ($this->h && !$this->w) { 1950 $this->w = ceil($this->h * ($this->source_width / $this->source_height)); 1951 } 1952 } 1953 $commandline .= ' -'.$IMresizeParameter.' '.phpthumb_functions::escapeshellarg_replacement($this->w.'x'.$this->h); 1954 } 1955 1956 } else { 1957 1958 if ($this->iar && ((int) $this->w > 0) && ((int) $this->h > 0)) { 1959 1960 list($nw, $nh) = phpthumb_functions::TranslateWHbyAngle($this->w, $this->h, $this->ra); 1961 $nw = ((round($nw) != 0) ? round($nw) : ''); 1962 $nh = ((round($nh) != 0) ? round($nh) : ''); 1963 $commandline .= ' -'.$IMresizeParameter.' '.phpthumb_functions::escapeshellarg_replacement($nw.'x'.$nh.'!'); 1964 1965 } elseif ($this->far && ((int) $this->w > 0) && ((int) $this->h > 0)) { 1966 1967 $commandline .= ' -'.$IMresizeParameter.' '.phpthumb_functions::escapeshellarg_replacement(phpthumb_functions::nonempty_min($this->w, $getimagesize[0]).'x'.phpthumb_functions::nonempty_min($this->h, $getimagesize[1])); 1968 $commandline .= ' -gravity center'; 1969 if ($this->bg) { 1970 $commandline .= ' -background ' . phpthumb_functions::escapeshellarg_replacement('#' . $this->bg); 1971 } else { 1972 $commandline .= ' -background none'; 1973 } 1974 $commandline .= ' -extent '.phpthumb_functions::escapeshellarg_replacement($this->w.'x'.$this->h); 1975 1976 } else { 1977 1978 $this->w = (($this->aoe && $this->w) ? $this->w : ($this->w ? phpthumb_functions::nonempty_min($this->w, $getimagesize[0]) : null)); 1979 $this->h = (($this->aoe && $this->h) ? $this->h : ($this->h ? phpthumb_functions::nonempty_min($this->h, $getimagesize[1]) : null)); 1980 if ($this->w || $this->h) { 1981 if ($IMuseExplicitImageOutputDimensions) { 1982 if ($this->w && !$this->h) { 1983 $this->h = ceil($this->w / ($this->source_width / $this->source_height)); 1984 } elseif ($this->h && !$this->w) { 1985 $this->w = ceil($this->h * ($this->source_width / $this->source_height)); 1986 } 1987 } 1988 list($nw, $nh) = phpthumb_functions::TranslateWHbyAngle($this->w, $this->h, $this->ra); 1989 $nw = ((round($nw) != 0) ? round($nw) : ''); 1990 $nh = ((round($nh) != 0) ? round($nh) : ''); 1991 $commandline .= ' -'.$IMresizeParameter.' '.phpthumb_functions::escapeshellarg_replacement($nw.'x'.$nh); 1992 } 1993 1994 } 1995 } 1996 } 1997 1998 } else { 1999 2000 $this->DebugMessage('getimagesize('.$this->sourceFilename.') failed', __FILE__, __LINE__); 2001 if ($this->w || $this->h) { 2002 $exactDimensionsBang = (($this->iar && ((int) $this->w > 0) && ((int) $this->h > 0)) ? '!' : ''); 2003 if ($IMuseExplicitImageOutputDimensions) { 2004 // unknown source aspect ratio, just put large number and hope IM figures it out 2005 $commandline .= ' -'.$IMresizeParameter.' '.phpthumb_functions::escapeshellarg_replacement(($this->w ? $this->w : '9999').'x'.($this->h ? $this->h : '9999').$exactDimensionsBang); 2006 } else { 2007 $commandline .= ' -'.$IMresizeParameter.' '.phpthumb_functions::escapeshellarg_replacement($this->w.'x'.$this->h.$exactDimensionsBang); 2008 } 2009 } 2010 2011 } 2012 2013 if ($this->ra) { 2014 $this->ra = (int) $this->ra; 2015 if ($this->ImageMagickSwitchAvailable('rotate')) { 2016 if (!preg_match('#('.implode('|', $this->AlphaCapableFormats).')#i', $outputFormat) || phpthumb_functions::version_compare_replacement($this->ImageMagickVersion(), '6.3.7', '>=')) { 2017 $this->DebugMessage('Using ImageMagick rotate', __FILE__, __LINE__); 2018 $commandline .= ' -rotate '.phpthumb_functions::escapeshellarg_replacement($this->ra); 2019 if (($this->ra % 90) != 0) { 2020 if (preg_match('#('.implode('|', $this->AlphaCapableFormats).')#i', $outputFormat)) { 2021 // alpha-capable format 2022 $commandline .= ' -background rgba(255,255,255,0)'; 2023 } else { 2024 $commandline .= ' -background '.phpthumb_functions::escapeshellarg_replacement('#'.($this->bg ? $this->bg : 'FFFFFF')); 2025 } 2026 } 2027 $this->ra = 0; 2028 } else { 2029 $this->DebugMessage('Not using ImageMagick rotate because alpha background buggy before v6.3.7', __FILE__, __LINE__); 2030 } 2031 } else { 2032 $this->DebugMessage('Not using ImageMagick rotate because not supported', __FILE__, __LINE__); 2033 } 2034 } 2035 2036 $successfullyProcessedFilters = array(); 2037 foreach ($this->fltr as $filterkey => $filtercommand) { 2038 @list($command, $parameter) = explode('|', $filtercommand, 2); 2039 switch ($command) { 2040 case 'brit': 2041 if ($this->ImageMagickSwitchAvailable('modulate')) { 2042 $commandline .= ' -modulate '.phpthumb_functions::escapeshellarg_replacement((100 + (int) $parameter).',100,100'); 2043 $successfullyProcessedFilters[] = $filterkey; 2044 } 2045 break; 2046 2047 case 'cont': 2048 if ($this->ImageMagickSwitchAvailable('contrast')) { 2049 $contDiv10 = round((int) $parameter / 10); 2050 if ($contDiv10 > 0) { 2051 $contDiv10 = min($contDiv10, 100); 2052 for ($i = 0; $i < $contDiv10; $i++) { 2053 $commandline .= ' -contrast'; // increase contrast by 10% 2054 } 2055 } elseif ($contDiv10 < 0) { 2056 $contDiv10 = max($contDiv10, -100); 2057 for ($i = $contDiv10; $i < 0; $i++) { 2058 $commandline .= ' +contrast'; // decrease contrast by 10% 2059 } 2060 } else { 2061 // do nothing 2062 } 2063 $successfullyProcessedFilters[] = $filterkey; 2064 } 2065 break; 2066 2067 case 'ds': 2068 if ($this->ImageMagickSwitchAvailable(array('colorspace', 'modulate'))) { 2069 if ($parameter == 100) { 2070 $commandline .= ' -colorspace GRAY'; 2071 $commandline .= ' -modulate 100,0,100'; 2072 } else { 2073 $commandline .= ' -modulate '.phpthumb_functions::escapeshellarg_replacement('100,'.(100 - (int) $parameter).',100'); 2074 } 2075 $successfullyProcessedFilters[] = $filterkey; 2076 } 2077 break; 2078 2079 case 'sat': 2080 if ($this->ImageMagickSwitchAvailable(array('colorspace', 'modulate'))) { 2081 if ($parameter == -100) { 2082 $commandline .= ' -colorspace GRAY'; 2083 $commandline .= ' -modulate 100,0,100'; 2084 } else { 2085 $commandline .= ' -modulate '.phpthumb_functions::escapeshellarg_replacement('100,'.(100 + (int) $parameter).',100'); 2086 } 2087 $successfullyProcessedFilters[] = $filterkey; 2088 } 2089 break; 2090 2091 case 'gray': 2092 if ($this->ImageMagickSwitchAvailable(array('colorspace', 'modulate'))) { 2093 $commandline .= ' -colorspace GRAY'; 2094 $commandline .= ' -modulate 100,0,100'; 2095 $successfullyProcessedFilters[] = $filterkey; 2096 } 2097 break; 2098 2099 case 'clr': 2100 if ($this->ImageMagickSwitchAvailable(array('fill', 'colorize'))) { 2101 @list($amount, $color) = explode('|', $parameter); 2102 $commandline .= ' -fill '.phpthumb_functions::escapeshellarg_replacement('#'.preg_replace('#[^0-9A-F]#i', '', $color)); 2103 $commandline .= ' -colorize '.phpthumb_functions::escapeshellarg_replacement(min(max((int) $amount, 0), 100)); 2104 $successfullyProcessedFilters[] = $filterkey; 2105 } 2106 break; 2107 2108 case 'sep': 2109 if ($this->ImageMagickSwitchAvailable('sepia-tone')) { 2110 @list($amount, $color) = explode('|', $parameter); 2111 $amount = ($amount ? $amount : 80); 2112 if (!$color) { 2113 $commandline .= ' -sepia-tone '.phpthumb_functions::escapeshellarg_replacement(min(max((int) $amount, 0), 100).'%'); 2114 $successfullyProcessedFilters[] = $filterkey; 2115 } 2116 } 2117 break; 2118 2119 case 'gam': 2120 @list($amount) = explode('|', $parameter); 2121 $amount = min(max((float) $amount, 0.001), 10); 2122 if (number_format($amount, 3) != '1.000') { 2123 if ($this->ImageMagickSwitchAvailable('gamma')) { 2124 $commandline .= ' -gamma '.phpthumb_functions::escapeshellarg_replacement($amount); 2125 $successfullyProcessedFilters[] = $filterkey; 2126 } 2127 } 2128 break; 2129 2130 case 'neg': 2131 if ($this->ImageMagickSwitchAvailable('negate')) { 2132 $commandline .= ' -negate'; 2133 $successfullyProcessedFilters[] = $filterkey; 2134 } 2135 break; 2136 2137 case 'th': 2138 @list($amount) = explode('|', $parameter); 2139 if ($this->ImageMagickSwitchAvailable(array('threshold', 'dither', 'monochrome'))) { 2140 $commandline .= ' -threshold '.phpthumb_functions::escapeshellarg_replacement(round(min(max((int) $amount, 0), 255) / 2.55).'%'); 2141 $commandline .= ' -dither'; 2142 $commandline .= ' -monochrome'; 2143 $successfullyProcessedFilters[] = $filterkey; 2144 } 2145 break; 2146 2147 case 'rcd': 2148 if ($this->ImageMagickSwitchAvailable(array('colors', 'dither'))) { 2149 @list($colors, $dither) = explode('|', $parameter); 2150 $colors = ($colors ? (int) $colors : 256); 2151 $dither = ((strlen($dither) > 0) ? (bool) $dither : true); 2152 $commandline .= ' -colors '.phpthumb_functions::escapeshellarg_replacement(max($colors, 8)); // ImageMagick will otherwise fail with "cannot quantize to fewer than 8 colors" 2153 $commandline .= ($dither ? ' -dither' : ' +dither'); 2154 $successfullyProcessedFilters[] = $filterkey; 2155 } 2156 break; 2157 2158 case 'flip': 2159 if ($this->ImageMagickSwitchAvailable(array('flip', 'flop'))) { 2160 if (strpos(strtolower($parameter), 'x') !== false) { 2161 $commandline .= ' -flop'; 2162 } 2163 if (strpos(strtolower($parameter), 'y') !== false) { 2164 $commandline .= ' -flip'; 2165 } 2166 $successfullyProcessedFilters[] = $filterkey; 2167 } 2168 break; 2169 2170 case 'edge': 2171 if ($this->ImageMagickSwitchAvailable('edge')) { 2172 $parameter = (!empty($parameter) ? $parameter : 2); 2173 $commandline .= ' -edge '.phpthumb_functions::escapeshellarg_replacement(!empty($parameter) ? (int) $parameter : 1); 2174 $successfullyProcessedFilters[] = $filterkey; 2175 } 2176 break; 2177 2178 case 'emb': 2179 if ($this->ImageMagickSwitchAvailable(array('emboss', 'negate'))) { 2180 $parameter = (!empty($parameter) ? $parameter : 2); 2181 $commandline .= ' -emboss '.phpthumb_functions::escapeshellarg_replacement((int) $parameter); 2182 if ($parameter < 2) { 2183 $commandline .= ' -negate'; // ImageMagick negates the image for some reason with '-emboss 1'; 2184 } 2185 $successfullyProcessedFilters[] = $filterkey; 2186 } 2187 break; 2188 2189 case 'lvl': 2190 @list($band, $method, $threshold) = explode('|', $parameter); 2191 $band = ($band ? preg_replace('#[^RGBA\\*]#', '', strtoupper($band)) : '*'); 2192 $method = ((strlen($method) > 0) ? (int) $method : 2); 2193 $threshold = ((strlen($threshold) > 0) ? min(max((float) $threshold, 0), 100) : 0.1); 2194 2195 $band = preg_replace('#[^RGBA\\*]#', '', strtoupper($band)); 2196 2197 if (($method > 1) && !$this->ImageMagickSwitchAvailable(array('channel', 'contrast-stretch'))) { 2198 // Because ImageMagick processing happens before PHP-GD filters, and because some 2199 // clipping is involved in the "lvl" filter, if "lvl" happens before "wb" then the 2200 // "wb" filter will have (almost) no effect. Therefore, if "wb" is enabled then 2201 // force the "lvl" filter to be processed by GD, not ImageMagick. 2202 foreach ($this->fltr as $fltr_key => $fltr_value) { 2203 list($fltr_cmd) = explode('|', $fltr_value); 2204 if ($fltr_cmd == 'wb') { 2205 $this->DebugMessage('Setting "lvl" filter method to "0" (from "'.$method.'") because white-balance filter also enabled', __FILE__, __LINE__); 2206 $method = 0; 2207 } 2208 } 2209 } 2210 2211 switch ($method) { 2212 case 0: // internal RGB 2213 case 1: // internal grayscale 2214 break; 2215 case 2: // ImageMagick "contrast-stretch" 2216 if ($this->ImageMagickSwitchAvailable('contrast-stretch')) { 2217 if ($band != '*') { 2218 $commandline .= ' -channel '.phpthumb_functions::escapeshellarg_replacement(strtoupper($band)); 2219 } 2220 $threshold = preg_replace('#[^0-9\\.]#', '', $threshold); // should be unneccesary, but just to be double-sure 2221 //$commandline .= ' -contrast-stretch '.phpthumb_functions::escapeshellarg_replacement($threshold.'%'); 2222 $commandline .= ' -contrast-stretch \''.$threshold.'%\''; 2223 if ($band != '*') { 2224 $commandline .= ' +channel'; 2225 } 2226 $successfullyProcessedFilters[] = $filterkey; 2227 } 2228 break; 2229 case 3: // ImageMagick "normalize" 2230 if ($this->ImageMagickSwitchAvailable('normalize')) { 2231 if ($band != '*') { 2232 $commandline .= ' -channel '.phpthumb_functions::escapeshellarg_replacement(strtoupper($band)); 2233 } 2234 $commandline .= ' -normalize'; 2235 if ($band != '*') { 2236 $commandline .= ' +channel'; 2237 } 2238 $successfullyProcessedFilters[] = $filterkey; 2239 } 2240 break; 2241 default: 2242 $this->DebugMessage('unsupported method ('.$method.') for "lvl" filter', __FILE__, __LINE__); 2243 break; 2244 } 2245 if (isset($this->fltr[$filterkey]) && ($method > 1)) { 2246 $this->fltr[$filterkey] = $command.'|'.$band.'|0|'.$threshold; 2247 $this->DebugMessage('filter "lvl" remapped from method "'.$method.'" to method "0" because ImageMagick support is missing', __FILE__, __LINE__); 2248 } 2249 break; 2250 2251 case 'wb': 2252 if ($this->ImageMagickSwitchAvailable(array('channel', 'contrast-stretch'))) { 2253 @list($threshold) = explode('|', $parameter); 2254 $threshold = (!empty($threshold) ? min(max((float) $threshold, 0), 100) : 0.1); 2255 $threshold = preg_replace('#[^0-9\\.]#', '', $threshold); // should be unneccesary, but just to be double-sure 2256 //$commandline .= ' -channel R -contrast-stretch '.phpthumb_functions::escapeshellarg_replacement($threshold.'%'); // doesn't work on Windows because most versions of PHP do not properly 2257 //$commandline .= ' -channel G -contrast-stretch '.phpthumb_functions::escapeshellarg_replacement($threshold.'%'); // escape special characters (such as %) and just replace them with spaces 2258 //$commandline .= ' -channel B -contrast-stretch '.phpthumb_functions::escapeshellarg_replacement($threshold.'%'); // https://bugs.php.net/bug.php?id=43261 2259 $commandline .= ' -channel R -contrast-stretch \''.$threshold.'%\''; 2260 $commandline .= ' -channel G -contrast-stretch \''.$threshold.'%\''; 2261 $commandline .= ' -channel B -contrast-stretch \''.$threshold.'%\''; 2262 $commandline .= ' +channel'; 2263 $successfullyProcessedFilters[] = $filterkey; 2264 } 2265 break; 2266 2267 case 'blur': 2268 if ($this->ImageMagickSwitchAvailable('blur')) { 2269 @list($radius) = explode('|', $parameter); 2270 $radius = (!empty($radius) ? min(max((int) $radius, 0), 25) : 1); 2271 $commandline .= ' -blur '.phpthumb_functions::escapeshellarg_replacement($radius); 2272 $successfullyProcessedFilters[] = $filterkey; 2273 } 2274 break; 2275 2276 case 'gblr': 2277 @list($radius) = explode('|', $parameter); 2278 $radius = (!empty($radius) ? min(max((int) $radius, 0), 25) : 1); 2279 // "-gaussian" changed to "-gaussian-blur" sometime around 2009 2280 if ($this->ImageMagickSwitchAvailable('gaussian-blur')) { 2281 $commandline .= ' -gaussian-blur '.phpthumb_functions::escapeshellarg_replacement($radius); 2282 $successfullyProcessedFilters[] = $filterkey; 2283 } elseif ($this->ImageMagickSwitchAvailable('gaussian')) { 2284 $commandline .= ' -gaussian '.phpthumb_functions::escapeshellarg_replacement($radius); 2285 $successfullyProcessedFilters[] = $filterkey; 2286 } 2287 break; 2288 2289 case 'usm': 2290 if ($this->ImageMagickSwitchAvailable('unsharp')) { 2291 @list($amount, $radius, $threshold) = explode('|', $parameter); 2292 $amount = ($amount ? min(max((int) $amount, 0), 255) : 80); 2293 $radius = ($radius ? min(max((int) $radius, 0), 10) : 0.5); 2294 $threshold = ('' !== $threshold ? min(max((int) $threshold, 0), 50) : 3); 2295 $commandline .= ' -unsharp '.phpthumb_functions::escapeshellarg_replacement(number_format(($radius * 2) - 1, 2, '.', '').'x1+'.number_format($amount / 100, 2, '.', '').'+'.number_format($threshold / 100, 2, '.', '')); 2296 $successfullyProcessedFilters[] = $filterkey; 2297 } 2298 break; 2299 2300 case 'bord': 2301 if ($this->ImageMagickSwitchAvailable(array('border', 'bordercolor', 'thumbnail', 'crop'))) { 2302 if (!$this->zc) { 2303 @list($width, $rX, $rY, $color) = explode('|', $parameter); 2304 $width = (int) $width; 2305 $rX = (int) $rX; 2306 $rY = (int) $rY; 2307 if ($width && !$rX && !$rY) { 2308 if (!phpthumb_functions::IsHexColor($color)) { 2309 $color = ((!empty($this->bc) && phpthumb_functions::IsHexColor($this->bc)) ? $this->bc : '000000'); 2310 } 2311 $commandline .= ' -border '.phpthumb_functions::escapeshellarg_replacement((int) $width); 2312 $commandline .= ' -bordercolor '.phpthumb_functions::escapeshellarg_replacement('#'.$color); 2313 2314 if (preg_match('# \\-crop "([\d]+)x([\d]+)\\+0\\+0" #', $commandline, $matches)) { 2315 $commandline = str_replace(' -crop "'.$matches[1].'x'.$matches[2].'+0+0" ', ' -crop '.phpthumb_functions::escapeshellarg_replacement(($matches[1] - (2 * $width)).'x'.($matches[2] - (2 * $width)).'+0+0').' ', $commandline); 2316 } elseif (preg_match('# \\-'.$IMresizeParameter.' "([0-9]+)x([0-9]+)" #', $commandline, $matches)) { 2317 $commandline = str_replace(' -'.$IMresizeParameter.' "'.$matches[1].'x'.$matches[2].'" ', ' -'.$IMresizeParameter.' '.phpthumb_functions::escapeshellarg_replacement(($matches[1] - (2 * $width)).'x'.($matches[2] - (2 * $width))).' ', $commandline); 2318 } 2319 $successfullyProcessedFilters[] = $filterkey; 2320 } 2321 } 2322 } 2323 break; 2324 2325 case 'crop': 2326 break; 2327 2328 case 'sblr': 2329 break; 2330 2331 case 'mean': 2332 break; 2333 2334 case 'smth': 2335 break; 2336 2337 case 'bvl': 2338 break; 2339 2340 case 'wmi': 2341 break; 2342 2343 case 'wmt': 2344 break; 2345 2346 case 'over': 2347 break; 2348 2349 case 'hist': 2350 break; 2351 2352 case 'fram': 2353 break; 2354 2355 case 'drop': 2356 break; 2357 2358 case 'mask': 2359 break; 2360 2361 case 'elip': 2362 break; 2363 2364 case 'ric': 2365 break; 2366 2367 case 'stc': 2368 break; 2369 2370 case 'size': 2371 break; 2372 2373 default: 2374 $this->DebugMessage('Unknown $this->fltr['.$filterkey.'] ('.$filtercommand.') -- deleting filter command', __FILE__, __LINE__); 2375 $successfullyProcessedFilters[] = $filterkey; 2376 break; 2377 } 2378 if (!isset($this->fltr[$filterkey])) { 2379 $this->DebugMessage('Processed $this->fltr['.$filterkey.'] ('.$filtercommand.') with ImageMagick', __FILE__, __LINE__); 2380 } else { 2381 $this->DebugMessage('Skipping $this->fltr['.$filterkey.'] ('.$filtercommand.') with ImageMagick', __FILE__, __LINE__); 2382 } 2383 } 2384 $this->DebugMessage('Remaining $this->fltr after ImageMagick: ('.$this->phpThumbDebugVarDump($this->fltr).')', __FILE__, __LINE__); 2385 if (count($this->fltr) > 0) { 2386 $this->useRawIMoutput = false; 2387 } 2388 2389 if (preg_match('#jpe?g#i', $outputFormat) && $this->q) { 2390 if ($this->ImageMagickSwitchAvailable(array('quality', 'interlace'))) { 2391 $commandline .= ' -quality '.phpthumb_functions::escapeshellarg_replacement($this->thumbnailQuality); 2392 if ($this->config_output_interlace) { 2393 // causes weird things with animated GIF... leave for JPEG only 2394 $commandline .= ' -interlace line '; // Use Line or Plane to create an interlaced PNG or GIF or progressive JPEG image 2395 } 2396 } 2397 } 2398 $commandline .= ' '.$outputFormat.':'.phpthumb_functions::escapeshellarg_replacement($IMtempfilename); 2399 if (!$this->iswindows) { 2400 $commandline .= ' 2>&1'; 2401 } 2402 $this->DebugMessage('ImageMagick called as ('.$commandline.')', __FILE__, __LINE__); 2403 $IMresult = phpthumb_functions::SafeExec($commandline); 2404 clearstatcache(); 2405 if (!@file_exists($IMtempfilename) || !@filesize($IMtempfilename)) { 2406 $this->FatalError('ImageMagick failed with message ('.trim($IMresult).')'); 2407 $this->DebugMessage('ImageMagick failed with message ('.trim($IMresult).')', __FILE__, __LINE__); 2408 if ($this->iswindows && !$IMresult) { 2409 $this->DebugMessage('Check to make sure that PHP has read+write permissions to "'.dirname($IMtempfilename).'"', __FILE__, __LINE__); 2410 } 2411 2412 } else { 2413 2414 foreach ($successfullyProcessedFilters as $dummy => $filterkey) { 2415 unset($this->fltr[$filterkey]); 2416 } 2417 $this->IMresizedData = file_get_contents($IMtempfilename); 2418 $getimagesize_imresized = @getimagesize($IMtempfilename); 2419 $this->DebugMessage('getimagesize('.$IMtempfilename.') returned [w='.$getimagesize_imresized[0].';h='.$getimagesize_imresized[1].';f='.$getimagesize_imresized[2].']', __FILE__, __LINE__); 2420 if (($this->config_max_source_pixels > 0) && (($getimagesize_imresized[0] * $getimagesize_imresized[1]) > $this->config_max_source_pixels)) { 2421 $this->DebugMessage('skipping ImageMagickThumbnailToGD::'.$ImageCreateFunction.'() because IM output is too large ('.$getimagesize_imresized[0].'x'.$getimagesize_imresized[0].' = '.($getimagesize_imresized[0] * $getimagesize_imresized[1]).' > '.$this->config_max_source_pixels.')', __FILE__, __LINE__); 2422 } elseif (function_exists(@$ImageCreateFunction) && ($this->gdimg_source = @$ImageCreateFunction($IMtempfilename))) { 2423 $this->source_width = imagesx($this->gdimg_source); 2424 $this->source_height = imagesy($this->gdimg_source); 2425 $this->DebugMessage('ImageMagickThumbnailToGD::'.$ImageCreateFunction.'() succeeded, $this->gdimg_source is now ('.$this->source_width.'x'.$this->source_height.')', __FILE__, __LINE__); 2426 $this->DebugMessage('ImageMagickThumbnailToGD() returning $this->IMresizedData ('.strlen($this->IMresizedData).' bytes)', __FILE__, __LINE__); 2427 } else { 2428 $this->useRawIMoutput = true; 2429 $this->DebugMessage('$this->useRawIMoutput set to TRUE because '.@$ImageCreateFunction.'('.$IMtempfilename.') failed', __FILE__, __LINE__); 2430 } 2431 if (file_exists($IMtempfilename)) { 2432 $this->DebugMessage('deleting "'.$IMtempfilename.'"', __FILE__, __LINE__); 2433 @unlink($IMtempfilename); 2434 } 2435 return true; 2436 2437 } 2438 if (file_exists($IMtempfilename)) { 2439 $this->DebugMessage('deleting "'.$IMtempfilename.'"', __FILE__, __LINE__); 2440 @unlink($IMtempfilename); 2441 } 2442 2443 } elseif ($this->issafemode) { 2444 $this->DebugMessage('ImageMagickThumbnailToGD() aborting because PHP safe_mode is enabled and phpThumb_tempnam() failed', __FILE__, __LINE__); 2445 $this->useRawIMoutput = false; 2446 } else { 2447 if (file_exists($IMtempfilename)) { 2448 $this->DebugMessage('deleting "'.$IMtempfilename.'"', __FILE__, __LINE__); 2449 @unlink($IMtempfilename); 2450 } 2451 $this->DebugMessage('ImageMagickThumbnailToGD() aborting, phpThumb_tempnam() failed', __FILE__, __LINE__); 2452 } 2453 } else { 2454 $this->DebugMessage('ImageMagickThumbnailToGD() aborting because ImageMagickCommandlineBase() failed', __FILE__, __LINE__); 2455 } 2456 $this->useRawIMoutput = false; 2457 return false; 2458 } 2459 2460 2461 public function Rotate() { 2462 if ($this->ra || $this->ar) { 2463 if (!function_exists('imagerotate')) { 2464 $this->DebugMessage('!function_exists(imagerotate)', __FILE__, __LINE__); 2465 return false; 2466 } 2467 if (!include_once __DIR__ .'/phpthumb.filters.php' ) { 2468 $this->DebugMessage('Error including "'. __DIR__ .'/phpthumb.filters.php" which is required for applying filters ('.implode(';', $this->fltr).')', __FILE__, __LINE__); 2469 return false; 2470 } 2471 2472 $this->config_background_hexcolor = ($this->bg ? $this->bg : $this->config_background_hexcolor); 2473 if (!phpthumb_functions::IsHexColor($this->config_background_hexcolor)) { 2474 return $this->ErrorImage('Invalid hex color string "'.$this->config_background_hexcolor.'" for parameter "bg"'); 2475 } 2476 2477 $rotate_angle = 0; 2478 if ($this->ra) { 2479 2480 $rotate_angle = (float) $this->ra; 2481 2482 } else { 2483 2484 if ($this->ar == 'x') { 2485 if (phpthumb_functions::version_compare_replacement(PHP_VERSION, '4.2.0', '>=')) { 2486 if ($this->sourceFilename) { 2487 if (function_exists('exif_read_data')) { 2488 if ($exif_data = @exif_read_data($this->sourceFilename, 'IFD0')) { 2489 // http://sylvana.net/jpegcrop/exif_orientation.html 2490 switch (@$exif_data['Orientation']) { 2491 case 1: 2492 $rotate_angle = 0; 2493 break; 2494 case 3: 2495 $rotate_angle = 180; 2496 break; 2497 case 6: 2498 $rotate_angle = 270; 2499 break; 2500 case 8: 2501 $rotate_angle = 90; 2502 break; 2503 2504 default: 2505 $this->DebugMessage('EXIF auto-rotate failed because unknown $exif_data[Orientation] "'.@$exif_data['Orientation'].'"', __FILE__, __LINE__); 2506 return false; 2507 break; 2508 } 2509 $this->DebugMessage('EXIF auto-rotate set to '.$rotate_angle.' degrees ($exif_data[Orientation] = "'.@$exif_data['Orientation'].'")', __FILE__, __LINE__); 2510 } else { 2511 $this->DebugMessage('failed: exif_read_data('.$this->sourceFilename.')', __FILE__, __LINE__); 2512 return false; 2513 } 2514 } else { 2515 $this->DebugMessage('!function_exists(exif_read_data)', __FILE__, __LINE__); 2516 return false; 2517 } 2518 } else { 2519 $this->DebugMessage('Cannot auto-rotate from EXIF data because $this->sourceFilename is empty', __FILE__, __LINE__); 2520 return false; 2521 } 2522 } else { 2523 $this->DebugMessage('Cannot auto-rotate from EXIF data because PHP is less than v4.2.0 ('. PHP_VERSION .')', __FILE__, __LINE__); 2524 return false; 2525 } 2526 } elseif (($this->ar == 'l') && ($this->source_height > $this->source_width)) { 2527 $rotate_angle = 270; 2528 } elseif (($this->ar == 'L') && ($this->source_height > $this->source_width)) { 2529 $rotate_angle = 90; 2530 } elseif (($this->ar == 'p') && ($this->source_width > $this->source_height)) { 2531 $rotate_angle = 90; 2532 } elseif (($this->ar == 'P') && ($this->source_width > $this->source_height)) { 2533 $rotate_angle = 270; 2534 } 2535 2536 } 2537 if ($rotate_angle % 90) { 2538 $this->is_alpha = true; 2539 } 2540 phpthumb_filters::ImprovedImageRotate($this->gdimg_source, $rotate_angle, $this->config_background_hexcolor, $this->bg, $this); 2541 $this->source_width = imagesx($this->gdimg_source); 2542 $this->source_height = imagesy($this->gdimg_source); 2543 } 2544 return true; 2545 } 2546 2547 2548 public function FixedAspectRatio() { 2549 // optional fixed-dimension images (regardless of aspect ratio) 2550 2551 if (!$this->far) { 2552 // do nothing 2553 return true; 2554 } 2555 2556 if (!$this->w || !$this->h) { 2557 return false; 2558 } 2559 $this->thumbnail_width = $this->w; 2560 $this->thumbnail_height = $this->h; 2561 $this->is_alpha = true; 2562 if ($this->thumbnail_image_width >= $this->thumbnail_width) { 2563 2564 $aspectratio = $this->thumbnail_image_height / $this->thumbnail_image_width; 2565 if ($this->w) { 2566 $this->thumbnail_image_height = round($this->thumbnail_image_width * $aspectratio); 2567 $this->thumbnail_height = ($this->h ? $this->h : $this->thumbnail_image_height); 2568 } elseif ($this->thumbnail_image_height < $this->thumbnail_height) { 2569 $this->thumbnail_image_height = $this->thumbnail_height; 2570 $this->thumbnail_image_width = round($this->thumbnail_image_height / $aspectratio); 2571 } 2572 2573 } else { 2574 2575 $aspectratio = $this->thumbnail_image_width / $this->thumbnail_image_height; 2576 if ($this->h) { 2577 $this->thumbnail_image_width = round($this->thumbnail_image_height * $aspectratio); 2578 } elseif ($this->thumbnail_image_width < $this->thumbnail_width) { 2579 $this->thumbnail_image_width = $this->thumbnail_width; 2580 $this->thumbnail_image_height = round($this->thumbnail_image_width / $aspectratio); 2581 } 2582 2583 } 2584 return true; 2585 } 2586 2587 2588 public function OffsiteDomainIsAllowed($hostname, $allowed_domains) { 2589 static $domain_is_allowed = array(); 2590 $hostname = strtolower($hostname); 2591 if (!isset($domain_is_allowed[$hostname])) { 2592 $domain_is_allowed[$hostname] = false; 2593 foreach ($allowed_domains as $valid_domain) { 2594 $starpos = strpos($valid_domain, '*'); 2595 if ($starpos !== false) { 2596 $valid_domain = substr($valid_domain, $starpos + 1); 2597 if (preg_match('#'.preg_quote($valid_domain).'$#', $hostname)) { 2598 $domain_is_allowed[$hostname] = true; 2599 break; 2600 } 2601 } else { 2602 if (strtolower($valid_domain) === $hostname) { 2603 $domain_is_allowed[$hostname] = true; 2604 break; 2605 } 2606 } 2607 } 2608 } 2609 return $domain_is_allowed[$hostname]; 2610 } 2611 2612 2613 public function AntiOffsiteLinking() { 2614 // Optional anti-offsite hijacking of the thumbnail script 2615 $allow = true; 2616 if ($allow && $this->config_nooffsitelink_enabled && (@$_SERVER['HTTP_REFERER'] || $this->config_nooffsitelink_require_refer)) { 2617 $this->DebugMessage('AntiOffsiteLinking() checking $_SERVER[HTTP_REFERER] "'.@$_SERVER['HTTP_REFERER'].'"', __FILE__, __LINE__); 2618 foreach ($this->config_nooffsitelink_valid_domains as $key => $valid_domain) { 2619 // $_SERVER['HTTP_HOST'] contains the port number, so strip it out here to make default configuration work 2620 list($clean_domain) = explode(':', $valid_domain); 2621 $this->config_nooffsitelink_valid_domains[$key] = $clean_domain; 2622 } 2623 $parsed_url = phpthumb_functions::ParseURLbetter(@$_SERVER['HTTP_REFERER']); 2624 if (!$this->OffsiteDomainIsAllowed(@$parsed_url['host'], $this->config_nooffsitelink_valid_domains)) { 2625 $allow = false; 2626 //$this->DebugMessage('AntiOffsiteLinking() - "'.@$parsed_url['host'].'" is NOT in $this->config_nooffsitelink_valid_domains ('.implode(';', $this->config_nooffsitelink_valid_domains).')', __FILE__, __LINE__); 2627 $this->ErrorImage('AntiOffsiteLinking() - "'.@$parsed_url['host'].'" is NOT in $this->config_nooffsitelink_valid_domains ('.implode(';', $this->config_nooffsitelink_valid_domains).')'); 2628 } else { 2629 $this->DebugMessage('AntiOffsiteLinking() - "'.@$parsed_url['host'].'" is in $this->config_nooffsitelink_valid_domains ('.implode(';', $this->config_nooffsitelink_valid_domains).')', __FILE__, __LINE__); 2630 } 2631 } 2632 2633 if ($allow && $this->config_nohotlink_enabled && preg_match('#^(f|ht)tps?\://#i', $this->src)) { 2634 $parsed_url = phpthumb_functions::ParseURLbetter($this->src); 2635 //if (!phpthumb_functions::CaseInsensitiveInArray(@$parsed_url['host'], $this->config_nohotlink_valid_domains)) { 2636 if (!$this->OffsiteDomainIsAllowed(@$parsed_url['host'], $this->config_nohotlink_valid_domains)) { 2637 // This domain is not allowed 2638 $allow = false; 2639 $this->DebugMessage('AntiOffsiteLinking() - "'.$parsed_url['host'].'" is NOT in $this->config_nohotlink_valid_domains ('.implode(';', $this->config_nohotlink_valid_domains).')', __FILE__, __LINE__); 2640 } else { 2641 $this->DebugMessage('AntiOffsiteLinking() - "'.$parsed_url['host'].'" is in $this->config_nohotlink_valid_domains ('.implode(';', $this->config_nohotlink_valid_domains).')', __FILE__, __LINE__); 2642 } 2643 } 2644 2645 if ($allow) { 2646 $this->DebugMessage('AntiOffsiteLinking() says this is allowed', __FILE__, __LINE__); 2647 return true; 2648 } 2649 2650 if (!phpthumb_functions::IsHexColor($this->config_error_bgcolor)) { 2651 return $this->ErrorImage('Invalid hex color string "'.$this->config_error_bgcolor.'" for $this->config_error_bgcolor'); 2652 } 2653 if (!phpthumb_functions::IsHexColor($this->config_error_textcolor)) { 2654 return $this->ErrorImage('Invalid hex color string "'.$this->config_error_textcolor.'" for $this->config_error_textcolor'); 2655 } 2656 if ($this->config_nooffsitelink_erase_image) { 2657 2658 return $this->ErrorImage($this->config_nooffsitelink_text_message, $this->thumbnail_width, $this->thumbnail_height); 2659 2660 } else { 2661 2662 $this->config_nooffsitelink_watermark_src = $this->ResolveFilenameToAbsolute($this->config_nooffsitelink_watermark_src); 2663 if (is_file($this->config_nooffsitelink_watermark_src)) { 2664 2665 if (!include_once __DIR__ .'/phpthumb.filters.php' ) { 2666 $this->DebugMessage('Error including "'. __DIR__ .'/phpthumb.filters.php" which is required for applying watermark', __FILE__, __LINE__); 2667 return false; 2668 } 2669 $watermark_img = $this->ImageCreateFromStringReplacement(file_get_contents($this->config_nooffsitelink_watermark_src)); 2670 $phpthumbFilters = new phpthumb_filters(); 2671 $phpthumbFilters->phpThumbObject = &$this; 2672 $opacity = 50; 2673 $margin = 5; 2674 $phpthumbFilters->WatermarkOverlay($this->gdimg_output, $watermark_img, '*', $opacity, $margin); 2675 imagedestroy($watermark_img); 2676 unset($phpthumbFilters); 2677 2678 } else { 2679 2680 $nohotlink_text_array = explode("\n", wordwrap($this->config_nooffsitelink_text_message, floor($this->thumbnail_width / imagefontwidth($this->config_error_fontsize)), "\n")); 2681 $nohotlink_text_color = phpthumb_functions::ImageHexColorAllocate($this->gdimg_output, $this->config_error_textcolor); 2682 2683 $topoffset = round(($this->thumbnail_height - (count($nohotlink_text_array) * imagefontheight($this->config_error_fontsize))) / 2); 2684 2685 $rowcounter = 0; 2686 $this->DebugMessage('AntiOffsiteLinking() writing '.count($nohotlink_text_array).' lines of text "'.$this->config_nooffsitelink_text_message.'" (in #'.$this->config_error_textcolor.') on top of image', __FILE__, __LINE__); 2687 foreach ($nohotlink_text_array as $textline) { 2688 $leftoffset = max(0, round(($this->thumbnail_width - (strlen($textline) * imagefontwidth($this->config_error_fontsize))) / 2)); 2689 imagestring($this->gdimg_output, $this->config_error_fontsize, $leftoffset, $topoffset + ($rowcounter++ * imagefontheight($this->config_error_fontsize)), $textline, $nohotlink_text_color); 2690 } 2691 2692 } 2693 2694 } 2695 return true; 2696 } 2697 2698 2699 public function AlphaChannelFlatten() { 2700 if (!$this->is_alpha) { 2701 // image doesn't have alpha transparency, no need to flatten 2702 $this->DebugMessage('skipping AlphaChannelFlatten() because !$this->is_alpha', __FILE__, __LINE__); 2703 return false; 2704 } 2705 switch ($this->thumbnailFormat) { 2706 case 'png': 2707 case 'webp': 2708 case 'avif': 2709 case 'ico': 2710 // image has alpha transparency, but output as PNG, WEBP, AVIF, ICO which can handle it 2711 $this->DebugMessage('skipping AlphaChannelFlatten() because ($this->thumbnailFormat == "'.$this->thumbnailFormat.'")', __FILE__, __LINE__); 2712 return false; 2713 break; 2714 2715 case 'gif': 2716 // image has alpha transparency, but output as GIF which can handle only single-color transparency 2717 $CurrentImageColorTransparent = imagecolortransparent($this->gdimg_output); 2718 if ($CurrentImageColorTransparent == -1) { 2719 // no transparent color defined 2720 2721 if (phpthumb_functions::gd_version() < 2.0) { 2722 $this->DebugMessage('AlphaChannelFlatten() failed because GD version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 2723 return false; 2724 } 2725 2726 if ($img_alpha_mixdown_dither = @imagecreatetruecolor(imagesx($this->gdimg_output), imagesy($this->gdimg_output))) { 2727 2728 $dither_color = array(); 2729 for ($i = 0; $i <= 255; $i++) { 2730 $dither_color[$i] = imagecolorallocate($img_alpha_mixdown_dither, $i, $i, $i); 2731 } 2732 2733 // scan through current truecolor image copy alpha channel to temp image as grayscale 2734 for ($x = 0; $x < $this->thumbnail_width; $x++) { 2735 for ($y = 0; $y < $this->thumbnail_height; $y++) { 2736 $PixelColor = phpthumb_functions::GetPixelColor($this->gdimg_output, $x, $y); 2737 imagesetpixel($img_alpha_mixdown_dither, $x, $y, $dither_color[ $PixelColor[ 'alpha'] * 2 ]); 2738 } 2739 } 2740 2741 // dither alpha channel grayscale version down to 2 colors 2742 imagetruecolortopalette($img_alpha_mixdown_dither, true, 2); 2743 2744 // reduce color palette to 256-1 colors (leave one palette position for transparent color) 2745 imagetruecolortopalette($this->gdimg_output, true, 255); 2746 2747 // allocate a new color for transparent color index 2748 $TransparentColor = imagecolorallocate($this->gdimg_output, 1, 254, 253); 2749 imagecolortransparent($this->gdimg_output, $TransparentColor); 2750 2751 // scan through alpha channel image and note pixels with >50% transparency 2752 for ($x = 0; $x < $this->thumbnail_width; $x++) { 2753 for ($y = 0; $y < $this->thumbnail_height; $y++) { 2754 $AlphaChannelPixel = phpthumb_functions::GetPixelColor($img_alpha_mixdown_dither, $x, $y); 2755 if ($AlphaChannelPixel['red'] > 127) { 2756 imagesetpixel($this->gdimg_output, $x, $y, $TransparentColor); 2757 } 2758 } 2759 } 2760 imagedestroy($img_alpha_mixdown_dither); 2761 2762 $this->DebugMessage('AlphaChannelFlatten() set image to 255+1 colors with transparency for GIF output', __FILE__, __LINE__); 2763 return true; 2764 2765 } else { 2766 $this->DebugMessage('AlphaChannelFlatten() failed imagecreate('.imagesx($this->gdimg_output).', '.imagesy($this->gdimg_output).')', __FILE__, __LINE__); 2767 return false; 2768 } 2769 2770 } else { 2771 // a single transparent color already defined, leave as-is 2772 $this->DebugMessage('skipping AlphaChannelFlatten() because ($this->thumbnailFormat == "'.$this->thumbnailFormat.'") and imagecolortransparent() returned "'.$CurrentImageColorTransparent.'"', __FILE__, __LINE__); 2773 return true; 2774 } 2775 break; 2776 } 2777 $this->DebugMessage('continuing AlphaChannelFlatten() for output format "'.$this->thumbnailFormat.'"', __FILE__, __LINE__); 2778 // image has alpha transparency, and is being output in a format that doesn't support it -- flatten 2779 if ($gdimg_flatten_temp = phpthumb_functions::ImageCreateFunction($this->thumbnail_width, $this->thumbnail_height)) { 2780 2781 $this->config_background_hexcolor = ($this->bg ? $this->bg : $this->config_background_hexcolor); 2782 if (!phpthumb_functions::IsHexColor($this->config_background_hexcolor)) { 2783 return $this->ErrorImage('Invalid hex color string "'.$this->config_background_hexcolor.'" for parameter "bg"'); 2784 } 2785 $background_color = phpthumb_functions::ImageHexColorAllocate($this->gdimg_output, $this->config_background_hexcolor); 2786 imagefilledrectangle($gdimg_flatten_temp, 0, 0, $this->thumbnail_width, $this->thumbnail_height, $background_color); 2787 imagecopy($gdimg_flatten_temp, $this->gdimg_output, 0, 0, 0, 0, $this->thumbnail_width, $this->thumbnail_height); 2788 2789 imagealphablending($this->gdimg_output, true); 2790 imagesavealpha($this->gdimg_output, false); 2791 imagecolortransparent($this->gdimg_output, -1); 2792 imagecopy($this->gdimg_output, $gdimg_flatten_temp, 0, 0, 0, 0, $this->thumbnail_width, $this->thumbnail_height); 2793 2794 imagedestroy($gdimg_flatten_temp); 2795 return true; 2796 2797 } else { 2798 $this->DebugMessage('ImageCreateFunction() failed', __FILE__, __LINE__); 2799 } 2800 return false; 2801 } 2802 2803 2804 public function ApplyFilters() { 2805 if ($this->fltr && is_array($this->fltr)) { 2806 if (!include_once __DIR__ .'/phpthumb.filters.php' ) { 2807 $this->DebugMessage('Error including "'. __DIR__ .'/phpthumb.filters.php" which is required for applying filters ('.implode(';', $this->fltr).')', __FILE__, __LINE__); 2808 return false; 2809 } 2810 $phpthumbFilters = new phpthumb_filters(); 2811 $phpthumbFilters->phpThumbObject = &$this; 2812 foreach ($this->fltr as $filtercommand) { 2813 @list($command, $parameter) = explode('|', $filtercommand, 2); 2814 $this->DebugMessage('Attempting to process filter command "'.$command.'('.$parameter.')"', __FILE__, __LINE__); 2815 switch ($command) { 2816 case 'brit': // Brightness 2817 $phpthumbFilters->Brightness($this->gdimg_output, $parameter); 2818 break; 2819 2820 case 'cont': // Contrast 2821 $phpthumbFilters->Contrast($this->gdimg_output, $parameter); 2822 break; 2823 2824 case 'ds': // Desaturation 2825 $phpthumbFilters->Desaturate($this->gdimg_output, $parameter, ''); 2826 break; 2827 2828 case 'sat': // Saturation 2829 $phpthumbFilters->Saturation($this->gdimg_output, $parameter, ''); 2830 break; 2831 2832 case 'gray': // Grayscale 2833 $phpthumbFilters->Grayscale($this->gdimg_output); 2834 break; 2835 2836 case 'clr': // Colorize 2837 if (phpthumb_functions::gd_version() < 2) { 2838 $this->DebugMessage('Skipping Colorize() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 2839 break; 2840 } 2841 @list($amount, $color) = explode('|', $parameter, 2); 2842 $phpthumbFilters->Colorize($this->gdimg_output, $amount, $color); 2843 break; 2844 2845 case 'sep': // Sepia 2846 if (phpthumb_functions::gd_version() < 2) { 2847 $this->DebugMessage('Skipping Sepia() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 2848 break; 2849 } 2850 @list($amount, $color) = explode('|', $parameter, 2); 2851 $phpthumbFilters->Sepia($this->gdimg_output, $amount, $color); 2852 break; 2853 2854 case 'gam': // Gamma correction 2855 $phpthumbFilters->Gamma($this->gdimg_output, $parameter); 2856 break; 2857 2858 case 'neg': // Negative colors 2859 $phpthumbFilters->Negative($this->gdimg_output); 2860 break; 2861 2862 case 'th': // Threshold 2863 $phpthumbFilters->Threshold($this->gdimg_output, $parameter); 2864 break; 2865 2866 case 'rcd': // ReduceColorDepth 2867 if (phpthumb_functions::gd_version() < 2) { 2868 $this->DebugMessage('Skipping ReduceColorDepth() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 2869 break; 2870 } 2871 @list($colors, $dither) = explode('|', $parameter, 2); 2872 $colors = ($colors ? (int) $colors : 256); 2873 $dither = ((strlen($dither) > 0) ? (bool) $dither : true); 2874 $phpthumbFilters->ReduceColorDepth($this->gdimg_output, $colors, $dither); 2875 break; 2876 2877 case 'flip': // Flip 2878 $phpthumbFilters->Flip($this->gdimg_output, strpos(strtolower($parameter), 'x') !== false, strpos(strtolower($parameter), 'y') !== false); 2879 break; 2880 2881 case 'edge': // EdgeDetect 2882 $phpthumbFilters->EdgeDetect($this->gdimg_output); 2883 break; 2884 2885 case 'emb': // Emboss 2886 $phpthumbFilters->Emboss($this->gdimg_output); 2887 break; 2888 2889 case 'bvl': // Bevel 2890 @list($width, $color1, $color2) = explode('|', $parameter, 3); 2891 $phpthumbFilters->Bevel($this->gdimg_output, $width, $color1, $color2); 2892 break; 2893 2894 case 'lvl': // autoLevels 2895 @list($band, $method, $threshold) = explode('|', $parameter, 3); 2896 $band = ($band ? preg_replace('#[^RGBA\\*]#', '', strtoupper($band)) : '*'); 2897 $method = ((strlen($method) > 0) ? (int) $method : 2); 2898 $threshold = ((strlen($threshold) > 0) ? (float) $threshold : 0.1); 2899 2900 $phpthumbFilters->HistogramStretch($this->gdimg_output, $band, $method, $threshold); 2901 break; 2902 2903 case 'wb': // WhiteBalance 2904 $phpthumbFilters->WhiteBalance($this->gdimg_output, $parameter); 2905 break; 2906 2907 case 'hist': // Histogram overlay 2908 if (phpthumb_functions::gd_version() < 2) { 2909 $this->DebugMessage('Skipping HistogramOverlay() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 2910 break; 2911 } 2912 @list($bands, $colors, $width, $height, $alignment, $opacity, $margin_x, $margin_y) = explode('|', $parameter, 8); 2913 $bands = ($bands ? $bands : '*'); 2914 $colors = ($colors ? $colors : ''); 2915 $width = ($width ? $width : 0.25); 2916 $height = ($height ? $height : 0.25); 2917 $alignment = ($alignment ? $alignment : 'BR'); 2918 $opacity = ($opacity ? $opacity : 50); 2919 $margin_x = ($margin_x ? $margin_x : 5); 2920 // $margin_y -- it wasn't forgotten, let the value always pass unchanged 2921 $phpthumbFilters->HistogramOverlay($this->gdimg_output, $bands, $colors, $width, $height, $alignment, $opacity, $margin_x, $margin_y); 2922 break; 2923 2924 case 'fram': // Frame 2925 @list($frame_width, $edge_width, $color_frame, $color1, $color2) = explode('|', $parameter, 5); 2926 $phpthumbFilters->Frame($this->gdimg_output, $frame_width, $edge_width, $color_frame, $color1, $color2); 2927 break; 2928 2929 case 'drop': // DropShadow 2930 if (phpthumb_functions::gd_version() < 2) { 2931 $this->DebugMessage('Skipping DropShadow() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 2932 return false; 2933 } 2934 $this->is_alpha = true; 2935 @list($distance, $width, $color, $angle, $fade) = explode('|', $parameter, 5); 2936 $phpthumbFilters->DropShadow($this->gdimg_output, $distance, $width, $color, $angle, $fade); 2937 break; 2938 2939 case 'mask': // Mask cropping 2940 if (phpthumb_functions::gd_version() < 2) { 2941 $this->DebugMessage('Skipping Mask() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 2942 return false; 2943 } 2944 @list($mask_filename, $invert) = explode('|', $parameter, 2); 2945 $mask_filename = $this->ResolveFilenameToAbsolute($mask_filename); 2946 if (@is_readable($mask_filename) && ($fp_mask = @fopen($mask_filename, 'rb'))) { 2947 $MaskImageData = ''; 2948 do { 2949 $buffer = fread($fp_mask, 8192); 2950 $MaskImageData .= $buffer; 2951 } while (strlen($buffer) > 0); 2952 fclose($fp_mask); 2953 if ($gdimg_mask = $this->ImageCreateFromStringReplacement($MaskImageData)) { 2954 if ($invert && phpthumb_functions::version_compare_replacement(PHP_VERSION, '5.0.0', '>=') && phpthumb_functions::gd_is_bundled()) { 2955 imagefilter($gdimg_mask, IMG_FILTER_NEGATE); 2956 } 2957 $this->is_alpha = true; 2958 $phpthumbFilters->ApplyMask($gdimg_mask, $this->gdimg_output); 2959 imagedestroy($gdimg_mask); 2960 } else { 2961 $this->DebugMessage('ImageCreateFromStringReplacement() failed for "'.$mask_filename.'"', __FILE__, __LINE__); 2962 } 2963 } else { 2964 $this->DebugMessage('Cannot open mask file "'.$mask_filename.'"', __FILE__, __LINE__); 2965 } 2966 break; 2967 2968 case 'elip': // Ellipse cropping 2969 if (phpthumb_functions::gd_version() < 2) { 2970 $this->DebugMessage('Skipping Ellipse() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 2971 return false; 2972 } 2973 $this->is_alpha = true; 2974 $phpthumbFilters->Ellipse($this->gdimg_output); 2975 break; 2976 2977 case 'ric': // RoundedImageCorners 2978 if (phpthumb_functions::gd_version() < 2) { 2979 $this->DebugMessage('Skipping RoundedImageCorners() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 2980 return false; 2981 } 2982 @list($radius_x, $radius_y) = explode('|', $parameter, 2); 2983 if (($radius_x < 1) || ($radius_y < 1)) { 2984 $this->DebugMessage('Skipping RoundedImageCorners('.$radius_x.', '.$radius_y.') because x/y radius is less than 1', __FILE__, __LINE__); 2985 break; 2986 } 2987 $this->is_alpha = true; 2988 $phpthumbFilters->RoundedImageCorners($this->gdimg_output, $radius_x, $radius_y); 2989 break; 2990 2991 case 'crop': // Crop 2992 @list($left, $right, $top, $bottom) = explode('|', $parameter, 4); 2993 $phpthumbFilters->Crop($this->gdimg_output, $left, $right, $top, $bottom); 2994 break; 2995 2996 case 'bord': // Border 2997 @list($border_width, $radius_x, $radius_y, $hexcolor_border) = explode('|', $parameter, 4); 2998 $this->is_alpha = true; 2999 $phpthumbFilters->ImageBorder($this->gdimg_output, $border_width, $radius_x, $radius_y, $hexcolor_border); 3000 break; 3001 3002 case 'over': // Overlay 3003 @list($filename, $underlay, $margin, $opacity) = explode('|', $parameter, 4); 3004 $underlay = (bool) ($underlay ? $underlay : false); 3005 $margin = ((strlen($margin) > 0) ? $margin : ($underlay ? 0.1 : 0.0)); 3006 $opacity = ((strlen($opacity) > 0) ? $opacity : 100); 3007 if (($margin > 0) && ($margin < 1)) { 3008 $margin = min(0.499, $margin); 3009 } elseif (($margin > -1) && ($margin < 0)) { 3010 $margin = max(-0.499, $margin); 3011 } 3012 3013 $filename = $this->ResolveFilenameToAbsolute($filename); 3014 if (@is_readable($filename) && ($fp_watermark = @fopen($filename, 'rb'))) { 3015 $WatermarkImageData = ''; 3016 do { 3017 $buffer = fread($fp_watermark, 8192); 3018 $WatermarkImageData .= $buffer; 3019 } while (strlen($buffer) > 0); 3020 fclose($fp_watermark); 3021 if ($img_watermark = $this->ImageCreateFromStringReplacement($WatermarkImageData)) { 3022 if (($margin > 0) && ($margin < 1)) { 3023 $resized_x = max(1, imagesx($this->gdimg_output) - round(2 * (imagesx($this->gdimg_output) * $margin))); 3024 $resized_y = max(1, imagesy($this->gdimg_output) - round(2 * (imagesy($this->gdimg_output) * $margin))); 3025 } else { 3026 $resized_x = max(1, imagesx($this->gdimg_output) - round(2 * $margin)); 3027 $resized_y = max(1, imagesy($this->gdimg_output) - round(2 * $margin)); 3028 } 3029 3030 if ($underlay) { 3031 3032 if ($img_watermark_resized = phpthumb_functions::ImageCreateFunction(imagesx($this->gdimg_output), imagesy($this->gdimg_output))) { 3033 imagealphablending($img_watermark_resized, false); 3034 imagesavealpha($img_watermark_resized, true); 3035 $this->ImageResizeFunction($img_watermark_resized, $img_watermark, 0, 0, 0, 0, imagesx($img_watermark_resized), imagesy($img_watermark_resized), imagesx($img_watermark), imagesy($img_watermark)); 3036 if ($img_source_resized = phpthumb_functions::ImageCreateFunction($resized_x, $resized_y)) { 3037 imagealphablending($img_source_resized, false); 3038 imagesavealpha($img_source_resized, true); 3039 $this->ImageResizeFunction($img_source_resized, $this->gdimg_output, 0, 0, 0, 0, imagesx($img_source_resized), imagesy($img_source_resized), imagesx($this->gdimg_output), imagesy($this->gdimg_output)); 3040 $phpthumbFilters->WatermarkOverlay($img_watermark_resized, $img_source_resized, 'C', $opacity, $margin); 3041 imagecopy($this->gdimg_output, $img_watermark_resized, 0, 0, 0, 0, imagesx($this->gdimg_output), imagesy($this->gdimg_output)); 3042 } else { 3043 $this->DebugMessage('phpthumb_functions::ImageCreateFunction('.$resized_x.', '.$resized_y.')', __FILE__, __LINE__); 3044 } 3045 imagedestroy($img_watermark_resized); 3046 } else { 3047 $this->DebugMessage('phpthumb_functions::ImageCreateFunction('.imagesx($this->gdimg_output).', '.imagesy($this->gdimg_output).')', __FILE__, __LINE__); 3048 } 3049 3050 } else { // overlay 3051 3052 if ($img_watermark_resized = phpthumb_functions::ImageCreateFunction($resized_x, $resized_y)) { 3053 imagealphablending($img_watermark_resized, false); 3054 imagesavealpha($img_watermark_resized, true); 3055 $this->ImageResizeFunction($img_watermark_resized, $img_watermark, 0, 0, 0, 0, imagesx($img_watermark_resized), imagesy($img_watermark_resized), imagesx($img_watermark), imagesy($img_watermark)); 3056 $phpthumbFilters->WatermarkOverlay($this->gdimg_output, $img_watermark_resized, 'C', $opacity, $margin); 3057 imagedestroy($img_watermark_resized); 3058 } else { 3059 $this->DebugMessage('phpthumb_functions::ImageCreateFunction('.$resized_x.', '.$resized_y.')', __FILE__, __LINE__); 3060 } 3061 3062 } 3063 imagedestroy($img_watermark); 3064 3065 } else { 3066 $this->DebugMessage('ImageCreateFromStringReplacement() failed for "'.$filename.'"', __FILE__, __LINE__); 3067 } 3068 } else { 3069 $this->DebugMessage('Cannot open overlay file "'.$filename.'"', __FILE__, __LINE__); 3070 } 3071 break; 3072 3073 case 'wmi': // WaterMarkImage 3074 @list($filename, $alignment, $opacity, $margin['x'], $margin['y'], $rotate_angle) = explode('|', $parameter, 6); 3075 // $margin can be pixel margin or percent margin if $alignment is text, or max width/height if $alignment is position like "50x75" 3076 $alignment = ($alignment ? $alignment : 'BR'); 3077 $opacity = ('' != $opacity ? (int) $opacity : 50); 3078 $rotate_angle = ('' != $rotate_angle ? (int) $rotate_angle : 0); 3079 if (!preg_match('#^([0-9\\.\\-]*)x([0-9\\.\\-]*)$#i', $alignment, $matches)) { 3080 $margins = array('x', 'y'); 3081 foreach ($margins as $xy) { 3082 $margin[$xy] = ('' !== $margin[ $xy ] ? $margin[ $xy] : 5); 3083 if (($margin[$xy] > 0) && ($margin[$xy] < 1)) { 3084 $margin[$xy] = min(0.499, $margin[$xy]); 3085 } elseif (($margin[$xy] > -1) && ($margin[$xy] < 0)) { 3086 $margin[$xy] = max(-0.499, $margin[$xy]); 3087 } 3088 } 3089 } 3090 3091 $filename = $this->ResolveFilenameToAbsolute($filename); 3092 if (@is_readable($filename)) { 3093 if ($img_watermark = $this->ImageCreateFromFilename($filename)) { 3094 if ($rotate_angle !== 0) { 3095 $phpthumbFilters->ImprovedImageRotate($img_watermark, $rotate_angle, 'FFFFFF', null, $this); 3096 } 3097 if (preg_match('#^([0-9\\.\\-]*)x([0-9\\.\\-]*)$#i', $alignment, $matches)) { 3098 $watermark_max_width = (int) ($margin[ 'x'] ? $margin[ 'x'] : imagesx($img_watermark)); 3099 $watermark_max_height = (int) ($margin[ 'y'] ? $margin[ 'y'] : imagesy($img_watermark)); 3100 $scale = phpthumb_functions::ScaleToFitInBox(imagesx($img_watermark), imagesy($img_watermark), $watermark_max_width, $watermark_max_height, true, true); 3101 $this->DebugMessage('Scaling watermark by a factor of '.number_format($scale, 4), __FILE__, __LINE__); 3102 if (($scale > 1) || ($scale < 1)) { 3103 if ($img_watermark2 = phpthumb_functions::ImageCreateFunction($scale * imagesx($img_watermark), $scale * imagesy($img_watermark))) { 3104 imagealphablending($img_watermark2, false); 3105 imagesavealpha($img_watermark2, true); 3106 $this->ImageResizeFunction($img_watermark2, $img_watermark, 0, 0, 0, 0, imagesx($img_watermark2), imagesy($img_watermark2), imagesx($img_watermark), imagesy($img_watermark)); 3107 $img_watermark = $img_watermark2; 3108 } else { 3109 $this->DebugMessage('ImageCreateFunction('.($scale * imagesx($img_watermark)).', '.($scale * imagesx($img_watermark)).') failed', __FILE__, __LINE__); 3110 } 3111 } 3112 $watermark_dest_x = round($matches[1] - (imagesx($img_watermark) / 2)); 3113 $watermark_dest_y = round($matches[2] - (imagesy($img_watermark) / 2)); 3114 $alignment = $watermark_dest_x.'x'.$watermark_dest_y; 3115 } 3116 $phpthumbFilters->WatermarkOverlay($this->gdimg_output, $img_watermark, $alignment, $opacity, $margin['x'], $margin['y']); 3117 imagedestroy($img_watermark); 3118 if (isset($img_watermark2) && (is_resource($img_watermark2) || (is_object($img_watermark2) && $img_watermark2 instanceOf \GdImage))) { 3119 imagedestroy($img_watermark2); 3120 } 3121 } else { 3122 $this->DebugMessage('ImageCreateFromFilename() failed for "'.$filename.'"', __FILE__, __LINE__); 3123 } 3124 } else { 3125 $this->DebugMessage('!is_readable('.$filename.')', __FILE__, __LINE__); 3126 } 3127 break; 3128 3129 case 'wmt': // WaterMarkText 3130 @list($text, $size, $alignment, $hex_color, $ttffont, $opacity, $margin, $angle, $bg_color, $bg_opacity, $fillextend, $lineheight) = explode('|', $parameter, 12); 3131 $text = ($text ? $text : ''); 3132 $size = ($size ? $size : 3); 3133 $alignment = ($alignment ? $alignment : 'BR'); 3134 $hex_color = ($hex_color ? $hex_color : '000000'); 3135 $ttffont = ($ttffont ? $ttffont : ''); 3136 $opacity = ('' != $opacity ? $opacity : 50); 3137 $margin = ('' != $margin ? $margin : 5); 3138 $angle = ('' != $angle ? $angle : 0); 3139 $bg_color = ($bg_color ? $bg_color : false); 3140 $bg_opacity = ($bg_opacity ? $bg_opacity : 0); 3141 $fillextend = ($fillextend ? $fillextend : ''); 3142 $lineheight = ($lineheight ? $lineheight : 1.0); 3143 3144 if (basename($ttffont) == $ttffont) { 3145 $ttffont = $this->realPathSafe($this->config_ttf_directory.DIRECTORY_SEPARATOR.$ttffont); 3146 } else { 3147 $ttffont = $this->ResolveFilenameToAbsolute($ttffont); 3148 } 3149 $phpthumbFilters->WatermarkText($this->gdimg_output, $text, $size, $alignment, $hex_color, $ttffont, $opacity, $margin, $angle, $bg_color, $bg_opacity, $fillextend, $lineheight); 3150 break; 3151 3152 case 'blur': // Blur 3153 @list($radius) = explode('|', $parameter, 1); 3154 $radius = ($radius ? $radius : 1); 3155 if (phpthumb_functions::gd_version() >= 2) { 3156 $phpthumbFilters->Blur($this->gdimg_output, $radius); 3157 } else { 3158 $this->DebugMessage('Skipping Blur() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 3159 } 3160 break; 3161 3162 case 'gblr': // Gaussian Blur 3163 $phpthumbFilters->BlurGaussian($this->gdimg_output); 3164 break; 3165 3166 case 'sblr': // Selective Blur 3167 $phpthumbFilters->BlurSelective($this->gdimg_output); 3168 break; 3169 3170 case 'mean': // MeanRemoval blur 3171 $phpthumbFilters->MeanRemoval($this->gdimg_output); 3172 break; 3173 3174 case 'smth': // Smooth blur 3175 $phpthumbFilters->Smooth($this->gdimg_output, $parameter); 3176 break; 3177 3178 case 'usm': // UnSharpMask sharpening 3179 @list($amount, $radius, $threshold) = explode('|', $parameter, 3); 3180 $amount = ($amount ? $amount : 80); 3181 $radius = ($radius ? $radius : 0.5); 3182 $threshold = ('' !== $threshold ? $threshold : 3); 3183 if (phpthumb_functions::gd_version() >= 2.0) { 3184 ob_start(); 3185 if (!@include_once __DIR__ .'/phpthumb.unsharp.php' ) { 3186 $include_error = ob_get_contents(); 3187 if ($include_error) { 3188 $this->DebugMessage('include_once("'. __DIR__ .'/phpthumb.unsharp.php") generated message: "'.$include_error.'"', __FILE__, __LINE__); 3189 } 3190 $this->DebugMessage('Error including "'. __DIR__ .'/phpthumb.unsharp.php" which is required for unsharp masking', __FILE__, __LINE__); 3191 ob_end_clean(); 3192 return false; 3193 } 3194 ob_end_clean(); 3195 phpUnsharpMask::applyUnsharpMask($this->gdimg_output, $amount, $radius, $threshold); 3196 } else { 3197 $this->DebugMessage('Skipping unsharp mask because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 3198 return false; 3199 } 3200 break; 3201 3202 case 'size': // Resize 3203 @list($newwidth, $newheight, $stretch) = explode('|', $parameter); 3204 $newwidth = (!$newwidth ? imagesx($this->gdimg_output) : ((($newwidth > 0) && ($newwidth < 1)) ? round($newwidth * imagesx($this->gdimg_output)) : round($newwidth))); 3205 $newheight = (!$newheight ? imagesy($this->gdimg_output) : ((($newheight > 0) && ($newheight < 1)) ? round($newheight * imagesy($this->gdimg_output)) : round($newheight))); 3206 $stretch = ($stretch ? true : false); 3207 if ($stretch) { 3208 $scale_x = phpthumb_functions::ScaleToFitInBox(imagesx($this->gdimg_output), imagesx($this->gdimg_output), $newwidth, $newwidth, true, true); 3209 $scale_y = phpthumb_functions::ScaleToFitInBox(imagesy($this->gdimg_output), imagesy($this->gdimg_output), $newheight, $newheight, true, true); 3210 } else { 3211 $scale_x = phpthumb_functions::ScaleToFitInBox(imagesx($this->gdimg_output), imagesy($this->gdimg_output), $newwidth, $newheight, true, true); 3212 $scale_y = $scale_x; 3213 } 3214 $this->DebugMessage('Scaling watermark ('.($stretch ? 'with' : 'without').' stretch) by a factor of "'.number_format($scale_x, 4).' x '.number_format($scale_y, 4).'"', __FILE__, __LINE__); 3215 if (($scale_x > 1) || ($scale_x < 1) || ($scale_y > 1) || ($scale_y < 1)) { 3216 if ($img_temp = phpthumb_functions::ImageCreateFunction(imagesx($this->gdimg_output), imagesy($this->gdimg_output))) { 3217 imagecopy($img_temp, $this->gdimg_output, 0, 0, 0, 0, imagesx($this->gdimg_output), imagesy($this->gdimg_output)); 3218 if ($this->gdimg_output = phpthumb_functions::ImageCreateFunction($scale_x * imagesx($img_temp), $scale_y * imagesy($img_temp))) { 3219 imagealphablending($this->gdimg_output, false); 3220 imagesavealpha($this->gdimg_output, true); 3221 $this->ImageResizeFunction($this->gdimg_output, $img_temp, 0, 0, 0, 0, imagesx($this->gdimg_output), imagesy($this->gdimg_output), imagesx($img_temp), imagesy($img_temp)); 3222 } else { 3223 $this->DebugMessage('ImageCreateFunction('.($scale_x * imagesx($img_temp)).', '.($scale_y * imagesy($img_temp)).') failed', __FILE__, __LINE__); 3224 } 3225 imagedestroy($img_temp); 3226 } else { 3227 $this->DebugMessage('ImageCreateFunction('.imagesx($this->gdimg_output).', '.imagesy($this->gdimg_output).') failed', __FILE__, __LINE__); 3228 } 3229 } 3230 break; 3231 3232 case 'rot': // ROTate 3233 @list($angle, $bgcolor) = explode('|', $parameter, 2); 3234 $phpthumbFilters->ImprovedImageRotate($this->gdimg_output, $angle, $bgcolor, null, $this); 3235 break; 3236 3237 case 'stc': // Source Transparent Color 3238 @list($hexcolor, $min_limit, $max_limit) = explode('|', $parameter, 3); 3239 if (!phpthumb_functions::IsHexColor($hexcolor)) { 3240 $this->DebugMessage('Skipping SourceTransparentColor hex color is invalid ('.$hexcolor.')', __FILE__, __LINE__); 3241 return false; 3242 } 3243 $min_limit = ('' !== $min_limit ? $min_limit : 5); 3244 $max_limit = ('' !== $max_limit ? $max_limit : 10); 3245 if ($gdimg_mask = $phpthumbFilters->SourceTransparentColorMask($this->gdimg_output, $hexcolor, $min_limit, $max_limit)) { 3246 $this->is_alpha = true; 3247 $phpthumbFilters->ApplyMask($gdimg_mask, $this->gdimg_output); 3248 imagedestroy($gdimg_mask); 3249 } else { 3250 $this->DebugMessage('SourceTransparentColorMask() failed for "'.$hexcolor.','.$min_limit.','.$max_limit.'"', __FILE__, __LINE__); 3251 } 3252 break; 3253 } 3254 $this->DebugMessage('Finished processing filter command "'.$command.'('.$parameter.')"', __FILE__, __LINE__); 3255 } 3256 } 3257 return true; 3258 } 3259 3260 3261 public function MaxFileSize() { 3262 if (phpthumb_functions::gd_version() < 2) { 3263 $this->DebugMessage('Skipping MaxFileSize() because gd_version is "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 3264 return false; 3265 } 3266 if ($this->maxb > 0) { 3267 switch ($this->thumbnailFormat) { 3268 case 'png': 3269 case 'gif': 3270 $imgRenderFunction = 'image'.$this->thumbnailFormat; 3271 3272 ob_start(); 3273 $imgRenderFunction($this->gdimg_output); 3274 $imgdata = ob_get_contents(); 3275 ob_end_clean(); 3276 3277 if (strlen($imgdata) > $this->maxb) { 3278 for ($i = 8; $i >= 1; $i--) { 3279 $tempIMG = imagecreatetruecolor(imagesx($this->gdimg_output), imagesy($this->gdimg_output)); 3280 imagecopy($tempIMG, $this->gdimg_output, 0, 0, 0, 0, imagesx($this->gdimg_output), imagesy($this->gdimg_output)); 3281 imagetruecolortopalette($tempIMG, true, pow(2, $i)); 3282 ob_start(); 3283 $imgRenderFunction($tempIMG); 3284 $imgdata = ob_get_contents(); 3285 ob_end_clean(); 3286 3287 if (strlen($imgdata) <= $this->maxb) { 3288 imagetruecolortopalette($this->gdimg_output, true, pow(2, $i)); 3289 break; 3290 } 3291 } 3292 } 3293 break; 3294 3295 case 'jpeg': 3296 ob_start(); 3297 imagejpeg($this->gdimg_output); 3298 $imgdata = ob_get_contents(); 3299 ob_end_clean(); 3300 3301 if (strlen($imgdata) > $this->maxb) { 3302 for ($i = 3; $i < 20; $i++) { 3303 $q = round(100 * (1 - log10($i / 2))); 3304 ob_start(); 3305 imagejpeg($this->gdimg_output, null, $q); 3306 $imgdata = ob_get_contents(); 3307 ob_end_clean(); 3308 3309 $this->thumbnailQuality = $q; 3310 if (strlen($imgdata) <= $this->maxb) { 3311 break; 3312 } 3313 } 3314 } 3315 if (strlen($imgdata) > $this->maxb) { 3316 return false; 3317 } 3318 break; 3319 3320 default: 3321 return false; 3322 } 3323 } 3324 return true; 3325 } 3326 3327 3328 public function CalculateThumbnailDimensions() { 3329 $this->DebugMessage('CalculateThumbnailDimensions() starting with [W,H,sx,sy,sw,sh] initially set to ['.$this->source_width.','.$this->source_height.','.$this->sx.','.$this->sy.','.$this->sw.','.$this->sh.']', __FILE__, __LINE__); 3330//echo $this->source_width.'x'.$this->source_height.'<hr>'; 3331 $this->thumbnailCropX = ($this->sx ? (($this->sx >= 2) ? $this->sx : round($this->sx * $this->source_width)) : 0); 3332//echo $this->thumbnailCropX.'<br>'; 3333 $this->thumbnailCropY = ($this->sy ? (($this->sy >= 2) ? $this->sy : round($this->sy * $this->source_height)) : 0); 3334//echo $this->thumbnailCropY.'<br>'; 3335 $this->thumbnailCropW = ($this->sw ? (($this->sw >= 2) ? $this->sw : round($this->sw * $this->source_width)) : $this->source_width); 3336//echo $this->thumbnailCropW.'<br>'; 3337 $this->thumbnailCropH = ($this->sh ? (($this->sh >= 2) ? $this->sh : round($this->sh * $this->source_height)) : $this->source_height); 3338//echo $this->thumbnailCropH.'<hr>'; 3339 3340 // limit source area to original image area 3341 $this->thumbnailCropW = max(1, min($this->thumbnailCropW, $this->source_width - $this->thumbnailCropX)); 3342 $this->thumbnailCropH = max(1, min($this->thumbnailCropH, $this->source_height - $this->thumbnailCropY)); 3343 3344 $this->DebugMessage('CalculateThumbnailDimensions() starting with [x,y,w,h] initially set to ['.$this->thumbnailCropX.','.$this->thumbnailCropY.','.$this->thumbnailCropW.','.$this->thumbnailCropH.']', __FILE__, __LINE__); 3345 3346 3347 if ($this->zc && $this->w && $this->h) { 3348 // Zoom Crop 3349 // retain proportional resizing we did above, but crop off larger dimension so smaller 3350 // dimension fully fits available space 3351 3352 $scaling_X = $this->source_width / $this->w; 3353 $scaling_Y = $this->source_height / $this->h; 3354 if ($scaling_X > $scaling_Y) { 3355 // some of the width will need to be cropped 3356 $allowable_width = $this->source_width / $scaling_X * $scaling_Y; 3357 $this->thumbnailCropW = round($allowable_width); 3358 $this->thumbnailCropX = round(($this->source_width - $allowable_width) / 2); 3359 3360 } elseif ($scaling_Y > $scaling_X) { 3361 // some of the height will need to be cropped 3362 $allowable_height = $this->source_height / $scaling_Y * $scaling_X; 3363 $this->thumbnailCropH = round($allowable_height); 3364 $this->thumbnailCropY = round(($this->source_height - $allowable_height) / 2); 3365 3366 } else { 3367 // image fits perfectly, no cropping needed 3368 } 3369 $this->thumbnail_width = $this->w; 3370 $this->thumbnail_height = $this->h; 3371 $this->thumbnail_image_width = $this->thumbnail_width; 3372 $this->thumbnail_image_height = $this->thumbnail_height; 3373 3374 } elseif ($this->iar && $this->w && $this->h) { 3375 3376 // Ignore Aspect Ratio 3377 // stretch image to fit exactly 'w' x 'h' 3378 $this->thumbnail_width = $this->w; 3379 $this->thumbnail_height = $this->h; 3380 $this->thumbnail_image_width = $this->thumbnail_width; 3381 $this->thumbnail_image_height = $this->thumbnail_height; 3382 3383 } else { 3384 3385 $original_aspect_ratio = $this->thumbnailCropW / $this->thumbnailCropH; 3386 if ($this->aoe) { 3387 if ($this->w && $this->h) { 3388 $maxwidth = min($this->w, $this->h * $original_aspect_ratio); 3389 $maxheight = min($this->h, $this->w / $original_aspect_ratio); 3390 } elseif ($this->w) { 3391 $maxwidth = $this->w; 3392 $maxheight = $this->w / $original_aspect_ratio; 3393 } elseif ($this->h) { 3394 $maxwidth = $this->h * $original_aspect_ratio; 3395 $maxheight = $this->h; 3396 } else { 3397 $maxwidth = $this->thumbnailCropW; 3398 $maxheight = $this->thumbnailCropH; 3399 } 3400 } else { 3401 $maxwidth = phpthumb_functions::nonempty_min($this->w, $this->thumbnailCropW, $this->config_output_maxwidth); 3402 $maxheight = phpthumb_functions::nonempty_min($this->h, $this->thumbnailCropH, $this->config_output_maxheight); 3403//echo $maxwidth.'x'.$maxheight.'<br>'; 3404 $maxwidth = min($maxwidth, $maxheight * $original_aspect_ratio); 3405 $maxheight = min($maxheight, $maxwidth / $original_aspect_ratio); 3406//echo $maxwidth.'x'.$maxheight.'<hr>'; 3407 } 3408 3409 $this->thumbnail_image_width = $maxwidth; 3410 $this->thumbnail_image_height = $maxheight; 3411 $this->thumbnail_width = $maxwidth; 3412 $this->thumbnail_height = $maxheight; 3413 3414 $this->FixedAspectRatio(); 3415 } 3416 3417 $this->thumbnail_width = max(1, floor($this->thumbnail_width)); 3418 $this->thumbnail_height = max(1, floor($this->thumbnail_height)); 3419 return true; 3420 } 3421 3422 3423 public function CreateGDoutput() { 3424 $this->CalculateThumbnailDimensions(); 3425 3426 // create the GD image (either true-color or 256-color, depending on GD version) 3427 $this->gdimg_output = phpthumb_functions::ImageCreateFunction($this->thumbnail_width, $this->thumbnail_height); 3428 3429 // images that have transparency must have the background filled with the configured 'bg' color otherwise the transparent color will appear as black 3430 imagesavealpha($this->gdimg_output, true); 3431 if ($this->is_alpha && phpthumb_functions::gd_version() >= 2) { 3432 3433 imagealphablending($this->gdimg_output, false); 3434 $output_full_alpha = phpthumb_functions::ImageColorAllocateAlphaSafe($this->gdimg_output, 255, 255, 255, 127); 3435 imagefilledrectangle($this->gdimg_output, 0, 0, $this->thumbnail_width, $this->thumbnail_height, $output_full_alpha); 3436 3437 } else { 3438 3439 $current_transparent_color = imagecolortransparent($this->gdimg_source); 3440 if ($this->bg || (@$current_transparent_color >= 0)) { 3441 3442 $this->config_background_hexcolor = ($this->bg ? $this->bg : $this->config_background_hexcolor); 3443 if (!phpthumb_functions::IsHexColor($this->config_background_hexcolor)) { 3444 return $this->ErrorImage('Invalid hex color string "'.$this->config_background_hexcolor.'" for parameter "bg"'); 3445 } 3446 $background_color = phpthumb_functions::ImageHexColorAllocate($this->gdimg_output, $this->config_background_hexcolor); 3447 imagefilledrectangle($this->gdimg_output, 0, 0, $this->thumbnail_width, $this->thumbnail_height, $background_color); 3448 3449 } 3450 3451 } 3452 $this->DebugMessage('CreateGDoutput() returning canvas "'.$this->thumbnail_width.'x'.$this->thumbnail_height.'"', __FILE__, __LINE__); 3453 return true; 3454 } 3455 3456 public function SetOrientationDependantWidthHeight() { 3457 $this->DebugMessage('SetOrientationDependantWidthHeight() starting with "'.$this->source_width.'"x"'.$this->source_height.'"', __FILE__, __LINE__); 3458 if ($this->source_height > $this->source_width) { 3459 // portrait 3460 $this->w = phpthumb_functions::OneOfThese($this->wp, $this->w, $this->ws, $this->wl); 3461 $this->h = phpthumb_functions::OneOfThese($this->hp, $this->h, $this->hs, $this->hl); 3462 } elseif ($this->source_height < $this->source_width) { 3463 // landscape 3464 $this->w = phpthumb_functions::OneOfThese($this->wl, $this->w, $this->ws, $this->wp); 3465 $this->h = phpthumb_functions::OneOfThese($this->hl, $this->h, $this->hs, $this->hp); 3466 } else { 3467 // square 3468 $this->w = phpthumb_functions::OneOfThese($this->ws, $this->w, $this->wl, $this->wp); 3469 $this->h = phpthumb_functions::OneOfThese($this->hs, $this->h, $this->hl, $this->hp); 3470 } 3471 //$this->w = round($this->w ? $this->w : (($this->h && $this->source_height) ? $this->h * $this->source_width / $this->source_height : $this->w)); 3472 //$this->h = round($this->h ? $this->h : (($this->w && $this->source_width) ? $this->w * $this->source_height / $this->source_width : $this->h)); 3473 $this->DebugMessage('SetOrientationDependantWidthHeight() setting w="'. (int) $this->w .'", h="'. (int) $this->h .'"', __FILE__, __LINE__); 3474 return true; 3475 } 3476 3477 public function ExtractEXIFgetImageSize() { 3478 $this->DebugMessage('starting ExtractEXIFgetImageSize()', __FILE__, __LINE__); 3479 3480 if (preg_match('#^http:#i', $this->src) && !$this->sourceFilename && $this->rawImageData) { 3481 $this->SourceDataToTempFile(); 3482 } 3483 if (null === $this->getimagesizeinfo) { 3484 if ($this->sourceFilename) { 3485 if ($this->getimagesizeinfo = @getimagesize($this->sourceFilename)) { 3486 $this->source_width = $this->getimagesizeinfo[0]; 3487 $this->source_height = $this->getimagesizeinfo[1]; 3488 $this->DebugMessage('getimagesize('.$this->sourceFilename.') says image is '.$this->source_width.'x'.$this->source_height, __FILE__, __LINE__); 3489 } else { 3490 $this->DebugMessage('getimagesize('.$this->sourceFilename.') failed', __FILE__, __LINE__); 3491 } 3492 } else { 3493 $this->DebugMessage('skipping getimagesize() because $this->sourceFilename is empty', __FILE__, __LINE__); 3494 } 3495 } else { 3496 $this->DebugMessage('skipping getimagesize() because !is_null($this->getimagesizeinfo)', __FILE__, __LINE__); 3497 } 3498 3499 if (is_resource($this->gdimg_source) || (is_object($this->gdimg_source) && $this->gdimg_source instanceOf \GdImage)) { 3500 3501 $this->source_width = imagesx($this->gdimg_source); 3502 $this->source_height = imagesy($this->gdimg_source); 3503 3504 $this->SetOrientationDependantWidthHeight(); 3505 3506 } elseif ($this->rawImageData && !$this->sourceFilename) { 3507 3508 if ($this->SourceImageIsTooLarge($this->source_width, $this->source_height)) { 3509 $this->DebugMessage('NOT bypassing EXIF and getimagesize sections because source image is too large for GD ('.$this->source_width.'x'.$this->source_width.'='.($this->source_width * $this->source_height * 5).'MB)', __FILE__, __LINE__); 3510 } else { 3511 $this->DebugMessage('bypassing EXIF and getimagesize sections because $this->rawImageData is set, and $this->sourceFilename is not set, and source image is not too large for GD ('.$this->source_width.'x'.$this->source_width.'='.($this->source_width * $this->source_height * 5).'MB)', __FILE__, __LINE__); 3512 } 3513 3514 } 3515 3516 if (!empty($this->getimagesizeinfo)) { 3517 // great 3518 $this->getimagesizeinfo['filesize'] = @filesize($this->sourceFilename); 3519 } elseif (!$this->rawImageData) { 3520 $this->DebugMessage('getimagesize("'.$this->sourceFilename.'") failed', __FILE__, __LINE__); 3521 } 3522 3523 if ($this->config_prefer_imagemagick) { 3524 if ($this->ImageMagickThumbnailToGD()) { 3525 return true; 3526 } 3527 $this->DebugMessage('ImageMagickThumbnailToGD() failed', __FILE__, __LINE__); 3528 } 3529 3530 if (isset($this->getimagesizeinfo[1])) { 3531 $this->source_width = $this->getimagesizeinfo[0]; 3532 $this->source_height = $this->getimagesizeinfo[1]; 3533 } 3534 3535 $this->SetOrientationDependantWidthHeight(); 3536 3537 if (phpthumb_functions::version_compare_replacement(PHP_VERSION, '4.2.0', '>=') && function_exists('exif_read_data')) { 3538 switch (@$this->getimagesizeinfo[2]) { 3539 case IMAGETYPE_JPEG: 3540 case IMAGETYPE_TIFF_II: 3541 case IMAGETYPE_TIFF_MM: 3542 $this->exif_raw_data = @exif_read_data($this->sourceFilename, 0, true); 3543 break; 3544 } 3545 } 3546 if (function_exists('exif_thumbnail') && (@$this->getimagesizeinfo[2] == IMAGETYPE_JPEG)) { 3547 // Extract EXIF info from JPEGs 3548 3549 $this->exif_thumbnail_width = ''; 3550 $this->exif_thumbnail_height = ''; 3551 $this->exif_thumbnail_type = ''; 3552 3553 // The parameters width, height and imagetype are available since PHP v4.3.0 3554 if (phpthumb_functions::version_compare_replacement(PHP_VERSION, '4.3.0', '>=')) { 3555 3556 $this->exif_thumbnail_data = @exif_thumbnail($this->sourceFilename, $this->exif_thumbnail_width, $this->exif_thumbnail_height, $this->exif_thumbnail_type); 3557 3558 } else { 3559 3560 // older versions of exif_thumbnail output an error message but NOT return false on failure 3561 ob_start(); 3562 $this->exif_thumbnail_data = exif_thumbnail($this->sourceFilename); 3563 $exit_thumbnail_error = ob_get_contents(); 3564 ob_end_clean(); 3565 if (!$exit_thumbnail_error && $this->exif_thumbnail_data) { 3566 3567 if ($gdimg_exif_temp = $this->ImageCreateFromStringReplacement($this->exif_thumbnail_data, false)) { 3568 $this->exif_thumbnail_width = imagesx($gdimg_exif_temp); 3569 $this->exif_thumbnail_height = imagesy($gdimg_exif_temp); 3570 $this->exif_thumbnail_type = 2; // (2 == JPEG) before PHP v4.3.0 only JPEG format EXIF thumbnails are returned 3571 unset($gdimg_exif_temp); 3572 } else { 3573 return $this->ErrorImage('Failed - $this->ImageCreateFromStringReplacement($this->exif_thumbnail_data) in '.__FILE__.' on line '.__LINE__); 3574 } 3575 3576 } 3577 3578 } 3579 3580 } elseif (!function_exists('exif_thumbnail')) { 3581 3582 $this->DebugMessage('exif_thumbnail() does not exist, cannot extract EXIF thumbnail', __FILE__, __LINE__); 3583 3584 } 3585 3586 $this->DebugMessage('EXIF thumbnail extraction: (size='.(!empty($this->exif_thumbnail_data) ? strlen((string) $this->exif_thumbnail_data) : 0).'; type="'.$this->exif_thumbnail_type.'"; '. (int) $this->exif_thumbnail_width .'x'. (int) $this->exif_thumbnail_height .')', __FILE__, __LINE__); 3587 3588 // see if EXIF thumbnail can be used directly with no processing 3589 if ($this->config_use_exif_thumbnail_for_speed && $this->exif_thumbnail_data) { 3590 while (true) { 3591 if (!$this->xto) { 3592 $source_ar = $this->source_width / $this->source_height; 3593 $exif_ar = $this->exif_thumbnail_width / $this->exif_thumbnail_height; 3594 if (number_format($source_ar, 2) != number_format($exif_ar, 2)) { 3595 $this->DebugMessage('not using EXIF thumbnail because $source_ar != $exif_ar ('.$source_ar.' != '.$exif_ar.')', __FILE__, __LINE__); 3596 break; 3597 } 3598 if ($this->w && ($this->w != $this->exif_thumbnail_width)) { 3599 $this->DebugMessage('not using EXIF thumbnail because $this->w != $this->exif_thumbnail_width ('.$this->w.' != '.$this->exif_thumbnail_width.')', __FILE__, __LINE__); 3600 break; 3601 } 3602 if ($this->h && ($this->h != $this->exif_thumbnail_height)) { 3603 $this->DebugMessage('not using EXIF thumbnail because $this->h != $this->exif_thumbnail_height ('.$this->h.' != '.$this->exif_thumbnail_height.')', __FILE__, __LINE__); 3604 break; 3605 } 3606 $CannotBeSetParameters = array('sx', 'sy', 'sh', 'sw', 'far', 'bg', 'bc', 'fltr', 'phpThumbDebug'); 3607 foreach ($CannotBeSetParameters as $parameter) { 3608 if ($this->$parameter) { 3609 break 2; 3610 } 3611 } 3612 } 3613 3614 $this->DebugMessage('setting $this->gdimg_source = $this->ImageCreateFromStringReplacement($this->exif_thumbnail_data)', __FILE__, __LINE__); 3615 $this->gdimg_source = $this->ImageCreateFromStringReplacement($this->exif_thumbnail_data); 3616 $this->source_width = imagesx($this->gdimg_source); 3617 $this->source_height = imagesy($this->gdimg_source); 3618 return true; 3619 } 3620 } 3621 3622 if (($this->config_max_source_pixels > 0) && (($this->source_width * $this->source_height) > $this->config_max_source_pixels)) { 3623 3624 // Source image is larger than would fit in available PHP memory. 3625 // If ImageMagick is installed, use it to generate the thumbnail. 3626 // Else, if an EXIF thumbnail is available, use that as the source image. 3627 // Otherwise, no choice but to fail with an error message 3628 $this->DebugMessage('image is '.$this->source_width.'x'.$this->source_height.' and therefore contains more pixels ('.($this->source_width * $this->source_height).') than $this->config_max_source_pixels setting ('.$this->config_max_source_pixels.')', __FILE__, __LINE__); 3629 if (!$this->config_prefer_imagemagick && $this->ImageMagickThumbnailToGD()) { 3630 // excellent, we have a thumbnailed source image 3631 return true; 3632 } 3633 3634 } 3635 return true; 3636 } 3637 3638 3639 public function SetCacheFilename() { 3640 if (null !== $this->cache_filename) { 3641 $this->DebugMessage('$this->cache_filename already set, skipping SetCacheFilename()', __FILE__, __LINE__); 3642 return true; 3643 } 3644 if (null === $this->config_cache_directory) { 3645 $this->setCacheDirectory(); 3646 if (!$this->config_cache_directory) { 3647 $this->DebugMessage('SetCacheFilename() failed because $this->config_cache_directory is empty', __FILE__, __LINE__); 3648 return false; 3649 } 3650 } 3651 $this->setOutputFormat(); 3652 3653 if (!$this->sourceFilename && !$this->rawImageData && $this->src) { 3654 $this->sourceFilename = $this->ResolveFilenameToAbsolute($this->src); 3655 } 3656 3657 if ($this->config_cache_default_only_suffix && $this->sourceFilename) { 3658 // simplified cache filenames: 3659 // only use default parameters in phpThumb.config.php 3660 // substitute source filename into * in $this->config_cache_default_only_suffix 3661 // (eg: '*_thumb' becomes 'picture_thumb.jpg') 3662 if (strpos($this->config_cache_default_only_suffix, '*') === false) { 3663 $this->DebugMessage('aborting simplified caching filename because no * in "'.$this->config_cache_default_only_suffix.'"', __FILE__, __LINE__); 3664 } else { 3665 preg_match('#(.+)(\\.[a-z0-9]+)?$#i', basename($this->sourceFilename), $matches); 3666 $this->cache_filename = $this->config_cache_directory.DIRECTORY_SEPARATOR.rawurlencode(str_replace('*', @$matches[1], $this->config_cache_default_only_suffix)).'.'.strtolower($this->thumbnailFormat); 3667 return true; 3668 } 3669 } 3670 3671 $this->cache_filename = ''; 3672 if ($this->new) { 3673 $broad_directory_name = strtolower(md5($this->new)); 3674 $this->cache_filename .= '_new'.$broad_directory_name; 3675 } elseif ($this->md5s) { 3676 // source image MD5 hash provided 3677 $this->DebugMessage('SetCacheFilename() _raw set from $this->md5s = "'.$this->md5s.'"', __FILE__, __LINE__); 3678 $broad_directory_name = $this->md5s; 3679 $this->cache_filename .= '_raw'.$this->md5s; 3680 } elseif (!$this->src && $this->rawImageData) { 3681 $this->DebugMessage('SetCacheFilename() _raw set from md5($this->rawImageData) = "'.md5($this->rawImageData).'"', __FILE__, __LINE__); 3682 $broad_directory_name = strtolower(md5($this->rawImageData)); 3683 $this->cache_filename .= '_raw'.$broad_directory_name; 3684 } else { 3685 $this->DebugMessage('SetCacheFilename() _src set from md5($this->sourceFilename) "'.$this->sourceFilename.'" = "'.md5($this->sourceFilename).'"', __FILE__, __LINE__); 3686 $broad_directory_name = strtolower(md5($this->sourceFilename)); 3687 $this->cache_filename .= '_src'.$broad_directory_name; 3688 } 3689 if (!empty($_SERVER['HTTP_REFERER']) && $this->config_nooffsitelink_enabled) { 3690 $parsed_url1 = @phpthumb_functions::ParseURLbetter(@$_SERVER['HTTP_REFERER']); 3691 $parsed_url2 = @phpthumb_functions::ParseURLbetter('http://'.@$_SERVER['HTTP_HOST']); 3692 if (@$parsed_url1['host'] && @$parsed_url2['host'] && ($parsed_url1['host'] != $parsed_url2['host'])) { 3693 // include "_offsite" only if nooffsitelink_enabled and if referrer doesn't match the domain of the current server 3694 $this->cache_filename .= '_offsite'; 3695 } 3696 } 3697 3698 $ParametersString = ''; 3699 if ($this->fltr && is_array($this->fltr)) { 3700 $ParametersString .= '_fltr'.implode('_fltr', $this->fltr); 3701 } 3702 $FilenameParameters1 = array('ar', 'bg', 'bc', 'far', 'sx', 'sy', 'sw', 'sh', 'zc'); 3703 foreach ($FilenameParameters1 as $key) { 3704 if ($this->$key) { 3705 $ParametersString .= '_'.$key.$this->$key; 3706 } 3707 } 3708 $FilenameParameters2 = array('h', 'w', 'wl', 'wp', 'ws', 'hp', 'hs', 'xto', 'ra', 'iar', 'aoe', 'maxb', 'sfn', 'dpi'); 3709 foreach ($FilenameParameters2 as $key) { 3710 if ($this->$key) { 3711 $ParametersString .= '_'.$key. (int) $this->$key; 3712 } 3713 } 3714 $FilenameParameters3 = array('ica'); 3715 foreach ($FilenameParameters3 as $key) { 3716 if ($this->$key) { 3717 $ParametersString .= '_'.$key.substr(md5($this->$key), 0, 4); 3718 } 3719 } 3720 if ($this->thumbnailFormat == 'jpeg') { 3721 // only JPEG output has variable quality option 3722 $ParametersString .= '_q'. (int) $this->thumbnailQuality; 3723 } 3724 $this->DebugMessage('SetCacheFilename() _par set from md5('.$ParametersString.')', __FILE__, __LINE__); 3725 $this->cache_filename .= '_par'.strtolower(md5($ParametersString)); 3726 3727 if ($this->md5s) { 3728 // source image MD5 hash provided 3729 // do not source image modification date -- 3730 // cached image will be used even if file was modified or removed 3731 } elseif (!$this->config_cache_source_filemtime_ignore_remote && preg_match('#^(f|ht)tps?\://#i', $this->src)) { 3732 $this->cache_filename .= '_dat'. (int) phpthumb_functions::filedate_remote($this->src); 3733 } elseif (!$this->config_cache_source_filemtime_ignore_local && $this->src && !$this->rawImageData) { 3734 $this->cache_filename .= '_dat'. (int) (@filemtime($this->sourceFilename)); 3735 } 3736 3737 $this->cache_filename .= '.'.strtolower($this->thumbnailFormat); 3738 $broad_directories = ''; 3739 for ($i = 0; $i < $this->config_cache_directory_depth; $i++) { 3740 $broad_directories .= DIRECTORY_SEPARATOR.substr($broad_directory_name, 0, $i + 1); 3741 } 3742 3743 $this->cache_filename = $this->config_cache_directory.$broad_directories.DIRECTORY_SEPARATOR.$this->config_cache_prefix.rawurlencode($this->cache_filename); 3744 return true; 3745 } 3746 3747 3748 public function SourceImageIsTooLarge($width, $height) { 3749 if (!$this->config_max_source_pixels) { 3750 return false; 3751 } 3752 if ($this->php_memory_limit && function_exists('memory_get_usage')) { 3753 $available_memory = $this->php_memory_limit - memory_get_usage(); 3754 return (bool) (($width * $height * 5) > $available_memory); 3755 } 3756 return (bool) (($width * $height) > $this->config_max_source_pixels); 3757 } 3758 3759 public function ImageCreateFromFilename($filename) { 3760 // try to create GD image source directly via GD, if possible, 3761 // rather than buffering to memory and creating with imagecreatefromstring 3762 $ImageCreateWasAttempted = false; 3763 $gd_image = false; 3764 3765 $this->DebugMessage('starting ImageCreateFromFilename('.$filename.')', __FILE__, __LINE__); 3766 if ($filename && ($getimagesizeinfo = @getimagesize($filename))) { 3767 if (!$this->SourceImageIsTooLarge($getimagesizeinfo[0], $getimagesizeinfo[1])) { 3768 $ImageCreateFromFunction = array( 3769 IMAGETYPE_GIF => 'imagecreatefromgif', 3770 IMAGETYPE_JPEG => 'imagecreatefromjpeg', 3771 IMAGETYPE_PNG => 'imagecreatefrompng', 3772 IMAGETYPE_WBMP => 'imagecreatefromwbmp', 3773 IMAGETYPE_WEBP => 'imagecreatefromwebp', 3774 IMAGETYPE_AVIF => 'imagecreatefromavif', 3775 ); 3776 $this->DebugMessage('ImageCreateFromFilename found ($getimagesizeinfo[2]=='.@$getimagesizeinfo[2].')', __FILE__, __LINE__); 3777 switch (@$getimagesizeinfo[2]) { 3778 case IMAGETYPE_GIF: 3779 case IMAGETYPE_JPEG: 3780 case IMAGETYPE_PNG: 3781 case IMAGETYPE_WBMP: 3782 case IMAGETYPE_WEBP: 3783 case IMAGETYPE_AVIF: 3784 $ImageCreateFromFunctionName = $ImageCreateFromFunction[$getimagesizeinfo[2]]; 3785 if (function_exists($ImageCreateFromFunctionName)) { 3786 $this->DebugMessage('Calling '.$ImageCreateFromFunctionName.'('.$filename.')', __FILE__, __LINE__); 3787 $ImageCreateWasAttempted = true; 3788 $gd_image = $ImageCreateFromFunctionName($filename); 3789 } else { 3790 $this->DebugMessage('NOT calling '.$ImageCreateFromFunctionName.'('.$filename.') because !function_exists('.$ImageCreateFromFunctionName.')', __FILE__, __LINE__); 3791 } 3792 break; 3793 3794 case IMAGETYPE_SWF: 3795 case IMAGETYPE_PSD: 3796 case IMAGETYPE_BMP: 3797 case IMAGETYPE_TIFF_II: 3798 case IMAGETYPE_TIFF_MM: 3799 case IMAGETYPE_JPC: 3800 case IMAGETYPE_JP2: 3801 case IMAGETYPE_JPX: 3802 case IMAGETYPE_JB2: 3803 case IMAGETYPE_SWC: 3804 case IMAGETYPE_IFF: 3805 case IMAGETYPE_XBM: 3806 case IMAGETYPE_ICO: 3807 $this->DebugMessage('No built-in image creation function for image type "'.@$getimagesizeinfo[2].'" ($getimagesizeinfo[2])', __FILE__, __LINE__); 3808 break; 3809 3810 default: 3811 $this->DebugMessage('Unknown value for $getimagesizeinfo[2]: "'.@$getimagesizeinfo[2].'"', __FILE__, __LINE__); 3812 break; 3813 } 3814 } else { 3815 $this->DebugMessage('image is '.$getimagesizeinfo[0].'x'.$getimagesizeinfo[1].' and therefore contains more pixels ('.($getimagesizeinfo[0] * $getimagesizeinfo[1]).') than $this->config_max_source_pixels setting ('.$this->config_max_source_pixels.')', __FILE__, __LINE__); 3816 return false; 3817 } 3818 } else { 3819 $this->DebugMessage('empty $filename or getimagesize('.$filename.') failed', __FILE__, __LINE__); 3820 } 3821 3822 if (!$gd_image) { 3823 // cannot create from filename, attempt to create source image with imagecreatefromstring, if possible 3824 if ($ImageCreateWasAttempted) { 3825 $this->DebugMessage($ImageCreateFromFunctionName.'() was attempted but FAILED', __FILE__, __LINE__); 3826 } 3827 $this->DebugMessage('Populating $rawimagedata', __FILE__, __LINE__); 3828 $rawimagedata = ''; 3829 if ($fp = @fopen($filename, 'rb')) { 3830 $filesize = filesize($filename); 3831 $blocksize = 8192; 3832 $blockreads = ceil($filesize / $blocksize); 3833 for ($i = 0; $i < $blockreads; $i++) { 3834 $rawimagedata .= fread($fp, $blocksize); 3835 } 3836 fclose($fp); 3837 } else { 3838 $this->DebugMessage('cannot fopen('.$filename.')', __FILE__, __LINE__); 3839 } 3840 if ($rawimagedata) { 3841 $this->DebugMessage('attempting ImageCreateFromStringReplacement($rawimagedata ('.strlen($rawimagedata).' bytes), true)', __FILE__, __LINE__); 3842 $gd_image = $this->ImageCreateFromStringReplacement($rawimagedata, true); 3843 } 3844 } 3845 return $gd_image; 3846 } 3847 3848 public function SourceImageToGD() { 3849 if (is_resource($this->gdimg_source) || (is_object($this->gdimg_source) && $this->gdimg_source instanceOf \GdImage)) { 3850 $this->source_width = imagesx($this->gdimg_source); 3851 $this->source_height = imagesy($this->gdimg_source); 3852 $this->DebugMessage('skipping SourceImageToGD() because $this->gdimg_source is already a resource ('.$this->source_width.'x'.$this->source_height.')', __FILE__, __LINE__); 3853 return true; 3854 } 3855 $this->DebugMessage('starting SourceImageToGD()', __FILE__, __LINE__); 3856 3857 if ($this->config_prefer_imagemagick) { 3858 if (empty($this->sourceFilename) && !empty($this->rawImageData)) { 3859 $this->DebugMessage('Copying raw image data to temp file and trying again with ImageMagick', __FILE__, __LINE__); 3860 if ($tempnam = $this->phpThumb_tempnam()) { 3861 if (file_put_contents($tempnam, $this->rawImageData)) { 3862 $this->sourceFilename = $tempnam; 3863 if ($this->ImageMagickThumbnailToGD()) { 3864 // excellent, we have a thumbnailed source image 3865 $this->DebugMessage('ImageMagickThumbnailToGD() succeeded', __FILE__, __LINE__); 3866 } else { 3867 $this->DebugMessage('ImageMagickThumbnailToGD() failed', __FILE__, __LINE__); 3868 } 3869 @chmod($tempnam, $this->getParameter('config_file_create_mask')); 3870 } else { 3871 $this->DebugMessage('failed to put $this->rawImageData into temp file "'.$tempnam.'"', __FILE__, __LINE__); 3872 } 3873 } else { 3874 $this->DebugMessage('failed to generate temp file name', __FILE__, __LINE__); 3875 } 3876 } 3877 } 3878 if (!$this->gdimg_source && $this->rawImageData) { 3879 3880 if ($this->SourceImageIsTooLarge($this->source_width, $this->source_height)) { 3881 $memory_get_usage = (function_exists('memory_get_usage') ? memory_get_usage() : 0); 3882 return $this->ErrorImage('Source image is too large ('.$this->source_width.'x'.$this->source_height.' = '.number_format($this->source_width * $this->source_height / 1000000, 1).'Mpx, max='.number_format($this->config_max_source_pixels / 1000000, 1).'Mpx) for GD creation (either install ImageMagick or increase PHP memory_limit to at least '.ceil(($memory_get_usage + (5 * $this->source_width * $this->source_height)) / 1048576).'M).'); 3883 } 3884 if ($this->md5s && ($this->md5s != md5($this->rawImageData))) { 3885 return $this->ErrorImage('$this->md5s != md5($this->rawImageData)'."\n".'"'.$this->md5s.'" != '."\n".'"'.md5($this->rawImageData).'"'); 3886 } 3887 //if ($this->issafemode) { 3888 // return $this->ErrorImage('Cannot generate thumbnails from raw image data when PHP SAFE_MODE enabled'); 3889 //} 3890 $this->gdimg_source = $this->ImageCreateFromStringReplacement($this->rawImageData); 3891 if (!$this->gdimg_source) { 3892 if (substr($this->rawImageData, 0, 2) === 'BM') { 3893 $this->getimagesizeinfo[2] = 6; // BMP 3894 } elseif (substr($this->rawImageData, 0, 4) === 'II'."\x2A\x00") { 3895 $this->getimagesizeinfo[2] = 7; // TIFF (littlendian) 3896 } elseif (substr($this->rawImageData, 0, 4) === 'MM'."\x00\x2A") { 3897 $this->getimagesizeinfo[2] = 8; // TIFF (bigendian) 3898 } 3899 $this->DebugMessage('SourceImageToGD.ImageCreateFromStringReplacement() failed with unknown image type "'.substr($this->rawImageData, 0, 4).'" ('.phpthumb_functions::HexCharDisplay(substr($this->rawImageData, 0, 4)).')', __FILE__, __LINE__); 3900// return $this->ErrorImage('Unknown image type identified by "'.substr($this->rawImageData, 0, 4).'" ('.phpthumb_functions::HexCharDisplay(substr($this->rawImageData, 0, 4)).') in SourceImageToGD()['.__LINE__.']'); 3901 } 3902 3903 } elseif (!$this->gdimg_source && $this->sourceFilename) { 3904 3905 if ($this->md5s && ($this->md5s != phpthumb_functions::md5_file_safe($this->sourceFilename))) { 3906 return $this->ErrorImage('$this->md5s != md5(sourceFilename)'."\n".'"'.$this->md5s.'" != '."\n".'"'.phpthumb_functions::md5_file_safe($this->sourceFilename).'"'); 3907 } 3908 switch (@$this->getimagesizeinfo[2]) { 3909 case 1: 3910 case 3: 3911 // GIF or PNG input file may have transparency 3912 $this->is_alpha = true; 3913 break; 3914 } 3915 if (!$this->SourceImageIsTooLarge($this->source_width, $this->source_height)) { 3916 $this->gdimg_source = $this->ImageCreateFromFilename($this->sourceFilename); 3917 } 3918 3919 } 3920 3921 while (true) { 3922 if ($this->gdimg_source) { 3923 $this->DebugMessage('Not using EXIF thumbnail data because $this->gdimg_source is already set', __FILE__, __LINE__); 3924 break; 3925 } 3926 if (!$this->exif_thumbnail_data) { 3927 $this->DebugMessage('Not using EXIF thumbnail data because $this->exif_thumbnail_data is empty', __FILE__, __LINE__); 3928 break; 3929 } 3930 if (ini_get('safe_mode')) { 3931 if (!$this->SourceImageIsTooLarge($this->source_width, $this->source_height)) { 3932 $this->DebugMessage('Using EXIF thumbnail data because source image too large and safe_mode enabled', __FILE__, __LINE__); 3933 $this->aoe = true; 3934 } else { 3935 break; 3936 } 3937 } else { 3938 if (!$this->config_use_exif_thumbnail_for_speed) { 3939 $this->DebugMessage('Not using EXIF thumbnail data because $this->config_use_exif_thumbnail_for_speed is FALSE', __FILE__, __LINE__); 3940 break; 3941 } 3942 if (($this->thumbnailCropX != 0) || ($this->thumbnailCropY != 0)) { 3943 $this->DebugMessage('Not using EXIF thumbnail data because source cropping is enabled ('.$this->thumbnailCropX.','.$this->thumbnailCropY.')', __FILE__, __LINE__); 3944 break; 3945 } 3946 if (($this->w > $this->exif_thumbnail_width) || ($this->h > $this->exif_thumbnail_height)) { 3947 $this->DebugMessage('Not using EXIF thumbnail data because EXIF thumbnail is too small ('.$this->exif_thumbnail_width.'x'.$this->exif_thumbnail_height.' vs '.$this->w.'x'.$this->h.')', __FILE__, __LINE__); 3948 break; 3949 } 3950 $source_ar = $this->source_width / $this->source_height; 3951 $exif_ar = $this->exif_thumbnail_width / $this->exif_thumbnail_height; 3952 if (number_format($source_ar, 2) != number_format($exif_ar, 2)) { 3953 $this->DebugMessage('not using EXIF thumbnail because $source_ar != $exif_ar ('.$source_ar.' != '.$exif_ar.')', __FILE__, __LINE__); 3954 break; 3955 } 3956 } 3957 3958 // EXIF thumbnail exists, and is equal to or larger than destination thumbnail, and will be use as source image 3959 $this->DebugMessage('Trying to use EXIF thumbnail as source image', __FILE__, __LINE__); 3960 3961 if ($gdimg_exif_temp = $this->ImageCreateFromStringReplacement($this->exif_thumbnail_data, false)) { 3962 3963 $this->DebugMessage('Successfully using EXIF thumbnail as source image', __FILE__, __LINE__); 3964 $this->gdimg_source = $gdimg_exif_temp; 3965 $this->source_width = $this->exif_thumbnail_width; 3966 $this->source_height = $this->exif_thumbnail_height; 3967 $this->thumbnailCropW = $this->source_width; 3968 $this->thumbnailCropH = $this->source_height; 3969 return true; 3970 3971 } else { 3972 $this->DebugMessage('$this->ImageCreateFromStringReplacement($this->exif_thumbnail_data, false) failed', __FILE__, __LINE__); 3973 } 3974 3975 break; 3976 } 3977 3978 if (!$this->gdimg_source) { 3979 $this->DebugMessage('$this->gdimg_source is still empty', __FILE__, __LINE__); 3980 3981 $this->DebugMessage('ImageMagickThumbnailToGD() failed', __FILE__, __LINE__); 3982 3983 $imageHeader = ''; 3984 $gd_info = gd_info(); 3985 $GDreadSupport = false; 3986 switch (@$this->getimagesizeinfo[2]) { 3987 case 1: 3988 $imageHeader = 'Content-Type: image/gif'; 3989 $GDreadSupport = (bool) @$gd_info['GIF Read Support']; 3990 break; 3991 case 2: 3992 $imageHeader = 'Content-Type: image/jpeg'; 3993 $GDreadSupport = (bool) @$gd_info['JPG Support']; 3994 break; 3995 case 3: 3996 $imageHeader = 'Content-Type: image/png'; 3997 $GDreadSupport = (bool) @$gd_info['PNG Support']; 3998 break; 3999 } 4000 if ($imageHeader) { 4001 // cannot create image for whatever reason (maybe imagecreatefromjpeg et al are not available?) 4002 // and ImageMagick is not available either, no choice but to output original (not resized/modified) data and exit 4003 if ($this->config_error_die_on_source_failure) { 4004 $errormessages = array(); 4005 $errormessages[] = 'All attempts to create GD image source failed.'; 4006 if ($this->fatalerror) { 4007 $errormessages[] = $this->fatalerror; 4008 } 4009 if ($this->issafemode) { 4010 $errormessages[] = 'Safe Mode enabled, therefore ImageMagick is unavailable. (disable Safe Mode if possible)'; 4011 } elseif (!$this->ImageMagickVersion()) { 4012 $errormessages[] = 'ImageMagick is not installed (it is highly recommended that you install it).'; 4013 } 4014 if ($this->SourceImageIsTooLarge($this->getimagesizeinfo[0], $this->getimagesizeinfo[1])) { 4015 $memory_get_usage = (function_exists('memory_get_usage') ? memory_get_usage() : 0); 4016 $errormessages[] = 'Source image is too large ('.$this->getimagesizeinfo[0].'x'.$this->getimagesizeinfo[1].' = '.number_format($this->getimagesizeinfo[0] * $this->getimagesizeinfo[1] / 1000000, 1).'Mpx, max='.number_format($this->config_max_source_pixels / 1000000, 1).'Mpx) for GD creation (either install ImageMagick or increase PHP memory_limit to at least '.ceil(($memory_get_usage + (5 * $this->getimagesizeinfo[0] * $this->getimagesizeinfo[1])) / 1048576).'M).'; 4017 } elseif (!$GDreadSupport) { 4018 $errormessages[] = 'GD does not have read support for "'.$imageHeader.'".'; 4019 } else { 4020 $errormessages[] = 'Source image probably corrupt.'; 4021 } 4022 $this->ErrorImage(implode("\n", $errormessages)); 4023 4024 } else { 4025 $this->DebugMessage('All attempts to create GD image source failed ('.(ini_get('safe_mode') ? 'Safe Mode enabled, ImageMagick unavailable and source image probably too large for GD': ($GDreadSupport ? 'source image probably corrupt' : 'GD does not have read support for "'.$imageHeader.'"')).'), cannot generate thumbnail'); 4026 //$this->DebugMessage('All attempts to create GD image source failed ('.($GDreadSupport ? 'source image probably corrupt' : 'GD does not have read support for "'.$imageHeader.'"').'), outputing raw image', __FILE__, __LINE__); 4027 //if (!$this->phpThumbDebug) { 4028 // header($imageHeader); 4029 // echo $this->rawImageData; 4030 // exit; 4031 //} 4032 return false; 4033 } 4034 } 4035 4036 //switch (substr($this->rawImageData, 0, 2)) { 4037 // case 'BM': 4038 switch (@$this->getimagesizeinfo[2]) { 4039 case 6: 4040 ob_start(); 4041 if (!@include_once __DIR__ .'/phpthumb.bmp.php' ) { 4042 ob_end_clean(); 4043 return $this->ErrorImage('include_once('. __DIR__ .'/phpthumb.bmp.php) failed'); 4044 } 4045 ob_end_clean(); 4046 if ($fp = @fopen($this->sourceFilename, 'rb')) { 4047 $this->rawImageData = ''; 4048 while (!feof($fp)) { 4049 $this->rawImageData .= fread($fp, 32768); 4050 } 4051 fclose($fp); 4052 } 4053 $phpthumb_bmp = new phpthumb_bmp(); 4054 $this->gdimg_source = $phpthumb_bmp->phpthumb_bmp2gd($this->rawImageData, phpthumb_functions::gd_version() >= 2.0); 4055 unset($phpthumb_bmp); 4056 if ($this->gdimg_source) { 4057 $this->DebugMessage('$phpthumb_bmp->phpthumb_bmp2gd() succeeded', __FILE__, __LINE__); 4058 } else { 4059 return $this->ErrorImage($this->ImageMagickVersion() ? 'ImageMagick failed on BMP source conversion' : 'phpthumb_bmp2gd() failed'); 4060 } 4061 break; 4062 //} 4063 //switch (substr($this->rawImageData, 0, 4)) { 4064 // case 'II'."\x2A\x00": 4065 // case 'MM'."\x00\x2A": 4066 case 7: 4067 case 8: 4068 return $this->ErrorImage($this->ImageMagickVersion() ? 'ImageMagick failed on TIFF source conversion' : 'ImageMagick is unavailable and phpThumb() does not support TIFF source images without it'); 4069 break; 4070 4071 //case "\xD7\xCD\xC6\x9A": 4072 // return $this->ErrorImage($this->ImageMagickVersion() ? 'ImageMagick failed on WMF source conversion' : 'ImageMagick is unavailable and phpThumb() does not support WMF source images without it'); 4073 // break; 4074 } 4075 4076 if (!$this->gdimg_source) { 4077 if ($this->rawImageData) { 4078 $HeaderFourBytes = substr($this->rawImageData, 0, 4); 4079 } elseif ($this->sourceFilename) { 4080 if ($fp = @fopen($this->sourceFilename, 'rb')) { 4081 $HeaderFourBytes = fread($fp, 4); 4082 fclose($fp); 4083 } else { 4084 return $this->ErrorImage('failed to open "'.$this->sourceFilename.'" SourceImageToGD() ['.__LINE__.']'); 4085 } 4086 } else { 4087 return $this->ErrorImage('Unable to create image, neither filename nor image data suppplied in SourceImageToGD() ['.__LINE__.']'); 4088 } 4089 if (!$this->ImageMagickVersion() && !phpthumb_functions::gd_version()) { 4090 return $this->ErrorImage('Neither GD nor ImageMagick seem to be installed on this server. At least one (preferably GD), or better both, MUST be installed for phpThumb to work.'); 4091 } elseif ($HeaderFourBytes == "\xD7\xCD\xC6\x9A") { // WMF 4092 return $this->ErrorImage($this->ImageMagickVersion() ? 'ImageMagick failed on WMF source conversion' : 'ImageMagick is unavailable and phpThumb() does not support WMF source images without it'); 4093 } elseif ($HeaderFourBytes == '%PDF') { // "%PDF" 4094 return $this->ErrorImage($this->ImageMagickVersion() ? 'ImageMagick and GhostScript are both required for PDF source images; GhostScript may not be properly configured' : 'ImageMagick and/or GhostScript are unavailable and phpThumb() does not support PDF source images without them'); 4095 } elseif (substr($HeaderFourBytes, 0, 3) == "\xFF\xD8\xFF") { // JPEG 4096 return $this->ErrorImage('Image (JPEG) is too large for PHP-GD memory_limit, please install ImageMagick or increase php.ini memory_limit setting'); 4097 } elseif ($HeaderFourBytes == '%PNG') { // "%PNG" 4098 return $this->ErrorImage('Image (PNG) is too large for PHP-GD memory_limit, please install ImageMagick or increase php.ini memory_limit setting'); 4099 } elseif (substr($HeaderFourBytes, 0, 3) == 'GIF') { // GIF 4100 return $this->ErrorImage('Image (GIF) is too large for PHP-GD memory_limit, please install ImageMagick or increase php.ini memory_limit setting'); 4101 } 4102 return $this->ErrorImage('Unknown image type identified by "'.$HeaderFourBytes.'" ('.phpthumb_functions::HexCharDisplay($HeaderFourBytes).') in SourceImageToGD() ['.__LINE__.']'); 4103 } 4104 } 4105 4106 if (!$this->gdimg_source) { 4107 if ($gdimg_exif_temp = $this->ImageCreateFromStringReplacement($this->exif_thumbnail_data, false)) { 4108 $this->DebugMessage('All other attempts failed, but successfully using EXIF thumbnail as source image', __FILE__, __LINE__); 4109 $this->gdimg_source = $gdimg_exif_temp; 4110 // override allow-enlarging setting if EXIF thumbnail is the only source available 4111 // otherwise thumbnails larger than the EXIF thumbnail will be created at EXIF size 4112 $this->aoe = true; 4113 return true; 4114 } 4115 return false; 4116 } 4117 4118 $this->source_width = imagesx($this->gdimg_source); 4119 $this->source_height = imagesy($this->gdimg_source); 4120 return true; 4121 } 4122 4123 private function ImageCropAuto() { 4124 // ImageCropAuto 4125 if (!is_null($this->ica)) { 4126 $this->DebugMessage('ImageCropAuto('.$this->ica.') starting', __FILE__, __LINE__); 4127 if (function_exists('imagecropauto')) { // (PHP 5 >= 5.5.0, PHP 7) 4128 // https://www.php.net/manual/en/function.imagecropauto.php 4129 // 0 = IMG_CROP_DEFAULT 4130 // 1 = IMG_CROP_TRANSPARENT 4131 // 2 = IMG_CROP_BLACK 4132 // 3 = IMG_CROP_WHITE 4133 // 4 = IMG_CROP_SIDES 4134 // 5 = IMG_CROP_THRESHOLD 4135 if (preg_match('#^(([0-4])|(5)\\|(0?\\.?[0-9]+)\\|([0-9A-F]{6}))$#i', $this->ica, $matches)) { 4136 @list($dummy, $dummy, $ica_mode1, $ica_mode2, $ica_threshold, $ica_color) = $matches; 4137 if ($ica_mode2) { 4138 $param_color = hexdec($ica_color); 4139 if (!imageistruecolor($this->gdimg_source)) { 4140 $param_color = imagecolorclosest($this->gdimg_source, hexdec(substr($ica_color, 0, 2)), hexdec(substr($ica_color, 2, 2)), hexdec(substr($ica_color, 4, 2))); 4141 } 4142 $cropped = imagecropauto($this->gdimg_source, intval($ica_mode2), floatval($ica_threshold), $param_color); 4143 } else { 4144 $cropped = imagecropauto($this->gdimg_source, intval($ica_mode1)); 4145 } 4146 if ($cropped !== false) { // in case a new image resource was returned 4147 $this->DebugMessage('ImageCropAuto changing source image size from '.imagesx($this->gdimg_source).'x'.imagesy($this->gdimg_source).' to '.imagesx($cropped).'x'.imagesy($cropped), __FILE__, __LINE__); 4148 imagedestroy($this->gdimg_source); // we destroy the original image 4149 $this->gdimg_source = $cropped; // and assign the cropped image to $im 4150 $this->source_width = imagesx($this->gdimg_source); 4151 $this->source_height = imagesy($this->gdimg_source); 4152 } else { 4153 $this->DebugMessage('imagecropauto failed', __FILE__, __LINE__); 4154 } 4155 } else { 4156 $this->DebugMessage('invalid "ica" parameter syntax, ignoring', __FILE__, __LINE__); 4157 } 4158 } else { 4159 $this->DebugMessage('!function_exists(imagecropauto), ignoring "ica" parameter', __FILE__, __LINE__); 4160 } 4161 } 4162 return true; 4163 } 4164 4165 public function phpThumbDebugVarDump($var) { 4166 if (null === $var) { 4167 return 'NULL'; 4168 } elseif (is_bool($var)) { 4169 return ($var ? 'TRUE' : 'FALSE'); 4170 } elseif (is_string($var)) { 4171 return 'string('.strlen($var).')'.str_repeat(' ', max(0, 3 - strlen(strlen($var)))).' "'.$var.'"'; 4172 } elseif (is_int($var)) { 4173 return 'integer '.$var; 4174 } elseif (is_float($var)) { 4175 return 'float '.$var; 4176 } elseif (is_array($var)) { 4177 ob_start(); 4178 var_dump($var); 4179 $vardumpoutput = ob_get_contents(); 4180 ob_end_clean(); 4181 return strtr($vardumpoutput, "\n\r\t", ' '); 4182 } 4183 return gettype($var); 4184 } 4185 4186 public function phpThumbDebug($level='') { 4187 if ($level && ($this->phpThumbDebug !== $level)) { 4188 return true; 4189 } 4190 if ($this->config_disable_debug) { 4191 return $this->ErrorImage('phpThumbDebug disabled'); 4192 } 4193 4194 $FunctionsExistance = array('exif_thumbnail', 'gd_info', 'image_type_to_mime_type', 'getimagesize', 'imagecopyresampled', 'imagecopyresized', 'imagecreate', 'imagecreatefromstring', 'imagecreatetruecolor', 'imageistruecolor', 'imagerotate', 'imagetypes', 'version_compare', 'imagecreatefromgif', 'imagecreatefromjpeg', 'imagecreatefrompng', 'imagecreatefromwbmp', 'imagecreatefromxbm', 'imagecreatefromxpm', 'imagecreatefromstring', 'imagecreatefromgd', 'imagecreatefromgd2', 'imagecreatefromgd2part', 'imagejpeg', 'imagegif', 'imagepng', 'imagewbmp'); 4195 $ParameterNames = array('src', 'new', 'w', 'h', 'f', 'q', 'sx', 'sy', 'sw', 'sh', 'far', 'bg', 'bc', 'zc', 'ica', 'file', 'goto', 'err', 'xto', 'ra', 'ar', 'aoe', 'iar', 'maxb'); 4196 $ConfigVariableNames = array('document_root', 'temp_directory', 'output_format', 'output_maxwidth', 'output_maxheight', 'error_message_image_default', 'error_bgcolor', 'error_textcolor', 'error_fontsize', 'error_die_on_error', 'error_silent_die_on_error', 'error_die_on_source_failure', 'nohotlink_enabled', 'nohotlink_valid_domains', 'nohotlink_erase_image', 'nohotlink_text_message', 'nooffsitelink_enabled', 'nooffsitelink_valid_domains', 'nooffsitelink_require_refer', 'nooffsitelink_erase_image', 'nooffsitelink_text_message', 'high_security_enabled', 'allow_src_above_docroot', 'allow_src_above_phpthumb', 'max_source_pixels', 'use_exif_thumbnail_for_speed', 'border_hexcolor', 'background_hexcolor', 'ttf_directory', 'disable_pathinfo_parsing', 'disable_imagecopyresampled'); 4197 $OtherVariableNames = array('phpThumbDebug', 'thumbnailQuality', 'thumbnailFormat', 'gdimg_output', 'gdimg_source', 'sourceFilename', 'source_width', 'source_height', 'thumbnailCropX', 'thumbnailCropY', 'thumbnailCropW', 'thumbnailCropH', 'exif_thumbnail_width', 'exif_thumbnail_height', 'exif_thumbnail_type', 'thumbnail_width', 'thumbnail_height', 'thumbnail_image_width', 'thumbnail_image_height'); 4198 4199 $DebugOutput = array(); 4200 $DebugOutput[] = 'phpThumb() version = '.$this->phpthumb_version; 4201 $DebugOutput[] = 'phpversion() = '.@PHP_VERSION; 4202 $DebugOutput[] = 'PHP_OS = '.PHP_OS; 4203 $DebugOutput[] = '$_SERVER[SERVER_SOFTWARE] = '.@$_SERVER['SERVER_SOFTWARE']; 4204 $DebugOutput[] = '__FILE__ = '.__FILE__; 4205 $DebugOutput[] = 'realpath(.) = '.@realpath('.'); 4206 $DebugOutput[] = '$_SERVER[PHP_SELF] = '.@$_SERVER['PHP_SELF']; 4207 $DebugOutput[] = '$_SERVER[HOST_NAME] = '.@$_SERVER['HOST_NAME']; 4208 $DebugOutput[] = '$_SERVER[HTTP_REFERER] = '.@$_SERVER['HTTP_REFERER']; 4209 $DebugOutput[] = '$_SERVER[QUERY_STRING] = '.@$_SERVER['QUERY_STRING']; 4210 $DebugOutput[] = '$_SERVER[PATH_INFO] = '.@$_SERVER['PATH_INFO']; 4211 $DebugOutput[] = '$_SERVER[DOCUMENT_ROOT] = '.@$_SERVER['DOCUMENT_ROOT']; 4212 $DebugOutput[] = 'getenv(DOCUMENT_ROOT) = '.@getenv('DOCUMENT_ROOT'); 4213 $DebugOutput[] = ''; 4214 4215 $DebugOutput[] = 'get_magic_quotes_gpc() = '.(function_exists('get_magic_quotes_gpc') ? $this->phpThumbDebugVarDump(@get_magic_quotes_gpc()) : false); 4216 $DebugOutput[] = 'get_magic_quotes_runtime() = '.(function_exists('get_magic_quotes_runtime') ? $this->phpThumbDebugVarDump(@get_magic_quotes_runtime()) : false); 4217 $DebugOutput[] = 'error_reporting() = '.$this->phpThumbDebugVarDump(error_reporting()); 4218 $DebugOutput[] = 'ini_get(error_reporting) = '.$this->phpThumbDebugVarDump(@ini_get('error_reporting')); 4219 $DebugOutput[] = 'ini_get(display_errors) = '.$this->phpThumbDebugVarDump(@ini_get('display_errors')); 4220 $DebugOutput[] = 'ini_get(allow_url_fopen) = '.$this->phpThumbDebugVarDump(@ini_get('allow_url_fopen')); 4221 $DebugOutput[] = 'ini_get(disable_functions) = '.$this->phpThumbDebugVarDump(@ini_get('disable_functions')); 4222 $DebugOutput[] = 'get_cfg_var(disable_functions) = '.$this->phpThumbDebugVarDump(@get_cfg_var('disable_functions')); 4223 $DebugOutput[] = 'ini_get(safe_mode) = '.$this->phpThumbDebugVarDump(@ini_get('safe_mode')); 4224 $DebugOutput[] = 'ini_get(open_basedir) = '.$this->phpThumbDebugVarDump(@ini_get('open_basedir')); 4225 $DebugOutput[] = 'ini_get(max_execution_time) = '.$this->phpThumbDebugVarDump(@ini_get('max_execution_time')); 4226 $DebugOutput[] = 'ini_get(memory_limit) = '.$this->phpThumbDebugVarDump(@ini_get('memory_limit')); 4227 $DebugOutput[] = 'get_cfg_var(memory_limit) = '.$this->phpThumbDebugVarDump(@get_cfg_var('memory_limit')); 4228 $DebugOutput[] = 'memory_get_usage() = '.(function_exists('memory_get_usage') ? $this->phpThumbDebugVarDump(@memory_get_usage()) : 'n/a'); 4229 $DebugOutput[] = ''; 4230 4231 $DebugOutput[] = '$this->config_prefer_imagemagick = '.$this->phpThumbDebugVarDump($this->config_prefer_imagemagick); 4232 $DebugOutput[] = '$this->config_imagemagick_path = '.$this->phpThumbDebugVarDump($this->config_imagemagick_path); 4233 $DebugOutput[] = '$this->ImageMagickWhichConvert() = '.$this->ImageMagickWhichConvert(); 4234 $IMpathUsed = ($this->config_imagemagick_path ? $this->config_imagemagick_path : $this->ImageMagickWhichConvert()); 4235 $DebugOutput[] = '[actual ImageMagick path used] = '.$this->phpThumbDebugVarDump($IMpathUsed); 4236 $DebugOutput[] = 'file_exists([actual ImageMagick path used]) = '.$this->phpThumbDebugVarDump(@file_exists($IMpathUsed)); 4237 $DebugOutput[] = 'ImageMagickVersion(false) = '.$this->ImageMagickVersion(false); 4238 $DebugOutput[] = 'ImageMagickVersion(true) = '.$this->ImageMagickVersion(true); 4239 $DebugOutput[] = ''; 4240 4241 $DebugOutput[] = '$this->config_cache_directory = '.$this->phpThumbDebugVarDump($this->config_cache_directory); 4242 $DebugOutput[] = '$this->config_cache_directory_depth = '.$this->phpThumbDebugVarDump($this->config_cache_directory_depth); 4243 $DebugOutput[] = '$this->config_cache_disable_warning = '.$this->phpThumbDebugVarDump($this->config_cache_disable_warning); 4244 $DebugOutput[] = '$this->config_cache_maxage = '.$this->phpThumbDebugVarDump($this->config_cache_maxage); 4245 $DebugOutput[] = '$this->config_cache_maxsize = '.$this->phpThumbDebugVarDump($this->config_cache_maxsize); 4246 $DebugOutput[] = '$this->config_cache_maxfiles = '.$this->phpThumbDebugVarDump($this->config_cache_maxfiles); 4247 $DebugOutput[] = '$this->config_cache_force_passthru = '.$this->phpThumbDebugVarDump($this->config_cache_force_passthru); 4248 $DebugOutput[] = '$this->cache_filename = '.$this->phpThumbDebugVarDump($this->cache_filename); 4249 $DebugOutput[] = 'is_readable($this->config_cache_directory) = '.$this->phpThumbDebugVarDump(@is_readable($this->config_cache_directory)); 4250 $DebugOutput[] = 'is_writable($this->config_cache_directory) = '.$this->phpThumbDebugVarDump(@is_writable($this->config_cache_directory)); 4251 $DebugOutput[] = 'is_readable($this->cache_filename) = '.$this->phpThumbDebugVarDump(@is_readable($this->cache_filename)); 4252 $DebugOutput[] = 'is_writable($this->cache_filename) = '.(@file_exists($this->cache_filename) ? $this->phpThumbDebugVarDump(@is_writable($this->cache_filename)) : 'n/a'); 4253 $DebugOutput[] = ''; 4254 4255 foreach ($ConfigVariableNames as $varname) { 4256 $varname = 'config_'.$varname; 4257 $value = $this->$varname; 4258 $DebugOutput[] = '$this->'.str_pad($varname, 37, ' ', STR_PAD_RIGHT).' = '.$this->phpThumbDebugVarDump($value); 4259 } 4260 $DebugOutput[] = ''; 4261 foreach ($OtherVariableNames as $varname) { 4262 $value = $this->$varname; 4263 $DebugOutput[] = '$this->'.str_pad($varname, 27, ' ', STR_PAD_RIGHT).' = '.$this->phpThumbDebugVarDump($value); 4264 } 4265 $DebugOutput[] = 'strlen($this->rawImageData) = '.(!empty($this->rawImageData) ? strlen($this->rawImageData) : ''); 4266 $DebugOutput[] = 'strlen($this->exif_thumbnail_data) = '.(!empty($this->exif_thumbnail_data) ? strlen($this->exif_thumbnail_data) : ''); 4267 $DebugOutput[] = ''; 4268 4269 foreach ($ParameterNames as $varname) { 4270 $value = $this->$varname; 4271 $DebugOutput[] = '$this->'.str_pad($varname, 4, ' ', STR_PAD_RIGHT).' = '.$this->phpThumbDebugVarDump($value); 4272 } 4273 $DebugOutput[] = ''; 4274 4275 foreach ($FunctionsExistance as $functionname) { 4276 $DebugOutput[] = 'builtin_function_exists('.$functionname.')'.str_repeat(' ', 23 - strlen($functionname)).' = '.$this->phpThumbDebugVarDump(phpthumb_functions::builtin_function_exists($functionname)); 4277 } 4278 $DebugOutput[] = ''; 4279 4280 $gd_info = gd_info(); 4281 foreach ($gd_info as $key => $value) { 4282 $DebugOutput[] = 'gd_info.'.str_pad($key, 34, ' ', STR_PAD_RIGHT).' = '.$this->phpThumbDebugVarDump($value); 4283 } 4284 $DebugOutput[] = ''; 4285 4286 $exif_info = phpthumb_functions::exif_info(); 4287 foreach ($exif_info as $key => $value) { 4288 $DebugOutput[] = 'exif_info.'.str_pad($key, 26, ' ', STR_PAD_RIGHT).' = '.$this->phpThumbDebugVarDump($value); 4289 } 4290 $DebugOutput[] = ''; 4291 4292 if ($ApacheLookupURIarray = phpthumb_functions::ApacheLookupURIarray(dirname(@$_SERVER['PHP_SELF']))) { 4293 foreach ($ApacheLookupURIarray as $key => $value) { 4294 $DebugOutput[] = 'ApacheLookupURIarray.'.str_pad($key, 15, ' ', STR_PAD_RIGHT).' = '.$this->phpThumbDebugVarDump($value); 4295 } 4296 } else { 4297 $DebugOutput[] = 'ApacheLookupURIarray() -- FAILED'; 4298 } 4299 $DebugOutput[] = ''; 4300 4301 if (isset($_GET) && is_array($_GET)) { 4302 foreach ($_GET as $key => $value) { 4303 $DebugOutput[] = '$_GET['.$key.']'.str_repeat(' ', 30 - strlen($key)).'= '.$this->phpThumbDebugVarDump($value); 4304 } 4305 } 4306 if (isset($_POST) && is_array($_POST)) { 4307 foreach ($_POST as $key => $value) { 4308 $DebugOutput[] = '$_POST['.$key.']'.str_repeat(' ', 29 - strlen($key)).'= '.$this->phpThumbDebugVarDump($value); 4309 } 4310 } 4311 $DebugOutput[] = ''; 4312 4313 $DebugOutput[] = '$this->debugmessages:'; 4314 foreach ($this->debugmessages as $errorstring) { 4315 $DebugOutput[] = ' * '.$errorstring; 4316 } 4317 $DebugOutput[] = ''; 4318 4319 $DebugOutput[] = '$this->debugtiming:'; 4320 foreach ($this->debugtiming as $timestamp => $timingstring) { 4321 $DebugOutput[] = ' * '.$timestamp.' '.$timingstring; 4322 } 4323 $DebugOutput[] = ' * Total processing time: '.number_format(max(array_keys($this->debugtiming)) - min(array_keys($this->debugtiming)), 6); 4324 4325 $this->f = (isset($_GET['f']) ? $_GET['f'] : $this->f); // debug modes 0-2 don't recognize text mode otherwise 4326 return $this->ErrorImage(implode("\n", $DebugOutput), 700, 500, true); 4327 } 4328 4329 public function FatalError($text) { 4330 if (null === $this->fatalerror) { 4331 $this->fatalerror = $text; 4332 } 4333 return true; 4334 } 4335 4336 public function ErrorImage($text, $width=0, $height=0, $forcedisplay=false) { 4337 $width = ($width ? $width : $this->config_error_image_width); 4338 $height = ($height ? $height : $this->config_error_image_height); 4339 4340 $text = 'phpThumb() v'.$this->phpthumb_version."\n".'http://phpthumb.sourceforge.net'."\n\n".($this->config_disable_debug ? 'Error messages disabled.'."\n\n".'edit phpThumb.config.php and (temporarily) set'."\n".'$PHPTHUMB_CONFIG[\'disable_debug\'] = false;'."\n".'to view the details of this error' : $text); 4341 4342 $this->FatalError($text); 4343 $this->DebugMessage($text, __FILE__, __LINE__); 4344 $this->purgeTempFiles(); 4345 if ($this->config_error_silent_die_on_error) { 4346 exit; 4347 } 4348 if ($this->phpThumbDebug && !$forcedisplay) { 4349 return false; 4350 } 4351 if (!$this->config_error_die_on_error && !$forcedisplay) { 4352 return false; 4353 } 4354 if ($this->err || $this->config_error_message_image_default) { 4355 // Show generic custom error image instead of error message 4356 // for use on production sites where you don't want debug messages 4357 if (($this->err == 'showerror') || $this->phpThumbDebug) { 4358 // fall through and actually show error message even if default error image is set 4359 } else { 4360 header('Location: '.($this->err ? $this->err : $this->config_error_message_image_default)); 4361 exit; 4362 } 4363 } 4364 $this->setOutputFormat(); 4365 if (!$this->thumbnailFormat || !$this->config_disable_debug || (phpthumb_functions::gd_version() < 1)) { 4366 $this->thumbnailFormat = 'text'; 4367 } 4368 if (@$this->thumbnailFormat == 'text') { 4369 // bypass all GD functions and output text error message 4370 if (!headers_sent()) { 4371 header('Content-type: text/plain'); 4372 echo $text; 4373 } else { 4374 echo '<pre>'.htmlspecialchars($text).'</pre>'; 4375 } 4376 exit; 4377 } 4378 4379 $FontWidth = imagefontwidth($this->config_error_fontsize); 4380 $FontHeight = imagefontheight($this->config_error_fontsize); 4381 4382 $LinesOfText = explode("\n", @wordwrap($text, floor($width / $FontWidth), "\n", true)); 4383 $height = max($height, count($LinesOfText) * $FontHeight); 4384 4385 $headers_file = ''; 4386 $headers_line = ''; 4387 if (phpthumb_functions::version_compare_replacement(PHP_VERSION, '4.3.0', '>=') && headers_sent($headers_file, $headers_line)) { 4388 4389 echo "\n".'**Headers already sent in file "'.$headers_file.'" on line "'.$headers_line.'", dumping error message as text:**<br><pre>'."\n\n".$text."\n".'</pre>'; 4390 4391 } elseif (headers_sent()) { 4392 4393 echo "\n".'**Headers already sent, dumping error message as text:**<br><pre>'."\n\n".$text."\n".'</pre>'; 4394 4395 } elseif ($gdimg_error = imagecreate($width, $height)) { 4396 4397 $background_color = phpthumb_functions::ImageHexColorAllocate($gdimg_error, $this->config_error_bgcolor, true); 4398 $text_color = phpthumb_functions::ImageHexColorAllocate($gdimg_error, $this->config_error_textcolor, true); 4399 imagefilledrectangle($gdimg_error, 0, 0, $width, $height, $background_color); 4400 $lineYoffset = 0; 4401 foreach ($LinesOfText as $line) { 4402 imagestring($gdimg_error, $this->config_error_fontsize, 2, $lineYoffset, $line, $text_color); 4403 $lineYoffset += $FontHeight; 4404 } 4405 if (function_exists('imagetypes')) { 4406 $imagetypes = imagetypes(); 4407 if ($imagetypes & IMG_PNG) { 4408 header('Content-Type: image/png'); 4409 imagepng($gdimg_error); 4410 } elseif ($imagetypes & IMG_GIF) { 4411 header('Content-Type: image/gif'); 4412 imagegif($gdimg_error); 4413 } elseif ($imagetypes & IMG_JPG) { 4414 header('Content-Type: image/jpeg'); 4415 imagejpeg($gdimg_error); 4416 } elseif ($imagetypes & IMG_WBMP) { 4417 header('Content-Type: image/vnd.wap.wbmp'); 4418 imagewbmp($gdimg_error); 4419 } 4420 } 4421 imagedestroy($gdimg_error); 4422 4423 } 4424 if (!headers_sent()) { 4425 echo "\n".'**Failed to send graphical error image, dumping error message as text:**<br>'."\n\n".$text; 4426 } 4427 exit; 4428 } 4429 4430 public function ImageCreateFromStringReplacement(&$RawImageData, $DieOnErrors=false) { 4431 // there are serious bugs in the non-bundled versions of GD which may cause 4432 // PHP to segfault when calling imagecreatefromstring() - avoid if at all possible 4433 // when not using a bundled version of GD2 4434 if (!phpthumb_functions::gd_version()) { 4435 if ($DieOnErrors) { 4436 if (!headers_sent()) { 4437 // base64-encoded error image in GIF format 4438 $ERROR_NOGD = 'R0lGODlhIAAgALMAAAAAABQUFCQkJDY2NkZGRldXV2ZmZnJycoaGhpSUlKWlpbe3t8XFxdXV1eTk5P7+/iwAAAAAIAAgAAAE/vDJSau9WILtTAACUinDNijZtAHfCojS4W5H+qxD8xibIDE9h0OwWaRWDIljJSkUJYsN4bihMB8th3IToAKs1VtYM75cyV8sZ8vygtOE5yMKmGbO4jRdICQCjHdlZzwzNW4qZSQmKDaNjhUMBX4BBAlmMywFSRWEmAI6b5gAlhNxokGhooAIK5o/pi9vEw4Lfj4OLTAUpj6IabMtCwlSFw0DCKBoFqwAB04AjI54PyZ+yY3TD0ss2YcVmN/gvpcu4TOyFivWqYJlbAHPpOntvxNAACcmGHjZzAZqzSzcq5fNjxFmAFw9iFRunD1epU6tsIPmFCAJnWYE0FURk7wJDA0MTKpEzoWAAskiAAA7'; 4439 header('Content-Type: image/gif'); 4440 echo base64_decode($ERROR_NOGD); 4441 } else { 4442 echo '*** ERROR: No PHP-GD support available ***'; 4443 } 4444 exit; 4445 } else { 4446 $this->DebugMessage('ImageCreateFromStringReplacement() failed: gd_version says "'.phpthumb_functions::gd_version().'"', __FILE__, __LINE__); 4447 return false; 4448 } 4449 } 4450 if (phpthumb_functions::gd_is_bundled()) { 4451 $this->DebugMessage('ImageCreateFromStringReplacement() calling built-in imagecreatefromstring()', __FILE__, __LINE__); 4452 return @imagecreatefromstring($RawImageData); 4453 } 4454 if ($this->issafemode) { 4455 $this->DebugMessage('ImageCreateFromStringReplacement() failed: cannot create temp file in SAFE_MODE', __FILE__, __LINE__); 4456 return false; 4457 } 4458 4459 switch (substr($RawImageData, 0, 3)) { 4460 case 'GIF': 4461 $ICFSreplacementFunctionName = 'imagecreatefromgif'; 4462 break; 4463 case "\xFF\xD8\xFF": 4464 $ICFSreplacementFunctionName = 'imagecreatefromjpeg'; 4465 break; 4466 case "\x89".'PN': 4467 $ICFSreplacementFunctionName = 'imagecreatefrompng'; 4468 break; 4469 default: 4470 $this->DebugMessage('ImageCreateFromStringReplacement() failed: unknown fileformat signature "'.phpthumb_functions::HexCharDisplay(substr($RawImageData, 0, 3)).'"', __FILE__, __LINE__); 4471 return false; 4472 break; 4473 } 4474 $ErrorMessage = ''; 4475 if ($tempnam = $this->phpThumb_tempnam()) { 4476 if ($fp_tempnam = @fopen($tempnam, 'wb')) { 4477 fwrite($fp_tempnam, $RawImageData); 4478 fclose($fp_tempnam); 4479 @chmod($tempnam, $this->getParameter('config_file_create_mask')); 4480 if (($ICFSreplacementFunctionName == 'imagecreatefromgif') && !function_exists($ICFSreplacementFunctionName)) { 4481 4482 // Need to create from GIF file, but imagecreatefromgif does not exist 4483 ob_start(); 4484 if (!@include_once __DIR__ .'/phpthumb.gif.php' ) { 4485 $ErrorMessage = 'Failed to include required file "'. __DIR__ .'/phpthumb.gif.php" in '.__FILE__.' on line '.__LINE__; 4486 $this->DebugMessage($ErrorMessage, __FILE__, __LINE__); 4487 } 4488 ob_end_clean(); 4489 // gif_loadFileToGDimageResource() cannot read from raw data, write to file first 4490 if ($tempfilename = $this->phpThumb_tempnam()) { 4491 if ($fp_tempfile = @fopen($tempfilename, 'wb')) { 4492 fwrite($fp_tempfile, $RawImageData); 4493 fclose($fp_tempfile); 4494 $gdimg_source = gif_loadFileToGDimageResource($tempfilename); 4495 $this->DebugMessage('gif_loadFileToGDimageResource('.$tempfilename.') completed', __FILE__, __LINE__); 4496 $this->DebugMessage('deleting "'.$tempfilename.'"', __FILE__, __LINE__); 4497 unlink($tempfilename); 4498 return $gdimg_source; 4499 } else { 4500 $ErrorMessage = 'Failed to open tempfile in '.__FILE__.' on line '.__LINE__; 4501 $this->DebugMessage($ErrorMessage, __FILE__, __LINE__); 4502 } 4503 } else { 4504 $ErrorMessage = 'Failed to open generate tempfile name in '.__FILE__.' on line '.__LINE__; 4505 $this->DebugMessage($ErrorMessage, __FILE__, __LINE__); 4506 } 4507 4508 } elseif (function_exists($ICFSreplacementFunctionName) && ($gdimg_source = @$ICFSreplacementFunctionName($tempnam))) { 4509 4510 // great 4511 $this->DebugMessage($ICFSreplacementFunctionName.'('.$tempnam.') succeeded', __FILE__, __LINE__); 4512 $this->DebugMessage('deleting "'.$tempnam.'"', __FILE__, __LINE__); 4513 unlink($tempnam); 4514 return $gdimg_source; 4515 4516 } else { 4517 4518 // GD functions not available, or failed to create image 4519 $this->DebugMessage($ICFSreplacementFunctionName.'('.$tempnam.') '.(function_exists($ICFSreplacementFunctionName) ? 'failed' : 'does not exist'), __FILE__, __LINE__); 4520 if (isset($_GET['phpThumbDebug'])) { 4521 $this->phpThumbDebug(); 4522 } 4523 4524 } 4525 } else { 4526 $ErrorMessage = 'Failed to fopen('.$tempnam.', "wb") in '.__FILE__.' on line '.__LINE__."\n".'You may need to set $PHPTHUMB_CONFIG[temp_directory] in phpThumb.config.php'; 4527 if ($this->issafemode) { 4528 $ErrorMessage = 'ImageCreateFromStringReplacement() failed in '.__FILE__.' on line '.__LINE__.': cannot create temp file in SAFE_MODE'; 4529 } 4530 $this->DebugMessage($ErrorMessage, __FILE__, __LINE__); 4531 } 4532 $this->DebugMessage('deleting "'.$tempnam.'"', __FILE__, __LINE__); 4533 @unlink($tempnam); 4534 } else { 4535 $ErrorMessage = 'Failed to generate phpThumb_tempnam() in '.__FILE__.' on line '.__LINE__."\n".'You may need to set $PHPTHUMB_CONFIG[temp_directory] in phpThumb.config.php'; 4536 if ($this->issafemode) { 4537 $ErrorMessage = 'ImageCreateFromStringReplacement() failed in '.__FILE__.' on line '.__LINE__.': cannot create temp file in SAFE_MODE'; 4538 } 4539 } 4540 if ($DieOnErrors && $ErrorMessage) { 4541 return $this->ErrorImage($ErrorMessage); 4542 } 4543 return false; 4544 } 4545 4546 public function ImageResizeFunction(&$dst_im, &$src_im, $dstX, $dstY, $srcX, $srcY, $dstW, $dstH, $srcW, $srcH) { 4547 $dstX = (int) round($dstX); 4548 $dstY = (int) round($dstY); 4549 $srcX = (int) round($srcX); 4550 $srcY = (int) round($srcY); 4551 $dstW = (int) round($dstW); 4552 $dstH = (int) round($dstH); 4553 $srcW = (int) round($srcW); 4554 $srcH = (int) round($srcH); 4555 4556 $this->DebugMessage('ImageResizeFunction($o, $s, '.$dstX.', '.$dstY.', '.$srcX.', '.$srcY.', '.$dstW.', '.$dstH.', '.$srcW.', '.$srcH.')', __FILE__, __LINE__); 4557 if (($dstW == $srcW) && ($dstH == $srcH)) { 4558 return imagecopy($dst_im, $src_im, $dstX, $dstY, $srcX, $srcY, $srcW, $srcH); 4559 } 4560 if (phpthumb_functions::gd_version() >= 2.0) { 4561 if ($this->config_disable_imagecopyresampled) { 4562 return phpthumb_functions::ImageCopyResampleBicubic($dst_im, $src_im, $dstX, $dstY, $srcX, $srcY, $dstW, $dstH, $srcW, $srcH); 4563 } 4564 return imagecopyresampled($dst_im, $src_im, $dstX, $dstY, $srcX, $srcY, $dstW, $dstH, $srcW, $srcH); 4565 } 4566 return imagecopyresized($dst_im, $src_im, $dstX, $dstY, $srcX, $srcY, $dstW, $dstH, $srcW, $srcH); 4567 } 4568 4569 public function InitializeTempDirSetting() { 4570 $this->config_temp_directory = ($this->config_temp_directory ? $this->config_temp_directory : $this->realPathSafe((function_exists('sys_get_temp_dir') ? sys_get_temp_dir() : ''))); // sys_get_temp_dir added in PHP v5.2.1 4571 $this->config_temp_directory = ($this->config_temp_directory ? $this->config_temp_directory : $this->realPathSafe(ini_get('upload_tmp_dir'))); 4572 $this->config_temp_directory = ($this->config_temp_directory ? $this->config_temp_directory : $this->realPathSafe(getenv('TMPDIR'))); 4573 $this->config_temp_directory = ($this->config_temp_directory ? $this->config_temp_directory : $this->realPathSafe(getenv('TMP'))); 4574 return true; 4575 } 4576 4577 public function phpThumb_tempnam() { 4578 $this->InitializeTempDirSetting(); 4579 $tempnam = $this->realPathSafe(tempnam($this->config_temp_directory, 'pThumb')); 4580 $this->tempFilesToDelete[$tempnam] = $tempnam; 4581 $this->DebugMessage('phpThumb_tempnam() returning "'.$tempnam.'"', __FILE__, __LINE__); 4582 return $tempnam; 4583 } 4584 4585 public function DebugMessage($message, $file='', $line='') { 4586 $this->debugmessages[] = $message.($file ? ' in file "'.(basename($file) ? basename($file) : $file).'"' : '').($line ? ' on line '.$line : ''); 4587 return true; 4588 } 4589 4590 public function DebugTimingMessage($message, $file='', $line='', $timestamp=0) { 4591 if (!$timestamp) { 4592 $timestamp = array_sum(explode(' ', microtime())); 4593 } 4594 $this->debugtiming[number_format($timestamp, 6, '.', '')] = ': '.$message.($file ? ' in file "'.(basename($file) ? basename($file) : $file).'"' : '').($line ? ' on line '.$line : ''); 4595 return true; 4596 } 4597 4598} 4599