Search moodle.org's
Developer Documentation

See Release Notes

  • Bug fixes for general core bugs in 3.10.x will end 8 November 2021 (12 months).
  • Bug fixes for security issues in 3.10.x will end 9 May 2022 (18 months).
  • PHP version: minimum PHP 7.2.0 Note: minimum PHP version has increased since Moodle 3.8. PHP 7.3.x and 7.4.x are supported too.

Differences Between: [Versions 310 and 311] [Versions 310 and 400] [Versions 310 and 401] [Versions 310 and 402] [Versions 310 and 403]

   1  <?php
   2  // This file is part of Moodle - http://moodle.org/
   3  //
   4  // Moodle is free software: you can redistribute it and/or modify
   5  // it under the terms of the GNU General Public License as published by
   6  // the Free Software Foundation, either version 3 of the License, or
   7  // (at your option) any later version.
   8  //
   9  // Moodle is distributed in the hope that it will be useful,
  10  // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11  // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  12  // GNU General Public License for more details.
  13  //
  14  // You should have received a copy of the GNU General Public License
  15  // along with Moodle.  If not, see <http://www.gnu.org/licenses/>.
  16  
  17  /**
  18   * Prepares PHPUnit environment, the phpunit.xml configuration
  19   * must specify this file as bootstrap.
  20   *
  21   * Exit codes: {@see phpunit_bootstrap_error()}
  22   *
  23   * @package    core
  24   * @category   phpunit
  25   * @copyright  2012 Petr Skoda {@link http://skodak.org}
  26   * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  27   */
  28  
  29  if (isset($_SERVER['REMOTE_ADDR'])) {
  30      die; // No access from web!
  31  }
  32  
  33  // we want to know about all problems
  34  error_reporting(E_ALL | E_STRICT);
  35  ini_set('display_errors', '1');
  36  ini_set('log_errors', '1');
  37  
  38  // Make sure OPcache does not strip comments, we need them in phpunit!
  39  if (ini_get('opcache.enable') and strtolower(ini_get('opcache.enable')) !== 'off') {
  40      if (!ini_get('opcache.save_comments') or strtolower(ini_get('opcache.save_comments')) === 'off') {
  41          ini_set('opcache.enable', 0);
  42      } else {
  43          ini_set('opcache.load_comments', 1);
  44      }
  45  }
  46  
  47  if (!defined('IGNORE_COMPONENT_CACHE')) {
  48      define('IGNORE_COMPONENT_CACHE', true);
  49  }
  50  
  51  require_once (__DIR__.'/bootstraplib.php');
  52  require_once (__DIR__.'/../testing/lib.php');
  53  require_once (__DIR__.'/classes/autoloader.php');
  54  
  55  if (isset($_SERVER['REMOTE_ADDR'])) {
  56      phpunit_bootstrap_error(1, 'Unit tests can be executed only from command line!');
  57  }
  58  
  59  if (defined('PHPUNIT_TEST')) {
  60      phpunit_bootstrap_error(1, "PHPUNIT_TEST constant must not be manually defined anywhere!");
  61  }
  62  /** PHPUnit testing framework active */
  63  define('PHPUNIT_TEST', true);
  64  
  65  if (!defined('PHPUNIT_UTIL')) {
  66      /** Identifies utility scripts - the database does not need to be initialised */
  67      define('PHPUNIT_UTIL', false);
  68  }
  69  
  70  if (defined('CLI_SCRIPT')) {
  71      phpunit_bootstrap_error(1, 'CLI_SCRIPT must not be manually defined in any PHPUnit test scripts');
  72  }
  73  define('CLI_SCRIPT', true);
  74  
  75  $phpunitversion = PHPUnit\Runner\Version::id();
  76  if ($phpunitversion === '@package_version@') {
  77      // library checked out from git, let's hope dev knows that 3.6.0 is required
  78  } else if (version_compare($phpunitversion, '3.6.0', 'lt')) {
  79      phpunit_bootstrap_error(PHPUNIT_EXITCODE_PHPUNITWRONG, $phpunitversion);
  80  }
  81  unset($phpunitversion);
  82  
  83  // only load CFG from config.php, stop ASAP in lib/setup.php
  84  define('ABORT_AFTER_CONFIG', true);
  85  require(__DIR__ . '/../../config.php');
  86  
  87  if (!defined('PHPUNIT_LONGTEST')) {
  88      /** Execute longer version of tests */
  89      define('PHPUNIT_LONGTEST', false);
  90  }
  91  
  92  // remove error handling overrides done in config.php
  93  error_reporting(E_ALL | E_STRICT);
  94  ini_set('display_errors', '1');
  95  ini_set('log_errors', '1');
  96  set_time_limit(0); // no time limit in CLI scripts, user may cancel execution
  97  
  98  // prepare dataroot
  99  umask(0);
 100  if (isset($CFG->phpunit_directorypermissions)) {
 101      $CFG->directorypermissions = $CFG->phpunit_directorypermissions;
 102  } else {
 103      $CFG->directorypermissions = 02777;
 104  }
 105  $CFG->filepermissions = ($CFG->directorypermissions & 0666);
 106  if (!isset($CFG->phpunit_dataroot)) {
 107      phpunit_bootstrap_error(PHPUNIT_EXITCODE_CONFIGERROR, 'Missing $CFG->phpunit_dataroot in config.php, can not run tests!');
 108  }
 109  
 110  // Create test dir if does not exists yet.
 111  if (!file_exists($CFG->phpunit_dataroot)) {
 112      mkdir($CFG->phpunit_dataroot, $CFG->directorypermissions);
 113  }
 114  if (!is_dir($CFG->phpunit_dataroot)) {
 115      phpunit_bootstrap_error(PHPUNIT_EXITCODE_CONFIGERROR, '$CFG->phpunit_dataroot directory can not be created, can not run tests!');
 116  }
 117  
 118  // Ensure we access to phpunit_dataroot realpath always.
 119  $CFG->phpunit_dataroot = realpath($CFG->phpunit_dataroot);
 120  
 121  if (isset($CFG->dataroot) and $CFG->phpunit_dataroot === $CFG->dataroot) {
 122      phpunit_bootstrap_error(PHPUNIT_EXITCODE_CONFIGERROR, '$CFG->dataroot and $CFG->phpunit_dataroot must not be identical, can not run tests!');
 123  }
 124  
 125  if (!is_writable($CFG->phpunit_dataroot)) {
 126      // try to fix permissions if possible
 127      if (function_exists('posix_getuid')) {
 128          $chmod = fileperms($CFG->phpunit_dataroot);
 129          if (fileowner($CFG->phpunit_dataroot) == posix_getuid()) {
 130              $chmod = $chmod | 0700;
 131              chmod($CFG->phpunit_dataroot, $chmod);
 132          }
 133      }
 134      if (!is_writable($CFG->phpunit_dataroot)) {
 135          phpunit_bootstrap_error(PHPUNIT_EXITCODE_CONFIGERROR, '$CFG->phpunit_dataroot directory is not writable, can not run tests!');
 136      }
 137  }
 138  if (!file_exists("$CFG->phpunit_dataroot/phpunittestdir.txt")) {
 139      if ($dh = opendir($CFG->phpunit_dataroot)) {
 140          while (($file = readdir($dh)) !== false) {
 141              if ($file === 'phpunit' or $file === '.' or $file === '..' or $file === '.DS_Store') {
 142                  continue;
 143              }
 144              phpunit_bootstrap_error(PHPUNIT_EXITCODE_CONFIGERROR, '$CFG->phpunit_dataroot directory is not empty, can not run tests! Is it used for anything else?');
 145          }
 146          closedir($dh);
 147          unset($dh);
 148          unset($file);
 149      }
 150  
 151      // now we are 100% sure this dir is used only for phpunit tests
 152      testing_initdataroot($CFG->phpunit_dataroot, 'phpunit');
 153  }
 154  
 155  // verify db prefix
 156  if (!isset($CFG->phpunit_prefix)) {
 157      phpunit_bootstrap_error(PHPUNIT_EXITCODE_CONFIGERROR, 'Missing $CFG->phpunit_prefix in config.php, can not run tests!');
 158  }
 159  if ($CFG->phpunit_prefix === '') {
 160      phpunit_bootstrap_error(PHPUNIT_EXITCODE_CONFIGERROR, '$CFG->phpunit_prefix can not be empty, can not run tests!');
 161  }
 162  if (isset($CFG->prefix) and $CFG->prefix === $CFG->phpunit_prefix) {
 163      phpunit_bootstrap_error(PHPUNIT_EXITCODE_CONFIGERROR, '$CFG->prefix and $CFG->phpunit_prefix must not be identical, can not run tests!');
 164  }
 165  
 166  // override CFG settings if necessary and throw away extra CFG settings
 167  $CFG->wwwroot   = 'https://www.example.com/moodle';
 168  $CFG->dataroot  = $CFG->phpunit_dataroot;
 169  $CFG->prefix    = $CFG->phpunit_prefix;
 170  $CFG->dbtype    = isset($CFG->phpunit_dbtype) ? $CFG->phpunit_dbtype : $CFG->dbtype;
 171  $CFG->dblibrary = isset($CFG->phpunit_dblibrary) ? $CFG->phpunit_dblibrary : $CFG->dblibrary;
 172  $CFG->dbhost    = isset($CFG->phpunit_dbhost) ? $CFG->phpunit_dbhost : $CFG->dbhost;
 173  $CFG->dbname    = isset($CFG->phpunit_dbname) ? $CFG->phpunit_dbname : $CFG->dbname;
 174  $CFG->dbuser    = isset($CFG->phpunit_dbuser) ? $CFG->phpunit_dbuser : $CFG->dbuser;
 175  $CFG->dbpass    = isset($CFG->phpunit_dbpass) ? $CFG->phpunit_dbpass : $CFG->dbpass;
 176  $CFG->prefix    = isset($CFG->phpunit_prefix) ? $CFG->phpunit_prefix : $CFG->prefix;
 177  $CFG->dboptions = isset($CFG->phpunit_dboptions) ? $CFG->phpunit_dboptions : $CFG->dboptions;
 178  
 179  $allowed = array('wwwroot', 'dataroot', 'dirroot', 'admin', 'directorypermissions', 'filepermissions',
 180                   'dbtype', 'dblibrary', 'dbhost', 'dbname', 'dbuser', 'dbpass', 'prefix', 'dboptions',
 181                   'proxyhost', 'proxyport', 'proxytype', 'proxyuser', 'proxypassword', 'proxybypass', // keep proxy settings from config.php
 182                   'altcacheconfigpath', 'pathtogs', 'pathtophp', 'pathtodu', 'aspellpath', 'pathtodot',
 183                   'pathtounoconv', 'alternative_file_system_class', 'pathtopython'
 184                  );
 185  $productioncfg = (array)$CFG;
 186  $CFG = new stdClass();
 187  foreach ($productioncfg as $key=>$value) {
 188      if (!in_array($key, $allowed) and strpos($key, 'phpunit_') !== 0) {
 189          // ignore
 190          continue;
 191      }
 192      $CFG->{$key} = $value;
 193  }
 194  unset($key);
 195  unset($value);
 196  unset($allowed);
 197  unset($productioncfg);
 198  
 199  // force the same CFG settings in all sites
 200  $CFG->debug = (E_ALL | E_STRICT); // can not use DEBUG_DEVELOPER yet
 201  $CFG->debugdeveloper = true;
 202  $CFG->debugdisplay = 1;
 203  error_reporting($CFG->debug);
 204  ini_set('display_errors', '1');
 205  ini_set('log_errors', '1');
 206  
 207  // some ugly hacks
 208  $CFG->themerev = 1;
 209  $CFG->jsrev = 1;
 210  
 211  // load test case stub classes and other stuff
 212  require_once("$CFG->dirroot/lib/phpunit/lib.php");
 213  
 214  // finish moodle init
 215  define('ABORT_AFTER_CONFIG_CANCEL', true);
 216  if (isset($CFG->phpunit_profilingenabled) && $CFG->phpunit_profilingenabled) {
 217      $CFG->profilingenabled = true;
 218      $CFG->profilingincluded = '*';
 219  }
 220  require("$CFG->dirroot/lib/setup.php");
 221  
 222  raise_memory_limit(MEMORY_HUGE);
 223  
 224  if (PHPUNIT_UTIL) {
 225      // we are not going to do testing, this is 'true' in utility scripts that only init database
 226      return;
 227  }
 228  
 229  // is database and dataroot ready for testing?
 230  list($errorcode, $message) = phpunit_util::testing_ready_problem();
 231  // print some version info
 232  phpunit_util::bootstrap_moodle_info();
 233  if ($errorcode) {
 234      phpunit_bootstrap_error($errorcode, $message);
 235  }
 236  
 237  // prepare for the first test run - store fresh globals, reset database and dataroot, etc.
 238  phpunit_util::bootstrap_init();