* Copyright (C) 2003 Jean-Louis Bergamo * Copyright (C) 2004-2024 Laurent Destailleur * Copyright (C) 2004 Eric Seigne * Copyright (C) 2005-2017 Regis Houssin * Copyright (C) 2011-2023 Juanjo Menent * Copyright (C) 2015 Jean-François Ferry * Copyright (C) 2015 Raphaël Doursenaud * Copyright (C) 2018 Nicolas ZABOURI * Copyright (C) 2021-2025 Frédéric France * Copyright (C) 2024 MDW * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ /** * \file htdocs/admin/modules.php * \brief Page to activate/disable all modules */ if (!defined('CSRFCHECK_WITH_TOKEN') && (empty($_GET['action']) || $_GET['action'] != 'reset')) { // We force security except to disable modules so we can do it if a problem occurs on a module define('CSRFCHECK_WITH_TOKEN', '1'); // Force use of CSRF protection with tokens even for GET } // Load Dolibarr environment require '../main.inc.php'; require_once DOL_DOCUMENT_ROOT.'/core/lib/admin.lib.php'; require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php'; require_once DOL_DOCUMENT_ROOT.'/core/lib/geturl.lib.php'; require_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php'; require_once DOL_DOCUMENT_ROOT.'/core/class/events.class.php'; require_once DOL_DOCUMENT_ROOT.'/core/modules/DolibarrModules.class.php'; require_once DOL_DOCUMENT_ROOT.'/admin/remotestore/class/externalModules.class.php'; ' @phan-var-force string $dolibarr_main_url_root_alt '; /** * @var Conf $conf * @var DoliDB $db * @var HookManager $hookmanager * @var Societe $mysoc * @var Translate $langs * @var User $user * * @var string $dolibarr_main_url_root_alt */ // Load translation files required by the page $langs->loadLangs(array("errors", "admin", "modulebuilder")); // if we set another view list mode, we keep it (till we change one more time) if (GETPOSTISSET('mode')) { $mode = GETPOST('mode', 'alpha'); if ($mode == 'common' || $mode == 'commonkanban') { dolibarr_set_const($db, "MAIN_MODULE_SETUP_ON_LIST_BY_DEFAULT", $mode, 'chaine', 0, '', $conf->entity); } } else { $mode = getDolGlobalString('MAIN_MODULE_SETUP_ON_LIST_BY_DEFAULT', 'commonkanban'); } $action = GETPOST('action', 'aZ09'); $page_y = GETPOSTINT('page_y'); $optioncss = GETPOST('optioncss', 'aZ09'); $value = GETPOST('value', 'alpha'); $search_keyword = GETPOST('search_keyword', 'alpha'); $search_status = GETPOST('search_status', 'alpha'); $search_nature = GETPOST('search_nature', 'alpha'); $search_version = GETPOST('search_version', 'alpha'); // For remotestore search $options = array(); $options['per_page'] = 11; $options['no_page'] = (GETPOSTINT('no_page') ? GETPOSTINT('no_page') : 1); $options['categorie'] = (GETPOSTINT('categorie') ? GETPOSTINT('categorie') : 0); $options['search'] = GETPOST('search_keyword', 'alpha'); // If it is a new search, we reset page to 1 if (GETPOST('buttonsubmit', 'alphanohtml', 2)) { $options['no_page'] = 1; } // MAIN_ENABLE_EXTERNALMODULES_DOLISTORE is 1 if we enabled the dolistore modules $options['search_source_dolistore'] = getDolGlobalInt('MAIN_ENABLE_EXTERNALMODULES_DOLISTORE'); // MAIN_ENABLE_EXTERNALMODULES_COMMUNITY is 1 if we enabled the community modules $options['search_source_github'] = getDolGlobalInt('MAIN_ENABLE_EXTERNALMODULES_COMMUNITY'); if (!$user->admin) { accessforbidden(); } $familyinfo = array( 'hr' => array('position' => '001', 'label' => $langs->trans("ModuleFamilyHr")), 'crm' => array('position' => '006', 'label' => $langs->trans("ModuleFamilyCrm")), 'srm' => array('position' => '007', 'label' => $langs->trans("ModuleFamilySrm")), 'financial' => array('position' => '009', 'label' => $langs->trans("ModuleFamilyFinancial")), 'products' => array('position' => '012', 'label' => $langs->trans("ModuleFamilyProducts")), 'projects' => array('position' => '015', 'label' => $langs->trans("ModuleFamilyProjects")), 'ecm' => array('position' => '018', 'label' => $langs->trans("ModuleFamilyECM")), 'technic' => array('position' => '021', 'label' => $langs->trans("ModuleFamilyTechnic")), 'portal' => array('position' => '040', 'label' => $langs->trans("ModuleFamilyPortal")), 'interface' => array('position' => '050', 'label' => $langs->trans("ModuleFamilyInterface")), 'base' => array('position' => '060', 'label' => $langs->trans("ModuleFamilyBase")), 'other' => array('position' => '100', 'label' => $langs->trans("ModuleFamilyOther")), ); $param = ''; if (!GETPOST('buttonreset', 'alpha')) { if ($search_keyword) { $param .= '&search_keyword='.urlencode($search_keyword); } if ($search_status && $search_status != '-1') { $param .= '&search_status='.urlencode($search_status); } if ($search_nature && $search_nature != '-1') { $param .= '&search_nature='.urlencode($search_nature); } if ($search_version && $search_version != '-1') { $param .= '&search_version='.urlencode($search_version); } } $dirins = DOL_DOCUMENT_ROOT.'/custom'; $urldolibarrmodules = 'https://www.dolistore.com/'; // Initialize a technical object to manage hooks of page. Note that conf->hooks_modules contains an array of hook context $hookmanager->initHooks(array('adminmodules', 'globaladmin')); // Increase limit of time. Works only if we are not in safe mode $max_execution_time_for_deploy = getDolGlobalInt('MODULE_UPLOAD_MAX_EXECUTION_TIME', 300); // 5mn if not defined if (!empty($max_execution_time_for_deploy)) { $err = error_reporting(); error_reporting(0); // Disable all errors //error_reporting(E_ALL); @set_time_limit($max_execution_time_for_deploy); error_reporting($err); } // Other method - TODO is this required ? $max_time = @ini_get("max_execution_time"); if ($max_time && $max_time < $max_execution_time_for_deploy) { dol_syslog("max_execution_time=".$max_time." is lower than max_execution_time_for_deploy=".$max_execution_time_for_deploy.". We try to increase it dynamically."); @ini_set("max_execution_time", $max_execution_time_for_deploy); // This work only if safe mode is off. also web servers has timeout of 300 } $dolibarrdataroot = preg_replace('/([\\/]+)$/i', '', DOL_DATA_ROOT); $allowonlineinstall = true; $allowfromweb = 1; if (dol_is_file($dolibarrdataroot.'/installmodules.lock')) { $allowonlineinstall = false; } $debug = false; $remotestore = new ExternalModules($debug); if ($mode == 'marketplace') { // Make remote calls if (GETPOSTINT('dol_resetcache')) { dol_delete_file($remotestore->cache_file); } $remotestore->loadRemoteSources(false); } $object = new stdClass(); $now = dol_now(); /* * Actions */ $formconfirm = ''; $parameters = array(); $reshook = $hookmanager->executeHooks('doActions', $parameters, $object, $action); // Note that $action and $object may have been modified by some hooks if ($reshook < 0) { setEventMessages($hookmanager->error, $hookmanager->errors, 'errors'); } if (GETPOST('buttonreset', 'alpha')) { $search_keyword = ''; $search_status = ''; $search_nature = ''; $search_version = ''; } if ($action == 'install' && $allowonlineinstall) { $error = 0; $modulenameval = ''; // $original_file should match format module_modulename-x.y[.z].zip $tmpfile = $_FILES['fileinstall']['tmp_name']; $original_file = basename($_FILES["fileinstall"]["name"]); $original_file = preg_replace('/\s*\(\d+\)\.zip$/i', '.zip', $original_file); $newfile = dol_sanitizePathName($conf->admin->dir_temp.'/'.$original_file.'/'.$original_file); if (!$original_file) { $langs->load("Error"); setEventMessages($langs->trans("ErrorModuleFileRequired"), null, 'warnings'); $error++; } else { if (!$error && !preg_match('/\.zip$/i', $original_file)) { $langs->load("errors"); setEventMessages($langs->trans("ErrorFileMustBeADolibarrPackage", $original_file), null, 'errors'); $error++; } if (!$error && !preg_match('/^(module[a-zA-Z0-9]*_|theme_|).*\-([0-9][0-9\.]*)(\s\(\d+\)\s)?\.zip$/i', $original_file)) { $langs->load("errors"); setEventMessages($langs->trans("ErrorFilenameDosNotMatchDolibarrPackageRules", $original_file, 'modulename-x[.y.z].zip'), null, 'errors'); $error++; } if (empty($tmpfile)) { $langs->load("errors"); setEventMessages($langs->trans("ErrorFileNotUploaded"), null, 'errors'); $error++; } } if (!$error) { if ($original_file) { @dol_delete_dir_recursive($conf->admin->dir_temp.'/'.$original_file); dol_mkdir($conf->admin->dir_temp.'/'.$original_file); } $tmpdir = preg_replace('/\.zip$/i', '', $original_file).'.dir'; if ($tmpdir) { @dol_delete_dir_recursive($conf->admin->dir_temp.'/'.$tmpdir); dol_mkdir($conf->admin->dir_temp.'/'.$tmpdir); } $result = dol_move_uploaded_file($tmpfile, $newfile, 1, 0, $_FILES['fileinstall']['error']); if ((int) $result > 0) { $resultuncompress = dol_uncompress($newfile, $conf->admin->dir_temp.'/'.$tmpdir); if (!empty($resultuncompress['error'])) { $langs->load("errors"); setEventMessages($langs->trans($resultuncompress['error'], $original_file), null, 'errors'); $error++; } else { // Now we move the dir of the module $modulename = preg_replace('/module_/', '', $original_file); $modulename = preg_replace('/\-([0-9][0-9\.]*)\.zip$/i', '', $modulename); // Search dir $modulename $modulenamedir = $conf->admin->dir_temp.'/'.$tmpdir.'/'.$modulename; // Example ./mymodule if (!dol_is_dir($modulenamedir)) { $modulenamedir = $conf->admin->dir_temp.'/'.$tmpdir.'/htdocs/'.$modulename; // Example ./htdocs/mymodule //var_dump($modulenamedir); if (!dol_is_dir($modulenamedir)) { setEventMessages($langs->trans("ErrorModuleFileSeemsToHaveAWrongFormat").'
'.$langs->trans("ErrorModuleFileSeemsToHaveAWrongFormat2", $modulename, 'htdocs/'.$modulename), null, 'errors'); $error++; } } dol_syslog("Uncompress of module file is a success."); // Load module into $objMod /* $modulesdir = array($modulenamedir.'/core/modules/'); foreach ($modulesdir as $dir) { // Load modules attributes in arrays (name, numero, orders) from dir directory //print $dir."\n
"; dol_syslog("Scan directory ".$dir." for module descriptor files (modXXX.class.php)"); $handle = @opendir($dir); if (is_resource($handle)) { while (($file = readdir($handle)) !== false) { print $dir." ".$file."\n
"; if (is_readable($dir.$file) && substr($file, 0, 3) == 'mod' && substr($file, dol_strlen($file) - 10) == '.class.php') { $modName = substr($file, 0, dol_strlen($file) - 10); if ($modName) { try { $res = include_once $dir.$file; // A class already exists in a different file will send a non catchable fatal error. $modName = substr($file, 0, dol_strlen($file) - 10); if ($modName) { if (class_exists($modName)) { $objMod = new $modName($db); '@phan-var-force DolibarrModules $objMod'; //var_dump($objMod); } } } catch(Exception $e) { // Nothing done } } } } } } */ // Check if module is in the remote malware list if (!$error) { if (GETPOST('checkforcompliance') == 'on') { try { $res = include_once DOL_DOCUMENT_ROOT.'/core/modules/DolibarrModules.class.php'; $dolibarrmodule = new DolibarrModules($db); $checkRes = $dolibarrmodule->checkForcompliance($modulename); if (!is_numeric($checkRes) && $checkRes != '') { $langs->load("errors"); setEventMessages($modulename.' : '.$langs->trans($checkRes), null, 'errors'); $error++; } } catch (Exception $e) { // Nothing done } } } if (!$error) { // TODO Make more test ??? } // We check if this is a metapackage (and wecomplete with child packages) $modulenamearrays = array(); if (dol_is_file($modulenamedir.'/metapackage.conf')) { // This is a meta package $metafile = file_get_contents($modulenamedir.'/metapackage.conf'); $modulenamearrays = explode("\n", $metafile); } $modulenamearrays[$modulename] = $modulename; //var_dump($modulenamearrays);exit; // Lop on each packages (can have several if package is a metapackage) if (! $error) { foreach ($modulenamearrays as $modulenameval) { if (strpos($modulenameval, '#') === 0) { continue; // Discard comments } if (strpos($modulenameval, '//') === 0) { continue; // Discard comments } if (!trim($modulenameval)) { continue; } // Now we install the module if (!$error) { @dol_delete_dir_recursive($dirins.'/'.$modulenameval); // delete the target directory $submodulenamedir = $conf->admin->dir_temp.'/'.$tmpdir.'/'.$modulenameval; if (!dol_is_dir($submodulenamedir)) { $submodulenamedir = $conf->admin->dir_temp.'/'.$tmpdir.'/htdocs/'.$modulenameval; } dol_syslog("We copy now directory ".$submodulenamedir." into target dir ".$dirins.'/'.$modulenameval); $resultcopy = dolCopyDir($submodulenamedir, $dirins.'/'.$modulenameval, '0444', 1); if ($resultcopy <= 0) { dol_syslog('Failed to call dolCopyDir result='.$resultcopy." with param ".$submodulenamedir." and ".$dirins.'/'.$modulenameval, LOG_WARNING); $langs->load("errors"); setEventMessages($langs->trans("ErrorFailToCopyDir", $submodulenamedir, $dirins.'/'.$modulenameval), null, 'errors'); $error++; } } } } } } else { setEventMessages($langs->trans("ErrorFailToRenameFile", $tmpfile, $newfile).' - code = '.$result, null, 'errors'); $error++; } } // Add event purge $securityevent = new Events($db); if ($error) { $text = $langs->trans("SecurityModuleDeploymentError", dol_sanitizePathName($_FILES["fileinstall"]["name"])); $securityevent->type = 'MODULE_DEPLOYMENT_ERROR'; } else { $text = $langs->trans("SecurityModuleDeploymentSuccess", dol_sanitizePathName($_FILES["fileinstall"]["name"])); $securityevent->type = 'MODULE_DEPLOYMENT_SUCCESS'; } $securityevent->dateevent = $now; $securityevent->description = $text; $resultcreateevent = $securityevent->create($user); if (!$error) { $searchParams = array( 'search_keyword' => $modulenameval, 'search_status' => '-1', 'search_nature' => '-1', 'search_version' => '-1' ); $queryString = http_build_query($searchParams); $redirectUrl = DOL_URL_ROOT . '/admin/modules.php?' . $queryString; $message = $langs->trans("SetupIsReadyForUse", $redirectUrl, $langs->transnoentitiesnoconv("Home").' - '.$langs->transnoentitiesnoconv("Setup").' - '.$langs->transnoentitiesnoconv("Modules")); setEventMessages($message, null, 'warnings'); } } elseif ($action == 'install' && !$allowonlineinstall) { httponly_accessforbidden("You try to bypass the protection to disallow deployment of an external module. Hack attempt ?"); } if ($action == 'set' && $user->admin) { // We made some check against evil eternal modules that try to low security options. $checkOldValue = getDolGlobalInt('CHECKLASTVERSION_EXTERNALMODULE'); $csrfCheckOldValue = getDolGlobalInt('MAIN_SECURITY_CSRF_WITH_TOKEN'); $resarray = activateModule($value); if ($checkOldValue != getDolGlobalInt('CHECKLASTVERSION_EXTERNALMODULE')) { setEventMessage($langs->trans('WarningModuleHasChangedLastVersionCheckParameter', $value), 'warnings'); } if ($csrfCheckOldValue != getDolGlobalInt('MAIN_SECURITY_CSRF_WITH_TOKEN')) { setEventMessage($langs->trans('WarningModuleHasChangedSecurityCsrfParameter', $value), 'warnings'); } dolibarr_set_const($db, "MAIN_IHM_PARAMS_REV", getDolGlobalInt('MAIN_IHM_PARAMS_REV') + 1, 'chaine', 0, '', $conf->entity); if (!empty($resarray['errors'])) { setEventMessages('', $resarray['errors'], 'errors'); } else { //var_dump($resarray);exit; if ($resarray['nbperms'] > 0) { $tmpsql = "SELECT COUNT(rowid) as nb FROM ".MAIN_DB_PREFIX."user WHERE admin <> 1"; $resqltmp = $db->query($tmpsql); if ($resqltmp) { $obj = $db->fetch_object($resqltmp); //var_dump($obj->nb);exit; if ($obj && $obj->nb > 1) { $msg = $langs->trans('ModuleEnabledAdminMustCheckRights'); setEventMessages($msg, null, 'warnings'); } } else { dol_print_error($db); } } } header("Location: ".$_SERVER["PHP_SELF"]."?mode=".$mode.$param.($page_y ? '&page_y='.$page_y : '')); exit; } elseif ($action == 'reset' && $user->admin && GETPOST('confirm') == 'yes') { $result = unActivateModule($value); dolibarr_set_const($db, "MAIN_IHM_PARAMS_REV", getDolGlobalInt('MAIN_IHM_PARAMS_REV') + 1, 'chaine', 0, '', $conf->entity); if ($result) { setEventMessages($result, null, 'errors'); } header("Location: ".$_SERVER["PHP_SELF"]."?mode=".$mode.$param.($page_y ? '&page_y='.$page_y : '')); exit; } elseif (getDolGlobalInt("MAIN_FEATURES_LEVEL") > 1 && $action == 'reload' && $user->admin && GETPOST('confirm') == 'yes') { $result = unActivateModule($value, 0); dolibarr_set_const($db, "MAIN_IHM_PARAMS_REV", getDolGlobalInt('MAIN_IHM_PARAMS_REV') + 1, 'chaine', 0, '', $conf->entity); if ($result) { setEventMessages($result, null, 'errors'); header("Location: ".$_SERVER["PHP_SELF"]."?mode=".$mode.$param.($page_y ? '&page_y='.$page_y : '')); } $resarray = activateModule($value, 0, 1); dolibarr_set_const($db, "MAIN_IHM_PARAMS_REV", (getDolGlobalInt('MAIN_IHM_PARAMS_REV') + 1), 'chaine', 0, '', $conf->entity); if (!empty($resarray['errors'])) { setEventMessages('', $resarray['errors'], 'errors'); } else { if ($resarray['nbperms'] > 0) { $tmpsql = "SELECT COUNT(rowid) as nb FROM ".MAIN_DB_PREFIX."user WHERE admin <> 1"; $resqltmp = $db->query($tmpsql); if ($resqltmp) { $obj = $db->fetch_object($resqltmp); if ($obj && $obj->nb > 1) { $msg = $langs->trans('ModuleEnabledAdminMustCheckRights'); setEventMessages($msg, null, 'warnings'); } } else { dol_print_error($db); } } } header("Location: ".$_SERVER["PHP_SELF"]."?mode=".$mode.$param.($page_y ? '&page_y='.$page_y : '')); exit; } /* * View */ $form = new Form($db); $morejs = array(); $morecss = array("/admin/remotestore/css/store.css"); // Set dir where external modules are installed if (!dol_is_dir($dirins)) { dol_mkdir($dirins); } $dirins_ok = (dol_is_dir($dirins)); $help_url = 'EN:First_setup|FR:Premiers_paramétrages|ES:Primeras_configuraciones'; llxHeader('', $langs->trans("Setup"), $help_url, '', 0, 0, $morejs, $morecss, '', 'mod-admin page-modules'); // Search modules dirs $modulesdir = dolGetModulesDirs(); $arrayofnatures = array( 'core' => array('label' => $langs->transnoentitiesnoconv("NativeModules")), 'external' => array('label' => $langs->transnoentitiesnoconv("External").' - ['.$langs->trans("AllPublishers").']') ); $arrayofwarnings = array(); // Array of warning each module want to show when activated $arrayofwarningsext = array(); // Array of warning each module want to show when we activate an external module $filename = array(); $modules = array(); $orders = array(); $categ = array(); $publisherlogoarray = array(); $i = 0; // is a sequencer of modules found $j = 0; // j is module number. Automatically affected if module number not defined. $modNameLoaded = array(); //if ($mode == 'common' || $mode == 'commonkanban') { // Load $modules (required for the badge count) foreach ($modulesdir as $dir) { // Load modules attributes in arrays (name, numero, orders) from dir directory //print $dir."\n
"; dol_syslog("Scan directory ".$dir." for module descriptor files (modXXX.class.php)"); $handle = @opendir($dir); if (is_resource($handle)) { while (($file = readdir($handle)) !== false) { //print "$i ".$file."\n
"; if (is_readable($dir.$file) && substr($file, 0, 3) == 'mod' && substr($file, dol_strlen($file) - 10) == '.class.php') { $modName = substr($file, 0, dol_strlen($file) - 10); if ($modName) { if (!empty($modNameLoaded[$modName])) { // In cache of already loaded modules ? $mesg = "Error: Module ".$modName." was found twice: Into ".$modNameLoaded[$modName]." and ".$dir.". You probably have an old file on your disk.
"; setEventMessages($mesg, null, 'warnings'); dol_syslog($mesg, LOG_ERR); continue; } try { $res = include_once $dir.$file; // A class already exists in a different file will send a non catchable fatal error. if (class_exists($modName)) { $objMod = new $modName($db); '@phan-var-force DolibarrModules $objMod'; /** @var DolibarrModules $objMod */ $modNameLoaded[$modName] = $dir; if (!$objMod->numero > 0 && $modName != 'modUser') { dol_syslog('The module descriptor '.$modName.' must have a numero property', LOG_ERR); } $j = $objMod->numero; $modulequalified = 1; // We discard modules according to features level (PS: if module is activated we always show it) $const_name = 'MAIN_MODULE_'.strtoupper(preg_replace('/^mod/i', '', get_class($objMod))); if ($objMod->version == 'development' && (!getDolGlobalString($const_name) && (getDolGlobalInt('MAIN_FEATURES_LEVEL') < 2))) { $modulequalified = 0; } if ($objMod->version == 'experimental' && (!getDolGlobalString($const_name) && (getDolGlobalInt('MAIN_FEATURES_LEVEL') < 1))) { $modulequalified = 0; } if (preg_match('/deprecated/', $objMod->version) && (!getDolGlobalString($const_name) && (getDolGlobalInt('MAIN_FEATURES_LEVEL') >= 0))) { $modulequalified = 0; } // We discard modules according to property ->hidden if (!empty($objMod->hidden)) { $modulequalified = 0; } if ($modulequalified > 0) { $publisher = dol_escape_htmltag($objMod->getPublisher()); $external = ($objMod->isCoreOrExternalModule() == 'external'); if ($external) { if ($publisher) { // Check if there is a logo forpublisher /* Do not show the company logo in combo. Make combo list dirty. if (!empty($objMod->editor_squarred_logo)) { $publisherlogoarray['external_'.$publisher] = img_picto('', $objMod->editor_squarred_logo, 'class="publisherlogoinline"'); } $publisherlogo = empty($publisherlogoarray['external_'.$publisher]) ? '' : $publisherlogoarray['external_'.$publisher]; */ $arrayofnatures['external_'.$publisher] = array('label' => $langs->trans("External").' - '.$publisher, 'data-html' => $langs->trans("External").' - '.$publisher.''); } else { $arrayofnatures['external_'] = array('label' => $langs->trans("External").' - ['.$langs->trans("UnknownPublishers").']'); } } ksort($arrayofnatures); // Define an array $categ with categ with at least one qualified module $filename[$i] = $modName; $modules[$modName] = $objMod; // Gives the possibility to the module, to provide his own family info and position of this family if (is_array($objMod->familyinfo) && !empty($objMod->familyinfo)) { $familyinfo = array_merge($familyinfo, $objMod->familyinfo); $familykey = key($objMod->familyinfo); } else { $familykey = $objMod->family; } '@phan-var-force string $familykey'; // if not, phan considers $familykey may be null $moduleposition = ($objMod->module_position ? $objMod->module_position : '50'); if ($objMod->isCoreOrExternalModule() == 'external' && $moduleposition < 100000) { // an external module should never return a value lower than '80'. $moduleposition = '80'; // External modules at end by default } // Add list of warnings to show into arrayofwarnings and arrayofwarningsext if (!empty($objMod->warnings_activation)) { $arrayofwarnings[$modName] = $objMod->warnings_activation; } if (!empty($objMod->warnings_activation_ext)) { $arrayofwarningsext[$modName] = $objMod->warnings_activation_ext; } $familyposition = (empty($familyinfo[$familykey]['position']) ? '0' : $familyinfo[$familykey]['position']); $listOfOfficialModuleGroups = array('hr', 'technic', 'interface', 'technic', 'portal', 'financial', 'crm', 'base', 'products', 'srm', 'ecm', 'projects', 'other'); if ($external && !in_array($familykey, $listOfOfficialModuleGroups)) { // If module is extern and into a custom group (not into an official predefined one), it must appear at end (custom groups should not be before official groups). if (is_numeric($familyposition)) { $familyposition = sprintf("%03d", (int) $familyposition + 100); } } $orders[$i] = $familyposition."_".$familykey."_".$moduleposition."_".$j; // Sort by family, then by module position then number // Set categ[$i] $specialstring = 'unknown'; if ($objMod->version == 'development' || $objMod->version == 'experimental') { $specialstring = 'expdev'; } if (isset($categ[$specialstring])) { $categ[$specialstring]++; // Array of all different modules categories } else { $categ[$specialstring] = 1; } $j++; $i++; } else { dol_syslog("Module ".get_class($objMod)." not qualified"); } } else { print info_admin("admin/modules.php Warning bad descriptor file : ".$dir.$file." (Class ".$modName." not found into file)", 0, 0, '1', 'warning'); } } catch (Exception $e) { dol_syslog("Failed to load ".$dir.$file." ".$e->getMessage(), LOG_ERR); } } } } closedir($handle); } else { dol_syslog("htdocs/admin/modules.php: Failed to open directory ".$dir.". See permission and open_basedir option.", LOG_WARNING); } } '@phan-var-force array $modules'; /** @var array $modules */ if ($action == 'reset_confirm' && $user->admin) { if (!empty($modules[$value])) { $objMod = $modules[$value]; if (!empty($objMod->langfiles)) { $langs->loadLangs($objMod->langfiles); } $form = new Form($db); $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?value='.$value.'&mode='.$mode.$param, $langs->trans('ConfirmUnactivation'), $langs->trans(GETPOST('confirm_message_code')), 'reset', '', 'no', 1); } } if ($action == 'reload_confirm' && $user->admin) { if (!empty($modules[$value])) { $objMod = $modules[$value]; if (!empty($objMod->langfiles)) { $langs->loadLangs($objMod->langfiles); } $form = new Form($db); $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?value='.$value.'&mode='.$mode.$param, $langs->trans('ConfirmReload'), $langs->trans(GETPOST('confirm_message_code')), 'reload', '', 'no', 1); } } print $formconfirm; asort($orders); //var_dump($orders); //var_dump($categ); //var_dump($modules); $nbofactivatedmodules = count($conf->modules); // Define $nbmodulesnotautoenabled - TODO This code is at different places $nbmodulesnotautoenabled = count($conf->modules); $listofmodulesautoenabled = array('agenda', 'fckeditor', 'export', 'import'); foreach ($listofmodulesautoenabled as $moduleautoenable) { if (in_array($moduleautoenable, $conf->modules)) { $nbmodulesnotautoenabled--; } } print load_fiche_titre($langs->trans("ModulesSetup"), '', 'title_setup'); // Start to show page $deschelp = ''; if ($mode == 'common' || $mode == 'commonkanban') { $desc = $langs->trans("ModulesDesc", '{picto}'); $desc .= ' '.$langs->trans("ModulesDesc2", '{picto2}'); $desc = str_replace('{picto}', img_picto('', 'switch_off', 'class="size15x"'), $desc); $desc = str_replace('{picto2}', img_picto('', 'setup', 'class="size15x"'), $desc); if ($nbmodulesnotautoenabled <= getDolGlobalInt('MAIN_MIN_NB_ENABLED_MODULE_FOR_WARNING', 1)) { // If only minimal initial modules enabled $deschelp .= '
'.$desc."
\n"; } if (getDolGlobalString('MAIN_SETUP_MODULES_INFO')) { // Show a custom message. A good usage for SaaS with option MAIN_MIN_NB_ENABLED_MODULE_FOR_WARNING. $deschelp .= '
'.$langs->trans(getDolGlobalString('MAIN_SETUP_MODULES_INFO'))."
\n"; } if ($deschelp) { $deschelp .= '
'; } } //if ($mode == 'marketplace') { // $deschelp = '
'.$langs->trans("ModulesMarketPlaceDesc")."

\n"; //} if ($mode == 'deploy') { $deschelp = '
'.$langs->trans("ModulesDeployDesc", $langs->transnoentitiesnoconv("AvailableModules"))."

\n"; } if ($mode == 'develop') { $deschelp = '
'.$langs->trans("ModulesDevelopDesc")."

\n"; } $head = modules_prepare_head($nbofactivatedmodules, count($modules), $nbmodulesnotautoenabled); if ($mode == 'common' || $mode == 'commonkanban') { dol_set_focus('#search_keyword'); print '
'; print ''; if (isset($optioncss) && $optioncss != '') { print ''; } if (isset($sortfield) && $sortfield != '') { print ''; } if (isset($sortorder) && $sortorder != '') { print ''; } if (isset($page) && $page != '') { print ''; } print ''; print dol_get_fiche_head($head, 'modules', '', -1); print $deschelp; $moreforfilter = '
'; $moreforfilter .= ''; $moreforfilter .= '
'; $moreforfilter .= '
'; $moreforfilter .= img_picto($langs->trans("Filter"), 'filter', 'class="paddingright opacityhigh hideonsmartphone"').''; $moreforfilter .= '
'; $moreforfilter .= '
'; $moreforfilter .= $form->selectarray('search_nature', $arrayofnatures, dol_escape_htmltag($search_nature), $langs->trans('Origin'), 0, 0, '', 0, 0, 0, '', 'maxwidth250', 1); $moreforfilter .= '
'; if (getDolGlobalInt('MAIN_FEATURES_LEVEL')) { $array_version = array('stable' => $langs->transnoentitiesnoconv("Stable")); if (getDolGlobalInt('MAIN_FEATURES_LEVEL') < 0) { $array_version['deprecated'] = $langs->trans("Deprecated"); } if (getDolGlobalInt('MAIN_FEATURES_LEVEL') > 0) { $array_version['experimental'] = $langs->trans("Experimental"); } if (getDolGlobalInt('MAIN_FEATURES_LEVEL') > 1) { $array_version['development'] = $langs->trans("Development"); } $moreforfilter .= '
'; $moreforfilter .= $form->selectarray('search_version', $array_version, $search_version, $langs->transnoentitiesnoconv('Version'), 0, 0, '', 0, 0, 0, '', 'maxwidth150', 1); $moreforfilter .= '
'; } $array_status = array('active' => $langs->transnoentitiesnoconv("Enabled"), 'disabled' => $langs->transnoentitiesnoconv("Disabled")); $moreforfilter .= '
'; $moreforfilter .= $form->selectarray('search_status', $array_status, $search_status, $langs->transnoentitiesnoconv('Status'), 0, 0, '', 0, 0, 0, '', 'maxwidth150', 1); $moreforfilter .= '
'; $moreforfilter .= ' '; $moreforfilter .= '
'; $moreforfilter .= ''; if ($search_keyword || ($search_nature && $search_nature != '-1') || ($search_version && $search_version != '-1') || ($search_status && $search_status != '-1')) { $moreforfilter .= ' '; $moreforfilter .= ''; } $moreforfilter .= '
'; $moreforfilter .= '
'; $moreforfilter .= '
'; print $moreforfilter; $parameters = array(); $reshook = $hookmanager->executeHooks('printFieldPreListTitle', $parameters, $object, $action); // Note that $action and $object may have been modified by hook print $hookmanager->resPrint; $moreforfilter = ''; print '


'; $object = new stdClass(); $parameters = array(); $reshook = $hookmanager->executeHooks('insertExtraHeader', $parameters, $object, $action); // Note that $action and $object may have been modified by some hooks if ($reshook < 0) { setEventMessages($hookmanager->error, $hookmanager->errors, 'errors'); } $disabled_modules = array(); if (!empty($_SESSION["disablemodules"])) { $disabled_modules = explode(',', $_SESSION["disablemodules"]); } // Show list of modules $oldfamily = ''; $foundoneexternalmodulewithupdate = 0; $linenum = 0; $atleastonequalified = 0; $atleastoneforfamily = 0; foreach ($orders as $key => $value) { $linenum++; $tab = explode('_', $value); $familykey = $tab[1]; $module_position = $tab[2]; $modName = $filename[$key]; /** @var DolibarrModules $objMod */ $objMod = $modules[$modName]; if (!is_object($objMod)) { continue; } //print $objMod->name." - ".$key." - ".$objMod->version."
"; if ($mode == 'expdev' && $objMod->version != 'development' && $objMod->version != 'experimental') { continue; // Discard if not for current tab } if (!$objMod->getName()) { dol_syslog("Error for module ".$key." - Property name of module looks empty", LOG_WARNING); continue; } $modulenameshort = strtolower(preg_replace('/^mod/i', '', get_class($objMod))); $const_name = 'MAIN_MODULE_'.strtoupper(preg_replace('/^mod/i', '', get_class($objMod))); // Check filters $modulename = $objMod->getName(); $moduletechnicalname = $objMod->name; $moduledesc = $objMod->getDesc(); $moduledesclong = $objMod->getDescLong(); $moduleauthor = $objMod->getPublisher(); // We discard showing according to filters if ($search_keyword) { $qualified = 0; if (preg_match('/'.preg_quote($search_keyword, '/').'/i', $modulename) || preg_match('/'.preg_quote($search_keyword, '/').'/i', $moduletechnicalname) || ($moduledesc && preg_match('/'.preg_quote($search_keyword, '/').'/i', $moduledesc)) || ($moduledesclong && preg_match('/'.preg_quote($search_keyword, '/').'/i', $moduledesclong)) || ($moduleauthor && preg_match('/'.preg_quote($search_keyword, '/').'/i', $moduleauthor)) ) { $qualified = 1; } if (!$qualified) { continue; } } if ($search_status) { if ($search_status == 'active' && !getDolGlobalString($const_name)) { continue; } if ($search_status == 'disabled' && getDolGlobalString($const_name)) { continue; } } if ($search_nature) { if (preg_match('/^external/', $search_nature) && $objMod->isCoreOrExternalModule() != 'external') { continue; } $reg = array(); if (preg_match('/^external_(.*)$/', $search_nature, $reg)) { //print $reg[1].'-'.dol_escape_htmltag($objMod->getPublisher()); $publisher = dol_escape_htmltag($objMod->getPublisher()); if ($reg[1] && dol_escape_htmltag($reg[1]) != $publisher) { continue; } if (!$reg[1] && !empty($publisher)) { continue; } } if ($search_nature == 'core' && $objMod->isCoreOrExternalModule() == 'external') { continue; } } if ($search_version) { if (($objMod->version == 'development' || $objMod->version == 'experimental' || preg_match('/deprecated/', $objMod->version)) && $search_version == 'stable') { continue; } if ($objMod->version != 'development' && ($search_version == 'development')) { continue; } if ($objMod->version != 'experimental' && ($search_version == 'experimental')) { continue; } if (!preg_match('/deprecated/', $objMod->version) && ($search_version == 'deprecated')) { continue; } } $atleastonequalified++; // Load all language files of the qualified module if (isset($objMod->langfiles) && is_array($objMod->langfiles)) { foreach ($objMod->langfiles as $domain) { $langs->load($domain); } } // Print a separator if we change family if ($familykey != $oldfamily) { if ($oldfamily) { print '
'; } $familytext = empty($familyinfo[$familykey]['label']) ? $familykey : $familyinfo[$familykey]['label']; print load_fiche_titre($familytext, '', '', 0, '', 'modulefamilygroup'); if ($mode == 'commonkanban') { print '
'; } else { print '
'; print ''."\n"; } $atleastoneforfamily = 0; } $atleastoneforfamily++; if ($familykey != $oldfamily) { $familytext = empty($familyinfo[$familykey]['label']) ? $familykey : $familyinfo[$familykey]['label']; $oldfamily = $familykey; } // Version (with picto warning or not) $version = $objMod->getVersion(0); $versiontrans = ''; $warningstring = ''; if (preg_match('/development/i', $version)) { $warningstring = $langs->trans("Development"); } if (preg_match('/experimental/i', $version)) { $warningstring = $langs->trans("Experimental"); } if (preg_match('/deprecated/i', $version)) { $warningstring = $langs->trans("Deprecated"); } if ($objMod->isCoreOrExternalModule() == 'external' || preg_match('/development|experimental|deprecated/i', $version)) { $versiontrans .= $objMod->getVersion(1); } if ($objMod->isCoreOrExternalModule() == 'external' && ($action == 'checklastversion' || getDolGlobalString('CHECKLASTVERSION_EXTERNALMODULE'))) { // Setting CHECKLASTVERSION_EXTERNALMODULE to on is a bad practice to activate a check on an external access during the building of the admin page. // 1 external module can hang the application. // Adding a cron job could be a good idea: see DolibarrModules::checkForUpdate() $checkRes = $objMod->checkForUpdate(); if ($checkRes > 0) { setEventMessages($objMod->getName().' : '.preg_replace('/[^a-z0-9_\.\-\s]/i', '', $versiontrans).' -> '.preg_replace('/[^a-z0-9_\.\-\s]/i', '', $objMod->lastVersion), null, 'warnings'); } elseif ($checkRes < 0) { setEventMessages($objMod->getName().' '.$langs->trans('CheckVersionFail'), null, 'errors'); } } if ($objMod->isCoreOrExternalModule() == 'external' && $action == 'checklastversion' && !getDolGlobalString('DISABLE_CHECK_ON_MALWARE_MODULES')) { $checkRes = $objMod->checkForCompliance(); // Check if module is reported as non compliant with Dolibarr rules and law if (!is_numeric($checkRes) && $checkRes != '') { $langs->load("errors"); setEventMessages($objMod->getName().' : '.$langs->trans($checkRes), null, 'errors'); } } // Define imginfo $imginfo = "info"; if ($objMod->isCoreOrExternalModule() == 'external') { $imginfo = "info_black"; } $codeenabledisable = ''; $codetoconfig = ''; // Force disable of module disabled into session (for demo for example) if (in_array($modulenameshort, $disabled_modules)) { $objMod->disabled = true; } // Activate/Disable and Setup (2 columns) if (getDolGlobalString($const_name)) { // If module is already activated // Set $codeenabledisable $disableSetup = 0; if (!empty($arrayofwarnings[$modName])) { $codeenabledisable .= ''."\n"; } if (!empty($objMod->disabled)) { $codeenabledisable .= $langs->trans("Disabled"); } elseif (is_object($objMod) && (!empty($objMod->always_enabled) || ((isModEnabled('multicompany') && $objMod->core_enabled) && ($user->entity || $conf->entity != 1)))) { // @phan-suppress-next-line PhanUndeclaredMethod if (method_exists($objMod, 'alreadyUsed') && $objMod->alreadyUsed()) { $codeenabledisable .= $langs->trans("Used"); } else { $codeenabledisable .= img_picto($langs->trans("Required"), 'switch_on', '', 0, 0, 0, '', 'opacitymedium valignmiddle'); //print $langs->trans("Required"); } if (isModEnabled('multicompany') && $user->entity) { $disableSetup++; } } else { // @phan-suppress-next-line PhanUndeclaredMethod if (is_object($objMod) && !empty($objMod->warnings_unactivation[$mysoc->country_code]) && method_exists($objMod, 'alreadyUsed') && $objMod->alreadyUsed()) { $codeenabledisable .= 'warnings_unactivation[$mysoc->country_code]).'&value='.$modName.'&mode='.$mode.$param.'">'; $codeenabledisable .= img_picto($langs->trans("Activated").($warningstring ? ' '.$warningstring : ''), 'switch_on'); $codeenabledisable .= ''; if (getDolGlobalInt("MAIN_FEATURES_LEVEL") > 1) { $codeenabledisable .= ' '; $codeenabledisable .= ''; $codeenabledisable .= img_picto($langs->trans("Reload"), 'refresh', 'class="opacitymedium"'); $codeenabledisable .= ''; } } else { $codeenabledisable .= ''; $codeenabledisable .= img_picto($langs->trans("Activated").($warningstring ? ' '.$warningstring : ''), 'switch_on'); $codeenabledisable .= ''; if (getDolGlobalInt("MAIN_FEATURES_LEVEL") > 1) { $codeenabledisable .= ' '; $codeenabledisable .= ''; $codeenabledisable .= img_picto($langs->trans("Reload"), 'refresh', 'class="opacitymedium"'); $codeenabledisable .= ''; } } } // Set $codetoconfig if (!empty($objMod->config_page_url) && !$disableSetup) { $backtourlparam = ''; if ($search_keyword != '') { $backtourlparam .= ($backtourlparam ? '&' : '?').'search_keyword='.urlencode($search_keyword); // No urlencode here, done later } if ($search_nature > -1) { $backtourlparam .= ($backtourlparam ? '&' : '?').'search_nature='.urlencode($search_nature); // No urlencode here, done later } if ($search_version > -1) { $backtourlparam .= ($backtourlparam ? '&' : '?').'search_version='.urlencode($search_version); // No urlencode here, done later } if ($search_status > -1) { $backtourlparam .= ($backtourlparam ? '&' : '?').'search_status='.urlencode($search_status); // No urlencode here, done later } $backtourl = $_SERVER["PHP_SELF"].$backtourlparam; $regs = array(); if (is_array($objMod->config_page_url)) { $i = 0; foreach ($objMod->config_page_url as $page) { $urlpage = $page; if ($i++) { $codetoconfig .= ''.img_picto(ucfirst($page), "setup").''; // print ''.ucfirst($page).' '; } else { if (preg_match('/^([^@]+)@([^@]+)$/i', $urlpage, $regs)) { $urltouse = dol_buildpath('/'.$regs[2].'/admin/'.$regs[1], 1); $codetoconfig .= ''.img_picto($langs->trans("Setup"), "setup", 'style="padding-right: 6px"', 0, 0, 0, '', 'fa-15').''; } else { $urltouse = $urlpage; $codetoconfig .= ''.img_picto($langs->trans("Setup"), "setup", 'style="padding-right: 6px"', 0, 0, 0, '', 'fa-15').''; } } } } elseif (preg_match('/^([^@]+)@([^@]+)$/i', (string) $objMod->config_page_url, $regs)) { $codetoconfig .= ''.img_picto($langs->trans("Setup"), "setup", 'style="padding-right: 6px"', 0, 0, 0, '', 'fa-15').''; } else { $codetoconfig .= ''.img_picto($langs->trans("Setup"), "setup", 'style="padding-right: 6px"', 0, 0, 0, '', 'fa-15').''; } } else { $codetoconfig .= img_picto($langs->trans("NothingToSetup"), "setup", 'class="opacitytransp" style="padding-right: 6px"', 0, 0, 0, '', 'fa-15'); } } else { // Module not yet activated // Set $codeenabledisable if (!empty($objMod->always_enabled)) { // A 'always_enabled' module should not never be disabled. If this happen, we keep a link to re-enable it. $codeenabledisable .= ''."\n"; $codeenabledisable .= 'trans("Disabled"), 'switch_off'); $codeenabledisable .= "\n"; } elseif (!empty($objMod->disabled)) { $codeenabledisable .= $langs->trans("Disabled"); } else { // Module qualified for activation $warningmessage = ''; if (!empty($arrayofwarnings[$modName])) { $codeenabledisable .= ''."\n"; foreach ($arrayofwarnings[$modName] as $keycountry => $cursorwarningmessage) { if (preg_match('/^always/', $keycountry) || ($mysoc->country_code && preg_match('/^'.$mysoc->country_code.'/', $keycountry))) { $warningmessage .= ($warningmessage ? "\n" : "").$langs->trans($cursorwarningmessage, $objMod->getName(), $mysoc->country_code); } } } if ($objMod->isCoreOrExternalModule() == 'external' && !empty($arrayofwarningsext)) { $codeenabledisable .= ''."\n"; foreach ($arrayofwarningsext as $keymodule => $arrayofwarningsextbycountry) { $keymodulelowercase = strtolower(preg_replace('/^mod/', '', $keymodule)); if (in_array($keymodulelowercase, $conf->modules)) { // If module that request warning is on foreach ($arrayofwarningsextbycountry as $keycountry => $cursorwarningmessage) { if (preg_match('/^always/', $keycountry) || ($mysoc->country_code && preg_match('/^'.$mysoc->country_code.'/', $keycountry))) { $warningmessage .= ($warningmessage ? "\n" : "").$langs->trans($cursorwarningmessage, $objMod->getName(), $mysoc->country_code, $modules[$keymodule]->getName()); $warningmessage .= ($warningmessage ? "\n" : "").($warningmessage ? "\n" : "").$langs->trans("Module").' : '.$objMod->getName(); if (!empty($objMod->editor_name)) { $warningmessage .= ($warningmessage ? "\n" : "").$langs->trans("Publisher").' : '.$objMod->editor_name; } if (!empty($objMod->editor_name)) { $warningmessage .= ($warningmessage ? "\n" : "").$langs->trans("ModuleTriggeringThisWarning").' : '.$modules[$keymodule]->getName(); } } } } } } $urltogo = $_SERVER["PHP_SELF"].'?id='.$objMod->numero.'&token='.newToken().'&module_position='.$module_position.'&action=set&token='.newToken().'&value='.$modName.'&mode='.$mode.$param; $popupwidth = 500; $codeenabledisable .= ''."\n"; $codeenabledisable .= 'numero.'\', '.$popupwidth.');"'; } $codeenabledisable .= '>'; $codeenabledisable .= img_picto($langs->trans("Disabled"), 'switch_off'); $codeenabledisable .= "\n"; } // Set $codetoconfig $codetoconfig .= img_picto($langs->trans("NothingToSetup"), "setup", 'class="opacitytransp" style="padding-right: 6px"'); } if ($mode == 'commonkanban') { // Output Kanban print $objMod->getKanbanView($codeenabledisable, $codetoconfig); } else { print ''."\n"; if (getDolGlobalString('MAIN_MODULES_SHOW_LINENUMBERS')) { print ''; } // Picto + Name of module print ' \n"; // Desc print '\n"; // Help print ''; // Version print '\n"; // Link enable/disable print '\n"; // Link config print ''; print "\n"; } if ($objMod->needUpdate) { $foundoneexternalmodulewithupdate++; } } if ($action == 'checklastversion') { if ($foundoneexternalmodulewithupdate) { setEventMessages($langs->trans("ModuleUpdateAvailable"), null, 'warnings', '', 0, 1); } else { setEventMessages($langs->trans("NoExternalModuleWithUpdate"), null, 'mesgs'); } } if ($oldfamily) { if ($mode == 'commonkanban') { print ''; } else { print "
'.$linenum.''; $alttext = ''; //if (is_array($objMod->need_dolibarr_version)) $alttext.=($alttext?' - ':'').'Dolibarr >= '.join('.',$objMod->need_dolibarr_version); //if (is_array($objMod->phpmin)) $alttext.=($alttext?' - ':'').'PHP >= '.join('.',$objMod->phpmin); if (!empty($objMod->picto)) { if (preg_match('/^\//i', $objMod->picto)) { print img_picto($alttext, $objMod->picto, 'class="valignmiddle pictomodule paddingrightonly"', 1); } else { print img_object($alttext, $objMod->picto, 'class="valignmiddle pictomodule paddingrightonly"'); } } else { print img_object($alttext, 'generic', 'class="valignmiddle paddingrightonly"'); } print ' '.$objMod->getName().''; print "'; print nl2br($objMod->getDesc()); print "'; //print $form->textwithpicto('', $text, 1, $imginfo, 'minheight20', 0, 2, 1); print ''.img_picto(($objMod->isCoreOrExternalModule() == 'external' ? $langs->trans("ExternalModule").' - ' : '').$langs->trans("ClickToShowDescription"), $imginfo).''; print ''; if ($objMod->needUpdate) { $versionTitle = $langs->trans('ModuleUpdateAvailable').' : '.$objMod->lastVersion; print ''.$versiontrans.''; } else { print $versiontrans; } print "'; print $codeenabledisable; print "'; print $codetoconfig; print '
\n"; print '
'; } } if (!$atleastonequalified) { print '
'.$langs->trans("NoDeployedModulesFoundWithThisSearchCriteria").'

