* @copyright 2007-2016 PrestaShop SA * @license http://opensource.org/licenses/osl-3.0.php Open Software License (OSL 3.0) * International Registered Trademark & Property of PrestaShop SA */ @ini_set('max_execution_time', 0); /** * @property PrestaShopBackup $object */ class AdminBackupControllerCore extends AdminController { /** @var string The field we are sorting on */ protected $sort_by = 'date'; public function __construct() { $this->bootstrap = true; $this->table = 'backup'; $this->className = 'PrestaShopBackup'; $this->identifier = 'filename'; parent::__construct(); $this->fields_list = array( 'date' => array('title' => $this->l('Date'), 'type' => 'datetime', 'class' => 'fixed-width-lg', 'orderby' => false, 'search' => false), 'age' => array('title' => $this->l('Age'), 'orderby' => false, 'search' => false), 'filename' => array('title' => $this->l('File name'), 'orderby' => false, 'search' => false), 'filesize' => array('title' => $this->l('File size'), 'class' => 'fixed-width-sm', 'orderby' => false, 'search' => false) ); $this->bulk_actions = array('delete' => array( 'text' => $this->l('Delete selected'), 'confirm' => $this->l('Delete selected items?'), 'icon' => 'icon-trash') ); $this->fields_options = array( 'general' => array( 'title' => $this->l('Backup options'), 'fields' => array( 'PS_BACKUP_ALL' => array( 'title' => $this->l('Ignore statistics tables'), 'desc' => $this->l('Drop existing tables during import.').'
'._DB_PREFIX_.'connections, '._DB_PREFIX_.'connections_page, '._DB_PREFIX_ .'connections_source, '._DB_PREFIX_.'guest, '._DB_PREFIX_.'statssearch', 'cast' => 'intval', 'type' => 'bool' ), 'PS_BACKUP_DROP_TABLE' => array( 'title' => $this->l('Drop existing tables during import'), 'hint' => array( $this->l('If enabled, the backup script will drop your tables prior to restoring data.'), $this->l('(ie. "DROP TABLE IF EXISTS")'), ), 'cast' => 'intval', 'type' => 'bool' ) ), 'submit' => array('title' => $this->l('Save')) ), ); } public function renderList() { $this->addRowAction('view'); $this->addRowAction('delete'); return parent::renderList(); } public function renderView() { if (!($object = $this->loadObject())) { $this->errors[] = Tools::displayError('The object could not be loaded.'); } if ($object->id) { $this->tpl_view_vars = array('url_backup' => $object->getBackupURL()); } elseif ($object->error) { $this->errors[] = $object->error; $this->tpl_view_vars = array('errors' => $this->errors); } return parent::renderView(); } public function initViewDownload() { $this->tpl_folder = $this->tpl_folder.'download/'; return parent::renderView(); } public function initToolbar() { switch ($this->display) { case 'add': case 'edit': case 'view': $this->toolbar_btn['cancel'] = array( 'href' => self::$currentIndex.'&token='.$this->token, 'desc' => $this->l('Cancel') ); break; case 'options': $this->toolbar_btn['save'] = array( 'href' => '#', 'desc' => $this->l('Save') ); break; } } public function initContent() { if ($this->display == 'add') { $this->display = 'list'; } return parent::initContent(); } /** * Load class object using identifier in $_GET (if possible) * otherwise return an empty object * This method overrides the one in AdminTab because AdminTab assumes the id is a UnsignedInt * "Backups" Directory in admin directory must be writeable (CHMOD 777) * @param bool $opt Return an empty object if load fail * @return object */ protected function loadObject($opt = false) { if (($id = Tools::getValue($this->identifier)) && PrestaShopBackup::backupExist($id)) { return new $this->className($id); } $obj = new $this->className(); $obj->error = Tools::displayError('The backup file does not exist'); return $obj; } public function postProcess() { /* PrestaShop demo mode */ if (_PS_MODE_DEMO_) { $this->errors[] = Tools::displayError('This functionality has been disabled.'); return; } /* PrestaShop demo mode*/ // Test if the backup dir is writable if (!is_writable(PrestaShopBackup::getBackupPath())) { $this->warnings[] = $this->l('The "Backups" directory located in the admin directory must be writable (CHMOD 755 / 777).'); } elseif ($this->display == 'add') { if (($object = $this->loadObject())) { if (!$object->add()) { $this->errors[] = $object->error; } else { $this->context->smarty->assign(array( 'conf' => $this->l('It appears the backup was successful, however you must download and carefully verify the backup file before proceeding.'), 'backup_url' => $object->getBackupURL(), 'backup_weight' => number_format((filesize($object->id) * 0.000001), 2, '.', '') )); } } } parent::postProcess(); } public function getList($id_lang, $order_by = null, $order_way = null, $start = 0, $limit = null, $id_lang_shop = null) { if (!Validate::isTableOrIdentifier($this->table)) { die('filter is corrupted'); } if (empty($order_by)) { $order_by = Tools::getValue($this->table.'Orderby', $this->_defaultOrderBy); } if (empty($order_way)) { $order_way = Tools::getValue($this->table.'Orderway', 'ASC'); } // Try and obtain getList arguments from $_GET $order_by = Tools::getValue($this->table.'Orderby'); $order_way = Tools::getValue($this->table.'Orderway'); // Validate the orderBy and orderWay fields switch ($order_by) { case 'filename': case 'filesize': case 'date': case 'age': break; default: $order_by = 'date'; } switch ($order_way) { case 'asc': case 'desc': break; default: $order_way = 'desc'; } if (empty($limit)) { $limit = ((!isset($this->context->cookie->{$this->table.'_pagination'})) ? $this->_pagination[0] : $limit = $this->context->cookie->{$this->table.'_pagination'}); } $limit = (int)Tools::getValue('pagination', $limit); $this->context->cookie->{$this->table.'_pagination'} = $limit; /* Determine offset from current page */ if (!empty($_POST['submitFilter'.$this->list_id]) && is_numeric($_POST['submitFilter'.$this->list_id])) { $start = (int)$_POST['submitFilter'.$this->list_id] - 1 * $limit; } $this->_lang = (int)$id_lang; $this->_orderBy = $order_by; $this->_orderWay = strtoupper($order_way); $this->_list = array(); // Find all the backups $dh = @opendir(PrestaShopBackup::getBackupPath()); if ($dh === false) { $this->errors[] = Tools::displayError('Unable to open your backup directory'); return; } while (($file = readdir($dh)) !== false) { if (preg_match('/^([_a-zA-Z0-9\-]*[\d]+-[a-z\d]+)\.sql(\.gz|\.bz2)?$/', $file, $matches) == 0) { continue; } $timestamp = (int)$matches[1]; $date = date('Y-m-d H:i:s', $timestamp); $age = time() - $timestamp; if ($age < 3600) { $age = '< 1 '.$this->l('Hour', 'AdminTab', false, false); } elseif ($age < 86400) { $age = floor($age / 3600); $age = $age.' '.(($age == 1) ? $this->l('Hour', 'AdminTab', false, false) : $this->l('Hours', 'AdminTab', false, false)); } else { $age = floor($age / 86400); $age = $age.' '.(($age == 1) ? $this->l('Day') : $this->l('Days', 'AdminTab', false, false)); } $size = filesize(PrestaShopBackup::getBackupPath($file)); $this->_list[] = array( 'filename' => $file, 'age' => $age, 'date' => $date, 'filesize' => number_format($size / 1000, 2).' Kb', 'timestamp' => $timestamp, 'filesize_sort' => $size, ); } closedir($dh); $this->_listTotal = count($this->_list); // Sort the _list based on the order requirements switch ($this->_orderBy) { case 'filename': $this->sort_by = 'filename'; $sorter = 'strSort'; break; case 'filesize': $this->sort_by = 'filesize_sort'; $sorter = 'intSort'; break; case 'age': case 'date': $this->sort_by = 'timestamp'; $sorter = 'intSort'; break; } usort($this->_list, array($this, $sorter)); $this->_list = array_slice($this->_list, $start, $limit); } public function intSort($a, $b) { return $this->_orderWay == 'ASC' ? $a[$this->sort_by] - $b[$this->sort_by] : $b[$this->sort_by] - $a[$this->sort_by]; } public function strSort($a, $b) { return $this->_orderWay == 'ASC' ? strcmp($a[$this->sort_by], $b[$this->sort_by]) : strcmp($b[$this->sort_by], $a[$this->sort_by]); } }