* @copyright 2007-2011 PrestaShop SA * @version Release: $Revision: 9790 $ * @license http://opensource.org/licenses/osl-3.0.php Open Software License (OSL 3.0) * International Registered Trademark & Property of PrestaShop SA */ class AdminModulesControllerCore extends AdminController { /* ** @var array map with $_GET keywords and their callback */ private $map = array( 'install' => 'install', 'uninstall' => 'uninstall', 'configure' => 'getContent', 'delete' => 'delete' ); private $list_modules_categories = array(); private $list_partners_modules = array(); private $list_natives_modules = array(); private $cache_file_modules_list = '/config/modules_list.xml'; private $xml_modules_list = 'http://api.prestashop.com/xml/modules_list.xml'; private $nb_modules_total = 0; private $nb_modules_installed = 0; private $nb_modules_activated = 0; private $serial_modules = ''; private $modules_authors = array(); private $id_employee; private $iso_default_country; private $filter_configuration = array(); private $addons_url = 'https://addons.prestashop.com/webservice/151/'; private $logged_on_addons = false; private $cache_file_default_country_modules_list = '/config/default_country_modules_list.xml'; private $cache_file_customer_modules_list = '/config/customer_modules_list.xml'; /* ** Admin Modules Controller Constructor ** Init list modules categories ** Load id employee ** Load filter configuration ** Load cache file */ public function __construct() { parent::__construct(); include_once(_PS_ADMIN_DIR_.'/../tools/tar/Archive_Tar.php'); // Set the modules categories $this->list_modules_categories['administration']['name'] = $this->l('Administration'); $this->list_modules_categories['advertising_marketing']['name'] = $this->l('Advertising & Marketing'); $this->list_modules_categories['analytics_stats']['name'] = $this->l('Analytics & Stats'); $this->list_modules_categories['billing_invoicing']['name'] = $this->l('Billing & Invoicing'); $this->list_modules_categories['checkout']['name'] = $this->l('Checkout'); $this->list_modules_categories['content_management']['name'] = $this->l('Content Management'); $this->list_modules_categories['export']['name'] = $this->l('Export'); $this->list_modules_categories['front_office_features']['name'] = $this->l('Front Office Features'); $this->list_modules_categories['i18n_localization']['name'] = $this->l('I18n & Localization'); $this->list_modules_categories['merchandizing']['name'] = $this->l('Merchandizing'); $this->list_modules_categories['migration_tools']['name'] = $this->l('Migration Tools'); $this->list_modules_categories['payments_gateways']['name'] = $this->l('Payments & Gateways'); $this->list_modules_categories['payment_security']['name'] = $this->l('Payment Security'); $this->list_modules_categories['pricing_promotion']['name'] = $this->l('Pricing & Promotion'); $this->list_modules_categories['quick_bulk_update']['name'] = $this->l('Quick / Bulk update'); $this->list_modules_categories['search_filter']['name'] = $this->l('Search & Filter'); $this->list_modules_categories['seo']['name'] = $this->l('SEO'); $this->list_modules_categories['shipping_logistics']['name'] = $this->l('Shipping & Logistics'); $this->list_modules_categories['slideshows']['name'] = $this->l('Slideshows'); $this->list_modules_categories['smart_shopping']['name'] = $this->l('Smart Shopping'); $this->list_modules_categories['market_place']['name'] = $this->l('Market Place'); $this->list_modules_categories['social_networks']['name'] = $this->l('Social Networks'); $this->list_modules_categories['others']['name'] = $this->l('Other Modules'); // Set Id Employee, Iso Default Country and Filter Configuration $this->id_employee = (int)$this->context->employee->id; $this->iso_default_country = $this->context->country->iso_code; $this->filter_configuration = Configuration::getMultiple(array( 'PS_SHOW_TYPE_MODULES_'.(int)$this->id_employee, 'PS_SHOW_COUNTRY_MODULES_'.(int)$this->id_employee, 'PS_SHOW_INSTALLED_MODULES_'.(int)$this->id_employee, 'PS_SHOW_ENABLED_MODULES_'.(int)$this->id_employee, 'PS_SHOW_CAT_MODULES_'.(int)$this->id_employee, )); // Load cache file modules list (natives and partners modules) $xmlModules = false; if (file_exists(_PS_ROOT_DIR_.$this->cache_file_modules_list)) $xmlModules = @simplexml_load_file(_PS_ROOT_DIR_.$this->cache_file_modules_list); if ($xmlModules) foreach($xmlModules->children() as $xmlModule) foreach($xmlModule->children() as $module) foreach($module->attributes() as $key => $value) { if ($xmlModule->attributes() == 'native' && $key == 'name') $this->list_natives_modules[] = (string)$value; if ($xmlModule->attributes() == 'partner' && $key == 'name') $this->list_partners_modules[] = (string)$value; } // Check if logged on Addons if (isset($this->context->cookie->username_addons) && isset($this->context->cookie->password_addons) && !empty($this->context->cookie->username_addons) && !empty($this->context->cookie->password_addons)) $this->logged_on_addons = true; } /* ** Ajax Request Methods ** ** if modules_list.xml is outdated, ** this function will re-upload it from prestashop.com ** ** @return null */ public function isFresh($file, $timeout = 604800000) { if (file_exists(_PS_ROOT_DIR_.$file)) return ((time() - filemtime(_PS_ROOT_DIR_.$this->cache_file_modules_list)) < $timeout); else return false; } public function refresh($file_to_refresh, $external_file) { return file_put_contents(_PS_ROOT_DIR_.$file_to_refresh, Tools::file_get_contents($external_file)); } public function ajaxProcessRefreshModuleList() { // Refresh modules_list.xml every week if (!$this->isFresh($this->cache_file_modules_list, 604800)) { if ($this->refresh($this->cache_file_modules_list, $this->xml_modules_list)) $this->status = 'refresh'; else $this->status = 'error'; } else $this->status = 'cache'; // If logged to Addons Webservices, refresh default country modules list every day if ($this->logged_on_addons && $this->status != 'error') { if (!$this->isFresh($this->cache_file_default_country_modules_list, 86400)) { if ($this->refresh($this->cache_file_default_country_modules_list, $this->addons_url.'listing/'.strtolower(Configuration::get('PS_LOCALE_COUNTRY')))) $this->status = 'refresh'; else $this->status = 'error'; } else $this->status = 'cache'; } // If logged to Addons Webservices, refresh customer modules list every day if ($this->logged_on_addons && $this->status != 'error') { if (!$this->isFresh($this->cache_file_customer_modules_list, 60)) { if ($this->refresh($this->cache_file_customer_modules_list, $this->addons_url.'listing/customer/'.pSQL(trim($this->context->cookie->username_addons)).'/'.pSQL(trim($this->context->cookie->password_addons)))) $this->status = 'refresh'; else $this->status = 'error'; } else $this->status = 'cache'; } } public function displayAjaxRefreshModuleList() { echo Tools::jsonEncode(array('status' => $this->status)); } public function ajaxProcessLogOnAddonsWebservices() { $content = Tools::file_get_contents($this->addons_url.'check_customer/'.pSQL(trim(Tools::getValue('username_addons'))).'/'.pSQL(trim(Tools::getValue('password_addons')))); $xml = @simplexml_load_string($content, NULL, LIBXML_NOCDATA); if (!$xml) die('KO'); $result = strtoupper((string)$xml->msg); if (!in_array($result, array('OK', 'KO'))) die ('KO'); if ($result == 'OK') { $this->context->cookie->username_addons = pSQL(trim(Tools::getValue('username_addons'))); $this->context->cookie->password_addons = pSQL(trim(Tools::getValue('password_addons'))); } die($result); } public function ajaxProcessReloadModulesList() { if (Tools::getValue('filterCategory')) Configuration::updateValue('PS_SHOW_CAT_MODULES_'.(int)$this->id_employee, Tools::getValue('filterCategory')); if (Tools::getValue('unfilterCategory')) Configuration::updateValue('PS_SHOW_CAT_MODULES_'.(int)$this->id_employee, ''); $this->initContent(); $this->context->smarty->display('modules/list.tpl'); exit; } /* ** Get current URL ** ** @param array $remove List of keys to remove from URL ** @return string */ protected function getCurrentUrl($remove = array()) { $url = $_SERVER['REQUEST_URI']; if (!$remove) return $url; if (!is_array($remove)) $remove = array($remove); $url = preg_replace('#(?<=&|\?)(' . implode('|', $remove) . ')=.*?(&|$)#i', '', $url); $len = strlen($url); if ($url[$len - 1] == '&') $url = substr($url, 0, $len - 1); return $url; } private function extractArchive($file) { $success = false; if (substr($file, -4) == '.zip') { if (Tools::ZipExtract($file, _PS_MODULE_DIR_)) $success = true; else $this->_errors[] = Tools::displayError('Error while extracting module (file may be corrupted).'); } else { $archive = new Archive_Tar($file); if ($archive->extract(_PS_MODULE_DIR_)) $success = true; else $this->_errors[] = Tools::displayError('Error while extracting module (file may be corrupted).'); } @unlink($file); if ($success) Tools::redirectAdmin(self::$currentIndex.'&conf=8'.'&token='.$this->token); } private function recursiveDeleteOnDisk($dir) { if (strpos(realpath($dir), realpath(_PS_MODULE_DIR_)) === false) return ; if (is_dir($dir)) { $objects = scandir($dir); foreach ($objects as $object) if ($object != "." && $object != "..") { if (filetype($dir."/".$object) == "dir") $this->recursiveDeleteOnDisk($dir."/".$object); else unlink($dir."/".$object); } reset($objects); rmdir($dir); } } /* ** Filter Configuration Methods ** Set and reset filter configuration */ private function setFilterModules($module_type, $country_module_value, $module_install, $module_status) { Configuration::updateValue('PS_SHOW_TYPE_MODULES_'.(int)$this->id_employee, $module_type); Configuration::updateValue('PS_SHOW_COUNTRY_MODULES_'.(int)$this->id_employee, $country_module_value); Configuration::updateValue('PS_SHOW_INSTALLED_MODULES_'.(int)$this->id_employee, $module_install); Configuration::updateValue('PS_SHOW_ENABLED_MODULES_'.(int)$this->id_employee, $module_status); } private function resetFilterModules() { Configuration::updateValue('PS_SHOW_TYPE_MODULES_'.(int)$this->id_employee, 'allModules'); Configuration::updateValue('PS_SHOW_COUNTRY_MODULES_'.(int)$this->id_employee, 0); Configuration::updateValue('PS_SHOW_INSTALLED_MODULES_'.(int)$this->id_employee, 'installedUninstalled'); Configuration::updateValue('PS_SHOW_ENABLED_MODULES_'.(int)$this->id_employee, 'enabledDisabled'); Configuration::updateValue('PS_SHOW_CAT_MODULES_'.(int)$this->id_employee, ''); } /* ** Post Process Filter ** */ public function postProcessFilterModules() { $this->setFilterModules(Tools::getValue('module_type'), Tools::getValue('country_module_value'), Tools::getValue('module_install'), Tools::getValue('module_status')); Tools::redirectAdmin(self::$currentIndex.'&token='.$this->token); } public function postProcessResetFilterModules() { $this->resetFilterModules(); Tools::redirectAdmin(self::$currentIndex.'&token='.$this->token); } public function postProcessFilterCategory() { // Save configuration and redirect employee Configuration::updateValue('PS_SHOW_CAT_MODULES_'.(int)$this->id_employee, Tools::getValue('filterCategory')); Tools::redirectAdmin(self::$currentIndex.'&token='.$this->token); } public function postProcessUnfilterCategory() { // Save configuration and redirect employee Configuration::updateValue('PS_SHOW_CAT_MODULES_'.(int)$this->id_employee, ''); Tools::redirectAdmin(self::$currentIndex.'&token='.$this->token); } /* ** Post Process Module CallBack ** */ public function postProcessReset() { if ($this->tabAccess['edit'] === '1') { $module = Module::getInstanceByName(Tools::getValue('module_name')); if (Validate::isLoadedObject($module)) { if (!$module->getPermission('configure')) $this->_errors[] = Tools::displayError('You do not have the permission to use this module'); else { if ($module->uninstall()) if ($module->install()) Tools::redirectAdmin(self::$currentIndex.'&conf=21'.'&token='.$this->token.'&tab_module='.$module->tab.'&module_name='.$module->name.'&anchor=anchor'.ucfirst($module->name)); else $this->_errors[] = Tools::displayError('Cannot install module'); else $this->_errors[] = Tools::displayError('Cannot uninstall module'); } } else $this->_errors[] = Tools::displayError('Cannot load module object'); } else $this->_errors[] = Tools::displayError('You do not have permission to add here.'); } public function postProcessDownload() { // PrestaShop demo mode if (_PS_MODE_DEMO_) { $this->_errors[] = Tools::displayError('This functionnality has been disabled.'); return; } // Try to upload and unarchive the module if ($this->tabAccess['add'] === '1') { if (!isset($_FILES['file']['tmp_name']) OR empty($_FILES['file']['tmp_name'])) $this->_errors[] = $this->l('no file selected'); elseif (substr($_FILES['file']['name'], -4) != '.tar' AND substr($_FILES['file']['name'], -4) != '.zip' AND substr($_FILES['file']['name'], -4) != '.tgz' AND substr($_FILES['file']['name'], -7) != '.tar.gz') $this->_errors[] = Tools::displayError('Unknown archive type'); elseif (!@copy($_FILES['file']['tmp_name'], _PS_MODULE_DIR_.$_FILES['file']['name'])) $this->_errors[] = Tools::displayError('An error occurred while copying archive to module directory.'); else $this->extractArchive(_PS_MODULE_DIR_.$_FILES['file']['name']); } else $this->_errors[] = Tools::displayError('You do not have permission to add here.'); } public function postProcessEnable() { if ($this->tabAccess['edit'] === '1') { $module = Module::getInstanceByName(Tools::getValue('module_name')); if (Validate::isLoadedObject($module)) { if (!$module->getPermission('configure')) $this->_errors[] = Tools::displayError('You do not have the permission to use this module'); else { if (Tools::getValue('enable')) $module->enable(); else $module->disable(); Tools::redirectAdmin($this->getCurrentUrl('enable')); } } else $this->_errors[] = Tools::displayError('Cannot load module object'); } else $this->_errors[] = Tools::displayError('You do not have permission to add here.'); } public function postProcessDelete() { if ($this->tabAccess['delete'] === '1') { if (Tools::getValue('module_name') != '') { $module = Module::getInstanceByName(Tools::getValue('module_name')); if (Validate::isLoadedObject($module) AND !$module->getPermission('configure')) $this->_errors[] = Tools::displayError('You do not have the permission to use this module'); else { $moduleDir = _PS_MODULE_DIR_.str_replace(array('.', '/', '\\'), array('', '', ''), Tools::getValue('module_name')); $this->recursiveDeleteOnDisk($moduleDir); Tools::redirectAdmin(self::$currentIndex.'&conf=22&token='.$this->token.'&tab_module='.Tools::getValue('tab_module').'&module_name='.Tools::getValue('module_name')); } } } else $this->_errors[] = Tools::displayError('You do not have permission to delete here.'); } public function postProcessCallback() { $return = false; foreach ($this->map as $key => $method) { $modules = Tools::getValue($key); if (strpos($modules, '|')) $modules = explode('|', $modules); else $modules = empty($modules) ? false : array($modules); $module_errors = array(); if ($modules) foreach ($modules AS $name) { // If Addons module, download and unzip it before installing it if ($this->logged_on_addons && !is_dir('../modules/'.$name.'/')) { $filesList = array($this->cache_file_default_country_modules_list, $this->cache_file_customer_modules_list); foreach ($filesList as $file) if (file_exists(_PS_ROOT_DIR_.$file)) { $content = Tools::file_get_contents(_PS_ROOT_DIR_.$file); $xml = @simplexml_load_string($content, NULL, LIBXML_NOCDATA); foreach ($xml->module as $modaddons) if ($name == $modaddons->name && isset($modaddons->id)) if (@copy($this->addons_url.'module/'.pSQL($modaddons->id).'/'.pSQL(trim($this->context->cookie->username_addons)).'/'.pSQL(trim($this->context->cookie->password_addons)), '../modules/'.$modaddons->name.'.zip')) $this->extractArchive('../modules/'.$modaddons->name.'.zip'); } } // Check potential error if (!($module = Module::getInstanceByName(urldecode($name)))) $this->_errors[] = $this->l('module not found'); elseif ($key == 'install' AND $this->tabAccess['add'] !== '1') $this->_errors[] = Tools::displayError('You do not have permission to install a module.'); elseif ($key == 'uninstall' AND ($this->tabAccess['delete'] !== '1' OR !$module->getPermission('configure'))) $this->_errors[] = Tools::displayError('You do not have permission to delete this module.'); elseif ($key == 'configure' AND ($this->tabAccess['edit'] !== '1' OR !$module->getPermission('configure'))) $this->_errors[] = Tools::displayError('You do not have permission to configure this module.'); elseif ($key == 'install' AND Module::isInstalled($module->name)) $this->_errors[] = Tools::displayError('This module is already installed:').' '.$module->name; elseif ($key == 'uninstall' AND !Module::isInstalled($module->name)) $this->_errors[] = Tools::displayError('This module is already uninstalled:').' '.$module->name; else { // If we install a module, force temporary global context for multishop if (Shop::isFeatureActive() && Context::shop() != Shop::CONTEXT_ALL) { // If we install a module, force temporary global context for multishop if (Shop::isFeatureActive() && Context::shop() != Shop::CONTEXT_ALL && $method != 'getContent') { Context::getContext()->tmpOldShop = clone(Context::getContext()->shop); Context::getContext()->shop = new Shop(); Configuration::updateValue('RSS_FEED_TITLE', 'lol'); } } if (((method_exists($module, $method) && ($echo = $module->{$method}())) || ($echo = ' ')) AND $key == 'configure' AND Module::isInstalled($module->name)) { $backlink = self::$currentIndex.'&token='.$this->token.'&tab_module='.$module->tab.'&module_name='.$module->name; $hooklink = 'index.php?tab=AdminModulesPositions&token='.Tools::getAdminTokenLite('AdminModulesPositions').'&show_modules='.(int)$module->id; $tradlink = 'index.php?tab=AdminTranslations&token='.Tools::getAdminTokenLite('AdminTranslations').'&type=modules&lang='; $toolbar = '
| '.$this->l('Module').' '.$module->name.' | '.$this->l('Back').' | '.$this->l('Manage hooks').' | '.$this->l('Manage translations:').' ';
foreach (Language::getLanguages(false) AS $language)
$toolbar .= ' ';
$toolbar .= ' |
|---|---|---|---|
| active) ? 'checked="checked"' : '').' '.$activateOnclick.' /> '.$this->l('Activate module for').' '; if ($this->context->shop->getContextType() == Shop::CONTEXT_SHOP) $toolbar .= 'shop '.$this->context->shop->name.''; elseif ($this->context->shop->getContextType() == Shop::CONTEXT_GROUP) $toolbar .= 'all shops of group shop '.$this->context->shop->getGroup()->name.''; else $toolbar .= 'all shops'; $toolbar .= ' | |||