'; } print dol_get_fiche_end(); print '
'; // Show warning about external users print info_admin(showModulesExludedForExternal($modules))."\n"; print ''; } if ($mode == 'marketplace') { print dol_get_fiche_head($head, $mode, '', -1); print $deschelp; print '
'; print ''; // Marketplace and community modules print '
'; print ''."\n"; print ''."\n"; print ''; print ''; print ''; print ''; print ''; $url = 'https://www.dolistore.com'; // Marketplace print ''."\n"; print ''; print ''; print ''; print ''; print ''; $url = 'https://github.com/Dolibarr/dolibarr-community-modules'; // Community print ''."\n"; print ''; print ''; print ''; print ''; print ''; print "
'.$form->textwithpicto($langs->trans("Provider"), $langs->trans("WebSiteDesc")).''; print '
'.$langs->trans("DoliStoreDesc").'
'; print img_picto('', 'url', 'class="pictofixedwidth"').''.$url.'
'; print ajax_constantonoff('MAIN_ENABLE_EXTERNALMODULES_DOLISTORE', array(), null, 0, 0, 1); print ''; if (!getDolGlobalString('MAIN_DISABLE_EXTERNALMODULES_DOLISTORE') && getDolGlobalInt('MAIN_ENABLE_EXTERNALMODULES_DOLISTORE')) { $messagetoadd = '
'; if ($remotestore->dolistoreApiStatus <= 0) { $messagetoadd = '
'.$remotestore->dolistoreApiError.'
Failed to get answer of remote API server
'; } $messagetoadd .= '
Using Shop address MAIN_MODULE_DOLISTORE_SHOP_URL = '.$remotestore->shop_url; $messagetoadd .= '
Using Remote API address MAIN_MODULE_DOLISTORE_API_URL = '.$remotestore->dolistore_api_url; $messagetoadd .= '
Using API public key MAIN_MODULE_DOLISTORE_API_KEY = '.$remotestore->dolistore_api_key; // Add basic auth if needed $basicAuthLogin = getDolGlobalString('MAIN_MODULE_DOLISTORE_BASIC_LOGIN'); $basicAuthPassword = getDolGlobalString('MAIN_MODULE_DOLISTORE_BASIC_PASSWORD'); if ($basicAuthLogin) { $messagetoadd .= '
Using basic auth login: base64('.$basicAuthLogin.':'.$basicAuthPassword.')'; } $messagetoadd .= '
'; print $remotestore->libStatus($remotestore->dolistoreApiStatus, 2, $messagetoadd); } print '
'.$langs->trans("CommunityModulesDesc").'
'; print img_picto('', 'url', 'class="pictofixedwidth"').''.$url.'
'; print ajax_constantonoff('MAIN_ENABLE_EXTERNALMODULES_COMMUNITY', array(), null, 0, 0, 1); print ''; if (!getDolGlobalString('MAIN_DISABLE_EXTERNALMODULES_COMMUNITY') && getDolGlobalInt('MAIN_ENABLE_EXTERNALMODULES_COMMUNITY')) { $messagetoadd = '

