File Manager Lite
Dir:
/home/u540325668/domains/dumbafarm.in/public_html/classes
Upload
[..]
.htaccess (170 B)
Edit
Rename
Del
Access.php (13.25 KB)
Edit
Rename
Del
AddressChecksumCore.php (1.81 KB)
Edit
Rename
Del
AddressFormat.php (24.54 KB)
Edit
Rename
Del
Alias.php (4.99 KB)
Edit
Rename
Del
Attachment.php (11.27 KB)
Edit
Rename
Del
AttributeGroup.php (13.17 KB)
Edit
Rename
Del
CMS.php (11.45 KB)
Edit
Rename
Del
CMSCategory.php (23.56 KB)
Edit
Rename
Del
CMSRole.php (1.78 KB)
Edit
Rename
Del
CSV.php (3.04 KB)
Edit
Rename
Del
Carrier.php (63.97 KB)
Edit
Rename
Del
Cart.php (212.95 KB)
Edit
Rename
Del
Category.php (87 KB)
Edit
Rename
Del
Chart.php (4.08 KB)
Edit
Rename
Del
ChecksumInterface.php (1.22 KB)
Edit
Rename
Del
Configuration.php (26.5 KB)
Edit
Rename
Del
ConfigurationKPI.php (8.91 KB)
Edit
Rename
Del
ConfigurationTest.php (12.02 KB)
Edit
Rename
Del
Connection.php (10.02 KB)
Edit
Rename
Del
ConnectionsSource.php (4.81 KB)
Edit
Rename
Del
Contact.php (3.83 KB)
Edit
Rename
Del
Context.php (17.94 KB)
Edit
Rename
Del
Cookie.php (17.39 KB)
Edit
Rename
Del
Country.php (16.33 KB)
Edit
Rename
Del
Currency.php (36.3 KB)
Edit
Rename
Del
Curve.php (2.92 KB)
Edit
Rename
Del
Customer.php (52.22 KB)
Edit
Rename
Del
CustomerAddress.php (1.26 KB)
Edit
Rename
Del
CustomerMessage.php (5.75 KB)
Edit
Rename
Del
CustomerSession.php (2.65 KB)
Edit
Rename
Del
CustomerThread.php (9.62 KB)
Edit
Rename
Del
Customization.php (13.43 KB)
Edit
Rename
Del
CustomizationField.php (2.85 KB)
Edit
Rename
Del
DateRange.php (2.47 KB)
Edit
Rename
Del
Delivery.php (3.24 KB)
Edit
Rename
Del
Dispatcher.php (47.06 KB)
Edit
Rename
Del
Employee.php (23.5 KB)
Edit
Rename
Del
EmployeeSession.php (2.65 KB)
Edit
Rename
Del
Feature.php (11.38 KB)
Edit
Rename
Del
FeatureFlag.php (3.15 KB)
Edit
Rename
Del
FeatureValue.php (8.06 KB)
Edit
Rename
Del
FileUploader.php (3.73 KB)
Edit
Rename
Del
Gender.php (2.81 KB)
Edit
Rename
Del
Group.php (14.96 KB)
Edit
Rename
Del
GroupReduction.php (8.81 KB)
Edit
Rename
Del
Guest.php (8.31 KB)
Edit
Rename
Del
Hook.php (46.57 KB)
Edit
Rename
Del
Image.php (31.24 KB)
Edit
Rename
Del
ImageManager.php (31.79 KB)
Edit
Rename
Del
ImageType.php (6.99 KB)
Edit
Rename
Del
Language.php (62.06 KB)
Edit
Rename
Del
Link.php (61.5 KB)
Edit
Rename
Del
LocalizationPack.php (23.05 KB)
Edit
Rename
Del
Mail.php (33.49 KB)
Edit
Rename
Del
Manufacturer.php (21.2 KB)
Edit
Rename
Del
ManufacturerAddress.php (1.28 KB)
Edit
Rename
Del
Media.php (31.43 KB)
Edit
Rename
Del
Message.php (6.47 KB)
Edit
Rename
Del
Meta.php (18.21 KB)
Edit
Rename
Del
Notification.php (8.27 KB)
Edit
Rename
Del
ObjectModel.php (81.56 KB)
Edit
Rename
Del
Pack.php (25.7 KB)
Edit
Rename
Del
Page.php (4.49 KB)
Edit
Rename
Del
PaymentFree.php (1.33 KB)
Edit
Rename
Del
PaymentModule.php (59.49 KB)
Edit
Rename
Del
PhpEncryption.php (3.21 KB)
Edit
Rename
Del
PhpEncryptionEngine.php (3.86 KB)
Edit
Rename
Del
PrestaShopAutoload.php (9.57 KB)
Edit
Rename
Del
PrestaShopBackup.php (11.43 KB)
Edit
Rename
Del
PrestaShopCollection.php (21.28 KB)
Edit
Rename
Del
PrestaShopLogger.php (8.3 KB)
Edit
Rename
Del
Product.php (309.93 KB)
Edit
Rename
Del
ProductAttribute.php (13.48 KB)
Edit
Rename
Del
ProductDownload.php (10.05 KB)
Edit
Rename
Del
ProductPresenterFactory.php (3.89 KB)
Edit
Rename
Del
ProductSale.php (12.7 KB)
Edit
Rename
Del
ProductSupplier.php (8.34 KB)
Edit
Rename
Del
Profile.php (8.32 KB)
Edit
Rename
Del
QqUploadedFileForm.php (5.19 KB)
Edit
Rename
Del
QqUploadedFileXhr.php (5.69 KB)
Edit
Rename
Del
QuickAccess.php (6.25 KB)
Edit
Rename
Del
RequestSql.php (20.93 KB)
Edit
Rename
Del
Risk.php (2.36 KB)
Edit
Rename
Del
Search.php (51.86 KB)
Edit
Rename
Del
SearchEngine.php (2.81 KB)
Edit
Rename
Del
Smarty/
Rename
Del
SpecificPrice.php (32.43 KB)
Edit
Rename
Del
SpecificPriceRule.php (14.04 KB)
Edit
Rename
Del
State.php (7.77 KB)
Edit
Rename
Del
Store.php (6.24 KB)
Edit
Rename
Del
Supplier.php (17.89 KB)
Edit
Rename
Del
SupplierAddress.php (1.27 KB)
Edit
Rename
Del
Tab.php (21.7 KB)
Edit
Rename
Del
Tag.php (13.24 KB)
Edit
Rename
Del
Tools.php (136.37 KB)
Edit
Rename
Del
TranslatedConfiguration.php (4.68 KB)
Edit
Rename
Del
Upgrader.php (11.58 KB)
Edit
Rename
Del
Uploader.php (10.84 KB)
Edit
Rename
Del
Validate.php (38.87 KB)
Edit
Rename
Del
ValidateConstraintTranslator.php (2.55 KB)
Edit
Rename
Del
WarehouseAddress.php (1.27 KB)
Edit
Rename
Del
Zone.php (3.42 KB)
Edit
Rename
Del
assets/
Rename
Del
cache/
Rename
Del
checkout/
Rename
Del
container/
Rename
Del
controller/
Rename
Del
db/
Rename
Del
exception/
Rename
Del
form/
Rename
Del
helper/
Rename
Del
index.php (1.34 KB)
Edit
Rename
Del
lang/
Rename
Del
log/
Rename
Del
module/
Rename
Del
order/
Rename
Del
pdf/
Rename
Del
product/
Rename
Del
proxy/
Rename
Del
range/
Rename
Del
shop/
Rename
Del
stock/
Rename
Del
tax/
Rename
Del
tree/
Rename
Del
webservice/
Rename
Del
Edit: Image.php
<?php /** * Copyright since 2007 PrestaShop SA and Contributors * PrestaShop is an International Registered Trademark & Property of PrestaShop SA * * NOTICE OF LICENSE * * This source file is subject to the Open Software License (OSL 3.0) * that is bundled with this package in the file LICENSE.md. * It is also available through the world-wide-web at this URL: * https://opensource.org/licenses/OSL-3.0 * If you did not receive a copy of the license and are unable to * obtain it through the world-wide-web, please send an email * to license@prestashop.com so we can send you a copy immediately. * * DISCLAIMER * * Do not edit or add to this file if you wish to upgrade PrestaShop to newer * versions in the future. If you wish to customize PrestaShop for your * needs please refer to https://devdocs.prestashop.com/ for more information. * * @author PrestaShop SA and Contributors <contact@prestashop.com> * @copyright Since 2007 PrestaShop SA and Contributors * @license https://opensource.org/licenses/OSL-3.0 Open Software License (OSL 3.0) */ use PrestaShop\PrestaShop\Adapter\ServiceLocator; use PrestaShop\PrestaShop\Core\Exception\InvalidArgumentException; use PrestaShop\PrestaShop\Core\Image\ImageFormatConfiguration; /** * Class ImageCore. */ class ImageCore extends ObjectModel { public $id; /** @var int Image ID */ public $id_image; /** @var int Product ID */ public $id_product; /** @var int Position used to order images of the same product */ public $position; /** @var bool|null Image is cover */ public $cover; /** @var array<int,string> Legend */ public $legend; /** @var string image extension */ public $image_format = 'jpg'; /** @var string path to index.php file to be copied to new image folders */ public $source_index; /** @var string image folder */ protected $folder; /** @var string image path without extension */ protected $existing_path; /** @var int access rights of created folders (octal) */ protected static $access_rights = 0775; /** * @see ObjectModel::$definition */ public static $definition = [ 'table' => 'image', 'primary' => 'id_image', 'multilang' => true, 'fields' => [ 'id_product' => ['type' => self::TYPE_INT, 'shop' => 'both', 'validate' => 'isUnsignedId', 'required' => true], 'position' => ['type' => self::TYPE_INT, 'validate' => 'isUnsignedInt'], 'cover' => ['type' => self::TYPE_BOOL, 'allow_null' => true, 'validate' => 'isBool', 'shop' => true], 'legend' => ['type' => self::TYPE_STRING, 'lang' => true, 'validate' => 'isGenericName', 'size' => 128], ], ]; /** * @var array */ protected static $_cacheGetSize = []; /** * ImageCore constructor. * * @param int|null $id * @param int|null $idLang * @param null $id_shop * @param null $translator * * @throws PrestaShopDatabaseException * @throws PrestaShopException */ public function __construct($id = null, $idLang = null, $id_shop = null, $translator = null) { parent::__construct($id, $idLang, $id_shop, $translator); $this->image_dir = _PS_PRODUCT_IMG_DIR_; $this->source_index = _PS_PRODUCT_IMG_DIR_ . 'index.php'; } /** * Adds current Image as a new Object to the database. * * @param bool $autoDate Automatically set `date_upd` and `date_add` columns * @param bool $nullValues Whether we want to use NULL values instead of empty quotes values * * @return bool Indicates whether the Image has been successfully added * * @throws PrestaShopDatabaseException * @throws PrestaShopException */ public function add($autoDate = true, $nullValues = false) { if ($this->position <= 0) { $this->position = Image::getHighestPosition($this->id_product) + 1; } if ($this->cover) { $this->cover = true; } else { $this->cover = null; } return parent::add($autoDate, $nullValues); } /** * This override is needed because we need to set 'id_product' => (int) $this->id_product, in $data array which is * a specific case for association between shop and image * * {@inheritDoc} */ public function associateTo($id_shops, int $productId = null) { if (!$this->id) { return; } $productId = $productId ?? $this->id_product; if (empty($productId)) { throw new InvalidArgumentException('You cannot associate an image to shop without specifying product ID'); } if (!is_array($id_shops)) { $id_shops = [$id_shops]; } $data = []; foreach ($id_shops as $id_shop) { if (!$this->isAssociatedToShop($id_shop)) { $data[] = [ $this->def['primary'] => (int) $this->id, 'id_shop' => (int) $id_shop, 'id_product' => $productId, ]; } } if ($data) { return Db::getInstance()->insert($this->def['table'] . '_shop', $data); } return true; } /** * Updates the current Image in the database. * * @param bool $nullValues Whether we want to use NULL values instead of empty quotes values * * @return bool Indicates whether the Image has been successfully updated * * @throws PrestaShopDatabaseException * @throws PrestaShopException */ public function update($nullValues = false) { if ($this->cover) { $this->cover = true; } else { $this->cover = null; } return parent::update($nullValues); } /** * Deletes current Image from the database. * * @return bool `true` if delete was successful * * @throws PrestaShopException */ public function delete() { if (!parent::delete()) { return false; } if ($this->hasMultishopEntries()) { return true; } if (!$this->deleteProductAttributeImage() || !$this->deleteImage()) { return false; } // update positions Db::getInstance()->execute('SET @position:=0', false); Db::getInstance()->execute('UPDATE `' . _DB_PREFIX_ . 'image` SET position=(@position:=@position+1) WHERE `id_product` = ' . (int) $this->id_product . ' ORDER BY position ASC'); return true; } /** * Return first image (by position) associated with a product attribute. * * @param int $idShop Shop ID * @param int $idLang Language ID * @param int $idProduct Product ID * @param int $idProductAttribute Product Attribute ID * * @return array */ public static function getBestImageAttribute($idShop, $idLang, $idProduct, $idProductAttribute) { $cacheId = 'Image::getBestImageAttribute' . '-' . (int) $idProduct . '-' . (int) $idProductAttribute . '-' . (int) $idLang . '-' . (int) $idShop; if (!Cache::isStored($cacheId)) { $row = Db::getInstance()->getRow(' SELECT image_shop.`id_image` id_image, il.`legend` FROM `' . _DB_PREFIX_ . 'image` i INNER JOIN `' . _DB_PREFIX_ . 'image_shop` image_shop ON (i.id_image = image_shop.id_image AND image_shop.id_shop = ' . (int) $idShop . ') INNER JOIN `' . _DB_PREFIX_ . 'product_attribute_image` pai ON (pai.`id_image` = i.`id_image` AND pai.`id_product_attribute` = ' . (int) $idProductAttribute . ') LEFT JOIN `' . _DB_PREFIX_ . 'image_lang` il ON (image_shop.`id_image` = il.`id_image` AND il.`id_lang` = ' . (int) $idLang . ') WHERE i.`id_product` = ' . (int) $idProduct . ' ORDER BY i.`position` ASC'); Cache::store($cacheId, $row); } else { $row = Cache::retrieve($cacheId); } return $row; } /** * Return available images for a product. * * @param int $idLang Language ID * @param int $idProduct Product ID * @param int $idProductAttribute Product Attribute ID * @param int $idShop Shop ID * * @return array Images */ public static function getImages($idLang, $idProduct, $idProductAttribute = null, $idShop = null) { $attributeFilter = ($idProductAttribute ? ' AND ai.`id_product_attribute` = ' . (int) $idProductAttribute : ''); $shopFilter = ($idShop ? ' AND ims.`id_shop` = ' . (int) $idShop : ''); $sql = 'SELECT * FROM `' . _DB_PREFIX_ . 'image` i LEFT JOIN `' . _DB_PREFIX_ . 'image_lang` il ON (i.`id_image` = il.`id_image`)'; if ($idProductAttribute) { $sql .= ' LEFT JOIN `' . _DB_PREFIX_ . 'product_attribute_image` ai ON (i.`id_image` = ai.`id_image`)'; } if ($idShop) { $sql .= ' LEFT JOIN `' . _DB_PREFIX_ . 'image_shop` ims ON (i.`id_image` = ims.`id_image`)'; } $sql .= ' WHERE i.`id_product` = ' . (int) $idProduct . ' AND il.`id_lang` = ' . (int) $idLang . $attributeFilter . $shopFilter . ' ORDER BY i.`position` ASC'; return Db::getInstance()->executeS($sql); } /** * Check if a product has an image available. * * @param int $idLang Language ID * @param int $idProduct Product ID * @param int $idProductAttribute Product Attribute ID * * @return bool */ public static function hasImages($idLang, $idProduct, $idProductAttribute = null) { $attribute_filter = ($idProductAttribute ? ' AND ai.`id_product_attribute` = ' . (int) $idProductAttribute : ''); $sql = 'SELECT 1 FROM `' . _DB_PREFIX_ . 'image` i LEFT JOIN `' . _DB_PREFIX_ . 'image_lang` il ON (i.`id_image` = il.`id_image`)'; if ($idProductAttribute) { $sql .= ' LEFT JOIN `' . _DB_PREFIX_ . 'product_attribute_image` ai ON (i.`id_image` = ai.`id_image`)'; } $sql .= ' WHERE i.`id_product` = ' . (int) $idProduct . ' AND il.`id_lang` = ' . (int) $idLang . $attribute_filter; return (bool) Db::getInstance()->getValue($sql); } /** * Return Images. * * @return array Images */ public static function getAllImages() { return Db::getInstance()->executeS(' SELECT `id_image`, `id_product` FROM `' . _DB_PREFIX_ . 'image` ORDER BY `id_image` ASC'); } /** * Return number of images for a product. * * @param int $idProduct Product ID * * @return int number of images */ public static function getImagesTotal($idProduct) { $result = Db::getInstance()->getRow(' SELECT COUNT(`id_image`) AS total FROM `' . _DB_PREFIX_ . 'image` WHERE `id_product` = ' . (int) $idProduct); return $result['total']; } /** * Return highest position of images for a product. * * @param int $idProduct Product ID * * @return int highest position of images */ public static function getHighestPosition($idProduct) { $result = Db::getInstance()->getRow(' SELECT MAX(`position`) AS max FROM `' . _DB_PREFIX_ . 'image` WHERE `id_product` = ' . (int) $idProduct); return $result['max']; } /** * Delete product cover. * * @param int $idProduct Product ID * * @return bool result */ public static function deleteCover($idProduct) { if (!Validate::isUnsignedId($idProduct)) { die(Tools::displayError('Product ID is invalid.')); } if (file_exists(_PS_TMP_IMG_DIR_ . 'product_' . $idProduct . '.jpg')) { unlink(_PS_TMP_IMG_DIR_ . 'product_' . $idProduct . '.jpg'); } return Db::getInstance()->execute( ' UPDATE `' . _DB_PREFIX_ . 'image` SET `cover` = NULL WHERE `id_product` = ' . (int) $idProduct ) && Db::getInstance()->execute( ' UPDATE `' . _DB_PREFIX_ . 'image_shop` image_shop SET image_shop.`cover` = NULL WHERE image_shop.id_shop IN (' . implode(',', array_map('intval', Shop::getContextListShopID())) . ') AND image_shop.`id_product` = ' . (int) $idProduct ); } /** *Get product cover. * * @param int $idProduct Product ID * * @return bool result */ public static function getCover($idProduct) { return Db::getInstance()->getRow(' SELECT * FROM `' . _DB_PREFIX_ . 'image_shop` image_shop WHERE image_shop.`id_product` = ' . (int) $idProduct . ' AND image_shop.`cover`= 1'); } /** *Get global product cover. * * @param int $idProduct Product ID * * @return bool result */ public static function getGlobalCover($idProduct) { return Db::getInstance()->getRow(' SELECT * FROM `' . _DB_PREFIX_ . 'image` i WHERE i.`id_product` = ' . (int) $idProduct . ' AND i.`cover`= 1'); } /** * Copy images from a product to another. * * @param int $idProductOld Source product ID * @param int $idProductNew Destination product ID */ public static function duplicateProductImages($idProductOld, $idProductNew, $combinationImages) { $imagesTypes = ImageType::getImagesTypes('products'); $result = Db::getInstance()->executeS(' SELECT `id_image` FROM `' . _DB_PREFIX_ . 'image` WHERE `id_product` = ' . (int) $idProductOld); foreach ($result as $row) { $imageOld = new Image($row['id_image']); $imageNew = clone $imageOld; unset($imageNew->id); $imageNew->id_product = (int) $idProductNew; // A new id is generated for the cloned image when calling add() if ($imageNew->add()) { $newPath = $imageNew->getPathForCreation(); foreach ($imagesTypes as $imageType) { if (file_exists(_PS_PRODUCT_IMG_DIR_ . $imageOld->getExistingImgPath() . '-' . $imageType['name'] . '.jpg')) { if (!Configuration::get('PS_LEGACY_IMAGES')) { $imageNew->createImgFolder(); } copy( _PS_PRODUCT_IMG_DIR_ . $imageOld->getExistingImgPath() . '-' . $imageType['name'] . '.jpg', $newPath . '-' . $imageType['name'] . '.jpg' ); if (Configuration::get('WATERMARK_HASH')) { $oldImagePath = _PS_PRODUCT_IMG_DIR_ . $imageOld->getExistingImgPath() . '-' . $imageType['name'] . '-' . Configuration::get('WATERMARK_HASH') . '.jpg'; if (file_exists($oldImagePath)) { copy($oldImagePath, $newPath . '-' . $imageType['name'] . '-' . Configuration::get('WATERMARK_HASH') . '.jpg'); } } } } if (file_exists(_PS_PRODUCT_IMG_DIR_ . $imageOld->getExistingImgPath() . '.jpg')) { copy(_PS_PRODUCT_IMG_DIR_ . $imageOld->getExistingImgPath() . '.jpg', $newPath . '.jpg'); } Image::replaceAttributeImageAssociationId($combinationImages, (int) $imageOld->id, (int) $imageNew->id); // Duplicate shop associations for images $imageNew->duplicateShops($idProductOld); } else { return false; } } return Image::duplicateAttributeImageAssociations($combinationImages); } /** * @param array $combinationImages * @param int $savedId * @param int $idImage */ protected static function replaceAttributeImageAssociationId(&$combinationImages, $savedId, $idImage) { if (!isset($combinationImages['new']) || !is_array($combinationImages['new'])) { return; } foreach ($combinationImages['new'] as $id_product_attribute => $image_ids) { foreach ($image_ids as $key => $imageId) { if ((int) $imageId == (int) $savedId) { $combinationImages['new'][$id_product_attribute][$key] = (int) $idImage; } } } } /** * Duplicate product attribute image associations. * * @param array $combinationImages * * @return bool */ public static function duplicateAttributeImageAssociations($combinationImages) { if (!isset($combinationImages['new']) || !is_array($combinationImages['new'])) { return true; } $query = 'INSERT INTO `' . _DB_PREFIX_ . 'product_attribute_image` (`id_product_attribute`, `id_image`) VALUES '; foreach ($combinationImages['new'] as $idProductAttribute => $imageIds) { foreach ($imageIds as $imageId) { $query .= '(' . (int) $idProductAttribute . ', ' . (int) $imageId . '), '; } } $query = rtrim($query, ', '); return Db::getInstance()->execute($query); } /** * Change an image position and update relative positions. * * @param int $way position is moved up if 0, moved down if 1 * @param int $position new position of the moved image * * @return bool success */ public function updatePosition($way, $position) { if (!isset($this->id) || !$position) { return false; } // < and > statements rather than BETWEEN operator // since BETWEEN is treated differently according to databases return Db::getInstance()->execute( 'UPDATE `' . _DB_PREFIX_ . 'image` SET `position`= `position` ' . ($way ? '- 1' : '+ 1') . ' WHERE `position` ' . ($way ? '> ' . (int) $this->position . ' AND `position` <= ' . (int) $position : '< ' . (int) $this->position . ' AND `position` >= ' . (int) $position) . ' AND `id_product`=' . (int) $this->id_product ) && Db::getInstance()->execute( 'UPDATE `' . _DB_PREFIX_ . 'image` SET `position` = ' . (int) $position . ' WHERE `id_image` = ' . (int) $this->id_image ) ; } /** * @param string $type * * @return mixed */ public static function getSize($type) { if (!isset(self::$_cacheGetSize[$type])) { self::$_cacheGetSize[$type] = Db::getInstance()->getRow(' SELECT `width`, `height` FROM ' . _DB_PREFIX_ . 'image_type WHERE `name` = \'' . pSQL($type) . '\' '); } return self::$_cacheGetSize[$type]; } /** * @param array $params * * @return mixed */ public static function getWidth($params) { $result = self::getSize($params['type']); return $result['width']; } /** * @param array $params * * @return mixed */ public static function getHeight($params) { $result = self::getSize($params['type']); return $result['height']; } /** * Clear all images in tmp dir. */ public static function clearTmpDir() { foreach (scandir(_PS_TMP_IMG_DIR_, SCANDIR_SORT_NONE) as $d) { if (preg_match('/(.*)\.jpg$/', $d)) { unlink(_PS_TMP_IMG_DIR_ . $d); } } } /** * Delete Image - Product attribute associations for this image. */ public function deleteProductAttributeImage() { return Db::getInstance()->execute( ' DELETE FROM `' . _DB_PREFIX_ . 'product_attribute_image` WHERE `id_image` = ' . (int) $this->id ); } /** * Delete the product image from disk and remove the containing folder if empty * Handles both legacy and new image filesystems. */ public function deleteImage($forceDelete = false) { if (!$this->id) { return false; } // Delete base image if (file_exists($this->image_dir . $this->getExistingImgPath() . '.' . $this->image_format)) { unlink($this->image_dir . $this->getExistingImgPath() . '.' . $this->image_format); } else { return false; } $filesToDelete = []; // Delete auto-generated images $image_types = ImageType::getImagesTypes(); // Get image formats we will be deleting. It would probably be easier to use ImageFormatConfiguration::SUPPORTED_FORMATS, // but we want to avoid any behavior change in minor/patch version. $configuredImageFormats = ServiceLocator::get(ImageFormatConfiguration::class)->getGenerationFormats(); foreach ($image_types as $imageType) { foreach ($configuredImageFormats as $imageFormat) { $filesToDelete = $this->deleteAutoGeneratedImage($imageType, $imageFormat, $filesToDelete); } } // Delete watermark image $filesToDelete[] = $this->image_dir . $this->getExistingImgPath() . '-watermark.' . $this->image_format; $filesToDelete[] = $this->image_dir . $this->getExistingImgPath() . '-watermark2x.' . $this->image_format; // delete index.php $filesToDelete[] = $this->image_dir . $this->getImgFolder() . 'index.php'; // delete fileType $filesToDelete[] = $this->image_dir . $this->getImgFolder() . 'fileType'; // Delete tmp images $filesToDelete[] = _PS_TMP_IMG_DIR_ . 'product_' . $this->id_product . '.' . $this->image_format; $filesToDelete[] = _PS_TMP_IMG_DIR_ . 'product_mini_' . $this->id_product . '.' . $this->image_format; foreach ($filesToDelete as $file) { if (file_exists($file) && !@unlink($file)) { return false; } } // Can we delete the image folder? if (is_dir($this->image_dir . $this->getImgFolder())) { $deleteFolder = true; foreach (scandir($this->image_dir . $this->getImgFolder(), SCANDIR_SORT_NONE) as $file) { if (($file != '.' && $file != '..')) { $deleteFolder = false; break; } } } if (isset($deleteFolder) && $deleteFolder) { @rmdir($this->image_dir . $this->getImgFolder()); } return true; } /** * Recursively deletes all product images in the given folder tree and removes empty folders. * * @param string $path folder containing the product images to delete * @param string $format image format * * @return bool success */ public static function deleteAllImages($path, $format = 'jpg') { if (!$path || !$format || !is_dir($path)) { return false; } foreach (scandir($path, SCANDIR_SORT_NONE) as $file) { if (preg_match('/^[0-9]+(\-(.*))?\.' . $format . '$/', $file)) { unlink($path . $file); } elseif (is_dir($path . $file) && (preg_match('/^[0-9]$/', $file))) { Image::deleteAllImages($path . $file . '/', $format); } } // Can we remove the image folder? if (is_numeric(basename($path))) { $removeFolder = true; foreach (scandir($path, SCANDIR_SORT_NONE) as $file) { if (($file != '.' && $file != '..' && $file != 'index.php')) { $removeFolder = false; break; } } if ($removeFolder) { // we're only removing index.php if it's a folder we want to delete if (file_exists($path . 'index.php')) { @unlink($path . 'index.php'); } @rmdir($path); } } return true; } /** * Returns image path in the old or in the new filesystem. * * @ returns string image path */ public function getExistingImgPath() { if (!$this->id) { return false; } if (!$this->existing_path) { if (Configuration::get('PS_LEGACY_IMAGES') && file_exists(_PS_PRODUCT_IMG_DIR_ . $this->id_product . '-' . $this->id . '.' . $this->image_format)) { $this->existing_path = $this->id_product . '-' . $this->id; } else { $this->existing_path = $this->getImgPath(); } } return $this->existing_path; } /** * Returns the path to the folder containing the image in the new filesystem. * * @return string|bool path to folder */ public function getImgFolder() { if (!$this->id) { return false; } if (!$this->folder) { $this->folder = Image::getImgFolderStatic($this->id); } return $this->folder; } /** * Create parent folders for the image in the new filesystem. * * @return bool success */ public function createImgFolder() { if (!$this->id) { return false; } if (!file_exists(_PS_PRODUCT_IMG_DIR_ . $this->getImgFolder())) { // Apparently sometimes mkdir cannot set the rights, and sometimes chmod can't. Trying both. $success = @mkdir(_PS_PRODUCT_IMG_DIR_ . $this->getImgFolder(), self::$access_rights, true); $chmod = @chmod(_PS_PRODUCT_IMG_DIR_ . $this->getImgFolder(), self::$access_rights); // Create an index.php file in the new folder if (($success || $chmod) && !file_exists(_PS_PRODUCT_IMG_DIR_ . $this->getImgFolder() . 'index.php') && file_exists($this->source_index)) { return @copy($this->source_index, _PS_PRODUCT_IMG_DIR_ . $this->getImgFolder() . 'index.php'); } } return true; } /** * Returns the path to the image without file extension. * * @return string|bool path */ public function getImgPath() { if (!$this->id) { return false; } return $this->getImgFolder() . $this->id; } /** * Returns the path to the folder containing the image in the new filesystem. * * @param mixed $idImage * * @return string|bool path to folder */ public static function getImgFolderStatic($idImage) { if (!is_numeric($idImage)) { return false; } $folders = str_split((string) $idImage); return implode('/', $folders) . '/'; } /** * Move all legacy product image files from the image folder root to their subfolder in the new filesystem. * If max_execution_time is provided, stops before timeout and returns string "timeout". * If any image cannot be moved, stops and returns "false". * * @param int $maxExecutionTime * * @return mixed success or timeout */ public static function moveToNewFileSystem($maxExecutionTime = 0) { $startTime = time(); $image = null; $tmpFolder = 'duplicates/'; foreach (scandir(_PS_PRODUCT_IMG_DIR_, SCANDIR_SORT_NONE) as $file) { // matches the base product image or the thumbnails if (preg_match('/^([0-9]+\-)([0-9]+)(\-(.*))?\.jpg$/', $file, $matches)) { // don't recreate an image object for each image type if (!$image || $image->id !== (int) $matches[2]) { $image = new Image((int) $matches[2]); } // image exists in DB and with the correct product? if (Validate::isLoadedObject($image) && $image->id_product == (int) rtrim($matches[1], '-')) { // create the new folder if it does not exist if (!$image->createImgFolder()) { return false; } // if there's already a file at the new image path, move it to a dump folder // most likely the preexisting image is a demo image not linked to a product and it's ok to replace it $newPath = _PS_PRODUCT_IMG_DIR_ . $image->getImgPath() . (isset($matches[3]) ? $matches[3] : '') . '.jpg'; if (file_exists($newPath)) { if (!file_exists(_PS_PRODUCT_IMG_DIR_ . $tmpFolder)) { @mkdir(_PS_PRODUCT_IMG_DIR_ . $tmpFolder, self::$access_rights); @chmod(_PS_PRODUCT_IMG_DIR_ . $tmpFolder, self::$access_rights); } $tmpPath = _PS_PRODUCT_IMG_DIR_ . $tmpFolder . basename($file); if (!@rename($newPath, $tmpPath) || !file_exists($tmpPath)) { return false; } } // move the image if (!@rename(_PS_PRODUCT_IMG_DIR_ . $file, $newPath) || !file_exists($newPath)) { return false; } } } if ((int) $maxExecutionTime != 0 && (time() - $startTime > (int) $maxExecutionTime - 4)) { return 'timeout'; } } return true; } /** * Try to create and delete some folders to check if moving images to new file system will be possible. * * @return bool success */ public static function testFileSystem() { $folder1 = _PS_PRODUCT_IMG_DIR_ . 'testfilesystem/'; $testFolder = $folder1 . 'testsubfolder/'; // check if folders are already existing from previous failed test if (file_exists($testFolder)) { @rmdir($testFolder); @rmdir($folder1); } if (file_exists($testFolder)) { return false; } @mkdir($testFolder, self::$access_rights, true); @chmod($testFolder, self::$access_rights); if (!is_writable($testFolder)) { return false; } @rmdir($testFolder); @rmdir($folder1); if (file_exists($folder1)) { return false; } return true; } /** * Returns the path where a product image should be created (without file format). * * @return string|bool path */ public function getPathForCreation() { if (!$this->id) { return false; } if (Configuration::get('PS_LEGACY_IMAGES')) { if (!$this->id_product) { return false; } $path = $this->id_product . '-' . $this->id; } else { $path = $this->getImgPath(); $this->createImgFolder(); } return _PS_PRODUCT_IMG_DIR_ . $path; } /** * @param array $imageType * @param string $imageFormat * @param array $filesToDelete * * @return array */ private function deleteAutoGeneratedImage(array $imageType, string $imageFormat, array $filesToDelete): array { $filesToDelete[] = $this->image_dir . $this->getExistingImgPath() . '-' . $imageType['name'] . '.' . $imageFormat; $filesToDelete[] = $this->image_dir . $this->getExistingImgPath() . '-' . $imageType['name'] . '2x.' . $imageFormat; if (Configuration::get('WATERMARK_HASH')) { $filesToDelete[] = $this->image_dir . $this->getExistingImgPath() . '-' . $imageType['name'] . '-' . Configuration::get('WATERMARK_HASH') . '.' . $imageFormat; } return $filesToDelete; } }
Simpan