Search moodle.org's
Developer Documentation

See Release Notes
Long Term Support Release

  • Bug fixes for general core bugs in 3.9.x will end* 10 May 2021 (12 months).
  • Bug fixes for security issues in 3.9.x will end* 8 May 2023 (36 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 39 and 310] [Versions 39 and 311] [Versions 39 and 400] [Versions 39 and 401] [Versions 39 and 402] [Versions 39 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   * Testing util classes
  19   *
  20   * @abstract
  21   * @package    core
  22   * @category   test
  23   * @copyright  2012 Petr Skoda {@link http://skodak.org}
  24   * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  25   */
  26  
  27  /**
  28   * Utils for test sites creation
  29   *
  30   * @package   core
  31   * @category  test
  32   * @copyright 2012 Petr Skoda {@link http://skodak.org}
  33   * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  34   */
  35  abstract class testing_util {
  36  
  37      /**
  38       * @var string dataroot (likely to be $CFG->dataroot).
  39       */
  40      private static $dataroot = null;
  41  
  42      /**
  43       * @var testing_data_generator
  44       */
  45      protected static $generator = null;
  46  
  47      /**
  48       * @var string current version hash from php files
  49       */
  50      protected static $versionhash = null;
  51  
  52      /**
  53       * @var array original content of all database tables
  54       */
  55      protected static $tabledata = null;
  56  
  57      /**
  58       * @var array original structure of all database tables
  59       */
  60      protected static $tablestructure = null;
  61  
  62      /**
  63       * @var array keep list of sequenceid used in a table.
  64       */
  65      private static $tablesequences = array();
  66  
  67      /**
  68       * @var array list of updated tables.
  69       */
  70      public static $tableupdated = array();
  71  
  72      /**
  73       * @var array original structure of all database tables
  74       */
  75      protected static $sequencenames = null;
  76  
  77      /**
  78       * @var string name of the json file where we store the list of dataroot files to not reset during reset_dataroot.
  79       */
  80      private static $originaldatafilesjson = 'originaldatafiles.json';
  81  
  82      /**
  83       * @var boolean set to true once $originaldatafilesjson file is created.
  84       */
  85      private static $originaldatafilesjsonadded = false;
  86  
  87      /**
  88       * @var int next sequence value for a single test cycle.
  89       */
  90      protected static $sequencenextstartingid = null;
  91  
  92      /**
  93       * Return the name of the JSON file containing the init filenames.
  94       *
  95       * @static
  96       * @return string
  97       */
  98      public static function get_originaldatafilesjson() {
  99          return self::$originaldatafilesjson;
 100      }
 101  
 102      /**
 103       * Return the dataroot. It's useful when mocking the dataroot when unit testing this class itself.
 104       *
 105       * @static
 106       * @return string the dataroot.
 107       */
 108      public static function get_dataroot() {
 109          global $CFG;
 110  
 111          //  By default it's the test framework dataroot.
 112          if (empty(self::$dataroot)) {
 113              self::$dataroot = $CFG->dataroot;
 114          }
 115  
 116          return self::$dataroot;
 117      }
 118  
 119      /**
 120       * Set the dataroot. It's useful when mocking the dataroot when unit testing this class itself.
 121       *
 122       * @param string $dataroot the dataroot of the test framework.
 123       * @static
 124       */
 125      public static function set_dataroot($dataroot) {
 126          self::$dataroot = $dataroot;
 127      }
 128  
 129      /**
 130       * Returns the testing framework name
 131       * @static
 132       * @return string
 133       */
 134      protected static final function get_framework() {
 135          $classname = get_called_class();
 136          return substr($classname, 0, strpos($classname, '_'));
 137      }
 138  
 139      /**
 140       * Get data generator
 141       * @static
 142       * @return testing_data_generator
 143       */
 144      public static function get_data_generator() {
 145          if (is_null(self::$generator)) {
 146              require_once (__DIR__.'/../generator/lib.php');
 147              self::$generator = new testing_data_generator();
 148          }
 149          return self::$generator;
 150      }
 151  
 152      /**
 153       * Does this site (db and dataroot) appear to be used for production?
 154       * We try very hard to prevent accidental damage done to production servers!!
 155       *
 156       * @static
 157       * @return bool
 158       */
 159      public static function is_test_site() {
 160          global $DB, $CFG;
 161  
 162          $framework = self::get_framework();
 163  
 164          if (!file_exists(self::get_dataroot() . '/' . $framework . 'testdir.txt')) {
 165              // this is already tested in bootstrap script,
 166              // but anyway presence of this file means the dataroot is for testing
 167              return false;
 168          }
 169  
 170          $tables = $DB->get_tables(false);
 171          if ($tables) {
 172              if (!$DB->get_manager()->table_exists('config')) {
 173                  return false;
 174              }
 175              if (!get_config('core', $framework . 'test')) {
 176                  return false;
 177              }
 178          }
 179  
 180          return true;
 181      }
 182  
 183      /**
 184       * Returns whether test database and dataroot were created using the current version codebase
 185       *
 186       * @return bool
 187       */
 188      public static function is_test_data_updated() {
 189          global $DB;
 190  
 191          $framework = self::get_framework();
 192  
 193          $datarootpath = self::get_dataroot() . '/' . $framework;
 194          if (!file_exists($datarootpath . '/tabledata.ser') or !file_exists($datarootpath . '/tablestructure.ser')) {
 195              return false;
 196          }
 197  
 198          if (!file_exists($datarootpath . '/versionshash.txt')) {
 199              return false;
 200          }
 201  
 202          $hash = core_component::get_all_versions_hash();
 203          $oldhash = file_get_contents($datarootpath . '/versionshash.txt');
 204  
 205          if ($hash !== $oldhash) {
 206              return false;
 207          }
 208  
 209          // A direct database request must be used to avoid any possible caching of an older value.
 210          $dbhash = $DB->get_field('config', 'value', array('name' => $framework . 'test'));
 211          if ($hash !== $dbhash) {
 212              return false;
 213          }
 214  
 215          return true;
 216      }
 217  
 218      /**
 219       * Stores the status of the database
 220       *
 221       * Serializes the contents and the structure and
 222       * stores it in the test framework space in dataroot
 223       */
 224      protected static function store_database_state() {
 225          global $DB, $CFG;
 226  
 227          $framework = self::get_framework();
 228  
 229          // store data for all tables
 230          $data = array();
 231          $structure = array();
 232          $tables = $DB->get_tables();
 233          foreach ($tables as $table) {
 234              $columns = $DB->get_columns($table);
 235              $structure[$table] = $columns;
 236              if (isset($columns['id']) and $columns['id']->auto_increment) {
 237                  $data[$table] = $DB->get_records($table, array(), 'id ASC');
 238              } else {
 239                  // there should not be many of these
 240                  $data[$table] = $DB->get_records($table, array());
 241              }
 242          }
 243          $data = serialize($data);
 244          $datafile = self::get_dataroot() . '/' . $framework . '/tabledata.ser';
 245          file_put_contents($datafile, $data);
 246          testing_fix_file_permissions($datafile);
 247  
 248          $structure = serialize($structure);
 249          $structurefile = self::get_dataroot() . '/' . $framework . '/tablestructure.ser';
 250          file_put_contents($structurefile, $structure);
 251          testing_fix_file_permissions($structurefile);
 252      }
 253  
 254      /**
 255       * Stores the version hash in both database and dataroot
 256       */
 257      protected static function store_versions_hash() {
 258          global $CFG;
 259  
 260          $framework = self::get_framework();
 261          $hash = core_component::get_all_versions_hash();
 262  
 263          // add test db flag
 264          set_config($framework . 'test', $hash);
 265  
 266          // hash all plugin versions - helps with very fast detection of db structure changes
 267          $hashfile = self::get_dataroot() . '/' . $framework . '/versionshash.txt';
 268          file_put_contents($hashfile, $hash);
 269          testing_fix_file_permissions($hashfile);
 270      }
 271  
 272      /**
 273       * Returns contents of all tables right after installation.
 274       * @static
 275       * @return array  $table=>$records
 276       */
 277      protected static function get_tabledata() {
 278          if (!isset(self::$tabledata)) {
 279              $framework = self::get_framework();
 280  
 281              $datafile = self::get_dataroot() . '/' . $framework . '/tabledata.ser';
 282              if (!file_exists($datafile)) {
 283                  // Not initialised yet.
 284                  return array();
 285              }
 286  
 287              $data = file_get_contents($datafile);
 288              self::$tabledata = unserialize($data);
 289          }
 290  
 291          if (!is_array(self::$tabledata)) {
 292              testing_error(1, 'Can not read dataroot/' . $framework . '/tabledata.ser or invalid format, reinitialize test database.');
 293          }
 294  
 295          return self::$tabledata;
 296      }
 297  
 298      /**
 299       * Returns structure of all tables right after installation.
 300       * @static
 301       * @return array $table=>$records
 302       */
 303      public static function get_tablestructure() {
 304          if (!isset(self::$tablestructure)) {
 305              $framework = self::get_framework();
 306  
 307              $structurefile = self::get_dataroot() . '/' . $framework . '/tablestructure.ser';
 308              if (!file_exists($structurefile)) {
 309                  // Not initialised yet.
 310                  return array();
 311              }
 312  
 313              $data = file_get_contents($structurefile);
 314              self::$tablestructure = unserialize($data);
 315          }
 316  
 317          if (!is_array(self::$tablestructure)) {
 318              testing_error(1, 'Can not read dataroot/' . $framework . '/tablestructure.ser or invalid format, reinitialize test database.');
 319          }
 320  
 321          return self::$tablestructure;
 322      }
 323  
 324      /**
 325       * Returns the names of sequences for each autoincrementing id field in all standard tables.
 326       * @static
 327       * @return array $table=>$sequencename
 328       */
 329      public static function get_sequencenames() {
 330          global $DB;
 331  
 332          if (isset(self::$sequencenames)) {
 333              return self::$sequencenames;
 334          }
 335  
 336          if (!$structure = self::get_tablestructure()) {
 337              return array();
 338          }
 339  
 340          self::$sequencenames = array();
 341          foreach ($structure as $table => $ignored) {
 342              $name = $DB->get_manager()->generator->getSequenceFromDB(new xmldb_table($table));
 343              if ($name !== false) {
 344                  self::$sequencenames[$table] = $name;
 345              }
 346          }
 347  
 348          return self::$sequencenames;
 349      }
 350  
 351      /**
 352       * Returns list of tables that are unmodified and empty.
 353       *
 354       * @static
 355       * @return array of table names, empty if unknown
 356       */
 357      protected static function guess_unmodified_empty_tables() {
 358          global $DB;
 359  
 360          $dbfamily = $DB->get_dbfamily();
 361  
 362          if ($dbfamily === 'mysql') {
 363              $empties = array();
 364              $prefix = $DB->get_prefix();
 365              $rs = $DB->get_recordset_sql("SHOW TABLE STATUS LIKE ?", array($prefix.'%'));
 366              foreach ($rs as $info) {
 367                  $table = strtolower($info->name);
 368                  if (strpos($table, $prefix) !== 0) {
 369                      // incorrect table match caused by _
 370                      continue;
 371                  }
 372  
 373                  if (!is_null($info->auto_increment) && $info->rows == 0 && ($info->auto_increment == 1)) {
 374                      $table = preg_replace('/^'.preg_quote($prefix, '/').'/', '', $table);
 375                      $empties[$table] = $table;
 376                  }
 377              }
 378              $rs->close();
 379              return $empties;
 380  
 381          } else if ($dbfamily === 'mssql') {
 382              $empties = array();
 383              $prefix = $DB->get_prefix();
 384              $sql = "SELECT t.name
 385                        FROM sys.identity_columns i
 386                        JOIN sys.tables t ON t.object_id = i.object_id
 387                       WHERE t.name LIKE ?
 388                         AND i.name = 'id'
 389                         AND i.last_value IS NULL";
 390              $rs = $DB->get_recordset_sql($sql, array($prefix.'%'));
 391              foreach ($rs as $info) {
 392                  $table = strtolower($info->name);
 393                  if (strpos($table, $prefix) !== 0) {
 394                      // incorrect table match caused by _
 395                      continue;
 396                  }
 397                  $table = preg_replace('/^'.preg_quote($prefix, '/').'/', '', $table);
 398                  $empties[$table] = $table;
 399              }
 400              $rs->close();
 401              return $empties;
 402  
 403          } else if ($dbfamily === 'oracle') {
 404              $sequences = self::get_sequencenames();
 405              $sequences = array_map('strtoupper', $sequences);
 406              $lookup = array_flip($sequences);
 407              $empties = array();
 408              list($seqs, $params) = $DB->get_in_or_equal($sequences);
 409              $sql = "SELECT sequence_name FROM user_sequences WHERE last_number = 1 AND sequence_name $seqs";
 410              $rs = $DB->get_recordset_sql($sql, $params);
 411              foreach ($rs as $seq) {
 412                  $table = $lookup[$seq->sequence_name];
 413                  $empties[$table] = $table;
 414              }
 415              $rs->close();
 416              return $empties;
 417  
 418          } else {
 419              return array();
 420          }
 421      }
 422  
 423      /**
 424       * Determine the next unique starting id sequences.
 425       *
 426       * @static
 427       * @param array $records The records to use to determine the starting value for the table.
 428       * @param string $table table name.
 429       * @return int The value the sequence should be set to.
 430       */
 431      private static function get_next_sequence_starting_value($records, $table) {
 432          if (isset(self::$tablesequences[$table])) {
 433              return self::$tablesequences[$table];
 434          }
 435  
 436          $id = self::$sequencenextstartingid;
 437  
 438          // If there are records, calculate the minimum id we can use.
 439          // It must be bigger than the last record's id.
 440          if (!empty($records)) {
 441              $lastrecord = end($records);
 442              $id = max($id, $lastrecord->id + 1);
 443          }
 444  
 445          self::$sequencenextstartingid = $id + 1000;
 446  
 447          self::$tablesequences[$table] = $id;
 448  
 449          return $id;
 450      }
 451  
 452      /**
 453       * Reset all database sequences to initial values.
 454       *
 455       * @static
 456       * @param array $empties tables that are known to be unmodified and empty
 457       * @return void
 458       */
 459      public static function reset_all_database_sequences(array $empties = null) {
 460          global $DB;
 461  
 462          if (!$data = self::get_tabledata()) {
 463              // Not initialised yet.
 464              return;
 465          }
 466          if (!$structure = self::get_tablestructure()) {
 467              // Not initialised yet.
 468              return;
 469          }
 470  
 471          $updatedtables = self::$tableupdated;
 472  
 473          // If all starting Id's are the same, it's difficult to detect coding and testing
 474          // errors that use the incorrect id in tests.  The classic case is cmid vs instance id.
 475          // To reduce the chance of the coding error, we start sequences at different values where possible.
 476          // In a attempt to avoid tables with existing id's we start at a high number.
 477          // Reset the value each time all database sequences are reset.
 478          if (defined('PHPUNIT_SEQUENCE_START') and PHPUNIT_SEQUENCE_START) {
 479              self::$sequencenextstartingid = PHPUNIT_SEQUENCE_START;
 480          } else {
 481              self::$sequencenextstartingid = 100000;
 482          }
 483  
 484          $dbfamily = $DB->get_dbfamily();
 485          if ($dbfamily === 'postgres') {
 486              $queries = array();
 487              $prefix = $DB->get_prefix();
 488              foreach ($data as $table => $records) {
 489                  // If table is not modified then no need to do anything.
 490                  if (!isset($updatedtables[$table])) {
 491                      continue;
 492                  }
 493                  if (isset($structure[$table]['id']) and $structure[$table]['id']->auto_increment) {
 494                      $nextid = self::get_next_sequence_starting_value($records, $table);
 495                      $queries[] = "ALTER SEQUENCE {$prefix}{$table}_id_seq RESTART WITH $nextid";
 496                  }
 497              }
 498              if ($queries) {
 499                  $DB->change_database_structure(implode(';', $queries));
 500              }
 501  
 502          } else if ($dbfamily === 'mysql') {
 503              $queries = array();
 504              $sequences = array();
 505              $prefix = $DB->get_prefix();
 506              $rs = $DB->get_recordset_sql("SHOW TABLE STATUS LIKE ?", array($prefix.'%'));
 507              foreach ($rs as $info) {
 508                  $table = strtolower($info->name);
 509                  if (strpos($table, $prefix) !== 0) {
 510                      // incorrect table match caused by _
 511                      continue;
 512                  }
 513                  if (!is_null($info->auto_increment)) {
 514                      $table = preg_replace('/^'.preg_quote($prefix, '/').'/', '', $table);
 515                      $sequences[$table] = $info->auto_increment;
 516                  }
 517              }
 518              $rs->close();
 519              $prefix = $DB->get_prefix();
 520              foreach ($data as $table => $records) {
 521                  // If table is not modified then no need to do anything.
 522                  if (!isset($updatedtables[$table])) {
 523                      continue;
 524                  }
 525                  if (isset($structure[$table]['id']) and $structure[$table]['id']->auto_increment) {
 526                      if (isset($sequences[$table])) {
 527                          $nextid = self::get_next_sequence_starting_value($records, $table);
 528                          if ($sequences[$table] != $nextid) {
 529                              $queries[] = "ALTER TABLE {$prefix}{$table} AUTO_INCREMENT = $nextid";
 530                          }
 531                      } else {
 532                          // some problem exists, fallback to standard code
 533                          $DB->get_manager()->reset_sequence($table);
 534                      }
 535                  }
 536              }
 537              if ($queries) {
 538                  $DB->change_database_structure(implode(';', $queries));
 539              }
 540  
 541          } else if ($dbfamily === 'oracle') {
 542              $sequences = self::get_sequencenames();
 543              $sequences = array_map('strtoupper', $sequences);
 544              $lookup = array_flip($sequences);
 545  
 546              $current = array();
 547              list($seqs, $params) = $DB->get_in_or_equal($sequences);
 548              $sql = "SELECT sequence_name, last_number FROM user_sequences WHERE sequence_name $seqs";
 549              $rs = $DB->get_recordset_sql($sql, $params);
 550              foreach ($rs as $seq) {
 551                  $table = $lookup[$seq->sequence_name];
 552                  $current[$table] = $seq->last_number;
 553              }
 554              $rs->close();
 555  
 556              foreach ($data as $table => $records) {
 557                  // If table is not modified then no need to do anything.
 558                  if (!isset($updatedtables[$table])) {
 559                      continue;
 560                  }
 561                  if (isset($structure[$table]['id']) and $structure[$table]['id']->auto_increment) {
 562                      $lastrecord = end($records);
 563                      if ($lastrecord) {
 564                          $nextid = $lastrecord->id + 1;
 565                      } else {
 566                          $nextid = 1;
 567                      }
 568                      if (!isset($current[$table])) {
 569                          $DB->get_manager()->reset_sequence($table);
 570                      } else if ($nextid == $current[$table]) {
 571                          continue;
 572                      }
 573                      // reset as fast as possible - alternatively we could use http://stackoverflow.com/questions/51470/how-do-i-reset-a-sequence-in-oracle
 574                      $seqname = $sequences[$table];
 575                      $cachesize = $DB->get_manager()->generator->sequence_cache_size;
 576                      $DB->change_database_structure("DROP SEQUENCE $seqname");
 577                      $DB->change_database_structure("CREATE SEQUENCE $seqname START WITH $nextid INCREMENT BY 1 NOMAXVALUE CACHE $cachesize");
 578                  }
 579              }
 580  
 581          } else {
 582              // note: does mssql support any kind of faster reset?
 583              // This also implies mssql will not use unique sequence values.
 584              if (is_null($empties) and (empty($updatedtables))) {
 585                  $empties = self::guess_unmodified_empty_tables();
 586              }
 587              foreach ($data as $table => $records) {
 588                  // If table is not modified then no need to do anything.
 589                  if (isset($empties[$table]) or (!isset($updatedtables[$table]))) {
 590                      continue;
 591                  }
 592                  if (isset($structure[$table]['id']) and $structure[$table]['id']->auto_increment) {
 593                      $DB->get_manager()->reset_sequence($table);
 594                  }
 595              }
 596          }
 597      }
 598  
 599      /**
 600       * Reset all database tables to default values.
 601       * @static
 602       * @return bool true if reset done, false if skipped
 603       */
 604      public static function reset_database() {
 605          global $DB;
 606  
 607          $tables = $DB->get_tables(false);
 608          if (!$tables or empty($tables['config'])) {
 609              // not installed yet
 610              return false;
 611          }
 612  
 613          if (!$data = self::get_tabledata()) {
 614              // not initialised yet
 615              return false;
 616          }
 617          if (!$structure = self::get_tablestructure()) {
 618              // not initialised yet
 619              return false;
 620          }
 621  
 622          $empties = array();
 623          // Use local copy of self::$tableupdated, as list gets updated in for loop.
 624          $updatedtables = self::$tableupdated;
 625  
 626          // If empty tablesequences list then it's the very first run.
 627          if (empty(self::$tablesequences) && (($DB->get_dbfamily() != 'mysql') && ($DB->get_dbfamily() != 'postgres'))) {
 628              // Only Mysql and Postgres support random sequence, so don't guess, just reset everything on very first run.
 629              $empties = self::guess_unmodified_empty_tables();
 630          }
 631  
 632          // Check if any table has been modified by behat selenium process.
 633          if (defined('BEHAT_SITE_RUNNING')) {
 634              // Crazy way to reset :(.
 635              $tablesupdatedfile = self::get_tables_updated_by_scenario_list_path();
 636              if ($tablesupdated = @json_decode(file_get_contents($tablesupdatedfile), true)) {
 637                  self::$tableupdated = array_merge(self::$tableupdated, $tablesupdated);
 638                  unlink($tablesupdatedfile);
 639              }
 640              $updatedtables = self::$tableupdated;
 641          }
 642  
 643          $borkedmysql = false;
 644          if ($DB->get_dbfamily() === 'mysql') {
 645              $version = $DB->get_server_info();
 646              // Everything that comes from Oracle is evil!
 647              //
 648              // See http://dev.mysql.com/doc/refman/5.6/en/alter-table.html
 649              // You cannot reset the counter to a value less than or equal to to the value that is currently in use.
 650              //
 651              // From 5.6.16 release notes:
 652              //   InnoDB: The ALTER TABLE INPLACE algorithm would fail to decrease the auto-increment value.
 653              //           (Bug #17250787, Bug #69882)
 654              if (version_compare($version['version'], '5.6.0', '>=') && version_compare($version['version'], '5.6.16', '<')) {
 655                  $borkedmysql = true;
 656              }
 657  
 658              if (version_compare($version['version'], '5.7.0', '>=') && version_compare($version['version'], '5.7.4', '<')) {
 659                  $borkedmysql = true;
 660              }
 661          }
 662  
 663          if ($borkedmysql) {
 664              $mysqlsequences = array();
 665              $prefix = $DB->get_prefix();
 666              $rs = $DB->get_recordset_sql("SHOW TABLE STATUS LIKE ?", array($prefix.'%'));
 667              foreach ($rs as $info) {
 668                  $table = strtolower($info->name);
 669                  if (strpos($table, $prefix) !== 0) {
 670                      // Incorrect table match caused by _ char.
 671                      continue;
 672                  }
 673                  if (!is_null($info->auto_increment)) {
 674                      $table = preg_replace('/^'.preg_quote($prefix, '/').'/', '', $table);
 675                      $mysqlsequences[$table] = $info->auto_increment;
 676                  }
 677              }
 678              $rs->close();
 679          }
 680  
 681          foreach ($data as $table => $records) {
 682              // If table is not modified then no need to do anything.
 683              // $updatedtables tables is set after the first run, so check before checking for specific table update.
 684              if (!empty($updatedtables) && !isset($updatedtables[$table])) {
 685                  continue;
 686              }
 687  
 688              if ($borkedmysql) {
 689                  if (empty($records)) {
 690                      if (!isset($empties[$table])) {
 691                          // Table has been modified and is not empty.
 692                          $DB->delete_records($table, null);
 693                      }
 694                      continue;
 695                  }
 696  
 697                  if (isset($structure[$table]['id']) and $structure[$table]['id']->auto_increment) {
 698                      $current = $DB->get_records($table, array(), 'id ASC');
 699                      if ($current == $records) {
 700                          if (isset($mysqlsequences[$table]) and $mysqlsequences[$table] == $structure[$table]['id']->auto_increment) {
 701                              continue;
 702                          }
 703                      }
 704                  }
 705  
 706                  // Use TRUNCATE as a workaround and reinsert everything.
 707                  $DB->delete_records($table, null);
 708                  foreach ($records as $record) {
 709                      $DB->import_record($table, $record, false, true);
 710                  }
 711                  continue;
 712              }
 713  
 714              if (empty($records)) {
 715                  if (!isset($empties[$table])) {
 716                      // Table has been modified and is not empty.
 717                      $DB->delete_records($table, array());
 718                  }
 719                  continue;
 720              }
 721  
 722              if (isset($structure[$table]['id']) and $structure[$table]['id']->auto_increment) {
 723                  $currentrecords = $DB->get_records($table, array(), 'id ASC');
 724                  $changed = false;
 725                  foreach ($records as $id => $record) {
 726                      if (!isset($currentrecords[$id])) {
 727                          $changed = true;
 728                          break;
 729                      }
 730                      if ((array)$record != (array)$currentrecords[$id]) {
 731                          $changed = true;
 732                          break;
 733                      }
 734                      unset($currentrecords[$id]);
 735                  }
 736                  if (!$changed) {
 737                      if ($currentrecords) {
 738                          $lastrecord = end($records);
 739                          $DB->delete_records_select($table, "id > ?", array($lastrecord->id));
 740                          continue;
 741                      } else {
 742                          continue;
 743                      }
 744                  }
 745              }
 746  
 747              $DB->delete_records($table, array());
 748              foreach ($records as $record) {
 749                  $DB->import_record($table, $record, false, true);
 750              }
 751          }
 752  
 753          // reset all next record ids - aka sequences
 754          self::reset_all_database_sequences($empties);
 755  
 756          // remove extra tables
 757          foreach ($tables as $table) {
 758              if (!isset($data[$table])) {
 759                  $DB->get_manager()->drop_table(new xmldb_table($table));
 760              }
 761          }
 762  
 763          self::reset_updated_table_list();
 764  
 765          return true;
 766      }
 767  
 768      /**
 769       * Purge dataroot directory
 770       * @static
 771       * @return void
 772       */
 773      public static function reset_dataroot() {
 774          global $CFG;
 775  
 776          $childclassname = self::get_framework() . '_util';
 777  
 778          // Do not delete automatically installed files.
 779          self::skip_original_data_files($childclassname);
 780  
 781          // Clear file status cache, before checking file_exists.
 782          clearstatcache();
 783  
 784          // Clean up the dataroot folder.
 785          $handle = opendir(self::get_dataroot());
 786          while (false !== ($item = readdir($handle))) {
 787              if (in_array($item, $childclassname::$datarootskiponreset)) {
 788                  continue;
 789              }
 790              if (is_dir(self::get_dataroot()."/$item")) {
 791                  remove_dir(self::get_dataroot()."/$item", false);
 792              } else {
 793                  unlink(self::get_dataroot()."/$item");
 794              }
 795          }
 796          closedir($handle);
 797  
 798          // Clean up the dataroot/filedir folder.
 799          if (file_exists(self::get_dataroot() . '/filedir')) {
 800              $handle = opendir(self::get_dataroot() . '/filedir');
 801              while (false !== ($item = readdir($handle))) {
 802                  if (in_array('filedir' . DIRECTORY_SEPARATOR . $item, $childclassname::$datarootskiponreset)) {
 803                      continue;
 804                  }
 805                  if (is_dir(self::get_dataroot()."/filedir/$item")) {
 806                      remove_dir(self::get_dataroot()."/filedir/$item", false);
 807                  } else {
 808                      unlink(self::get_dataroot()."/filedir/$item");
 809                  }
 810              }
 811              closedir($handle);
 812          }
 813  
 814          make_temp_directory('');
 815          make_backup_temp_directory('');
 816          make_cache_directory('');
 817          make_localcache_directory('');
 818          // Purge all data from the caches. This is required for consistency between tests.
 819          // Any file caches that happened to be within the data root will have already been clearer (because we just deleted cache)
 820          // and now we will purge any other caches as well.  This must be done before the cache_factory::reset() as that
 821          // removes all definitions of caches and purge does not have valid caches to operate on.
 822          cache_helper::purge_all();
 823          // Reset the cache API so that it recreates it's required directories as well.
 824          cache_factory::reset();
 825      }
 826  
 827      /**
 828       * Gets a text-based site version description.
 829       *
 830       * @return string The site info
 831       */
 832      public static function get_site_info() {
 833          global $CFG;
 834  
 835          $output = '';
 836  
 837          // All developers have to understand English, do not localise!
 838          $env = self::get_environment();
 839  
 840          $output .= "Moodle ".$env['moodleversion'];
 841          if ($hash = self::get_git_hash()) {
 842              $output .= ", $hash";
 843          }
 844          $output .= "\n";
 845  
 846          // Add php version.
 847          require_once($CFG->libdir.'/environmentlib.php');
 848          $output .= "Php: ". normalize_version($env['phpversion']);
 849  
 850          // Add database type and version.
 851          $output .= ", " . $env['dbtype'] . ": " . $env['dbversion'];
 852  
 853          // OS details.
 854          $output .= ", OS: " . $env['os'] . "\n";
 855  
 856          return $output;
 857      }
 858  
 859      /**
 860       * Try to get current git hash of the Moodle in $CFG->dirroot.
 861       * @return string null if unknown, sha1 hash if known
 862       */
 863      public static function get_git_hash() {
 864          global $CFG;
 865  
 866          // This is a bit naive, but it should mostly work for all platforms.
 867  
 868          if (!file_exists("$CFG->dirroot/.git/HEAD")) {
 869              return null;
 870          }
 871  
 872          $headcontent = file_get_contents("$CFG->dirroot/.git/HEAD");
 873          if ($headcontent === false) {
 874              return null;
 875          }
 876  
 877          $headcontent = trim($headcontent);
 878  
 879          // If it is pointing to a hash we return it directly.
 880          if (strlen($headcontent) === 40) {
 881              return $headcontent;
 882          }
 883  
 884          if (strpos($headcontent, 'ref: ') !== 0) {
 885              return null;
 886          }
 887  
 888          $ref = substr($headcontent, 5);
 889  
 890          if (!file_exists("$CFG->dirroot/.git/$ref")) {
 891              return null;
 892          }
 893  
 894          $hash = file_get_contents("$CFG->dirroot/.git/$ref");
 895  
 896          if ($hash === false) {
 897              return null;
 898          }
 899  
 900          $hash = trim($hash);
 901  
 902          if (strlen($hash) != 40) {
 903              return null;
 904          }
 905  
 906          return $hash;
 907      }
 908  
 909      /**
 910       * Set state of modified tables.
 911       *
 912       * @param string $sql sql which is updating the table.
 913       */
 914      public static function set_table_modified_by_sql($sql) {
 915          global $DB;
 916  
 917          $prefix = $DB->get_prefix();
 918  
 919          preg_match('/( ' . $prefix . '\w*)(.*)/', $sql, $matches);
 920          // Ignore random sql for testing like "XXUPDATE SET XSSD".
 921          if (!empty($matches[1])) {
 922              $table = trim($matches[1]);
 923              $table = preg_replace('/^' . preg_quote($prefix, '/') . '/', '', $table);
 924              self::$tableupdated[$table] = true;
 925  
 926              if (defined('BEHAT_SITE_RUNNING')) {
 927                  $tablesupdatedfile = self::get_tables_updated_by_scenario_list_path();
 928                  $tablesupdated = @json_decode(file_get_contents($tablesupdatedfile), true);
 929                  if (!isset($tablesupdated[$table])) {
 930                      $tablesupdated[$table] = true;
 931                      @file_put_contents($tablesupdatedfile, json_encode($tablesupdated, JSON_PRETTY_PRINT));
 932                  }
 933              }
 934          }
 935      }
 936  
 937      /**
 938       * Reset updated table list. This should be done after every reset.
 939       */
 940      public static function reset_updated_table_list() {
 941          self::$tableupdated = array();
 942      }
 943  
 944      /**
 945       * Delete tablesupdatedbyscenario file. This should be called before suite,
 946       * to ensure full db reset.
 947       */
 948      public static function clean_tables_updated_by_scenario_list() {
 949          $tablesupdatedfile = self::get_tables_updated_by_scenario_list_path();
 950          if (file_exists($tablesupdatedfile)) {
 951              unlink($tablesupdatedfile);
 952          }
 953  
 954          // Reset static cache of cli process.
 955          self::reset_updated_table_list();
 956      }
 957  
 958      /**
 959       * Returns the path to the file which holds list of tables updated in scenario.
 960       * @return string
 961       */
 962      protected final static function get_tables_updated_by_scenario_list_path() {
 963          return self::get_dataroot() . '/tablesupdatedbyscenario.json';
 964      }
 965  
 966      /**
 967       * Drop the whole test database
 968       * @static
 969       * @param bool $displayprogress
 970       */
 971      protected static function drop_database($displayprogress = false) {
 972          global $DB;
 973  
 974          $tables = $DB->get_tables(false);
 975          if (isset($tables['config'])) {
 976              // config always last to prevent problems with interrupted drops!
 977              unset($tables['config']);
 978              $tables['config'] = 'config';
 979          }
 980  
 981          if ($displayprogress) {
 982              echo "Dropping tables:\n";
 983          }
 984          $dotsonline = 0;
 985          foreach ($tables as $tablename) {
 986              $table = new xmldb_table($tablename);
 987              $DB->get_manager()->drop_table($table);
 988  
 989              if ($dotsonline == 60) {
 990                  if ($displayprogress) {
 991                      echo "\n";
 992                  }
 993                  $dotsonline = 0;
 994              }
 995              if ($displayprogress) {
 996                  echo '.';
 997              }
 998              $dotsonline += 1;
 999          }
1000          if ($displayprogress) {
1001              echo "\n";
1002          }
1003      }
1004  
1005      /**
1006       * Drops the test framework dataroot
1007       * @static
1008       */
1009      protected static function drop_dataroot() {
1010          global $CFG;
1011  
1012          $framework = self::get_framework();
1013          $childclassname = $framework . '_util';
1014  
1015          $files = scandir(self::get_dataroot() . '/'  . $framework);
1016          foreach ($files as $file) {
1017              if (in_array($file, $childclassname::$datarootskipondrop)) {
1018                  continue;
1019              }
1020              $path = self::get_dataroot() . '/' . $framework . '/' . $file;
1021              if (is_dir($path)) {
1022                  remove_dir($path, false);
1023              } else {
1024                  unlink($path);
1025              }
1026          }
1027  
1028          $jsonfilepath = self::get_dataroot() . '/' . self::$originaldatafilesjson;
1029          if (file_exists($jsonfilepath)) {
1030              // Delete the json file.
1031              unlink($jsonfilepath);
1032              // Delete the dataroot filedir.
1033              remove_dir(self::get_dataroot() . '/filedir', false);
1034          }
1035      }
1036  
1037      /**
1038       * Skip the original dataroot files to not been reset.
1039       *
1040       * @static
1041       * @param string $utilclassname the util class name..
1042       */
1043      protected static function skip_original_data_files($utilclassname) {
1044          $jsonfilepath = self::get_dataroot() . '/' . self::$originaldatafilesjson;
1045          if (file_exists($jsonfilepath)) {
1046  
1047              $listfiles = file_get_contents($jsonfilepath);
1048  
1049              // Mark each files as to not be reset.
1050              if (!empty($listfiles) && !self::$originaldatafilesjsonadded) {
1051                  $originaldatarootfiles = json_decode($listfiles);
1052                  // Keep the json file. Only drop_dataroot() should delete it.
1053                  $originaldatarootfiles[] = self::$originaldatafilesjson;
1054                  $utilclassname::$datarootskiponreset = array_merge($utilclassname::$datarootskiponreset,
1055                      $originaldatarootfiles);
1056                  self::$originaldatafilesjsonadded = true;
1057              }
1058          }
1059      }
1060  
1061      /**
1062       * Save the list of the original dataroot files into a json file.
1063       */
1064      protected static function save_original_data_files() {
1065          global $CFG;
1066  
1067          $jsonfilepath = self::get_dataroot() . '/' . self::$originaldatafilesjson;
1068  
1069          // Save the original dataroot files if not done (only executed the first time).
1070          if (!file_exists($jsonfilepath)) {
1071  
1072              $listfiles = array();
1073              $currentdir = 'filedir' . DIRECTORY_SEPARATOR . '.';
1074              $parentdir = 'filedir' . DIRECTORY_SEPARATOR . '..';
1075              $listfiles[$currentdir] = $currentdir;
1076              $listfiles[$parentdir] = $parentdir;
1077  
1078              $filedir = self::get_dataroot() . '/filedir';
1079              if (file_exists($filedir)) {
1080                  $directory = new RecursiveDirectoryIterator($filedir);
1081                  foreach (new RecursiveIteratorIterator($directory) as $file) {
1082                      if ($file->isDir()) {
1083                          $key = substr($file->getPath(), strlen(self::get_dataroot() . '/'));
1084                      } else {
1085                          $key = substr($file->getPathName(), strlen(self::get_dataroot() . '/'));
1086                      }
1087                      $listfiles[$key] = $key;
1088                  }
1089              }
1090  
1091              // Save the file list in a JSON file.
1092              $fp = fopen($jsonfilepath, 'w');
1093              fwrite($fp, json_encode(array_values($listfiles)));
1094              fclose($fp);
1095          }
1096      }
1097  
1098      /**
1099       * Return list of environment versions on which tests will run.
1100       * Environment includes:
1101       * - moodleversion
1102       * - phpversion
1103       * - dbtype
1104       * - dbversion
1105       * - os
1106       *
1107       * @return array
1108       */
1109      public static function get_environment() {
1110          global $CFG, $DB;
1111  
1112          $env = array();
1113  
1114          // Add moodle version.
1115          $release = null;
1116          require("$CFG->dirroot/version.php");
1117          $env['moodleversion'] = $release;
1118  
1119          // Add php version.
1120          $phpversion = phpversion();
1121          $env['phpversion'] = $phpversion;
1122  
1123          // Add database type and version.
1124          $dbtype = $CFG->dbtype;
1125          $dbinfo = $DB->get_server_info();
1126          $dbversion = $dbinfo['version'];
1127          $env['dbtype'] = $dbtype;
1128          $env['dbversion'] = $dbversion;
1129  
1130          // OS details.
1131          $osdetails = php_uname('s') . " " . php_uname('r') . " " . php_uname('m');
1132          $env['os'] = $osdetails;
1133  
1134          return $env;
1135      }
1136  }