Content of the repository index file '.$remotestore->file_source_url.' is in the local cache file '.$remotestore->cache_file.' (Date: '.dol_print_date(dol_filemtime($remotestore->cache_file), 'dayhour', 'tzuserrel').')'; print $remotestore->libStatus($remotestore->githubFileStatus, 2, $messagetoadd); } print '
\n"; print '
'; print dol_get_fiche_end(); print '
'; if ($remotestore->numberOfProviders > 0) { // $options is array with filter criteria $nbmaxtoshow = $options['per_page']; $options['per_page']++; //$remotestore->getRemoteCategories(); //$remotestore->getRemoteProducts($options); print ''.$langs->trans('DOLISTOREdescriptionLong').'

'; $categories_tree = $remotestore->getCategories($options['categorie']); // Call API to get the categories $products_list = $remotestore->getProducts($options); $previouslink = $remotestore->get_previous_link(); $nextlink = $remotestore->get_next_link(); print '
'; print '
'; ?>
'.$langs->trans('Reset').''; } ?>  
'; $totalnboflines .= $langs->trans("itemFound", $remotestore->numberTotalOfProducts); $totalnboflines .= ''; print $totalnboflines; print $remotestore->getPagination(); print '
'; print '
'; print '
'; ?>
style="width:100%;"> get_products($nbmaxtoshow); ?>
getPagination(); ?>
'.$urldolibarrmodules.''; $message = ''; if ($allowonlineinstall) { if (!in_array('/custom', explode(',', $dolibarr_main_url_root_alt))) { $message = info_admin($langs->trans("ConfFileMustContainCustom", DOL_DOCUMENT_ROOT.'/custom', DOL_DOCUMENT_ROOT)); $allowfromweb = -1; } else { if ($dirins_ok) { if (!is_writable(dol_osencode($dirins))) { $langs->load("errors"); $message = info_admin($langs->trans("ErrorFailedToWriteInDir", $dirins), 0, 0, '1', 'warning'); $allowfromweb = 0; } } else { $message = info_admin($langs->trans("NotExistsDirect", $dirins).$langs->trans("InfDirAlt").$langs->trans("InfDirExample")); $allowfromweb = 0; } } } else { if (getDolGlobalString('MAIN_MESSAGE_INSTALL_MODULES_DISABLED_CONTACT_US')) { // Show clean message if (!is_numeric(getDolGlobalString('MAIN_MESSAGE_INSTALL_MODULES_DISABLED_CONTACT_US'))) { $message = info_admin($langs->trans(getDolGlobalString('MAIN_MESSAGE_INSTALL_MODULES_DISABLED_CONTACT_US')), 0, 0, 'warning'); } else { $message = info_admin($langs->trans('InstallModuleFromWebHasBeenDisabledContactUs'), 0, 0, 'warning'); } } else { // Show technical message $message = info_admin($langs->trans("InstallModuleFromWebHasBeenDisabledByFile", $dolibarrdataroot.'/installmodules.lock'), 0, 0, 'warning'); } $allowfromweb = 0; } print $deschelp; if ($allowfromweb < 1) { print $langs->trans("SomethingMakeInstallFromWebNotPossible"); print $message; //print $langs->trans("SomethingMakeInstallFromWebNotPossible2"); print '
'; } // $allowfromweb = -1 if installation or setup not correct, 0 if not allowed, 1 if allowed if ($allowfromweb >= 0) { if ($allowfromweb == 1) { //print $langs->trans("ThisIsProcessToFollow").'
'; } else { print '
'; print $langs->trans("ThisIsAlternativeProcessToFollow").'
'; print ''.$langs->trans("StepNb", 1).': '; print str_replace('{s1}', $fullurl, $langs->trans("FindPackageFromWebSite", '{s1}')).'
'; print ''.$langs->trans("StepNb", 2).': '; print str_replace('{s1}', $fullurl, $langs->trans("DownloadPackageFromWebSite", '{s1}')).'
'; print ''.$langs->trans("StepNb", 3).': '; } if ($allowfromweb == 1) { print '
'; print ''; print ''; print ''; print $langs->trans("YouCanSubmitFile").'


