'product_download', 'primary' => 'id_product_download', 'fields' => [ 'id_product' => ['type' => self::TYPE_INT, 'validate' => 'isUnsignedId', 'required' => true], 'display_filename' => ['type' => self::TYPE_STRING, 'validate' => 'isGenericName', 'size' => VirtualProductFileSettings::MAX_DISPLAY_FILENAME_LENGTH], 'filename' => ['type' => self::TYPE_STRING, 'validate' => 'isSha1', 'size' => VirtualProductFileSettings::MAX_FILENAME_LENGTH], 'date_add' => ['type' => self::TYPE_DATE, 'validate' => 'isDate'], 'date_expiration' => ['type' => self::TYPE_DATE, 'validate' => 'isDateOrNull'], 'nb_days_accessible' => ['type' => self::TYPE_INT, 'validate' => 'isUnsignedInt', 'size' => 10], 'nb_downloadable' => ['type' => self::TYPE_INT, 'validate' => 'isUnsignedInt', 'size' => 10], 'active' => ['type' => self::TYPE_BOOL, 'validate' => 'isBool'], 'is_shareable' => ['type' => self::TYPE_BOOL, 'validate' => 'isBool'], ], ]; /** * Build a virtual product. * * @param int $idProductDownload Existing productDownload id in order to load object (optional) */ public function __construct($idProductDownload = null) { parent::__construct($idProductDownload); // @TODO check if the file is present on hard drive } /** * @see ObjectModel::getFields() * * @return array */ public function getFields() { $fields = parent::getFields(); if (!$fields['date_expiration']) { $fields['date_expiration'] = '0000-00-00 00:00:00'; } return $fields; } public function add($autoDate = true, $nullValues = false) { return (bool) parent::add($autoDate, $nullValues); } public function update($nullValues = false) { if (parent::update($nullValues)) { // Refresh cache of feature detachable because the row can be deactive // Configuration::updateGlobalValue('PS_VIRTUAL_PROD_FEATURE_ACTIVE', ProductDownload::isCurrentlyUsed($this->def['table'], true)); return true; } return false; } public function delete($deleteFile = false) { $result = parent::delete(); if ($result && $deleteFile) { return $this->deleteFile(); } return $result; } /** * Delete the file. * * @param int $idProductDownload : if we need to delete a specific product attribute file * * @return bool */ public function deleteFile($idProductDownload = null) { if (!$this->checkFile()) { return false; } return unlink(_PS_DOWNLOAD_DIR_ . $this->filename) && Db::getInstance()->delete('product_download', 'id_product_download = ' . (int) $idProductDownload); } /** * Check if file exists. * * @return bool */ public function checkFile() { if (!$this->filename) { return false; } return file_exists(_PS_DOWNLOAD_DIR_ . $this->filename); } /** * Check if download repository is writable. * * @return bool */ public static function checkWritableDir() { return is_writable(_PS_DOWNLOAD_DIR_); } /** * Return the id_product_download from an id_product. * * @param int $idProduct Product the id * @param bool $active * * @return bool|int Product the id for this virtual product */ public static function getIdFromIdProduct($idProduct, $active = true) { if (!ProductDownload::isFeatureActive()) { return false; } self::$_productIds[$idProduct] = (int) Db::getInstance()->getValue(' SELECT `id_product_download` FROM `' . _DB_PREFIX_ . 'product_download` WHERE `id_product` = ' . (int) $idProduct . ' ' . ($active ? ' AND `active` = 1' : '') . ' ORDER BY `id_product_download` DESC'); return self::$_productIds[$idProduct]; } /** * Return the display filename from a physical filename. * * @param string $filename Filename physically * * @return int Product the id for this virtual product */ public static function getIdFromFilename($filename) { return (int) Db::getInstance(_PS_USE_SQL_SLAVE_)->getValue( 'SELECT `id_product_download` FROM `' . _DB_PREFIX_ . 'product_download` WHERE `filename` = \'' . pSQL($filename) . '\'' ); } /** * Return the filename from a Product ID. * * @param int $idProduct Product ID * * @return string Filename the filename for this virtual product */ public static function getFilenameFromIdProduct($idProduct) { return Db::getInstance(_PS_USE_SQL_SLAVE_)->getValue(' SELECT `filename` FROM `' . _DB_PREFIX_ . 'product_download` WHERE `id_product` = ' . (int) $idProduct . ' AND `active` = 1 '); } /** * Return the display filename from a physical filename. * * @param string $filename Filename physically * * @return string Filename the display filename for this virtual product */ public static function getFilenameFromFilename($filename) { return Db::getInstance(_PS_USE_SQL_SLAVE_)->getValue(' SELECT `display_filename` FROM `' . _DB_PREFIX_ . 'product_download` WHERE `filename` = \'' . pSQL($filename) . '\''); } /** * Return text link. * * @param string|false $hash hash code in table order detail (optional) * * @return string Html all the code for print a link to the file */ public function getTextLink($hash = false) { $key = $this->filename . '-' . ($hash ? $hash : 'orderdetail'); return Context::getContext()->link->getPageLink('get-file&key=' . $key); } /** * Return html link. * * @param string|bool $class CSS selector * @param string|bool $hash hash code in table order detail * * @return string Html all the code for print a link to the file */ public function getHtmlLink($class = false, $hash = false) { $link = $this->getTextLink($hash); $html = 'display_filename . ''; return $html; } /** * Return a deadline. * * @return string Datetime in SQL format */ public function getDeadline() { if (!(int) $this->nb_days_accessible) { return '0000-00-00 00:00:00'; } $timestamp = strtotime('+' . (int) $this->nb_days_accessible . ' day'); return date('Y-m-d H:i:s', $timestamp); } /** * Return a hash for control download access. * * @return string Hash ready to insert in database */ public function getHash() { // TODO check if this hash not already in database return sha1(microtime() . $this->id); } /** * Return a sha1 filename. * * @return string Sha1 unique filename */ public static function getNewFilename() { do { $filename = sha1(microtime()); } while (file_exists(_PS_DOWNLOAD_DIR_ . $filename)); return $filename; } /** * This method is allow to know if a feature is used or active. * * @return bool */ public static function isFeatureActive() { return Configuration::get('PS_VIRTUAL_PROD_FEATURE_ACTIVE'); } }