See Release Notes
Long Term Support Release
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 }
title
Description
Body
title
Description
Body
title
Description
Body
title
Body