'; print ''; print ''; print '

'; $max = getDolGlobalString('MAIN_UPLOAD_DOC'); // In Kb $maxphp = @ini_get('upload_max_filesize'); // In unknown if (preg_match('/k$/i', $maxphp)) { $maxphp = preg_replace('/k$/i', '', $maxphp); $maxphp *= 1; } if (preg_match('/m$/i', $maxphp)) { $maxphp = preg_replace('/m$/i', '', $maxphp); $maxphp *= 1024; } if (preg_match('/g$/i', $maxphp)) { $maxphp = preg_replace('/g$/i', '', $maxphp); $maxphp *= 1024 * 1024; } if (preg_match('/t$/i', $maxphp)) { $maxphp = preg_replace('/t$/i', '', $maxphp); $maxphp *= 1024 * 1024 * 1024; } $maxphp2 = @ini_get('post_max_size'); // In unknown if (preg_match('/k$/i', $maxphp2)) { $maxphp2 = preg_replace('/k$/i', '', $maxphp2); $maxphp2 *= 1; } if (preg_match('/m$/i', $maxphp2)) { $maxphp2 = preg_replace('/m$/i', '', $maxphp2); $maxphp2 *= 1024; } if (preg_match('/g$/i', $maxphp2)) { $maxphp2 = preg_replace('/g$/i', '', $maxphp2); $maxphp2 *= 1024 * 1024; } if (preg_match('/t$/i', $maxphp2)) { $maxphp2 = preg_replace('/t$/i', '', $maxphp2); $maxphp2 *= 1024 * 1024 * 1024; } // Now $max and $maxphp and $maxphp2 are in Kb $maxmin = $max; $maxphptoshow = $maxphptoshowparam = ''; if ($maxphp > 0) { $maxmin = min($max, $maxphp); $maxphptoshow = $maxphp; $maxphptoshowparam = 'upload_max_filesize'; } if ($maxphp2 > 0) { $maxmin = min($max, $maxphp2); if ($maxphp2 < $maxphp) { $maxphptoshow = $maxphp2; $maxphptoshowparam = 'post_max_size'; } } if ($maxmin > 0) { print ''."\n"; // MAX_FILE_SIZE doit précéder le champ input de type file print ''; } print ''; print ''; if (getDolGlobalString('MAIN_UPLOAD_DOC')) { if ($user->admin) { $langs->load('other'); print ' '; print info_admin($langs->trans("ThisLimitIsDefinedInSetup", $max, $maxphptoshow, $maxphptoshowparam), 1); } } else { print ' ('.$langs->trans("UploadDisabled").')'; } print '
'; print '
'; print '
'; print '
'; } else { print $langs->trans("UnpackPackageInModulesRoot", $dirins).'
'; print ''.$langs->trans("StepNb", 4).': '; print $langs->trans("SetupIsReadyForUse", DOL_URL_ROOT.'/admin/modules.php?mainmenu=home', $langs->transnoentitiesnoconv("Home").' - '.$langs->transnoentitiesnoconv("Setup").' - '.$langs->transnoentitiesnoconv("Modules")).'
'; } } print dol_get_fiche_end(); } if ($mode == 'develop') { print dol_get_fiche_head($head, $mode, '', -1); print $deschelp; print '
'; // Marketplace print ''."\n"; print ''."\n"; //print ''; print ''; print ''; print ''; print ''."\n"; print ''; print ''; print ''; print ''; print ''."\n"; $url = 'https://partners.dolibarr.org'; print ''; print ''; print ''; print ''; print "
'.$langs->trans("Logo").''.$langs->trans("DevelopYourModuleDesc").''.$langs->trans("URL").'
'; print '
'; print '
'.$langs->trans("TryToUseTheModuleBuilder", $langs->transnoentitiesnoconv("ModuleBuilder")).''; if (isModEnabled('modulebuilder')) { print $langs->trans("SeeTopRightMenu"); } else { print ''.$langs->trans("ModuleMustBeEnabledFirst", $langs->transnoentitiesnoconv("ModuleBuilder")).''; } print '
'; print''; print ''.$langs->trans("DoliPartnersDesc").''; print img_picto('', 'url', 'class="pictofixedwidth"'); print $url.'
\n"; print dol_get_fiche_end(); } // End of page llxFooter(); $db->close();