730 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			730 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| namespace ShortPixel\Controller;
 | |
| 
 | |
| if ( ! defined( 'ABSPATH' ) ) {
 | |
|  exit; // Exit if accessed directly.
 | |
| }
 | |
| 
 | |
| use ShortPixel\ShortPixelLogger\ShortPixelLogger as Log;
 | |
| 
 | |
| class ApiController
 | |
| {
 | |
|     const STATUS_ENQUEUED = 10;
 | |
| 		const STATUS_PARTIAL_SUCCESS = 3;
 | |
| 		const STATUS_SUCCESS = 2;
 | |
| 		const STATUS_WAITING = 1;
 | |
|     const STATUS_UNCHANGED = 0;
 | |
|     const STATUS_ERROR = -1;
 | |
|     const STATUS_FAIL = -2;
 | |
|     const STATUS_QUOTA_EXCEEDED = -3;
 | |
|     const STATUS_SKIP = -4;
 | |
|     const STATUS_NOT_FOUND = -5;
 | |
|     const STATUS_NO_KEY = -6;
 | |
|    // const STATUS_RETRY = -7;
 | |
|    // const STATUS_SEARCHING = -8; // when the Queue is looping over images, but in batch none were   found.
 | |
| 	 const STATUS_OPTIMIZED_BIGGER = -9;
 | |
| 	 const STATUS_CONVERTED = -10;
 | |
| 
 | |
|     const STATUS_QUEUE_FULL = -404;
 | |
|     const STATUS_MAINTENANCE = -500;
 | |
| 		const STATUS_CONNECTION_ERROR = -503; // Not official, error connection in WP.
 | |
|     const STATUS_NOT_API = -1000; // Not an API process, i.e restore / migrate. Don't handle as optimized
 | |
| 
 | |
| 		// Moved these numbers higher to prevent conflict with STATUS
 | |
|     const ERR_FILE_NOT_FOUND = -902;
 | |
|     const ERR_TIMEOUT = -903;
 | |
|     const ERR_SAVE = -904;
 | |
|     const ERR_SAVE_BKP = -905;
 | |
|     const ERR_INCORRECT_FILE_SIZE = -906;
 | |
|     const ERR_DOWNLOAD = -907;
 | |
|     const ERR_PNG2JPG_MEMORY = -908;
 | |
|     const ERR_POSTMETA_CORRUPT = -909;
 | |
|     const ERR_UNKNOWN = -999;
 | |
| 
 | |
| 
 | |
|     const DOWNLOAD_ARCHIVE = 7;
 | |
| 
 | |
|     private static $instance;
 | |
| 
 | |
|     private $apiEndPoint;
 | |
|     private $apiDumpEndPoint;
 | |
| 
 | |
|     protected static $temporaryFiles = array();
 | |
|     protected static $temporaryDirs = array();
 | |
| 
 | |
|     public function __construct()
 | |
|     {
 | |
|       $settings = \wpSPIO()->settings();
 | |
|       $this->apiEndPoint = $settings->httpProto . '://' . SHORTPIXEL_API . '/v2/reducer.php';
 | |
|       $this->apiDumpEndPoint = $settings->httpProto . '://' . SHORTPIXEL_API . '/v2/cleanup.php';
 | |
|     }
 | |
| 
 | |
|   public static function getInstance()
 | |
|   {
 | |
|      if (is_null(self::$instance))
 | |
|        self::$instance = new ApiController();
 | |
| 
 | |
|       return self::$instance;
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|   * @param Object $item Item of stdClass
 | |
|   * @return Returns same Item with Result of request
 | |
|   */
 | |
|   public function processMediaItem($item, $imageObj)
 | |
|   {
 | |
| 		 	if (! is_object($imageObj))
 | |
| 			{
 | |
| 				$item->result = $this->returnFailure(self::STATUS_FAIL, __('Item seems invalid, removed or corrupted.', 'shortpixel-image-optimiser'));
 | |
| 				return $item;
 | |
| 			}
 | |
| 		 	elseif (false === $imageObj->isProcessable() || $imageObj->isOptimizePrevented() == true)
 | |
| 			{
 | |
| 					if ($imageObj->isOptimized()) // This only looks at main item
 | |
| 					{
 | |
| 						 $item->result = $this->returnFailure(self::STATUS_FAIL, __('Item is already optimized', 'shortpixel-image-optimiser'));
 | |
| 						 return $item;
 | |
| 					}
 | |
| 					else {
 | |
| 						 $item->result = $this->returnFailure(self::STATUS_FAIL, __('Item is not processable and not optimized', 'shortpixel-image-optimiser'));
 | |
| 						 return $item;
 | |
| 					}
 | |
| 			}
 | |
| 
 | |
|       if (! is_array($item->urls) || count($item->urls) == 0)
 | |
|       {
 | |
|           $item->result = $this->returnFailure(self::STATUS_FAIL, __('No Urls given for this Item', 'shortpixel-image-optimiser'));
 | |
|           return $item;
 | |
|       }
 | |
| 			else { // if ok, urlencode them.
 | |
| 					$list = array();
 | |
| 				  foreach($item->urls as $url)
 | |
| 					{
 | |
| 							$parsed_url = parse_url($url);
 | |
| 							if (false !== $parsed_url)
 | |
| 							{
 | |
| 									//$url = $this->encodeURL($parsed_url, $url);
 | |
| 							}
 | |
| 							$list[] = $url;
 | |
| 					}
 | |
| 					$item->urls = $list;
 | |
| 			}
 | |
| 
 | |
|       $requestArgs = array('urls' => $item->urls); // obligatory
 | |
|       if (property_exists($item, 'compressionType'))
 | |
|         $requestArgs['compressionType'] = $item->compressionType;
 | |
|       $requestArgs['blocking'] =  ($item->tries == 0) ? false : true;
 | |
|       $requestArgs['item_id'] = $item->item_id;
 | |
|       $requestArgs['refresh'] = (property_exists($item, 'refresh') && $item->refresh) || $item->tries == 0 ? true : false;
 | |
|       $requestArgs['flags'] = (property_exists($item, 'flags')) ? $item->flags : array();
 | |
| 
 | |
| 			$requestArgs['paramlist']  = property_exists($item, 'paramlist') ? $item->paramlist : null;
 | |
| 			$requestArgs['returndatalist']  = property_exists($item, 'returndatalist') ? $item->returndatalist : null;
 | |
| 
 | |
|       $request = $this->getRequest($requestArgs);
 | |
|       $item = $this->doRequest($item, $request);
 | |
| 
 | |
| 			ResponseController::addData($item->item_id, 'images_total', count($item->urls));
 | |
| 
 | |
| 			// If error has occured, but it's not related to connection.
 | |
| 			if ($item->result->is_error === true && $item->result->is_done === true)
 | |
| 			{
 | |
| 				 $this->dumpMediaItem($item); // item failed, directly dump anything from server.
 | |
| 			}
 | |
| 
 | |
|       return $item;
 | |
|   }
 | |
| 
 | |
| 	/* Ask to remove the items from the remote cache.
 | |
| 	  @param $item Must be object, with URLS set as array of urllist. - Secretly not a mediaItem - shame
 | |
| 	*/
 | |
| 	public function dumpMediaItem($item)
 | |
| 	{
 | |
|      $settings = \wpSPIO()->settings();
 | |
|      $keyControl = ApiKeyController::getInstance();
 | |
| 
 | |
| 		 if (property_exists($item, 'urls') === false || ! is_array($item->urls) || count($item->urls) == 0)
 | |
| 		 {
 | |
| 			  Log::addWarn('Media Item without URLS cannnot be dumped ', $item);
 | |
| 				return false;
 | |
| 		 }
 | |
| 
 | |
| 		 $request = $this->getRequest();
 | |
| 
 | |
| 		 $request['body'] = json_encode(
 | |
| 			 			array(
 | |
|                 'plugin_version' => SHORTPIXEL_IMAGE_OPTIMISER_VERSION,
 | |
|                 'key' => $keyControl->forceGetApiKey(),
 | |
|                 'urllist' => $item->urls	)	, JSON_UNESCAPED_UNICODE);
 | |
| 
 | |
| 		 Log::addDebug('Dumping Media Item ', $item->urls);
 | |
| 
 | |
| 		 $ret = wp_remote_post($this->apiDumpEndPoint, $request);
 | |
| 
 | |
|      return $ret;
 | |
| 
 | |
| 	}
 | |
| 
 | |
|   /** Former, prepare Request in API */
 | |
|   private function getRequest($args = array())
 | |
|   {
 | |
|     $settings = \wpSPIO()->settings();
 | |
|     $keyControl = ApiKeyController::getInstance();
 | |
| 
 | |
|     $defaults = array(
 | |
|           'urls' => null,
 | |
| 					'paramlist' => null,
 | |
| 					'returndatalist' => null,
 | |
|           'compressionType' => $settings->compressionType,
 | |
|           'blocking' => true,
 | |
|           'item_id' => null,
 | |
|           'refresh' => false,
 | |
|           'flags' => array(),
 | |
|     );
 | |
| 
 | |
|     $args = wp_parse_args($args, $defaults);
 | |
|     $convertTo = implode("|", $args['flags']);
 | |
| 
 | |
|     $requestParameters = array(
 | |
|         'plugin_version' => SHORTPIXEL_IMAGE_OPTIMISER_VERSION,
 | |
|         'key' => $keyControl->forceGetApiKey(),
 | |
|         'lossy' => $args['compressionType'],
 | |
|         'cmyk2rgb' => $settings->CMYKtoRGBconversion,
 | |
|         'keep_exif' => ($settings->keepExif ? "1" : "0"),
 | |
|         'convertto' => $convertTo,
 | |
|         'resize' => $settings->resizeImages ? 1 + 2 * ($settings->resizeType == 'inner' ? 1 : 0) : 0,
 | |
|         'resize_width' => $settings->resizeWidth,
 | |
|         'resize_height' => $settings->resizeHeight,
 | |
|         'urllist' => $args['urls'],
 | |
|     );
 | |
| 
 | |
| 		if (! is_null($args['paramlist']))
 | |
| 		{
 | |
| 			 $requestParameters['paramlist'] = $args['paramlist'];
 | |
| 		}
 | |
| 
 | |
| 		if (! is_null($args['returndatalist']))
 | |
| 		{
 | |
| 			 $requestParameters['returndatalist'] = $args['returndatalist'];
 | |
| 		}
 | |
| 
 | |
|     if($args['refresh']) { // @todo if previous status was ShortPixelAPI::ERR_INCORRECT_FILE_SIZE; then refresh.
 | |
|         $requestParameters['refresh'] = 1;
 | |
|     }
 | |
| 
 | |
| 		$requestParameters = apply_filters('shortpixel/api/request', $requestParameters, $args['item_id']);
 | |
| 
 | |
|     $arguments = array(
 | |
|         'method' => 'POST',
 | |
|         'timeout' => 15,
 | |
|         'redirection' => 3,
 | |
|         'sslverify' => apply_filters('shortpixel/system/sslverify', true),
 | |
|         'httpversion' => '1.0',
 | |
|         'blocking' => $args['blocking'],
 | |
|         'headers' => array(),
 | |
|         'body' => json_encode($requestParameters, JSON_UNESCAPED_UNICODE),
 | |
|         'cookies' => array()
 | |
|     );
 | |
|     //add this explicitely only for https, otherwise (for http) it slows down the request
 | |
|     if($settings->httpProto !== 'https') {
 | |
|         unset($arguments['sslverify']);
 | |
|     }
 | |
| 
 | |
|     return $arguments;
 | |
|   }
 | |
| 
 | |
| 
 | |
| 	/** DoRequest : Does a remote_post to the API
 | |
| 	*
 | |
| 	* @param Object $item  The QueueItemObject
 | |
| 	* @param Array $requestParameters  The HTTP parameters for the remote post (arguments in getRequest)
 | |
| 	*/
 | |
|   protected function doRequest($item, $requestParameters )
 | |
|   {
 | |
|     $response = wp_remote_post($this->apiEndPoint, $requestParameters );
 | |
|     Log::addDebug('ShortPixel API Request sent', $requestParameters['body']);
 | |
| 
 | |
|     //only if $Blocking is true analyze the response
 | |
|     if ( $requestParameters['blocking'] )
 | |
|     {
 | |
|         if ( is_object($response) && get_class($response) == 'WP_Error' )
 | |
|         {
 | |
|             $errorMessage = $response->errors['http_request_failed'][0];
 | |
|             $errorCode = self::STATUS_CONNECTION_ERROR;
 | |
|             $item->result = $this->returnRetry($errorCode, $errorMessage);
 | |
|         }
 | |
|         elseif ( isset($response['response']['code']) && $response['response']['code'] <> 200 )
 | |
|         {
 | |
|             $errorMessage = $response['response']['code'] . " - " . $response['response']['message'];
 | |
|             $errorCode = $response['response']['code'];
 | |
|             $item->result = $this->returnFailure($errorCode, $errorMessage);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|            $item->result = $this->handleResponse($item, $response);
 | |
|         }
 | |
| 
 | |
|     }
 | |
|     else // This should be only non-blocking the FIRST time it's send off.
 | |
|     {
 | |
| 			 if ($item->tries > 0)
 | |
| 			 {
 | |
| 			 		Log::addWarn('DOREQUEST sent item non-blocking with multiple tries!', $item);
 | |
| 			 }
 | |
| 
 | |
| 			 $urls = count($item->urls);
 | |
| 			 $flags = property_exists($item, 'flags') ? $item->flags : array();
 | |
| 			 $flags = implode("|", $flags);
 | |
| 			 $text = sprintf(__('New item #%d sent for processing ( %d URLS %s)  ', 'shortpixel-image-optimiser'), $item->item_id, $urls, $flags );
 | |
| 
 | |
|        $item->result = $this->returnOK(self::STATUS_ENQUEUED, $text );
 | |
|     }
 | |
| 
 | |
|     return $item;
 | |
|   }
 | |
| 
 | |
| 	/**
 | |
| 	* @param $parsed_url Array  Result of parse_url
 | |
| 	*/
 | |
| 	private function encodeURL($parsed_url, $url)
 | |
| 	{
 | |
| 		//str_replace($parsed_url['path'], urlencode($parsed_url['path']), $url);
 | |
| 		$path = $parsed_url['path'];
 | |
| 		//echo strrpos($parsed_url, ',');
 | |
| 		$filename = substr($path, strrpos($path, '/') + 1); //strrpos($path, '/');
 | |
| 
 | |
| 		$path = str_replace($filename, urlencode($filename), $url);
 | |
| 
 | |
| 		return $path;
 | |
| 
 | |
| 	}
 | |
| 
 | |
|   private function parseResponse($response)
 | |
|   {
 | |
|     $data = $response['body'];
 | |
| 
 | |
|     $data = json_decode($data);
 | |
|     return (array)$data;
 | |
|   }
 | |
| 
 | |
| 	/**
 | |
| 	*
 | |
| 	**/
 | |
|   private function handleResponse($item, $response)
 | |
|   {
 | |
| 
 | |
|     $APIresponse = $this->parseResponse($response);//get the actual response from API, its an array
 | |
|     $settings = \wpSPIO()->settings();
 | |
| 
 | |
| 		// Don't know if it's this or that.
 | |
| 		$status = false;
 | |
| 		if (isset($APIresponse['Status']))
 | |
| 		{
 | |
| 			$status = $APIresponse['Status'];
 | |
| 		}
 | |
| 		elseif(is_array($APIresponse) && isset($APIresponse[0]) && property_exists($APIresponse[0], 'Status'))
 | |
| 		{
 | |
| 			$status = $APIresponse[0]->Status;
 | |
| 		}
 | |
| 		elseif ( is_array($APIresponse)) // This is a workaround for some obscure PHP 5.6 bug. @todo Remove when dropping support PHP < 7.
 | |
| 		{
 | |
| 			 foreach($APIresponse as $key => $data)
 | |
| 			 {
 | |
| 				 // Running the whole array, because handleSuccess enums on key index as well :/
 | |
| 				 // we are not just looking for status here, but also replacing the whole array, because of obscure bug.
 | |
| 				  if (property_exists($data, 'Status'))
 | |
| 					{
 | |
| 						 if ($status === false)
 | |
| 						 {
 | |
| 						 	$status = $data->Status;
 | |
| 						 }
 | |
| 						 $APIresponse[$key] = $data; // reset it, so it can read the index.  This should be 0.
 | |
| 					}
 | |
| 			 }
 | |
| 		}
 | |
| 
 | |
| 			if (isset($APIresponse['returndatalist']))
 | |
| 			{
 | |
| 				$returnDataList = (array) $APIresponse['returndatalist'];
 | |
| 				if (isset($returnDataList['sizes']) && is_object($returnDataList['sizes']))
 | |
| 					$returnDataList['sizes'] = (array) $returnDataList['sizes'];
 | |
| 
 | |
| 				if (isset($returnDataList['doubles']) && is_object($returnDataList['doubles']))
 | |
| 						$returnDataList['doubles'] = (array) $returnDataList['doubles'];
 | |
| 
 | |
| 				if (isset($returnDataList['duplicates']) && is_object($returnDataList['duplicates']))
 | |
| 							$returnDataList['duplicates'] = (array) $returnDataList['duplicates'];
 | |
| 
 | |
| 				if (isset($returnDataList['fileSizes']) && is_object($returnDataList['fileSizes']))
 | |
| 										$returnDataList['fileSizes'] = (array) $returnDataList['fileSizes'];
 | |
| 
 | |
| 				unset($APIresponse['returndatalist']);
 | |
| 			}
 | |
| 			else {
 | |
| 				$returnDataList = array();
 | |
| 			}
 | |
| 
 | |
|     // This is only set if something is up, otherwise, ApiResponse returns array
 | |
|     if (is_object($status))
 | |
|     {
 | |
|         // Check for known errors. : https://shortpixel.com/api-docs
 | |
| 				Log::addDebug('Api Response Status :' . $status->Code  );
 | |
|         switch($status->Code)
 | |
|         {
 | |
|               case -102: // Invalid URL
 | |
|               case -105: // URL missing
 | |
|               case -106: // Url is inaccessible
 | |
|               case -113: // Too many inaccessible URLs
 | |
|               case -201: // Invalid image format
 | |
|               case -202: // Invalid image or unsupported format
 | |
|               case -203: // Could not download file
 | |
|                  return $this->returnFailure( self::STATUS_ERROR, $status->Message);
 | |
|               break;
 | |
|               case -403: // Quota Exceeded
 | |
|               case -301: // The file is larger than remaining quota
 | |
| 									// legacy
 | |
|                   @delete_option('bulkProcessingStatus');
 | |
| 									QuotaController::getInstance()->setQuotaExceeded();
 | |
| 
 | |
|                   return $this->returnRetry( self::STATUS_QUOTA_EXCEEDED, __('Quota exceeded.','shortpixel-image-optimiser'));
 | |
|                   break;
 | |
| 							case -306:
 | |
| 									return $this->returnFailure( self::STATUS_FAIL, __('Files need to be from a single domain per request.', 'shortpixel-image-optimiser'));
 | |
| 							break;
 | |
|               case -401: // Invalid Api Key
 | |
| 							case -402: // Wrong API key
 | |
|                   return $this->returnFailure( self::STATUS_NO_KEY, $status->Message);
 | |
|               break;
 | |
|               case -404: // Maximum number in optimization queue (remote)
 | |
|                   //return array("Status" => self::STATUS_QUEUE_FULL, "Message" => $APIresponse['Status']->Message);
 | |
|                   return $this->returnRetry( self::STATUS_QUEUE_FULL, $status->Message);
 | |
|               case -500: // API in maintenance.
 | |
|                   //return array("Status" => self::STATUS_MAINTENANCE, "Message" => $APIresponse['Status']->Message);
 | |
|                   return $this->returnRetry( self::STATUS_MAINTENANCE, $status->Message);
 | |
|           }
 | |
|     }
 | |
| 
 | |
|     $neededURLS = $item->urls; // URLS we are waiting for.
 | |
| 
 | |
|     if ( is_array($APIresponse) && isset($APIresponse[0]) ) //API returned image details
 | |
|     {
 | |
| 
 | |
| 				if (! isset($returnDataList['sizes']))
 | |
| 				{
 | |
| 					   return $this->returnFailure(self::STATUS_FAIL,  __('Item did not return image size information. This might be a failed queue item. Reset the queue if this persists or contact support','shortpixel-image-optimiser'));
 | |
| 				}
 | |
| 			//        return $this->returnFailure(self::STATUS_FAIL,  __('Unrecognized API response. Please contact support.','shortpixel-image-optimiser'));
 | |
| 
 | |
| 				$analyze = array('total' => count($item->urls), 'ready' => 0, 'waiting' => 0);
 | |
| 				$waitingDebug = array();
 | |
| 
 | |
| 				$imageList = array();
 | |
| 				$partialSuccess = false;
 | |
| 				$imageNames = array_keys($returnDataList['sizes']);
 | |
| 				$fileNames = array_values($returnDataList['sizes']);
 | |
| 
 | |
| 				foreach($APIresponse as $index => $imageObject)
 | |
| 				{
 | |
| 						if (! property_exists($imageObject, 'Status'))
 | |
| 						{
 | |
| 							Log::addWarn('Result without Status', $imageObject);
 | |
| 							continue; // can't do nothing with that, probably not an image.
 | |
| 						}
 | |
| 						elseif ($imageObject->Status->Code == self::STATUS_UNCHANGED || $imageObject->Status->Code == self::STATUS_WAITING)
 | |
| 						{
 | |
| 							 $analyze['waiting']++;
 | |
| 							 $partialSuccess = true; // Not the whole job has been done.
 | |
| 						}
 | |
| 						elseif ($imageObject->Status->Code == self::STATUS_SUCCESS)
 | |
| 						{
 | |
| 							 $analyze['ready']++;
 | |
| 							 $imageName = $imageNames[$index];
 | |
| 							 $fileName = $fileNames[$index];
 | |
| 							 $data = array(
 | |
| 								 'fileName' => $fileName,
 | |
| 								 'imageName' => $imageName,
 | |
| 							 );
 | |
| 
 | |
|                // Filesize might not be present, but also imageName ( only if smartcrop is done, might differ per image)
 | |
| 							 if (isset($returnDataList['fileSizes']) && isset($returnDataList['fileSizes'][$imageName]))
 | |
| 							 {
 | |
| 								  $data['fileSize'] = $returnDataList['fileSizes'][$imageName];
 | |
| 							 }
 | |
| 
 | |
| 							 if (! isset($item->files[$imageName]))
 | |
| 							 {
 | |
| 							 	  $imageList[$imageName] = $this->handleNewSuccess($item, $imageObject, $data);
 | |
| 							 }
 | |
| 							 else {
 | |
| 							 }
 | |
| 						}
 | |
| 
 | |
| 				}
 | |
| 
 | |
| 				$imageData = array(
 | |
| 						'images_done' => $analyze['ready'],
 | |
| 						'images_waiting' => $analyze['waiting'],
 | |
| 						'images_total' => $analyze['total']
 | |
| 				);
 | |
| 				ResponseController::addData($item->item_id, $imageData);
 | |
| 
 | |
| 				if (count($imageList) > 0)
 | |
| 				{
 | |
| 						$data = array(
 | |
| 							'files' => $imageList,
 | |
| 							'data' => $returnDataList,
 | |
| 						);
 | |
| 					  if (false === $partialSuccess)
 | |
| 						{
 | |
| 							return $this->returnSuccess($data, self::STATUS_SUCCESS, false);
 | |
| 						}
 | |
| 						else {
 | |
| 							return $this->returnSuccess($data, self::STATUS_PARTIAL_SUCCESS, false);
 | |
| 						}
 | |
| 				}
 | |
| 				elseif ($analyze['waiting'] > 0) {
 | |
| 					return $this->returnOK(self::STATUS_UNCHANGED, sprintf(__('Item is waiting', 'shortpixel-image-optimiser')));
 | |
| 				}
 | |
| 				else {
 | |
| 					// Theoretically this should not be needed.
 | |
| 					Log::addWarn('ApiController Response not handled before default case');
 | |
| 					if ( isset($APIresponse[0]->Status->Message) ) {
 | |
| 
 | |
| 							$err = array("Status" => self::STATUS_FAIL, "Code" => (isset($APIresponse[0]->Status->Code) ? $APIresponse[0]->Status->Code : self::ERR_UNKNOWN),
 | |
| 													 "Message" => __('There was an error and your request was not processed.','shortpixel-image-optimiser')
 | |
| 																				. " (" . wp_basename($APIresponse[0]->OriginalURL) . ": " . $APIresponse[0]->Status->Message . ")");
 | |
| 							return $this->returnRetry($err['Code'], $err['Message']);
 | |
| 					} else {
 | |
| 							$err = array("Status" => self::STATUS_FAIL, "Message" => __('There was an error and your request was not processed.','shortpixel-image-optimiser'),
 | |
| 													 "Code" => (isset($APIresponse[0]->Status->Code) ? $APIresponse[0]->Status->Code : self::ERR_UNKNOWN));
 | |
| 							return $this->returnRetry($err['Code'], $err['Message']);
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
|     } // ApiResponse[0]
 | |
| 
 | |
|     // If this code reaches here, something is wrong.
 | |
|     if(!isset($APIresponse['Status'])) {
 | |
| 
 | |
|         Log::addError('API returned Unknown Status/Response ', $response);
 | |
|         return $this->returnFailure(self::STATUS_FAIL,  __('Unrecognized API response. Please contact support.','shortpixel-image-optimiser'));
 | |
| 
 | |
|     } else {
 | |
| 
 | |
|       //sometimes the response array can be different
 | |
|       if (is_numeric($APIresponse['Status']->Code)) {
 | |
|           $message = $APIresponse['Status']->Message;
 | |
|       } else {
 | |
|           $message = $APIresponse[0]->Status->Message;
 | |
|       }
 | |
| 
 | |
| 			if (! isset($message) || is_null($message) || $message == '')
 | |
| 			{
 | |
| 				 $message = __('Unrecognized API message. Please contact support.','shortpixel-image-optimiser');
 | |
| 			}
 | |
|       return $this->returnRetry(self::STATUS_FAIL, $message);
 | |
|     } // else
 | |
|   }
 | |
|   // handleResponse function
 | |
| 
 | |
| 
 | |
| 	private function handleNewSuccess($item, $fileData, $data)
 | |
| 	{
 | |
| 			$compressionType = property_exists($item, 'compressionType') ? $item->compressionType : $settings->compressionType;
 | |
| 			//$savedSpace =  $originalSpace =  $optimizedSpace = $fileCount  = 0;
 | |
| 
 | |
| 			$defaults = array(
 | |
| 					'fileName' => false,
 | |
| 					'imageName' => false,
 | |
| 					'fileSize' => false,
 | |
| 			);
 | |
| 
 | |
| 			$data = wp_parse_args($data, $defaults);
 | |
| 
 | |
| 			if (false === $data['fileName'] || false === $data['imageName'])
 | |
| 			{
 | |
| 				 Log::addError('Failure! HandleSuccess did not receive filename or imagename! ', $data);
 | |
| 				 Log::addError('Error Item:', $item);
 | |
| 
 | |
| 				 return $this->returnFailure(self::STATUS_FAIL, __('Internal error, missing variables'));
 | |
| 			}
 | |
| 
 | |
| 			$originalFileSize = (false === $data['fileSize']) ? intval($fileData->OriginalSize) : $data['fileSize'];
 | |
| 
 | |
| 		 	$image = array(
 | |
| 				 'image' => array(
 | |
| 				 'url' => false,
 | |
| 				 'originalSize' => $originalFileSize,
 | |
| 				 'optimizedSize' => false,
 | |
| 				 'status' => self::STATUS_SUCCESS,
 | |
| 				  ),
 | |
| 					'webp' => array(
 | |
| 				 'url' => false,
 | |
| 				 'size' => false,
 | |
| 				 'status' => self::STATUS_SKIP,
 | |
| 			 		),
 | |
| 					'avif' => array(
 | |
| 				 'url' => false,
 | |
| 				 'size' => false,
 | |
| 				 'status' => self::STATUS_SKIP,
 | |
| 			 		),
 | |
| 			);
 | |
| 
 | |
| 			$fileType = ($compressionType > 0) ? 'LossyURL' : 'LosslessURL';
 | |
| 			$fileSize = ($compressionType > 0) ? 'LossySize' : 'LosslessSize';
 | |
| 
 | |
|       // if originalURL and OptimizedURL is the same, API is returning it as the same item, aka not optimized.
 | |
|       if ($fileData->$fileType === $fileData->OriginalURL)
 | |
|       {
 | |
|         $image['image']['status'] = self::STATUS_UNCHANGED;
 | |
|       }
 | |
|       else
 | |
|       {
 | |
|         $image['image']['url'] = $fileData->$fileType;
 | |
|   			$image['image']['optimizedSize']  = intval($fileData->$fileSize);
 | |
|       }
 | |
| 
 | |
| 			// Don't download if the originalSize / OptimizedSize is the same ( same image ) . This can be non-opt result or it was not asked to be optimized( webp/avif only job i.e. )
 | |
| 			if ($image['image']['originalSize'] == $image['image']['optimizedSize'])
 | |
| 			{
 | |
| 				$image['image']['status'] = self::STATUS_UNCHANGED;
 | |
| 			}
 | |
| 
 | |
| 			$checkFileSize = intval($fileData->$fileSize); // Size of optimized image to check against Avif/Webp
 | |
| 
 | |
| 			if (false === $this->checkFileSizeMargin($originalFileSize, $checkFileSize))
 | |
| 			{
 | |
| 				 $image['image']['status'] = self::STATUS_OPTIMIZED_BIGGER;
 | |
| 				 $checkFileSize = $originalFileSize;
 | |
| 			}
 | |
| 
 | |
| 			if (property_exists($fileData, "WebP" . $fileType))
 | |
| 			{
 | |
| 				$type = "WebP" . $fileType;
 | |
| 				$size = "WebP" . $fileSize;
 | |
| 
 | |
| 				if ($fileData->$type != 'NA')
 | |
| 				{
 | |
| 						$image['webp']['url'] = $fileData->$type;
 | |
| 						$image['webp']['size'] = $fileData->$size;
 | |
| 						if (false === $this->checkFileSizeMargin($checkFileSize, $fileData->$size))
 | |
| 						{
 | |
| 							$image['webp']['status'] = self::STATUS_OPTIMIZED_BIGGER;
 | |
| 						}
 | |
| 						else {
 | |
| 							$image['webp']['status'] = self::STATUS_SUCCESS;
 | |
| 						}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if (property_exists($fileData, "AVIF" . $fileType))
 | |
| 			{
 | |
| 				$type = "AVIF" . $fileType;
 | |
| 				$size = "AVIF" . $fileSize;
 | |
| 
 | |
| 				if ($fileData->$type != 'NA')
 | |
| 				{
 | |
| 						$image['avif']['url'] = $fileData->$type;
 | |
| 						$image['avif']['size'] = $fileData->$size;
 | |
| 						if (false === $this->checkFileSizeMargin($checkFileSize, $fileData->$size))
 | |
| 						{
 | |
| 							$image['avif']['status'] = self::STATUS_OPTIMIZED_BIGGER;
 | |
| 						}
 | |
| 						else {
 | |
| 							$image['avif']['status'] = self::STATUS_SUCCESS;
 | |
| 						}
 | |
| 				}
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 			return $image;
 | |
| 	}
 | |
| 
 | |
|   private function getResultObject()
 | |
|   {
 | |
|         $result = new \stdClass;
 | |
|         $result->apiStatus = null;
 | |
|         $result->message = '';
 | |
|         $result->is_error = false;
 | |
|         $result->is_done = false;
 | |
|         //$result->errors = array();
 | |
| 
 | |
|         return $result;
 | |
|   }
 | |
| 
 | |
|   private function returnFailure($status, $message)
 | |
|   {
 | |
|         $result = $this->getResultObject();
 | |
|         $result->apiStatus = $status;
 | |
|         $result->message = $message;
 | |
|         $result->is_error = true;
 | |
|         $result->is_done = true;
 | |
| 
 | |
|         return $result;  // fatal.
 | |
|   }
 | |
| 
 | |
|   // Temporary Error, retry.
 | |
|   private function returnRetry($status, $message)
 | |
|   {
 | |
| 
 | |
|     $result = $this->getResultObject();
 | |
|     $result->apiStatus = $status;
 | |
|     $result->message = $message;
 | |
| 
 | |
|     //$result->errors[] = array('status' => $status, 'message' => $message);
 | |
|     $result->is_error = true;
 | |
| 
 | |
|     return $result;
 | |
|   }
 | |
| 
 | |
|   private function returnOK($status = self::STATUS_UNCHANGED, $message = false)
 | |
|   {
 | |
|       $result = $this->getResultObject();
 | |
|       $result->apiStatus = $status;
 | |
|       $result->is_error = false;
 | |
|       $result->message = $message;
 | |
| 
 | |
|       return $result;
 | |
|   }
 | |
| 
 | |
|   /** Returns a success status. This is succeseption, each file gives it's own status, bundled. */
 | |
|   private function returnSuccess($file, $status = self::STATUS_SUCCESS, $message = false)
 | |
|   {
 | |
|       $result = $this->getResultObject();
 | |
|       $result->apiStatus = $status;
 | |
|       $result->message = $message;
 | |
| 
 | |
| 			if (self::STATUS_SUCCESS === $status)
 | |
|       	$result->is_done = true;
 | |
| 
 | |
| 			if (is_array($file))
 | |
|         $result->files = $file;
 | |
|       else
 | |
|         $result->file = $file; // this file is being used in imageModel
 | |
| 
 | |
|       return $result;
 | |
|   }
 | |
| 
 | |
| 	// If this returns false, the resultSize is bigger, thus should be oversize.
 | |
| 	private function checkFileSizeMargin($fileSize, $resultSize)
 | |
| 	{
 | |
| 			// This is ok.
 | |
| 			if ($fileSize >= $resultSize)
 | |
| 				return true;
 | |
| 
 | |
| 			// Fine suppose, but crashes the increase
 | |
| 			if ($fileSize == 0)
 | |
| 				return true;
 | |
| 
 | |
| 		  $percentage = apply_filters('shortpixel/api/filesizeMargin', 5);
 | |
| 
 | |
| 			$increase = (($resultSize - $fileSize) / $fileSize) * 100;
 | |
| 
 | |
| 			if ($increase <= $percentage)
 | |
| 				return true;
 | |
| 
 | |
| 		  return false;
 | |
| 	}
 | |
| 
 | |
| } // class
 |