PageRenderTime 47ms CodeModel.GetById 3ms RepoModel.GetById 0ms app.codeStats 1ms

/htdocs/admin/modulehelp.php

http://github.com/Dolibarr/dolibarr
PHP | 587 lines | 436 code | 102 blank | 49 comment | 150 complexity | dd8e1518fea1af45024742679913d55a MD5 | raw file
Possible License(s): GPL-2.0, AGPL-3.0, LGPL-2.0, CC-BY-SA-4.0, BSD-3-Clause, MPL-2.0-no-copyleft-exception, LGPL-3.0, GPL-3.0, LGPL-2.1, MIT
  1. <?php
  2. /* Copyright (C) 2017 Laurent Destailleur <eldy@users.sourceforge.net>
  3. * Copyright (C) 2017 Regis Houssin <regis.houssin@inodbox.com>
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation; either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  17. */
  18. /**
  19. * \file htdocs/admin/modulehelp.php
  20. * \brief Page to activate/disable all modules
  21. */
  22. if (!defined('NOREQUIREMENU')) {
  23. define('NOREQUIREMENU', '1'); // If there is no need to load and show top and left menu
  24. }
  25. if (!defined('NOTOKENRENEWAL')) {
  26. define('NOTOKENRENEWAL', '1'); // Disabled because this page is into a popup on module search page and we want to avoid to have an Anti CSRF token error (done if MAIN_SECURITY_CSRF_WITH_TOKEN is on) when we make a second search after closing popup.
  27. }
  28. require '../main.inc.php';
  29. require_once DOL_DOCUMENT_ROOT.'/core/lib/admin.lib.php';
  30. require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
  31. require_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
  32. // Load translation files required by the page
  33. $langs->loadLangs(array('errors', 'admin', 'modulebuilder', 'exports'));
  34. $mode = GETPOST('mode', 'alpha');
  35. $action = GETPOST('action', 'aZ09');
  36. $id = GETPOST('id', 'int');
  37. if (empty($mode)) {
  38. $mode = 'desc';
  39. }
  40. if (!$user->admin) {
  41. accessforbidden();
  42. }
  43. /*
  44. * Actions
  45. */
  46. // Nothing
  47. /*
  48. * View
  49. */
  50. $form = new Form($db);
  51. $help_url = 'EN:First_setup|FR:Premiers_paramétrages|ES:Primeras_configuraciones';
  52. llxHeader('', $langs->trans("Setup"), $help_url);
  53. print '<!-- Force style container -->'."\n".'<style>
  54. .id-container {
  55. width: 100%;
  56. }
  57. </style>';
  58. $arrayofnatures = array('core'=>$langs->transnoentitiesnoconv("Core"), 'external'=>$langs->transnoentitiesnoconv("External").' - '.$langs->trans("AllPublishers"));
  59. // Search modules dirs
  60. $modulesdir = dolGetModulesDirs();
  61. $filename = array();
  62. $modules = array();
  63. $orders = array();
  64. $categ = array();
  65. $dirmod = array();
  66. $i = 0; // is a sequencer of modules found
  67. $j = 0; // j is module number. Automatically affected if module number not defined.
  68. $modNameLoaded = array();
  69. foreach ($modulesdir as $dir) {
  70. // Load modules attributes in arrays (name, numero, orders) from dir directory
  71. //print $dir."\n<br>";
  72. dol_syslog("Scan directory ".$dir." for module descriptor files (modXXX.class.php)");
  73. $handle = @opendir($dir);
  74. if (is_resource($handle)) {
  75. while (($file = readdir($handle)) !== false) {
  76. //print "$i ".$file."\n<br>";
  77. if (is_readable($dir.$file) && substr($file, 0, 3) == 'mod' && substr($file, dol_strlen($file) - 10) == '.class.php') {
  78. $modName = substr($file, 0, dol_strlen($file) - 10);
  79. if ($modName) {
  80. if (!empty($modNameLoaded[$modName])) {
  81. $mesg = "Error: Module ".$modName." was found twice: Into ".$modNameLoaded[$modName]." and ".$dir.". You probably have an old file on your disk.<br>";
  82. setEventMessages($mesg, null, 'warnings');
  83. dol_syslog($mesg, LOG_ERR);
  84. continue;
  85. }
  86. try {
  87. $res = include_once $dir.$file;
  88. if (class_exists($modName)) {
  89. try {
  90. $objMod = new $modName($db);
  91. $modNameLoaded[$modName] = $dir;
  92. if (!$objMod->numero > 0 && $modName != 'modUser') {
  93. dol_syslog('The module descriptor '.$modName.' must have a numero property', LOG_ERR);
  94. }
  95. $j = $objMod->numero;
  96. $modulequalified = 1;
  97. // We discard modules according to features level (PS: if module is activated we always show it)
  98. $const_name = 'MAIN_MODULE_'.strtoupper(preg_replace('/^mod/i', '', get_class($objMod)));
  99. if ($objMod->version == 'development' && (empty($conf->global->$const_name) && ($conf->global->MAIN_FEATURES_LEVEL < 2))) {
  100. $modulequalified = 0;
  101. }
  102. if ($objMod->version == 'experimental' && (empty($conf->global->$const_name) && ($conf->global->MAIN_FEATURES_LEVEL < 1))) {
  103. $modulequalified = 0;
  104. }
  105. if (preg_match('/deprecated/', $objMod->version) && (empty($conf->global->$const_name) && ($conf->global->MAIN_FEATURES_LEVEL >= 0))) {
  106. $modulequalified = 0;
  107. }
  108. // We discard modules according to property disabled
  109. //if (! empty($objMod->hidden)) $modulequalified=0;
  110. if ($modulequalified > 0) {
  111. $publisher = dol_escape_htmltag($objMod->getPublisher());
  112. $external = ($objMod->isCoreOrExternalModule() == 'external');
  113. if ($external) {
  114. if ($publisher) {
  115. $arrayofnatures['external_'.$publisher] = $langs->trans("External").' - '.$publisher;
  116. } else {
  117. $arrayofnatures['external_'] = $langs->trans("External").' - '.$langs->trans("UnknownPublishers");
  118. }
  119. }
  120. ksort($arrayofnatures);
  121. }
  122. // Define array $categ with categ with at least one qualified module
  123. if ($modulequalified > 0) {
  124. $modules[$i] = $objMod;
  125. $filename[$i] = $modName;
  126. // Gives the possibility to the module, to provide his own family info and position of this family
  127. if (is_array($objMod->familyinfo) && !empty($objMod->familyinfo)) {
  128. if (!is_array($familyinfo)) {
  129. $familyinfo = array();
  130. }
  131. $familyinfo = array_merge($familyinfo, $objMod->familyinfo);
  132. $familykey = key($objMod->familyinfo);
  133. } else {
  134. $familykey = $objMod->family;
  135. }
  136. $moduleposition = ($objMod->module_position ? $objMod->module_position : '50');
  137. if ($moduleposition == '50' && ($objMod->isCoreOrExternalModule() == 'external')) {
  138. $moduleposition = '80'; // External modules at end by default
  139. }
  140. $orders[$i] = $familyinfo[$familykey]['position']."_".$familykey."_".$moduleposition."_".$j; // Sort by family, then by module position then number
  141. $dirmod[$i] = $dir;
  142. //print $i.'-'.$dirmod[$i].'<br>';
  143. // Set categ[$i]
  144. $specialstring = 'unknown';
  145. if ($objMod->version == 'development' || $objMod->version == 'experimental') {
  146. $specialstring = 'expdev';
  147. }
  148. if (isset($categ[$specialstring])) {
  149. $categ[$specialstring]++; // Array of all different modules categories
  150. } else {
  151. $categ[$specialstring] = 1;
  152. }
  153. $j++;
  154. $i++;
  155. } else {
  156. dol_syslog("Module ".get_class($objMod)." not qualified");
  157. }
  158. } catch (Exception $e) {
  159. dol_syslog("Failed to load ".$dir.$file." ".$e->getMessage(), LOG_ERR);
  160. }
  161. } else {
  162. print "Warning bad descriptor file : ".$dir.$file." (Class ".$modName." not found into file)<br>";
  163. }
  164. } catch (Exception $e) {
  165. dol_syslog("Failed to load ".$dir.$file." ".$e->getMessage(), LOG_ERR);
  166. }
  167. }
  168. }
  169. }
  170. closedir($handle);
  171. } else {
  172. dol_syslog("htdocs/admin/modulehelp.php: Failed to open directory ".$dir.". See permission and open_basedir option.", LOG_WARNING);
  173. }
  174. }
  175. asort($orders);
  176. //var_dump($orders);
  177. //var_dump($categ);
  178. //var_dump($modules);
  179. unset($objMod);
  180. $i = 0;
  181. foreach ($orders as $tmpkey => $tmpvalue) {
  182. $tmpMod = $modules[$tmpkey];
  183. if ($tmpMod->numero == $id) {
  184. $key = $i;
  185. $modName = $filename[$tmpkey];
  186. $dirofmodule = $dirmod[$tmpkey];
  187. $objMod = $tmpMod;
  188. break;
  189. }
  190. $i++;
  191. }
  192. $value = $orders[$key];
  193. $tab = explode('_', $value);
  194. $familyposition = $tab[0]; $familykey = $tab[1]; $module_position = $tab[2]; $numero = $tab[3];
  195. $head = modulehelp_prepare_head($objMod);
  196. // Check filters
  197. $modulename = $objMod->getName();
  198. $moduledesc = $objMod->getDesc();
  199. $moduleauthor = $objMod->getPublisher();
  200. $moduledir = strtolower(preg_replace('/^mod/i', '', get_class($objMod)));
  201. $const_name = 'MAIN_MODULE_'.strtoupper(preg_replace('/^mod/i', '', get_class($objMod)));
  202. $text = '<span class="opacitymedium">'.$langs->trans("LastActivationDate").':</span> ';
  203. if (!empty($conf->global->$const_name)) {
  204. $text .= dol_print_date($objMod->getLastActivationDate(), 'dayhour');
  205. } else {
  206. $text .= $langs->trans("Disabled");
  207. }
  208. $tmp = $objMod->getLastActivationInfo();
  209. $authorid = $tmp['authorid'];
  210. if ($authorid > 0) {
  211. $tmpuser = new User($db);
  212. $tmpuser->fetch($authorid);
  213. $text .= '<br><span class="opacitymedium">'.$langs->trans("LastActivationAuthor").':</span> ';
  214. $text .= $tmpuser->getNomUrl(1);
  215. }
  216. $ip = $tmp['ip'];
  217. if ($ip) {
  218. $text .= '<br><span class="opacitymedium">'.$langs->trans("LastActivationIP").':</span> ';
  219. $text .= $ip;
  220. }
  221. $moreinfo = $text;
  222. $title = ($modulename ? $modulename : $moduledesc);
  223. print '<div class="centpercent">';
  224. $picto = 'object_'.$objMod->picto;
  225. print load_fiche_titre(($modulename ? $modulename : $moduledesc), $moreinfo, $picto, 0, '', 'titlemodulehelp');
  226. print '<br>';
  227. print dol_get_fiche_head($head, $mode, '', -1);
  228. if (!$modulename) {
  229. dol_syslog("Error for module ".$key." - Property name of module looks empty", LOG_WARNING);
  230. }
  231. // Load all lang files of module
  232. if (isset($objMod->langfiles) && is_array($objMod->langfiles)) {
  233. foreach ($objMod->langfiles as $domain) {
  234. $langs->load($domain);
  235. }
  236. }
  237. // Version (with picto warning or not)
  238. $version = $objMod->getVersion(0);
  239. $versiontrans = '';
  240. if (preg_match('/development/i', $version)) {
  241. $versiontrans .= img_warning($langs->trans("Development"), 'style="float: left"');
  242. }
  243. if (preg_match('/experimental/i', $version)) {
  244. $versiontrans .= img_warning($langs->trans("Experimental"), 'style="float: left"');
  245. }
  246. if (preg_match('/deprecated/i', $version)) {
  247. $versiontrans .= img_warning($langs->trans("Deprecated"), 'style="float: left"');
  248. }
  249. $versiontrans .= $objMod->getVersion(1);
  250. // Define imginfo
  251. $imginfo = "info";
  252. if ($objMod->isCoreOrExternalModule() == 'external') {
  253. $imginfo = "info_black";
  254. }
  255. // Define text of description of module
  256. $text = '';
  257. if ($mode == 'desc') {
  258. if ($moduledesc) {
  259. $text .= '<br>'.$moduledesc.'<br><br><br>';
  260. }
  261. $text .= '<span class="opacitymedium">'.$langs->trans("Version").':</span> '.$version;
  262. $moduledescriptorfile = get_class($objMod).'.class.php';
  263. $text .= '<br><span class="opacitymedium">'.$langs->trans("DescriptorFile").':</span> '.$moduledescriptorfile;
  264. $textexternal = '';
  265. if ($objMod->isCoreOrExternalModule() == 'external') {
  266. $textexternal .= '<br><span class="opacitymedium">'.$langs->trans("Origin").':</span> '.$langs->trans("ExternalModule").' - '.$langs->trans("InstalledInto", $dirofmodule);
  267. if ($objMod->editor_name != 'dolibarr') {
  268. $textexternal .= '<br><span class="opacitymedium">'.$langs->trans("Publisher").':</span> '.(empty($objMod->editor_name) ? $langs->trans("Unknown") : $objMod->editor_name);
  269. }
  270. $editor_url = $objMod->editor_url;
  271. if (!preg_match('/^http/', $editor_url)) {
  272. $editor_url = 'http://'.$editor_url;
  273. }
  274. if (!empty($objMod->editor_url) && !preg_match('/dolibarr\.org/i', $objMod->editor_url)) {
  275. $textexternal .= ($objMod->editor_name != 'dolibarr' ? ' - ' : '').img_picto('', 'globe').' <a href="'.$editor_url.'" target="_blank">'.$objMod->editor_url.'</a>';
  276. }
  277. $text .= $textexternal;
  278. $text .= '<br>';
  279. } else {
  280. $text .= '<br><span class="opacitymedium">'.$langs->trans("Origin").':</span> '.$langs->trans("Core").'<br>';
  281. }
  282. $moduledesclong = $objMod->getDescLong();
  283. if ($moduledesclong) {
  284. $text .= '<br><hr><div class="moduledesclong">'.$moduledesclong.'<div>';
  285. }
  286. }
  287. if ($mode == 'feature') {
  288. $text .= '<br><strong>'.$langs->trans("DependsOn").':</strong> ';
  289. if (count($objMod->depends)) {
  290. $text .= join(',', $objMod->depends);
  291. } else {
  292. $text .= $langs->trans("None");
  293. }
  294. $text .= '<br><strong>'.$langs->trans("RequiredBy").':</strong> ';
  295. if (count($objMod->requiredby)) {
  296. $text .= join(',', $objMod->requiredby);
  297. } else {
  298. $text .= $langs->trans("None");
  299. }
  300. $text .= '<br><br>';
  301. $text .= '<br><strong>'.$langs->trans("AddDataTables").':</strong> ';
  302. $sqlfiles = dol_dir_list(dol_buildpath($moduledir.'/sql/'), 'files', 0, 'llx.*\.sql', array('\.key\.sql', '\.sql\.back'));
  303. if (count($sqlfiles) > 0) {
  304. $text .= $langs->trans("Yes").' (';
  305. $i = 0;
  306. foreach ($sqlfiles as $val) {
  307. $text .= ($i ? ', ' : '').preg_replace('/\.sql$/', '', preg_replace('/llx_/', '', $val['name']));
  308. $i++;
  309. }
  310. $text .= ')';
  311. } else {
  312. $text .= $langs->trans("No");
  313. }
  314. $text .= '<br>';
  315. $text .= '<br><strong>'.$langs->trans("AddDictionaries").':</strong> ';
  316. if (isset($objMod->dictionaries) && isset($objMod->dictionaries['tablib']) && is_array($objMod->dictionaries['tablib']) && count($objMod->dictionaries['tablib'])) {
  317. $i = 0;
  318. foreach ($objMod->dictionaries['tablib'] as $val) {
  319. $text .= ($i ? ', ' : '').$val;
  320. $i++;
  321. }
  322. } else {
  323. $text .= $langs->trans("No");
  324. }
  325. $text .= '<br>';
  326. $text .= '<br><strong>'.$langs->trans("AddData").':</strong> ';
  327. $filedata = dol_buildpath($moduledir.'/sql/data.sql');
  328. if (dol_is_file($filedata)) {
  329. $text .= $langs->trans("Yes").' ('.$moduledir.'/sql/data.sql)';
  330. } else {
  331. $text .= $langs->trans("No");
  332. }
  333. $text .= '<br>';
  334. $text .= '<br><strong>'.$langs->trans("AddRemoveTabs").':</strong> ';
  335. if (isset($objMod->tabs) && is_array($objMod->tabs) && count($objMod->tabs)) {
  336. $i = 0;
  337. foreach ($objMod->tabs as $val) {
  338. if (is_array($val)) {
  339. $val = $val['data'];
  340. }
  341. if (is_string($val)) {
  342. $tmp = explode(':', $val, 3);
  343. $text .= ($i ? ', ' : '').$tmp[0].':'.$tmp[1];
  344. $i++;
  345. }
  346. }
  347. } else {
  348. $text .= $langs->trans("No");
  349. }
  350. $text .= '<br>';
  351. $text .= '<br><strong>'.$langs->trans("AddModels").':</strong> ';
  352. if (isset($objMod->module_parts) && isset($objMod->module_parts['models']) && $objMod->module_parts['models']) {
  353. $text .= $langs->trans("Yes");
  354. } else {
  355. $text .= $langs->trans("No");
  356. }
  357. $text .= '<br>';
  358. $text .= '<br><strong>'.$langs->trans("AddSubstitutions").':</strong> ';
  359. if (isset($objMod->module_parts) && isset($objMod->module_parts['substitutions']) && $objMod->module_parts['substitutions']) {
  360. $text .= $langs->trans("Yes");
  361. } else {
  362. $text .= $langs->trans("No");
  363. }
  364. $text .= '<br>';
  365. $text .= '<br><strong>'.$langs->trans("AddSheduledJobs").':</strong> ';
  366. if (isset($objMod->cronjobs) && is_array($objMod->cronjobs) && count($objMod->cronjobs)) {
  367. $i = 0;
  368. foreach ($objMod->cronjobs as $val) {
  369. $text .= ($i ? ', ' : '').($val['label']);
  370. $i++;
  371. }
  372. } else {
  373. $text .= $langs->trans("No");
  374. }
  375. $text .= '<br>';
  376. $text .= '<br><strong>'.$langs->trans("AddTriggers").':</strong> ';
  377. $moreinfoontriggerfile = '';
  378. if (isset($objMod->module_parts) && isset($objMod->module_parts['triggers']) && $objMod->module_parts['triggers']) {
  379. $yesno = 'Yes';
  380. } else {
  381. $yesno = 'No';
  382. }
  383. require_once DOL_DOCUMENT_ROOT.'/core/class/interfaces.class.php';
  384. $interfaces = new Interfaces($db);
  385. $triggers = $interfaces->getTriggersList(array((($objMod->isCoreOrExternalModule() == 'external') ? '/'.$moduledir : '').'/core/triggers'));
  386. foreach ($triggers as $triggercursor) {
  387. if ($triggercursor['module'] == $moduledir) {
  388. $yesno = 'Yes';
  389. $moreinfoontriggerfile = ' ('.$triggercursor['relpath'].')';
  390. }
  391. }
  392. $text .= $langs->trans($yesno).$moreinfoontriggerfile;
  393. $text .= '<br>';
  394. $text .= '<br><strong>'.$langs->trans("AddBoxes").':</strong> ';
  395. if (isset($objMod->boxes) && is_array($objMod->boxes) && count($objMod->boxes)) {
  396. $i = 0;
  397. foreach ($objMod->boxes as $val) {
  398. $text .= ($i ? ', ' : '').($val['file'] ? $val['file'] : $val[0]);
  399. $i++;
  400. }
  401. } else {
  402. $text .= $langs->trans("No");
  403. }
  404. $text .= '<br>';
  405. $text .= '<br><strong>'.$langs->trans("AddHooks").':</strong> ';
  406. if (isset($objMod->module_parts) && is_array($objMod->module_parts['hooks']) && count($objMod->module_parts['hooks'])) {
  407. $i = 0;
  408. foreach ($objMod->module_parts['hooks'] as $key => $val) {
  409. if ($key === 'entity') {
  410. continue;
  411. }
  412. // For special values
  413. if ($key === 'data') {
  414. if (is_array($val)) {
  415. foreach ($val as $value) {
  416. $text .= ($i ? ', ' : '').($value);
  417. $i++;
  418. }
  419. continue;
  420. }
  421. }
  422. $text .= ($i ? ', ' : '').($val);
  423. $i++;
  424. }
  425. } else {
  426. $text .= $langs->trans("No");
  427. }
  428. $text .= '<br>';
  429. $text .= '<br><strong>'.$langs->trans("AddPermissions").':</strong> ';
  430. if (isset($objMod->rights) && is_array($objMod->rights) && count($objMod->rights)) {
  431. $i = 0;
  432. foreach ($objMod->rights as $val) {
  433. $text .= ($i ? ', ' : '').($val[1]);
  434. $i++;
  435. }
  436. } else {
  437. $text .= $langs->trans("No");
  438. }
  439. $text .= '<br>';
  440. $text .= '<br><strong>'.$langs->trans("AddMenus").':</strong> ';
  441. if (isset($objMod->menu) && !empty($objMod->menu)) { // objMod can be an array or just an int 1
  442. $text .= $langs->trans("Yes");
  443. } else {
  444. $text .= $langs->trans("No");
  445. }
  446. $text .= '<br>';
  447. $text .= '<br><strong>'.$langs->trans("AddExportProfiles").':</strong> ';
  448. if (isset($objMod->export_label) && is_array($objMod->export_label) && count($objMod->export_label)) {
  449. $i = 0;
  450. foreach ($objMod->export_label as $val) {
  451. $text .= ($i ? ', ' : '').($val);
  452. $i++;
  453. }
  454. } else {
  455. $text .= $langs->trans("No");
  456. }
  457. $text .= '<br>';
  458. $text .= '<br><strong>'.$langs->trans("AddImportProfiles").':</strong> ';
  459. if (isset($objMod->import_label) && is_array($objMod->import_label) && count($objMod->import_label)) {
  460. $i = 0;
  461. foreach ($objMod->import_label as $val) {
  462. $text .= ($i ? ', ' : '').($val);
  463. $i++;
  464. }
  465. } else {
  466. $text .= $langs->trans("No");
  467. }
  468. $text .= '<br>';
  469. $text .= '<br><strong>'.$langs->trans("AddOtherPagesOrServices").':</strong> ';
  470. $text .= $langs->trans("DetectionNotPossible");
  471. }
  472. if ($mode == 'changelog') {
  473. $changelog = $objMod->getChangeLog();
  474. if ($changelog) {
  475. $text .= '<div class="moduledesclong">'.$changelog.'<div>';
  476. } else {
  477. $text .= '<div class="moduledesclong">'.$langs->trans("NotAvailable").'</div>';
  478. }
  479. }
  480. print $text;
  481. print dol_get_fiche_end();
  482. print '</div>';
  483. // End of page
  484. llxFooter();
  485. $db->close();