weekly release 1.9.12+
[moodle.git] / lib / environmentlib.php
blobd547df7d32abc76db1d10986a84af753af923c52
1 <?php //$Id$
3 ///////////////////////////////////////////////////////////////////////////
4 // //
5 // NOTICE OF COPYRIGHT //
6 // //
7 // Moodle - Modular Object-Oriented Dynamic Learning Environment //
8 // http://moodle.com //
9 // //
10 // Copyright (C) 1999 onwards Martin Dougiamas http://dougiamas.com //
11 // (C) 2001-3001 Eloy Lafuente (stronk7) http://contiento.com //
12 // //
13 // This program is free software; you can redistribute it and/or modify //
14 // it under the terms of the GNU General Public License as published by //
15 // the Free Software Foundation; either version 2 of the License, or //
16 // (at your option) any later version. //
17 // //
18 // This program is distributed in the hope that it will be useful, //
19 // but WITHOUT ANY WARRANTY; without even the implied warranty of //
20 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the //
21 // GNU General Public License for more details: //
22 // //
23 // http://www.gnu.org/copyleft/gpl.html //
24 // //
25 ///////////////////////////////////////////////////////////////////////////
27 // This library includes all the necessary stuff to execute some standard
28 // tests of required versions and libraries to run Moodle. It can be
29 // used from the admin interface, and both at install and upgrade.
31 // All the info is stored in the admin/environment.xml file,
32 // supporting to have an updated version in dataroot/environment
34 /// Add required files
35 require_once($CFG->libdir.'/xmlize.php');
37 /// Define a buch of XML processing errors
38 define('NO_ERROR', 0);
39 define('NO_VERSION_DATA_FOUND', 1);
40 define('NO_DATABASE_SECTION_FOUND', 2);
41 define('NO_DATABASE_VENDORS_FOUND', 3);
42 define('NO_DATABASE_VENDOR_MYSQL_FOUND', 4);
43 define('NO_DATABASE_VENDOR_POSTGRES_FOUND', 5);
44 define('NO_PHP_SECTION_FOUND', 6);
45 define('NO_PHP_VERSION_FOUND', 7);
46 define('NO_PHP_EXTENSIONS_SECTION_FOUND', 8);
47 define('NO_PHP_EXTENSIONS_NAME_FOUND', 9);
48 define('NO_DATABASE_VENDOR_VERSION_FOUND', 10);
49 define('NO_UNICODE_SECTION_FOUND', 11);
50 define('NO_CUSTOM_CHECK_FOUND', 12);
51 define('CUSTOM_CHECK_FILE_MISSING', 13);
52 define('CUSTOM_CHECK_FUNCTION_MISSING', 14);
54 /**
55 * This function will perform the whole check, returning
56 * true or false as final result. Also, he full array of
57 * environment_result will be returned in the parameter list.
58 * The function looks for the best version to compare and
59 * everything. This is the only function that should be called
60 * ever from the rest of Moodle.
61 * @param string version version to check.
62 * @param array results array of results checked.
63 * @param boolean true/false, whether to print the table or just return results array
64 * @return boolean true/false, depending of results
66 function check_moodle_environment($version, &$environment_results, $print_table=true) {
68 $status = true;
70 /// This are cached per request
71 static $result = true;
72 static $env_results;
73 static $cache_exists = false;
75 /// if we have results cached, use them
76 if ($cache_exists) {
77 $environment_results = $env_results;
78 /// No cache exists, calculate everything
79 } else {
80 /// Get the more recent version before the requested
81 if (!$version = get_latest_version_available($version)) {
82 $status = false;
85 /// Perform all the checks
86 if (!($environment_results = environment_check($version)) && $status) {
87 $status = false;
90 /// Iterate over all the results looking for some error in required items
91 /// or some error_code
92 if ($status) {
93 foreach ($environment_results as $environment_result) {
94 if (!$environment_result->getStatus() && $environment_result->getLevel() == 'required'
95 && !$environment_result->getBypassStr()) {
96 $result = false; // required item that is not bypased
97 } else if ($environment_result->getStatus() && $environment_result->getLevel() == 'required'
98 && $environment_result->getRestrictStr()) {
99 $result = false; // required item that is restricted
100 } else if ($environment_result->getErrorCode()) {
101 $result = false;
105 /// Going to end, we store environment_results to cache
106 $env_results = $environment_results;
107 $cache_exists = true;
108 } ///End of cache block
110 /// If we have decided to print all the information, just do it
111 if ($print_table) {
112 print_moodle_environment($result && $status, $environment_results);
115 return ($result && $status);
119 * This function will print one beautiful table with all the environmental
120 * configuration and how it suits Moodle needs.
121 * @param boolean final result of the check (true/false)
122 * @param array environment_results array of results gathered
124 function print_moodle_environment($result, $environment_results) {
125 /// Get some strings
126 $strname = get_string('name');
127 $strinfo = get_string('info');
128 $strreport = get_string('report');
129 $strstatus = get_string('status');
130 $strok = get_string('ok');
131 $strerror = get_string('error');
132 $strcheck = get_string('check');
133 $strbypassed = get_string('bypassed');
134 $strrestricted = get_string('restricted');
135 $strenvironmenterrortodo = get_string('environmenterrortodo', 'admin');
136 /// Table headers
137 $servertable = new stdClass;//table for server checks
138 $servertable->head = array ($strname, $strinfo, $strreport, $strstatus);
139 $servertable->align = array ('center', 'center', 'left', 'center');
140 $servertable->wrap = array ('nowrap', '', '', 'nowrap');
141 $servertable->size = array ('10', 10, '100%', '10');
142 $servertable->width = '90%';
143 $servertable->class = 'environmenttable generaltable';
145 $serverdata = array('ok'=>array(), 'warn'=>array(), 'error'=>array());
147 $othertable = new stdClass;//table for custom checks
148 $othertable->head = array ($strinfo, $strreport, $strstatus);
149 $othertable->align = array ('center', 'left', 'center');
150 $othertable->wrap = array ('', '', 'nowrap');
151 $othertable->size = array (10, '100%', '10');
152 $othertable->width = '90%';
153 $othertable->class = 'environmenttable generaltable';
155 $otherdata = array('ok'=>array(), 'warn'=>array(), 'error'=>array());
157 /// Iterate over each environment_result
158 $continue = true;
159 foreach ($environment_results as $environment_result) {
160 $errorline = false;
161 $warningline = false;
162 if ($continue) {
163 $type = $environment_result->getPart();
164 $info = $environment_result->getInfo();
165 $status = $environment_result->getStatus();
166 $error_code = $environment_result->getErrorCode();
167 /// Process Report field
168 $rec = new stdClass();
169 /// Something has gone wrong at parsing time
170 if ($error_code) {
171 $stringtouse = 'environmentxmlerror';
172 $rec->error_code = $error_code;
173 $status = $strerror;
174 $errorline = true;
175 $continue = false;
178 if ($continue) {
179 /// We are comparing versions
180 if ($rec->needed = $environment_result->getNeededVersion()) {
181 $rec->current = $environment_result->getCurrentVersion();
182 if ($environment_result->getLevel() == 'required') {
183 $stringtouse = 'environmentrequireversion';
184 } else {
185 $stringtouse = 'environmentrecommendversion';
187 /// We are checking installed & enabled things
188 } else if ($environment_result->getPart() == 'custom_check') {
189 if ($environment_result->getLevel() == 'required') {
190 $stringtouse = 'environmentrequirecustomcheck';
191 } else {
192 $stringtouse = 'environmentrecommendcustomcheck';
194 } else {
195 if ($environment_result->getLevel() == 'required') {
196 $stringtouse = 'environmentrequireinstall';
197 } else {
198 $stringtouse = 'environmentrecommendinstall';
201 /// Calculate the status value
202 if ($environment_result->getBypassStr() != '') { //Handle bypassed result (warning)
203 $status = $strbypassed;
204 $warningline = true;
205 } else if ($environment_result->getRestrictStr() != '') { //Handle restricted result (error)
206 $status = $strrestricted;
207 $errorline = true;
208 } else {
209 if ($status) { //Handle ok result (ok)
210 $status = $strok;
211 } else {
212 if ($environment_result->getLevel() == 'optional') {//Handle check result (warning)
213 $status = $strcheck;
214 $warningline = true;
215 } else { //Handle error result (error)
216 $status = $strcheck;
217 $errorline = true;
223 /// Build the text
224 $linkparts = array();
225 $linkparts[] = 'admin/environment';
226 $linkparts[] = $type;
227 if (!empty($info)){
228 $linkparts[] = $info;
230 $report = doc_link(join($linkparts, '/'), get_string($stringtouse, 'admin', $rec));
233 /// Format error or warning line
234 if ($errorline || $warningline) {
235 $messagetype = $errorline? 'error':'warn';
236 } else {
237 $messagetype = 'ok';
239 $status = '<span class="'.$messagetype.'">'.$status.'</span>';
240 /// Here we'll store all the feedback found
241 $feedbacktext = '';
242 ///Append the feedback if there is some
243 $feedbacktext .= $environment_result->strToReport($environment_result->getFeedbackStr(), $messagetype);
244 ///Append the bypass if there is some
245 $feedbacktext .= $environment_result->strToReport($environment_result->getBypassStr(), 'warn');
246 ///Append the restrict if there is some
247 $feedbacktext .= $environment_result->strToReport($environment_result->getRestrictStr(), 'error');
249 $report .= $feedbacktext;
250 /// Add the row to the table
252 if ($environment_result->getPart() == 'custom_check'){
253 $otherdata[$messagetype][] = array ($info, $report, $status);
254 } else {
255 $serverdata[$messagetype][] = array ($type, $info, $report, $status);
259 //put errors first in
260 $servertable->data = array_merge($serverdata['error'], $serverdata['warn'], $serverdata['ok']);
261 $othertable->data = array_merge($otherdata['error'], $otherdata['warn'], $otherdata['ok']);
263 /// Print table
264 print_heading(get_string('serverchecks', 'admin'));
265 print_table($servertable);
266 if (count($othertable->data)){
267 print_heading(get_string('customcheck', 'admin'));
268 print_table($othertable);
271 /// Finally, if any error has happened, print the summary box
272 if (!$result) {
273 print_simple_box($strenvironmenterrortodo, 'center', '', '', '', 'environmentbox errorbox');
279 * This function will normalize any version to just a serie of numbers
280 * separated by dots. Everything else will be removed.
281 * @param string $version the original version
282 * @return string the normalized version
284 function normalize_version($version) {
286 /// 1.9 Beta 2 should be read 1.9 on enviromental checks, not 1.9.2
287 /// we can discard everything after the first space
288 $version = trim($version);
289 $versionarr = explode(" ",$version);
290 if (!empty($versionarr)) {
291 $version = $versionarr[0];
293 /// Replace everything but numbers and dots by dots
294 $version = preg_replace('/[^\.\d]/', '.', $version);
295 /// Combine multiple dots in one
296 $version = preg_replace('/(\.{2,})/', '.', $version);
297 /// Trim possible leading and trailing dots
298 $version = trim($version, '.');
300 return $version;
305 * This function will load the environment.xml file and xmlize it
306 * @return mixed the xmlized structure or false on error
308 function load_environment_xml() {
310 global $CFG;
312 static $data; //Only load and xmlize once by request
314 if (!empty($data)) {
315 return $data;
318 /// First of all, take a look inside $CFG->dataroot/environment/environment.xml
319 $file = $CFG->dataroot.'/environment/environment.xml';
320 $internalfile = $CFG->dirroot.'/'.$CFG->admin.'/environment.xml';
321 if (!is_file($file) || !is_readable($file) || filemtime($file) < filemtime($internalfile) ||
322 !$contents = file_get_contents($file)) {
323 /// Fallback to fixed $CFG->admin/environment.xml
324 if (!is_file($internalfile) || !is_readable($internalfile) || !$contents = file_get_contents($internalfile)) {
325 return false;
328 /// XML the whole file
329 $data = xmlize($contents);
331 return $data;
336 * This function will return the list of Moodle versions available
337 * @return mixed array of versions. False on error.
339 function get_list_of_environment_versions ($contents) {
341 static $versions = array();
343 if (!empty($versions)) {
344 return $versions;
347 if (isset($contents['COMPATIBILITY_MATRIX']['#']['MOODLE'])) {
348 foreach ($contents['COMPATIBILITY_MATRIX']['#']['MOODLE'] as $version) {
349 $versions[] = $version['@']['version'];
353 return $versions;
358 * This function will return the most recent version in the environment.xml
359 * file previous or equal to the version requested
360 * @param string version top version from which we start to look backwards
361 * @return string more recent version or false if not found
363 function get_latest_version_available ($version) {
365 /// Normalize the version requested
366 $version = normalize_version($version);
368 /// Load xml file
369 if (!$contents = load_environment_xml()) {
370 return false;
373 /// Detect available versions
374 if (!$versions = get_list_of_environment_versions($contents)) {
375 return false;
377 /// First we look for exact version
378 if (in_array($version, $versions)) {
379 return $version;
380 } else {
381 $found_version = false;
382 /// Not exact match, so we are going to iterate over the list searching
383 /// for the latest version before the requested one
384 foreach ($versions as $arrversion) {
385 if (version_compare($arrversion, $version, '<')) {
386 $found_version = $arrversion;
391 return $found_version;
396 * This function will return the xmlized data belonging to one Moodle version
397 * @return mixed the xmlized structure or false on error
399 function get_environment_for_version($version) {
401 /// Normalize the version requested
402 $version = normalize_version($version);
404 /// Load xml file
405 if (!$contents = load_environment_xml()) {
406 return false;
409 /// Detect available versions
410 if (!$versions = get_list_of_environment_versions($contents)) {
411 return false;
414 /// If the version requested is available
415 if (!in_array($version, $versions)) {
416 return false;
419 /// We now we have it. Extract from full contents.
420 $fl_arr = array_flip($versions);
422 return $contents['COMPATIBILITY_MATRIX']['#']['MOODLE'][$fl_arr[$version]];
427 * This function will check for everything (DB, PHP and PHP extensions for now)
428 * returning an array of environment_result objects.
429 * @param string $version xml version we are going to use to test this server
430 * @return array array of results encapsulated in one environment_result object
432 function environment_check($version) {
434 global $CFG;
436 /// Normalize the version requested
437 $version = normalize_version($version);
439 $results = array(); //To store all the results
441 /// Only run the moodle versions checker on upgrade, not on install
442 if (empty($CFG->running_installer)) {
443 $results[] = environment_check_moodle($version);
445 $results[] = environment_check_unicode($version);
446 $results[] = environment_check_database($version);
447 $results[] = environment_check_php($version);
449 $phpext_results = environment_check_php_extensions($version);
450 $results = array_merge($results, $phpext_results);
452 $custom_results = environment_custom_checks($version);
453 $results = array_merge($results, $custom_results);
455 return $results;
460 * This function will check if php extensions requirements are satisfied
461 * @param string $version xml version we are going to use to test this server
462 * @return array array of results encapsulated in one environment_result object
464 function environment_check_php_extensions($version) {
466 $results = array();
468 /// Get the enviroment version we need
469 if (!$data = get_environment_for_version($version)) {
470 /// Error. No version data found
471 $result = new environment_results('php_extension');
472 $result->setStatus(false);
473 $result->setErrorCode(NO_VERSION_DATA_FOUND);
474 return array($result);
477 /// Extract the php_extension part
478 if (!isset($data['#']['PHP_EXTENSIONS']['0']['#']['PHP_EXTENSION'])) {
479 /// Error. No PHP section found
480 $result = new environment_results('php_extension');
481 $result->setStatus(false);
482 $result->setErrorCode(NO_PHP_EXTENSIONS_SECTION_FOUND);
483 return array($result);
485 /// Iterate over extensions checking them and creating the needed environment_results
486 foreach($data['#']['PHP_EXTENSIONS']['0']['#']['PHP_EXTENSION'] as $extension) {
487 $result = new environment_results('php_extension');
488 /// Check for level
489 $level = get_level($extension);
490 /// Check for extension name
491 if (!isset($extension['@']['name'])) {
492 $result->setStatus(false);
493 $result->setErrorCode(NO_PHP_EXTENSIONS_NAME_FOUND);
494 } else {
495 $extension_name = $extension['@']['name'];
496 /// The name exists. Just check if it's an installed extension
497 if (!extension_loaded($extension_name)) {
498 $result->setStatus(false);
499 } else {
500 $result->setStatus(true);
502 $result->setLevel($level);
503 $result->setInfo($extension_name);
506 /// Do any actions defined in the XML file.
507 process_environment_result($extension, $result);
509 /// Add the result to the array of results
510 $results[] = $result;
514 return $results;
518 * This function will do the custom checks.
519 * @param string $version xml version we are going to use to test this server.
520 * @return array array of results encapsulated in environment_result objects.
522 function environment_custom_checks($version) {
523 global $CFG;
525 $results = array();
527 /// Get current Moodle version (release) for later compare
528 $release = isset($CFG->release) ? $CFG->release : $version; /// In case $CFG fails (at install) use $version
529 $current_version = normalize_version($release);
531 /// Get the enviroment version we need
532 if (!$data = get_environment_for_version($version)) {
533 /// Error. No version data found - but this will already have been reported.
534 return $results;
537 /// Extract the CUSTOM_CHECKS part
538 if (!isset($data['#']['CUSTOM_CHECKS']['0']['#']['CUSTOM_CHECK'])) {
539 /// No custom checks found - not a problem
540 return $results;
543 /// Iterate over extensions checking them and creating the needed environment_results
544 foreach($data['#']['CUSTOM_CHECKS']['0']['#']['CUSTOM_CHECK'] as $check) {
545 $result = new environment_results('custom_check');
547 /// Check for level
548 $level = get_level($check);
550 /// Check for extension name
551 if (isset($check['@']['file']) && isset($check['@']['function'])) {
552 $file = $CFG->dirroot . '/' . $check['@']['file'];
553 $function = $check['@']['function'];
554 if (is_readable($file)) {
555 include_once($file);
556 if (function_exists($function)) {
557 $result->setLevel($level);
558 $result->setInfo($function);
559 $result = $function($result);
560 } else {
561 /// Only show error for current version (where function MUST exist)
562 /// else, we are performing custom checks against future versiosn
563 /// and function MAY not exist, so it doesn't cause error, just skip
564 /// custom check by returning null. MDL-15939
565 if (version_compare($current_version, $version, '>=')) {
566 $result->setStatus(false);
567 $result->setInfo($function);
568 $result->setErrorCode(CUSTOM_CHECK_FUNCTION_MISSING);
569 } else {
570 $result = null;
573 } else {
574 /// Only show error for current version (where function MUST exist)
575 /// else, we are performing custom checks against future versiosn
576 /// and function MAY not exist, so it doesn't cause error, just skip
577 /// custom check by returning null. MDL-15939
578 if (version_compare($current_version, $version, '>=')) {
579 $result->setStatus(false);
580 $result->setInfo($function);
581 $result->setErrorCode(CUSTOM_CHECK_FILE_MISSING);
582 } else {
583 $result = null;
586 } else {
587 $result->setStatus(false);
588 $result->setErrorCode(NO_CUSTOM_CHECK_FOUND);
591 if (!is_null($result)) {
592 /// Do any actions defined in the XML file.
593 process_environment_result($check, $result);
595 /// Add the result to the array of results
596 $results[] = $result;
600 return $results;
604 * This function will check if Moodle requirements are satisfied
605 * @param string $version xml version we are going to use to test this server
606 * @return object results encapsulated in one environment_result object
608 function environment_check_moodle($version) {
610 $result = new environment_results('moodle');
612 /// Get the enviroment version we need
613 if (!$data = get_environment_for_version($version)) {
614 /// Error. No version data found
615 $result->setStatus(false);
616 $result->setErrorCode(NO_VERSION_DATA_FOUND);
617 return $result;
620 /// Extract the moodle part
621 if (!isset($data['@']['requires'])) {
622 $needed_version = '1.0'; /// Default to 1.0 if no moodle requires is found
623 } else {
624 /// Extract required moodle version
625 $needed_version = $data['@']['requires'];
628 /// Now search the version we are using
629 $current_version = normalize_version(get_config('', 'release'));
631 /// And finally compare them, saving results
632 if (version_compare($current_version, $needed_version, '>=')) {
633 $result->setStatus(true);
634 } else {
635 $result->setStatus(false);
637 $result->setLevel('required');
638 $result->setCurrentVersion($current_version);
639 $result->setNeededVersion($needed_version);
641 return $result;
645 * This function will check if php requirements are satisfied
646 * @param string $version xml version we are going to use to test this server
647 * @return object results encapsulated in one environment_result object
649 function environment_check_php($version) {
651 $result = new environment_results('php');
653 /// Get the enviroment version we need
654 if (!$data = get_environment_for_version($version)) {
655 /// Error. No version data found
656 $result->setStatus(false);
657 $result->setErrorCode(NO_VERSION_DATA_FOUND);
658 return $result;
661 /// Extract the php part
662 if (!isset($data['#']['PHP'])) {
663 /// Error. No PHP section found
664 $result->setStatus(false);
665 $result->setErrorCode(NO_PHP_SECTION_FOUND);
666 return $result;
667 } else {
668 /// Extract level and version
669 $level = get_level($data['#']['PHP']['0']);
670 if (!isset($data['#']['PHP']['0']['@']['version'])) {
671 $result->setStatus(false);
672 $result->setErrorCode(NO_PHP_VERSION_FOUND);
673 return $result;
674 } else {
675 $needed_version = $data['#']['PHP']['0']['@']['version'];
679 /// Now search the version we are using
680 $current_version = normalize_version(phpversion());
682 /// And finally compare them, saving results
683 if (version_compare($current_version, $needed_version, '>=')) {
684 $result->setStatus(true);
685 } else {
686 $result->setStatus(false);
688 $result->setLevel($level);
689 $result->setCurrentVersion($current_version);
690 $result->setNeededVersion($needed_version);
692 /// Do any actions defined in the XML file.
693 process_environment_result($data['#']['PHP'][0], $result);
695 return $result;
700 * This function will check if unicode database requirements are satisfied
701 * @param string $version xml version we are going to use to test this server
702 * @return object results encapsulated in one environment_result object
704 function environment_check_unicode($version) {
705 global $db;
707 $result = new environment_results('unicode');
709 /// Get the enviroment version we need
710 if (!$data = get_environment_for_version($version)) {
711 /// Error. No version data found
712 $result->setStatus(false);
713 $result->setErrorCode(NO_VERSION_DATA_FOUND);
714 return $result;
717 /// Extract the unicode part
719 if (!isset($data['#']['UNICODE'])) {
720 /// Error. No UNICODE section found
721 $result->setStatus(false);
722 $result->setErrorCode(NO_UNICODE_SECTION_FOUND);
723 return $result;
724 } else {
725 /// Extract level
726 $level = get_level($data['#']['UNICODE']['0']);
729 if (!$unicodedb = setup_is_unicodedb()) {
730 $result->setStatus(false);
731 } else {
732 $result->setStatus(true);
735 $result->setLevel($level);
737 /// Do any actions defined in the XML file.
738 process_environment_result($data['#']['UNICODE'][0], $result);
740 return $result;
744 * This function will check if database requirements are satisfied
745 * @param string $version xml version we are going to use to test this server
746 * @return object results encapsulated in one environment_result object
748 function environment_check_database($version) {
750 global $db;
752 $result = new environment_results('database');
754 $vendors = array(); //Array of vendors in version
756 /// Get the enviroment version we need
757 if (!$data = get_environment_for_version($version)) {
758 /// Error. No version data found
759 $result->setStatus(false);
760 $result->setErrorCode(NO_VERSION_DATA_FOUND);
761 return $result;
764 /// Extract the database part
765 if (!isset($data['#']['DATABASE'])) {
766 /// Error. No DATABASE section found
767 $result->setStatus(false);
768 $result->setErrorCode(NO_DATABASE_SECTION_FOUND);
769 return $result;
770 } else {
771 /// Extract level
772 $level = get_level($data['#']['DATABASE']['0']);
775 /// Extract DB vendors. At least 2 are mandatory (mysql & postgres)
776 if (!isset($data['#']['DATABASE']['0']['#']['VENDOR'])) {
777 /// Error. No VENDORS found
778 $result->setStatus(false);
779 $result->setErrorCode(NO_DATABASE_VENDORS_FOUND);
780 return $result;
781 } else {
782 /// Extract vendors
783 foreach ($data['#']['DATABASE']['0']['#']['VENDOR'] as $vendor) {
784 if (isset($vendor['@']['name']) && isset($vendor['@']['version'])) {
785 $vendors[$vendor['@']['name']] = $vendor['@']['version'];
786 $vendorsxml[$vendor['@']['name']] = $vendor;
790 /// Check we have the mysql vendor version
791 if (empty($vendors['mysql'])) {
792 $result->setStatus(false);
793 $result->setErrorCode(NO_DATABASE_VENDOR_MYSQL_FOUND);
794 return $result;
796 /// Check we have the postgres vendor version
797 if (empty($vendors['postgres'])) {
798 $result->setStatus(false);
799 $result->setErrorCode(NO_DATABASE_VENDOR_POSTGRES_FOUND);
800 return $result;
803 /// Now search the version we are using (depending of vendor)
804 $current_vendor = set_dbfamily();
806 $dbinfo = $db->ServerInfo();
807 $current_version = normalize_version($dbinfo['version']);
808 $needed_version = $vendors[$current_vendor];
810 /// Check we have a needed version
811 if (!$needed_version) {
812 $result->setStatus(false);
813 $result->setErrorCode(NO_DATABASE_VENDOR_VERSION_FOUND);
814 return $result;
817 /// And finally compare them, saving results
818 if (version_compare($current_version, $needed_version, '>=')) {
819 $result->setStatus(true);
820 } else {
821 $result->setStatus(false);
823 $result->setLevel($level);
824 $result->setCurrentVersion($current_version);
825 $result->setNeededVersion($needed_version);
826 $result->setInfo($current_vendor);
828 /// Do any actions defined in the XML file.
829 process_environment_result($vendorsxml[$current_vendor], $result);
831 return $result;
836 * This function will post-process the result record by executing the specified
837 * function, modifying it as necessary, also a custom message will be added
838 * to the result object to be printed by the display layer.
839 * Every bypass function must be defined in this file and it'll return
840 * true/false to decide if the original test is bypassed or no. Also
841 * such bypass functions are able to directly handling the result object
842 * although it should be only under exceptional conditions.
844 * @param string xmldata containing the bypass data
845 * @param object result object to be updated
847 function process_environment_bypass($xml, &$result) {
849 /// Only try to bypass if we were in error and it was required
850 if ($result->getStatus() || $result->getLevel() == 'optional') {
851 return;
854 /// It there is bypass info (function and message)
855 if (is_array($xml['#']) && isset($xml['#']['BYPASS'][0]['@']['function']) && isset($xml['#']['BYPASS'][0]['@']['message'])) {
856 $function = $xml['#']['BYPASS'][0]['@']['function'];
857 $message = $xml['#']['BYPASS'][0]['@']['message'];
858 /// Look for the function
859 if (function_exists($function)) {
860 /// Call it, and if bypass = true is returned, apply meesage
861 if ($function($result)) {
862 /// We only set the bypass message if the function itself hasn't defined it before
863 if (empty($result->getBypassStr)) {
864 $result->setBypassStr($message);
872 * This function will post-process the result record by executing the specified
873 * function, modifying it as necessary, also a custom message will be added
874 * to the result object to be printed by the display layer.
875 * Every restrict function must be defined in this file and it'll return
876 * true/false to decide if the original test is restricted or no. Also
877 * such restrict functions are able to directly handling the result object
878 * although it should be only under exceptional conditions.
880 * @param string xmldata containing the restrict data
881 * @param object result object to be updated
883 function process_environment_restrict($xml, &$result) {
885 /// Only try to restrict if we were not in error and it was required
886 if (!$result->getStatus() || $result->getLevel() == 'optional') {
887 return;
889 /// It there is restrict info (function and message)
890 if (is_array($xml['#']) && isset($xml['#']['RESTRICT'][0]['@']['function']) && isset($xml['#']['RESTRICT'][0]['@']['message'])) {
891 $function = $xml['#']['RESTRICT'][0]['@']['function'];
892 $message = $xml['#']['RESTRICT'][0]['@']['message'];
893 /// Look for the function
894 if (function_exists($function)) {
895 /// Call it, and if restrict = true is returned, apply meesage
896 if ($function($result)) {
897 /// We only set the restrict message if the function itself hasn't defined it before
898 if (empty($result->getRestrictStr)) {
899 $result->setRestrictStr($message);
907 * This function will detect if there is some message available to be added to the
908 * result in order to clarify enviromental details.
909 * @param string xmldata containing the feedback data
910 * @param object reult object to be updated
912 function process_environment_messages($xml, &$result) {
914 /// If there is feedback info
915 if (is_array($xml['#']) && isset($xml['#']['FEEDBACK'][0]['#'])) {
916 $feedbackxml = $xml['#']['FEEDBACK'][0]['#'];
918 if (!$result->status and $result->getLevel() == 'required') {
919 if (isset($feedbackxml['ON_ERROR'][0]['@']['message'])) {
920 $result->setFeedbackStr($feedbackxml['ON_ERROR'][0]['@']['message']);
922 } else if (!$result->status and $result->getLevel() == 'optional') {
923 if (isset($feedbackxml['ON_CHECK'][0]['@']['message'])) {
924 $result->setFeedbackStr($feedbackxml['ON_CHECK'][0]['@']['message']);
926 } else {
927 if (isset($feedbackxml['ON_OK'][0]['@']['message'])) {
928 $result->setFeedbackStr($feedbackxml['ON_OK'][0]['@']['message']);
935 //--- Helper Class to return results to caller ---//
939 * This class is used to return the results of the environment
940 * main functions (environment_check_xxxx)
942 class environment_results {
944 var $part; //which are we checking (database, php, php_extension)
945 var $status; //true/false
946 var $error_code; //integer. See constants at the beginning of the file
947 var $level; //required/optional
948 var $current_version; //current version detected
949 var $needed_version; //version needed
950 var $info; //Aux. info (DB vendor, library...)
951 var $feedback_str; //String to show on error|on check|on ok
952 var $bypass_str; //String to show if some bypass has happened
953 var $restrict_str; //String to show if some restrict has happened
956 * Constructor of the environment_result class. Just set default values
958 function environment_results($part) {
959 $this->part=$part;
960 $this->status=false;
961 $this->error_code=NO_ERROR;
962 $this->level='required';
963 $this->current_version='';
964 $this->needed_version='';
965 $this->info='';
966 $this->feedback_str='';
967 $this->bypass_str='';
968 $this->restrict_str='';
972 * Set the status
973 * @param boolean the status (true/false)
975 function setStatus($status) {
976 $this->status=$status;
977 if ($status) {
978 $this->setErrorCode(NO_ERROR);
983 * Set the error_code
984 * @param integer the error code (see constants above)
986 function setErrorCode($error_code) {
987 $this->error_code=$error_code;
991 * Set the level
992 * @param string the level (required, optional)
994 function setLevel($level) {
995 $this->level=$level;
999 * Set the current version
1000 * @param string the current version
1002 function setCurrentVersion($current_version) {
1003 $this->current_version=$current_version;
1007 * Set the needed version
1008 * @param string the needed version
1010 function setNeededVersion($needed_version) {
1011 $this->needed_version=$needed_version;
1015 * Set the auxiliary info
1016 * @param string the auxiliary info
1018 function setInfo($info) {
1019 $this->info=$info;
1023 * Set the feedback string
1024 * @param mixed the feedback string that will be fetched from the admin lang file.
1025 * pass just the string or pass an array of params for get_string
1026 * You always should put your string in admin.php but a third param is useful
1027 * to pass an $a object / string to get_string
1029 function setFeedbackStr($str) {
1030 $this->feedback_str=$str;
1035 * Set the bypass string
1036 * @param string the bypass string that will be fetched from the admin lang file.
1037 * pass just the string or pass an array of params for get_string
1038 * You always should put your string in admin.php but a third param is useful
1039 * to pass an $a object / string to get_string
1041 function setBypassStr($str) {
1042 $this->bypass_str=$str;
1046 * Set the restrict string
1047 * @param string the restrict string that will be fetched from the admin lang file.
1048 * pass just the string or pass an array of params for get_string
1049 * You always should put your string in admin.php but a third param is useful
1050 * to pass an $a object / string to get_string
1052 function setRestrictStr($str) {
1053 $this->restrict_str=$str;
1057 * Get the status
1058 * @return boolean result
1060 function getStatus() {
1061 return $this->status;
1065 * Get the error code
1066 * @return integer error code
1068 function getErrorCode() {
1069 return $this->error_code;
1073 * Get the level
1074 * @return string level
1076 function getLevel() {
1077 return $this->level;
1081 * Get the current version
1082 * @return string current version
1084 function getCurrentVersion() {
1085 return $this->current_version;
1089 * Get the needed version
1090 * @return string needed version
1092 function getNeededVersion() {
1093 return $this->needed_version;
1097 * Get the aux info
1098 * @return string info
1100 function getInfo() {
1101 return $this->info;
1105 * Get the part this result belongs to
1106 * @return string part
1108 function getPart() {
1109 return $this->part;
1113 * Get the feedback string
1114 * @return mixed feedback string (can be an array of params for get_string or a single string to fetch from
1115 * admin.php lang file).
1117 function getFeedbackStr() {
1118 return $this->feedback_str;
1122 * Get the bypass string
1123 * @return mixed bypass string (can be an array of params for get_string or a single string to fetch from
1124 * admin.php lang file).
1126 function getBypassStr() {
1127 return $this->bypass_str;
1131 * Get the restrict string
1132 * @return mixed restrict string (can be an array of params for get_string or a single string to fetch from
1133 * admin.php lang file).
1135 function getRestrictStr() {
1136 return $this->restrict_str;
1140 * @param mixed $string params for get_string, either a string to fetch from admin.php or an array of
1141 * params for get_string.
1142 * @param string $class css class(es) for message.
1143 * @return string feedback string fetched from lang file wrapped in p tag with class $class or returns
1144 * empty string if $string is empty.
1146 function strToReport($string, $class){
1147 if (!empty($string)){
1148 if (is_array($string)){
1149 $str = call_user_func_array('get_string', $string);
1150 } else {
1151 $str = get_string($string, 'admin');
1153 return '<p class="'.$class.'">'.$str.'</p>';
1154 } else {
1155 return '';
1160 /// Here all the bypass functions are coded to be used by the environment
1161 /// checker. All those functions will receive the result object and will
1162 /// return it modified as needed (status and bypass string)
1165 * This function will bypass MySQL 4.1.16 reqs if:
1166 * - We are using MySQL > 4.1.12, informing about problems with non latin chars in the future
1168 * @param object result object to handle
1169 * @return boolean true/false to determinate if the bypass has to be performed (true) or no (false)
1171 function bypass_mysql416_reqs ($result) {
1172 /// See if we are running MySQL >= 4.1.12
1173 if (version_compare($result->getCurrentVersion(), '4.1.12', '>=')) {
1174 return true;
1177 return false;
1180 /// Here all the restrict functions are coded to be used by the environment
1181 /// checker. All those functions will receive the result object and will
1182 /// return it modified as needed (status and bypass string)
1185 * This function will restrict PHP reqs if:
1186 * - We are using PHP 5.0.x, informing about the buggy version
1188 * @param object result object to handle
1189 * @return boolean true/false to determinate if the restrict has to be performed (true) or no (false)
1191 function restrict_php50_version($result) {
1192 if (version_compare($result->getCurrentVersion(), '5.0.0', '>=')
1193 and version_compare($result->getCurrentVersion(), '5.0.99', '<')) {
1194 return true;
1196 return false;
1200 * @param array $element the element from the environment.xml file that should have
1201 * either a level="required" or level="optional" attribute.
1202 * @read string "required" or "optional".
1204 function get_level($element) {
1205 $level = 'required';
1206 if (isset($element['@']['level'])) {
1207 $level = $element['@']['level'];
1208 if (!in_array($level, array('required', 'optional'))) {
1209 debugging('The level of a check in the environment.xml file must be "required" or "optional".', DEBUG_DEVELOPER);
1210 $level = 'required';
1212 } else {
1213 debugging('Checks in the environment.xml file must have a level="required" or level="optional" attribute.', DEBUG_DEVELOPER);
1215 return $level;
1219 * Once the result has been determined, look in the XML for any
1220 * messages, or other things that should be done depending on the outcome.
1221 * @param array $element the element from the environment.xml file which
1222 * may have children defining what should be done with the outcome.
1223 * @param object $result the result of the test, which may be modified by
1224 * this function as specified in the XML.
1226 function process_environment_result($element, &$result) {
1227 /// Process messages, modifying the $result if needed.
1228 process_environment_messages($element, $result);
1229 /// Process bypass, modifying $result if needed.
1230 process_environment_bypass($element, $result);
1231 /// Process restrict, modifying $result if needed.
1232 process_environment_restrict($element, $result);