Search moodle.org's
Developer Documentation

See Release Notes

  • Bug fixes for general core bugs in 4.2.x will end 22 April 2024 (12 months).
  • Bug fixes for security issues in 4.2.x will end 7 October 2024 (18 months).
  • PHP version: minimum PHP 8.0.0 Note: minimum PHP version has increased since Moodle 4.1. PHP 8.1.x is supported too.

Differences Between: [Versions 310 and 402] [Versions 39 and 402]

   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  namespace core;
  18  
  19  use database_column_info;
  20  use moodle_database;
  21  use sql_generator;
  22  use xmldb_field;
  23  use xmldb_index;
  24  use xmldb_key;
  25  use xmldb_structure;
  26  use xmldb_table;
  27  
  28  /**
  29   * DDL layer tests.
  30   *
  31   * @package    core_ddl
  32   * @category   test
  33   * @copyright  2008 Nicolas Connault
  34   * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  35   */
  36  class ddl_test extends \database_driver_testcase {
  37      /** @var xmldb_table[] keys are table name. Created in setUp. */
  38      private $tables = array();
  39      /** @var array table name => array of stdClass test records loaded into that table. Created in setUp. */
  40      private $records = array();
  41  
  42      protected function setUp(): void {
  43          parent::setUp();
  44          $dbman = $this->tdb->get_manager(); // Loads DDL libs.
  45  
  46          $table = new xmldb_table('test_table0');
  47          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
  48          $table->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  49          $table->add_field('type', XMLDB_TYPE_CHAR, '20', null, XMLDB_NOTNULL, null, 'general');
  50          $table->add_field('name', XMLDB_TYPE_CHAR, '255', null, XMLDB_NOTNULL, null);
  51          $table->add_field('intro', XMLDB_TYPE_TEXT, 'small', null, XMLDB_NOTNULL, null, null);
  52          $table->add_field('logo', XMLDB_TYPE_BINARY, 'big', null, null, null);
  53          $table->add_field('assessed', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  54          $table->add_field('assesstimestart', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  55          $table->add_field('assesstimefinish', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  56          $table->add_field('scale', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  57          $table->add_field('maxbytes', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  58          $table->add_field('forcesubscribe', XMLDB_TYPE_INTEGER, '1', null, XMLDB_NOTNULL, null, '0');
  59          $table->add_field('trackingtype', XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, null, '1');
  60          $table->add_field('rsstype', XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, null, '0');
  61          $table->add_field('rssarticles', XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, null, '0');
  62          $table->add_field('timemodified', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  63          $table->add_field('grade', XMLDB_TYPE_NUMBER, '20,0', null, null, null, null);
  64          $table->add_field('percent', XMLDB_TYPE_NUMBER, '5,2', null, null, null, 66.6);
  65          $table->add_field('bignum', XMLDB_TYPE_NUMBER, '38,18', null, null, null, 1234567890.1234);
  66          $table->add_field('warnafter', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  67          $table->add_field('blockafter', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  68          $table->add_field('blockperiod', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  69          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
  70          $table->add_key('course', XMLDB_KEY_UNIQUE, array('course'));
  71          $table->add_index('type-name', XMLDB_INDEX_UNIQUE, array('type', 'name'));
  72          $table->add_index('rsstype', XMLDB_INDEX_NOTUNIQUE, array('rsstype'));
  73          $table->setComment("This is a test'n drop table. You can drop it safely");
  74  
  75          $this->tables[$table->getName()] = $table;
  76  
  77          // Define 2 initial records for this table.
  78          $this->records[$table->getName()] = array(
  79              (object)array(
  80                  'course' => '1',
  81                  'type'   => 'general',
  82                  'name'   => 'record',
  83                  'intro'  => 'first record'),
  84              (object)array(
  85                  'course' => '2',
  86                  'type'   => 'social',
  87                  'name'   => 'record',
  88                  'intro'  => 'second record'));
  89  
  90          // Second, smaller table.
  91          $table = new xmldb_table ('test_table1');
  92          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
  93          $table->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
  94          $table->add_field('name', XMLDB_TYPE_CHAR, '30', null, null, null, 'Moodle');
  95          $table->add_field('secondname', XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, null);
  96          $table->add_field('thirdname', XMLDB_TYPE_CHAR, '30', null, null, null, ''); // Nullable column with empty default.
  97          $table->add_field('intro', XMLDB_TYPE_TEXT, 'medium', null, XMLDB_NOTNULL, null, null);
  98          $table->add_field('avatar', XMLDB_TYPE_BINARY, 'medium', null, null, null, null);
  99          $table->add_field('grade', XMLDB_TYPE_NUMBER, '20,10', null, null, null);
 100          $table->add_field('gradefloat', XMLDB_TYPE_FLOAT, '20,0', null, null, null, null);
 101          $table->add_field('percentfloat', XMLDB_TYPE_FLOAT, '5,2', null, null, null, 99.9);
 102          $table->add_field('userid', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
 103          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 104          $table->add_key('course', XMLDB_KEY_FOREIGN_UNIQUE, array('course'), 'test_table0', array('course'));
 105          $table->setComment("This is a test'n drop table. You can drop it safely");
 106  
 107          $this->tables[$table->getName()] = $table;
 108  
 109          // Define 2 initial records for this table.
 110          $this->records[$table->getName()] = array(
 111              (object)array(
 112                  'course' => '1',
 113                  'secondname'   => 'first record', // Less than 10 cc, please don't modify. Some tests below depend of this.
 114                  'intro'  => 'first record'),
 115              (object)array(
 116                  'course'       => '2',
 117                  'secondname'   => 'second record', // More than 10 cc, please don't modify. Some tests below depend of this.
 118                  'intro'  => 'second record'));
 119      }
 120  
 121      private function create_deftable($tablename) {
 122          $dbman = $this->tdb->get_manager();
 123  
 124          if (!isset($this->tables[$tablename])) {
 125              return null;
 126          }
 127  
 128          $table = $this->tables[$tablename];
 129  
 130          if ($dbman->table_exists($table)) {
 131              $dbman->drop_table($table);
 132          }
 133          $dbman->create_table($table);
 134  
 135          return $table;
 136      }
 137  
 138      /**
 139       * Fill the given test table with some records, as far as
 140       * DDL behaviour must be tested both with real data and
 141       * with empty tables
 142       * @param string $tablename
 143       * @return int count of records
 144       */
 145      private function fill_deftable($tablename) {
 146          $DB = $this->tdb; // Do not use global $DB!
 147          $dbman = $this->tdb->get_manager();
 148  
 149          if (!isset($this->records[$tablename])) {
 150              return null;
 151          }
 152  
 153          if ($dbman->table_exists($tablename)) {
 154              foreach ($this->records[$tablename] as $row) {
 155                  $DB->insert_record($tablename, $row);
 156              }
 157          } else {
 158              return null;
 159          }
 160  
 161          return count($this->records[$tablename]);
 162      }
 163  
 164      /**
 165       * Test behaviour of table_exists()
 166       */
 167      public function test_table_exists() {
 168          $DB = $this->tdb; // Do not use global $DB!
 169          $dbman = $this->tdb->get_manager();
 170  
 171          // First make sure it returns false if table does not exist.
 172          $table = $this->tables['test_table0'];
 173  
 174          try {
 175              $result = $DB->get_records('test_table0');
 176          } catch (\dml_exception $e) {
 177              $result = false;
 178          }
 179          $this->resetDebugging();
 180  
 181          $this->assertFalse($result);
 182  
 183          $this->assertFalse($dbman->table_exists('test_table0')); // By name..
 184          $this->assertFalse($dbman->table_exists($table));        // By xmldb_table..
 185  
 186          // Create table and test again.
 187          $dbman->create_table($table);
 188  
 189          $this->assertSame(array(), $DB->get_records('test_table0'));
 190          $this->assertTrue($dbman->table_exists('test_table0')); // By name.
 191          $this->assertTrue($dbman->table_exists($table));        // By xmldb_table.
 192  
 193          // Drop table and test again.
 194          $dbman->drop_table($table);
 195  
 196          try {
 197              $result = $DB->get_records('test_table0');
 198          } catch (\dml_exception $e) {
 199              $result = false;
 200          }
 201          $this->resetDebugging();
 202  
 203          $this->assertFalse($result);
 204  
 205          $this->assertFalse($dbman->table_exists('test_table0')); // By name.
 206          $this->assertFalse($dbman->table_exists($table));        // By xmldb_table.
 207      }
 208  
 209      /**
 210       * Test behaviour of create_table()
 211       */
 212      public function test_create_table() {
 213  
 214          $DB = $this->tdb; // Do not use global $DB!
 215          $dbman = $this->tdb->get_manager();
 216  
 217          // Create table.
 218          $table = $this->tables['test_table1'];
 219  
 220          $dbman->create_table($table);
 221          $this->assertTrue($dbman->table_exists($table));
 222  
 223          // Basic get_tables() test.
 224          $tables = $DB->get_tables();
 225          $this->assertArrayHasKey('test_table1', $tables);
 226  
 227          // Basic get_columns() tests.
 228          $columns = $DB->get_columns('test_table1');
 229          $this->assertSame('R', $columns['id']->meta_type);
 230          $this->assertSame('I', $columns['course']->meta_type);
 231          $this->assertSame('C', $columns['name']->meta_type);
 232          $this->assertSame('C', $columns['secondname']->meta_type);
 233          $this->assertSame('C', $columns['thirdname']->meta_type);
 234          $this->assertSame('X', $columns['intro']->meta_type);
 235          $this->assertSame('B', $columns['avatar']->meta_type);
 236          $this->assertSame('N', $columns['grade']->meta_type);
 237          $this->assertSame('N', $columns['percentfloat']->meta_type);
 238          $this->assertSame('I', $columns['userid']->meta_type);
 239          // Some defaults.
 240          $this->assertTrue($columns['course']->has_default);
 241          $this->assertEquals(0, $columns['course']->default_value);
 242          $this->assertTrue($columns['name']->has_default);
 243          $this->assertSame('Moodle', $columns['name']->default_value);
 244          $this->assertTrue($columns['secondname']->has_default);
 245          $this->assertSame('', $columns['secondname']->default_value);
 246          $this->assertTrue($columns['thirdname']->has_default);
 247          $this->assertSame('', $columns['thirdname']->default_value);
 248          $this->assertTrue($columns['percentfloat']->has_default);
 249          $this->assertEquals(99.9, $columns['percentfloat']->default_value);
 250          $this->assertTrue($columns['userid']->has_default);
 251          $this->assertEquals(0, $columns['userid']->default_value);
 252  
 253          // Basic get_indexes() test.
 254          $indexes = $DB->get_indexes('test_table1');
 255          $courseindex = reset($indexes);
 256          $this->assertEquals(1, $courseindex['unique']);
 257          $this->assertSame('course', $courseindex['columns'][0]);
 258  
 259          // Check sequence returns 1 for first insert.
 260          $rec = (object)array(
 261              'course'     => 10,
 262              'secondname' => 'not important',
 263              'intro'      => 'not important');
 264          $this->assertSame(1, $DB->insert_record('test_table1', $rec));
 265  
 266          // Check defined defaults are working ok.
 267          $dbrec = $DB->get_record('test_table1', array('id' => 1));
 268          $this->assertSame('Moodle', $dbrec->name);
 269          $this->assertSame('', $dbrec->thirdname);
 270  
 271          // Check exceptions if multiple R columns.
 272          $table = new xmldb_table ('test_table2');
 273          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 274          $table->add_field('rid', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 275          $table->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
 276          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 277          $table->add_key('primaryx', XMLDB_KEY_PRIMARY, array('id'));
 278          $table->setComment("This is a test'n drop table. You can drop it safely");
 279  
 280          $this->tables[$table->getName()] = $table;
 281  
 282          try {
 283              $dbman->create_table($table);
 284              $this->fail('Exception expected');
 285          } catch (\moodle_exception $e) {
 286              $this->assertInstanceOf('ddl_exception', $e);
 287          }
 288  
 289          // Check exceptions missing primary key on R column.
 290          $table = new xmldb_table ('test_table2');
 291          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 292          $table->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
 293          $table->setComment("This is a test'n drop table. You can drop it safely");
 294  
 295          $this->tables[$table->getName()] = $table;
 296  
 297          try {
 298              $dbman->create_table($table);
 299              $this->fail('Exception expected');
 300          } catch (\moodle_exception $e) {
 301              $this->assertInstanceOf('ddl_exception', $e);
 302          }
 303  
 304          // Long table name names - the largest allowed by the configuration which exclude the prefix to ensure it's created.
 305          $tablechars = str_repeat('a', xmldb_table::NAME_MAX_LENGTH);
 306          $table = new xmldb_table($tablechars);
 307          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 308          $table->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '2');
 309          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 310          $table->setComment("This is a test'n drop table. You can drop it safely");
 311  
 312          $this->tables[$table->getName()] = $table;
 313  
 314          $dbman->create_table($table);
 315          $this->assertTrue($dbman->table_exists($table));
 316          $dbman->drop_table($table);
 317  
 318          // Table name is too long, ignoring any prefix size set.
 319          $tablechars = str_repeat('a', xmldb_table::NAME_MAX_LENGTH + 1);
 320          $table = new xmldb_table($tablechars);
 321          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 322          $table->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '2');
 323          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 324          $table->setComment("This is a test'n drop table. You can drop it safely");
 325  
 326          $this->tables[$table->getName()] = $table;
 327  
 328          try {
 329              $dbman->create_table($table);
 330              $this->fail('Exception expected');
 331          } catch (\moodle_exception $e) {
 332              $this->assertInstanceOf('coding_exception', $e);
 333          }
 334  
 335          // Invalid table name.
 336          $table = new xmldb_table('test_tableCD');
 337          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 338          $table->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '2');
 339          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 340          $table->setComment("This is a test'n drop table. You can drop it safely");
 341  
 342          $this->tables[$table->getName()] = $table;
 343  
 344          try {
 345              $dbman->create_table($table);
 346              $this->fail('Exception expected');
 347          } catch (\moodle_exception $e) {
 348              $this->assertInstanceOf('coding_exception', $e);
 349          }
 350  
 351          // Weird column names - the largest allowed.
 352          $table = new xmldb_table('test_table3');
 353          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 354          $table->add_field(str_repeat('b', xmldb_field::NAME_MAX_LENGTH), XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '2');
 355          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 356          $table->setComment("This is a test'n drop table. You can drop it safely");
 357  
 358          $this->tables[$table->getName()] = $table;
 359  
 360          $dbman->create_table($table);
 361          $this->assertTrue($dbman->table_exists($table));
 362          $dbman->drop_table($table);
 363  
 364          // Too long field name.
 365          $table = new xmldb_table('test_table4');
 366          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 367          $table->add_field(str_repeat('a', xmldb_field::NAME_MAX_LENGTH + 1), XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '2');
 368          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 369          $table->setComment("This is a test'n drop table. You can drop it safely");
 370  
 371          $this->tables[$table->getName()] = $table;
 372  
 373          try {
 374              $dbman->create_table($table);
 375              $this->fail('Exception expected');
 376          } catch (\moodle_exception $e) {
 377              $this->assertInstanceOf('coding_exception', $e);
 378          }
 379  
 380          // Invalid field name.
 381          $table = new xmldb_table('test_table4');
 382          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 383          $table->add_field('abCD', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '2');
 384          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 385          $table->setComment("This is a test'n drop table. You can drop it safely");
 386  
 387          $this->tables[$table->getName()] = $table;
 388  
 389          try {
 390              $dbman->create_table($table);
 391              $this->fail('Exception expected');
 392          } catch (\moodle_exception $e) {
 393              $this->assertInstanceOf('coding_exception', $e);
 394          }
 395  
 396          // Invalid integer length.
 397          $table = new xmldb_table('test_table4');
 398          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 399          $table->add_field('course', XMLDB_TYPE_INTEGER, '21', null, XMLDB_NOTNULL, null, '2');
 400          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 401          $table->setComment("This is a test'n drop table. You can drop it safely");
 402  
 403          $this->tables[$table->getName()] = $table;
 404  
 405          try {
 406              $dbman->create_table($table);
 407              $this->fail('Exception expected');
 408          } catch (\moodle_exception $e) {
 409              $this->assertInstanceOf('coding_exception', $e);
 410          }
 411  
 412          // Invalid integer default.
 413          $table = new xmldb_table('test_table4');
 414          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 415          $table->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, 'x');
 416          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 417          $table->setComment("This is a test'n drop table. You can drop it safely");
 418  
 419          $this->tables[$table->getName()] = $table;
 420  
 421          try {
 422              $dbman->create_table($table);
 423              $this->fail('Exception expected');
 424          } catch (\moodle_exception $e) {
 425              $this->assertInstanceOf('coding_exception', $e);
 426          }
 427  
 428          // Invalid decimal length - max precision is 38 digits.
 429          $table = new xmldb_table('test_table4');
 430          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 431          $table->add_field('num', XMLDB_TYPE_NUMBER, '39,19', null, XMLDB_NOTNULL, null, null);
 432          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 433          $table->setComment("This is a test'n drop table. You can drop it safely");
 434  
 435          $this->tables[$table->getName()] = $table;
 436  
 437          try {
 438              $dbman->create_table($table);
 439              $this->fail('Exception expected');
 440          } catch (\moodle_exception $e) {
 441              $this->assertInstanceOf('coding_exception', $e);
 442          }
 443  
 444          // Invalid decimal decimals - number of decimals can't be higher than total number of digits.
 445          $table = new xmldb_table('test_table4');
 446          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 447          $table->add_field('num', XMLDB_TYPE_NUMBER, '10,11', null, XMLDB_NOTNULL, null, null);
 448          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 449          $table->setComment("This is a test'n drop table. You can drop it safely");
 450  
 451          $this->tables[$table->getName()] = $table;
 452  
 453          try {
 454              $dbman->create_table($table);
 455              $this->fail('Exception expected');
 456          } catch (\moodle_exception $e) {
 457              $this->assertInstanceOf('coding_exception', $e);
 458          }
 459  
 460          // Invalid decimal whole number - the whole number part can't have more digits than integer fields.
 461          $table = new xmldb_table('test_table4');
 462          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 463          $table->add_field('num', XMLDB_TYPE_NUMBER, '38,17', null, XMLDB_NOTNULL, null, null);
 464          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 465          $table->setComment("This is a test'n drop table. You can drop it safely");
 466  
 467          $this->tables[$table->getName()] = $table;
 468  
 469          try {
 470              $dbman->create_table($table);
 471              $this->fail('Exception expected');
 472          } catch (\moodle_exception $e) {
 473              $this->assertInstanceOf('coding_exception', $e);
 474          }
 475  
 476          // Invalid decimal decimals - negative scale not supported.
 477          $table = new xmldb_table('test_table4');
 478          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 479          $table->add_field('num', XMLDB_TYPE_NUMBER, '30,-5', null, XMLDB_NOTNULL, null, null);
 480          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 481          $table->setComment("This is a test'n drop table. You can drop it safely");
 482  
 483          $this->tables[$table->getName()] = $table;
 484  
 485          try {
 486              $dbman->create_table($table);
 487              $this->fail('Exception expected');
 488          } catch (\moodle_exception $e) {
 489              $this->assertInstanceOf('coding_exception', $e);
 490          }
 491  
 492          // Invalid decimal default.
 493          $table = new xmldb_table('test_table4');
 494          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 495          $table->add_field('num', XMLDB_TYPE_NUMBER, '10,5', null, XMLDB_NOTNULL, null, 'x');
 496          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 497          $table->setComment("This is a test'n drop table. You can drop it safely");
 498  
 499          $this->tables[$table->getName()] = $table;
 500  
 501          try {
 502              $dbman->create_table($table);
 503              $this->fail('Exception expected');
 504          } catch (\moodle_exception $e) {
 505              $this->assertInstanceOf('coding_exception', $e);
 506          }
 507  
 508          // Invalid float length.
 509          $table = new xmldb_table('test_table4');
 510          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 511          $table->add_field('num', XMLDB_TYPE_FLOAT, '21,10', null, XMLDB_NOTNULL, null, null);
 512          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 513          $table->setComment("This is a test'n drop table. You can drop it safely");
 514  
 515          $this->tables[$table->getName()] = $table;
 516  
 517          try {
 518              $dbman->create_table($table);
 519              $this->fail('Exception expected');
 520          } catch (\moodle_exception $e) {
 521              $this->assertInstanceOf('coding_exception', $e);
 522          }
 523  
 524          // Invalid float decimals.
 525          $table = new xmldb_table('test_table4');
 526          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 527          $table->add_field('num', XMLDB_TYPE_FLOAT, '10,11', null, XMLDB_NOTNULL, null, null);
 528          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 529          $table->setComment("This is a test'n drop table. You can drop it safely");
 530  
 531          $this->tables[$table->getName()] = $table;
 532  
 533          try {
 534              $dbman->create_table($table);
 535              $this->fail('Exception expected');
 536          } catch (\moodle_exception $e) {
 537              $this->assertInstanceOf('coding_exception', $e);
 538          }
 539  
 540          // Invalid float default.
 541          $table = new xmldb_table('test_table4');
 542          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 543          $table->add_field('num', XMLDB_TYPE_FLOAT, '10,5', null, XMLDB_NOTNULL, null, 'x');
 544          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 545          $table->setComment("This is a test'n drop table. You can drop it safely");
 546  
 547          $this->tables[$table->getName()] = $table;
 548  
 549          try {
 550              $dbman->create_table($table);
 551              $this->fail('Exception expected');
 552          } catch (\moodle_exception $e) {
 553              $this->assertInstanceOf('coding_exception', $e);
 554          }
 555      }
 556  
 557      /**
 558       * Test if database supports tables with many TEXT fields,
 559       * InnoDB is known to failed during data insertion instead
 560       * of table creation when text fields contain actual data.
 561       */
 562      public function test_row_size_limits() {
 563  
 564          $DB = $this->tdb; // Do not use global $DB!
 565          $dbman = $this->tdb->get_manager();
 566  
 567          $text = str_repeat('Å¡', 1333);
 568  
 569          $data = new \stdClass();
 570          $data->name = 'test';
 571          $table = new xmldb_table('test_innodb');
 572          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 573          $table->add_field('name', XMLDB_TYPE_CHAR, '30', null, null, null, null);
 574          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 575          for ($i = 0; $i < 20; $i++) {
 576              $table->add_field('text'.$i, XMLDB_TYPE_TEXT, null, null, null, null, null);
 577              $data->{'text'.$i} = $text;
 578          }
 579          $dbman->create_table($table);
 580  
 581          try {
 582              $id = $DB->insert_record('test_innodb', $data);
 583              $expected = (array)$data;
 584              $expected['id'] = (string)$id;
 585              $this->assertEqualsCanonicalizing($expected, (array)$DB->get_record('test_innodb', array('id' => $id)));
 586          } catch (\dml_exception $e) {
 587              // Give some nice error message when known problematic MySQL with InnoDB detected.
 588              if ($DB->get_dbfamily() === 'mysql') {
 589                  $engine = strtolower($DB->get_dbengine());
 590                  if ($engine === 'innodb' or $engine === 'xtradb') {
 591                      if (!$DB->is_compressed_row_format_supported()) {
 592                          $this->fail("Row size limit reached in MySQL using InnoDB, configure server to use innodb_file_format=Barracuda and innodb_file_per_table=1");
 593                      }
 594                  }
 595              }
 596              throw $e;
 597          }
 598  
 599          $dbman->drop_table($table);
 600  
 601          $data = new \stdClass();
 602          $data->name = 'test';
 603          $table = new xmldb_table('test_innodb');
 604          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 605          $table->add_field('name', XMLDB_TYPE_CHAR, '30', null, null, null, null);
 606          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 607          $dbman->create_table($table);
 608          $DB->insert_record('test_innodb', array('name' => 'test'));
 609  
 610          for ($i = 0; $i < 20; $i++) {
 611              $field = new xmldb_field('text'.$i, XMLDB_TYPE_TEXT, null, null, null, null, null);
 612              $dbman->add_field($table, $field);
 613              $data->{'text'.$i} = $text;
 614  
 615              $id = $DB->insert_record('test_innodb', $data);
 616              $expected = (array)$data;
 617              $expected['id'] = (string)$id;
 618              $this->assertEqualsCanonicalizing($expected, (array)$DB->get_record('test_innodb', array('id' => $id)));
 619          }
 620  
 621          $dbman->drop_table($table);
 622  
 623          // MySQL VARCHAR fields may hit a different 65535 row size limit when creating tables.
 624          $data = new \stdClass();
 625          $data->name = 'test';
 626          $table = new xmldb_table('test_innodb');
 627          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
 628          $table->add_field('name', XMLDB_TYPE_CHAR, '30', null, null, null, null);
 629          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
 630          for ($i = 0; $i < 12; $i++) {
 631              $table->add_field('text'.$i, XMLDB_TYPE_CHAR, '1333', null, null, null, null);
 632              $data->{'text'.$i} = $text;
 633          }
 634          $dbman->create_table($table);
 635  
 636          $id = $DB->insert_record('test_innodb', $data);
 637          $expected = (array)$data;
 638          $expected['id'] = (string)$id;
 639          $this->assertEqualsCanonicalizing($expected, (array)$DB->get_record('test_innodb', array('id' => $id)));
 640  
 641          $dbman->drop_table($table);
 642      }
 643  
 644      /**
 645       * Test behaviour of drop_table()
 646       */
 647      public function test_drop_table() {
 648          $DB = $this->tdb; // Do not use global $DB!
 649          $dbman = $this->tdb->get_manager();
 650  
 651          // Initially table doesn't exist.
 652          $this->assertFalse($dbman->table_exists('test_table0'));
 653  
 654          // Create table with contents.
 655          $table = $this->create_deftable('test_table0');
 656          $this->assertTrue($dbman->table_exists('test_table0'));
 657  
 658          // Fill the table with some records before dropping it.
 659          $this->fill_deftable('test_table0');
 660  
 661          // Drop by xmldb_table object.
 662          $dbman->drop_table($table);
 663          $this->assertFalse($dbman->table_exists('test_table0'));
 664  
 665          // Basic get_tables() test.
 666          $tables = $DB->get_tables();
 667          $this->assertArrayNotHasKey('test_table0', $tables);
 668  
 669          // Columns cache must be empty.
 670          $columns = $DB->get_columns('test_table0');
 671          $this->assertEmpty($columns);
 672  
 673          $indexes = $DB->get_indexes('test_table0');
 674          $this->assertEmpty($indexes);
 675      }
 676  
 677      /**
 678       * Test behaviour of rename_table()
 679       */
 680      public function test_rename_table() {
 681          $DB = $this->tdb; // Do not use global $DB!
 682          $dbman = $this->tdb->get_manager();
 683  
 684          $table = $this->create_deftable('test_table1');
 685  
 686          // Fill the table with some records before renaming it.
 687          $insertedrows = $this->fill_deftable('test_table1');
 688  
 689          $this->assertFalse($dbman->table_exists('test_table_cust1'));
 690          $dbman->rename_table($table, 'test_table_cust1');
 691          $this->assertTrue($dbman->table_exists('test_table_cust1'));
 692  
 693          // Check sequence returns $insertedrows + 1 for this insert (after rename).
 694          $rec = (object)array(
 695              'course'     => 20,
 696              'secondname' => 'not important',
 697              'intro'      => 'not important');
 698          $this->assertSame($insertedrows+1, $DB->insert_record('test_table_cust1', $rec));
 699  
 700          // Verify behavior when target table already exists.
 701          $sourcetable = $this->create_deftable('test_table0');
 702          $targettable = $this->create_deftable('test_table1');
 703          try {
 704              $dbman->rename_table($sourcetable, $targettable->getName());
 705              $this->fail('Exception expected');
 706          } catch (\moodle_exception $e) {
 707              $this->assertInstanceOf('ddl_exception', $e);
 708              $this->assertEquals('Table "test_table1" already exists (can not rename table)', $e->getMessage());
 709          }
 710      }
 711  
 712      /**
 713       * Test behaviour of field_exists()
 714       */
 715      public function test_field_exists() {
 716          $dbman = $this->tdb->get_manager();
 717  
 718          $table = $this->create_deftable('test_table0');
 719  
 720          // String params.
 721          // Give a nonexistent table as first param (throw exception).
 722          try {
 723              $dbman->field_exists('nonexistenttable', 'id');
 724              $this->fail('Exception expected');
 725          } catch (\moodle_exception $e) {
 726              $this->assertInstanceOf('moodle_exception', $e);
 727          }
 728  
 729          // Give a nonexistent field as second param (return false).
 730          $this->assertFalse($dbman->field_exists('test_table0', 'nonexistentfield'));
 731  
 732          // Correct string params.
 733          $this->assertTrue($dbman->field_exists('test_table0', 'id'));
 734  
 735          // Object params.
 736          $realfield = $table->getField('id');
 737  
 738          // Give a nonexistent table as first param (throw exception).
 739          $nonexistenttable = new xmldb_table('nonexistenttable');
 740          try {
 741              $dbman->field_exists($nonexistenttable, $realfield);
 742              $this->fail('Exception expected');
 743          } catch (\moodle_exception $e) {
 744              $this->assertInstanceOf('moodle_exception', $e);
 745          }
 746  
 747          // Give a nonexistent field as second param (return false).
 748          $nonexistentfield = new xmldb_field('nonexistentfield');
 749          $this->assertFalse($dbman->field_exists($table, $nonexistentfield));
 750  
 751          // Correct object params.
 752          $this->assertTrue($dbman->field_exists($table, $realfield));
 753  
 754          // Mix string and object params.
 755          // Correct ones.
 756          $this->assertTrue($dbman->field_exists($table, 'id'));
 757          $this->assertTrue($dbman->field_exists('test_table0', $realfield));
 758          // Non existing tables (throw exception).
 759          try {
 760              $this->assertFalse($dbman->field_exists($nonexistenttable, 'id'));
 761              $this->fail('Exception expected');
 762          } catch (\moodle_exception $e) {
 763              $this->assertInstanceOf('moodle_exception', $e);
 764          }
 765          try {
 766              $this->assertFalse($dbman->field_exists('nonexistenttable', $realfield));
 767              $this->fail('Exception expected');
 768          } catch (\moodle_exception $e) {
 769              $this->assertInstanceOf('moodle_exception', $e);
 770          }
 771          // Non existing fields (return false).
 772          $this->assertFalse($dbman->field_exists($table, 'nonexistentfield'));
 773          $this->assertFalse($dbman->field_exists('test_table0', $nonexistentfield));
 774      }
 775  
 776      /**
 777       * Test behaviour of add_field()
 778       */
 779      public function test_add_field() {
 780          $DB = $this->tdb; // Do not use global $DB!
 781          $dbman = $this->tdb->get_manager();
 782  
 783          $table = $this->create_deftable('test_table1');
 784  
 785          // Fill the table with some records before adding fields.
 786          $this->fill_deftable('test_table1');
 787  
 788          // Add one not null field without specifying default value (throws ddl_exception).
 789          $field = new xmldb_field('onefield');
 790          $field->set_attributes(XMLDB_TYPE_INTEGER, '6', null, XMLDB_NOTNULL, null, null);
 791          try {
 792              $dbman->add_field($table, $field);
 793              $this->fail('Exception expected');
 794          } catch (\moodle_exception $e) {
 795              $this->assertInstanceOf('ddl_exception', $e);
 796          }
 797  
 798          // Add one existing field (throws ddl_exception).
 799          $field = new xmldb_field('course');
 800          $field->set_attributes(XMLDB_TYPE_INTEGER, '6', null, XMLDB_NOTNULL, null, 2);
 801          try {
 802              $dbman->add_field($table, $field);
 803              $this->fail('Exception expected');
 804          } catch (\moodle_exception $e) {
 805              $this->assertInstanceOf('ddl_exception', $e);
 806          }
 807  
 808          // TODO: add one field with invalid type, must throw exception.
 809          // TODO: add one text field with default, must throw exception.
 810          // TODO: add one binary field with default, must throw exception.
 811  
 812          // Add one integer field and check it.
 813          $field = new xmldb_field('oneinteger');
 814          $field->set_attributes(XMLDB_TYPE_INTEGER, '6', null, XMLDB_NOTNULL, null, 2);
 815          $dbman->add_field($table, $field);
 816          $this->assertTrue($dbman->field_exists($table, 'oneinteger'));
 817          $columns = $DB->get_columns('test_table1');
 818          $this->assertEquals('oneinteger', $columns['oneinteger']->name);
 819          $this->assertTrue($columns['oneinteger']->not_null);
 820          // Max_length and scale cannot be checked under all DBs at all for integer fields.
 821          $this->assertFalse($columns['oneinteger']->primary_key);
 822          $this->assertFalse($columns['oneinteger']->binary);
 823          $this->assertTrue($columns['oneinteger']->has_default);
 824          $this->assertEquals(2, $columns['oneinteger']->default_value);
 825          $this->assertSame('I', $columns['oneinteger']->meta_type);
 826          $this->assertEquals(2, $DB->get_field('test_table1', 'oneinteger', array(), IGNORE_MULTIPLE)); // Check default has been applied.
 827  
 828          // Add one numeric field and check it.
 829          $field = new xmldb_field('onenumber');
 830          $field->set_attributes(XMLDB_TYPE_NUMBER, '6,3', null, XMLDB_NOTNULL, null, 2.55);
 831          $dbman->add_field($table, $field);
 832          $this->assertTrue($dbman->field_exists($table, 'onenumber'));
 833          $columns = $DB->get_columns('test_table1');
 834          $this->assertSame('onenumber', $columns['onenumber']->name);
 835          $this->assertEquals(6, $columns['onenumber']->max_length);
 836          $this->assertEquals(3, $columns['onenumber']->scale);
 837          $this->assertTrue($columns['onenumber']->not_null);
 838          $this->assertFalse($columns['onenumber']->primary_key);
 839          $this->assertFalse($columns['onenumber']->binary);
 840          $this->assertTrue($columns['onenumber']->has_default);
 841          $this->assertEquals(2.550, $columns['onenumber']->default_value);
 842          $this->assertSame('N', $columns['onenumber']->meta_type);
 843          $this->assertEquals(2.550, $DB->get_field('test_table1', 'onenumber', array(), IGNORE_MULTIPLE)); // Check default has been applied.
 844  
 845          // Add one numeric field with scale of 0 and check it.
 846          $field = new xmldb_field('onenumberwith0scale');
 847          $field->set_attributes(XMLDB_TYPE_NUMBER, '6,0', null, XMLDB_NOTNULL, null, 2);
 848          $dbman->add_field($table, $field);
 849          $this->assertTrue($dbman->field_exists($table, 'onenumberwith0scale'));
 850          $columns = $DB->get_columns('test_table1');
 851          $this->assertEquals(6, $columns['onenumberwith0scale']->max_length);
 852          // We can not use assertEquals as that accepts null/false as a valid value.
 853          $this->assertSame('0', strval($columns['onenumberwith0scale']->scale));
 854  
 855          // Add one float field and check it (not official type - must work as number).
 856          $field = new xmldb_field('onefloat');
 857          $field->set_attributes(XMLDB_TYPE_FLOAT, '6,3', null, XMLDB_NOTNULL, null, 3.550);
 858          $dbman->add_field($table, $field);
 859          $this->assertTrue($dbman->field_exists($table, 'onefloat'));
 860          $columns = $DB->get_columns('test_table1');
 861          $this->assertSame('onefloat', $columns['onefloat']->name);
 862          $this->assertTrue($columns['onefloat']->not_null);
 863          // Max_length and scale cannot be checked under all DBs at all for float fields.
 864          $this->assertFalse($columns['onefloat']->primary_key);
 865          $this->assertFalse($columns['onefloat']->binary);
 866          $this->assertTrue($columns['onefloat']->has_default);
 867          $this->assertEquals(3.550, $columns['onefloat']->default_value);
 868          $this->assertSame('N', $columns['onefloat']->meta_type);
 869          // Just rounding DB information to 7 decimal digits. Fair enough to test 3.550 and avoids one nasty bug
 870          // in MSSQL core returning wrong floats (http://social.msdn.microsoft.com/Forums/en-US/sqldataaccess/thread/5e08de63-16bb-4f24-b645-0cf8fc669de3)
 871          // In any case, floats aren't officially supported by Moodle, with number/decimal type being the correct ones, so
 872          // this isn't a real problem at all.
 873          $this->assertEquals(3.550, round($DB->get_field('test_table1', 'onefloat', array(), IGNORE_MULTIPLE), 7)); // Check default has been applied.
 874  
 875          // Add one char field and check it.
 876          $field = new xmldb_field('onechar');
 877          $field->set_attributes(XMLDB_TYPE_CHAR, '25', null, XMLDB_NOTNULL, null, 'Nice dflt!');
 878          $dbman->add_field($table, $field);
 879          $this->assertTrue($dbman->field_exists($table, 'onechar'));
 880          $columns = $DB->get_columns('test_table1');
 881          $this->assertSame('onechar', $columns['onechar']->name);
 882          $this->assertEquals(25, $columns['onechar']->max_length);
 883          $this->assertNull($columns['onechar']->scale);
 884          $this->assertTrue($columns['onechar']->not_null);
 885          $this->assertFalse($columns['onechar']->primary_key);
 886          $this->assertFalse($columns['onechar']->binary);
 887          $this->assertTrue($columns['onechar']->has_default);
 888          $this->assertSame('Nice dflt!', $columns['onechar']->default_value);
 889          $this->assertSame('C', $columns['onechar']->meta_type);
 890          $this->assertEquals('Nice dflt!', $DB->get_field('test_table1', 'onechar', array(), IGNORE_MULTIPLE)); // Check default has been applied.
 891  
 892          // Add one big text field and check it.
 893          $field = new xmldb_field('onetext');
 894          $field->set_attributes(XMLDB_TYPE_TEXT, 'big');
 895          $dbman->add_field($table, $field);
 896          $this->assertTrue($dbman->field_exists($table, 'onetext'));
 897          $columns = $DB->get_columns('test_table1');
 898          $this->assertSame('onetext', $columns['onetext']->name);
 899          $this->assertEquals(-1, $columns['onetext']->max_length); // -1 means unknown or big.
 900          $this->assertNull($columns['onetext']->scale);
 901          $this->assertFalse($columns['onetext']->not_null);
 902          $this->assertFalse($columns['onetext']->primary_key);
 903          $this->assertFalse($columns['onetext']->binary);
 904          $this->assertFalse($columns['onetext']->has_default);
 905          $this->assertNull($columns['onetext']->default_value);
 906          $this->assertSame('X', $columns['onetext']->meta_type);
 907  
 908          // Add one medium text field and check it.
 909          $field = new xmldb_field('mediumtext');
 910          $field->set_attributes(XMLDB_TYPE_TEXT, 'medium');
 911          $dbman->add_field($table, $field);
 912          $columns = $DB->get_columns('test_table1');
 913          $this->assertTrue(($columns['mediumtext']->max_length == -1) or ($columns['mediumtext']->max_length >= 16777215)); // -1 means unknown or big.
 914  
 915          // Add one small text field and check it.
 916          $field = new xmldb_field('smalltext');
 917          $field->set_attributes(XMLDB_TYPE_TEXT, 'small');
 918          $dbman->add_field($table, $field);
 919          $columns = $DB->get_columns('test_table1');
 920          $this->assertTrue(($columns['smalltext']->max_length == -1) or ($columns['smalltext']->max_length >= 65535)); // -1 means unknown or big.
 921  
 922          // Add one binary field and check it.
 923          $field = new xmldb_field('onebinary');
 924          $field->set_attributes(XMLDB_TYPE_BINARY);
 925          $dbman->add_field($table, $field);
 926          $this->assertTrue($dbman->field_exists($table, 'onebinary'));
 927          $columns = $DB->get_columns('test_table1');
 928          $this->assertSame('onebinary', $columns['onebinary']->name);
 929          $this->assertEquals(-1, $columns['onebinary']->max_length);
 930          $this->assertNull($columns['onebinary']->scale);
 931          $this->assertFalse($columns['onebinary']->not_null);
 932          $this->assertFalse($columns['onebinary']->primary_key);
 933          $this->assertTrue($columns['onebinary']->binary);
 934          $this->assertFalse($columns['onebinary']->has_default);
 935          $this->assertNull($columns['onebinary']->default_value);
 936          $this->assertSame('B', $columns['onebinary']->meta_type);
 937  
 938          // TODO: check datetime type. Although unused should be fully supported.
 939      }
 940  
 941      /**
 942       * Test behaviour of drop_field()
 943       */
 944      public function test_drop_field() {
 945          $DB = $this->tdb; // Do not use global $DB!
 946          $dbman = $this->tdb->get_manager();
 947  
 948          $table = $this->create_deftable('test_table0');
 949  
 950          // Fill the table with some records before dropping fields.
 951          $this->fill_deftable('test_table0');
 952  
 953          // Drop field with simple xmldb_field having indexes, must return exception.
 954          $field = new xmldb_field('type'); // Field has indexes and default clause.
 955          $this->assertTrue($dbman->field_exists($table, 'type'));
 956          try {
 957              $dbman->drop_field($table, $field);
 958              $this->fail('Exception expected');
 959          } catch (\moodle_exception $e) {
 960              $this->assertInstanceOf('ddl_dependency_exception', $e);
 961          }
 962          $this->assertTrue($dbman->field_exists($table, 'type')); // Continues existing, drop aborted.
 963  
 964          // Drop field with complete xmldb_field object and related indexes, must return exception.
 965          $field = $table->getField('course'); // Field has indexes and default clause.
 966          $this->assertTrue($dbman->field_exists($table, $field));
 967          try {
 968              $dbman->drop_field($table, $field);
 969              $this->fail('Exception expected');
 970          } catch (\moodle_exception $e) {
 971              $this->assertInstanceOf('ddl_dependency_exception', $e);
 972          }
 973          $this->assertTrue($dbman->field_exists($table, $field)); // Continues existing, drop aborted.
 974  
 975          // Drop one non-existing field, must return exception.
 976          $field = new xmldb_field('nonexistingfield');
 977          $this->assertFalse($dbman->field_exists($table, $field));
 978          try {
 979              $dbman->drop_field($table, $field);
 980              $this->fail('Exception expected');
 981          } catch (\moodle_exception $e) {
 982              $this->assertInstanceOf('ddl_field_missing_exception', $e);
 983          }
 984  
 985          // Drop field with simple xmldb_field, not having related indexes.
 986          $field = new xmldb_field('forcesubscribe'); // Field has default clause.
 987          $this->assertTrue($dbman->field_exists($table, 'forcesubscribe'));
 988          $dbman->drop_field($table, $field);
 989          $this->assertFalse($dbman->field_exists($table, 'forcesubscribe'));
 990  
 991          // Drop field with complete xmldb_field object, not having related indexes.
 992          $field = new xmldb_field('trackingtype'); // Field has default clause.
 993          $this->assertTrue($dbman->field_exists($table, $field));
 994          $dbman->drop_field($table, $field);
 995          $this->assertFalse($dbman->field_exists($table, $field));
 996      }
 997  
 998      /**
 999       * Test behaviour of change_field_type()
1000       */
1001      public function test_change_field_type() {
1002          $DB = $this->tdb; // Do not use global $DB!
1003          $dbman = $this->tdb->get_manager();
1004  
1005          // Create table with indexed field and not indexed field to
1006          // perform tests in both fields, both having defaults.
1007          $table = new xmldb_table('test_table_cust0');
1008          $table->add_field('id', XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1009          $table->add_field('onenumber', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '2');
1010          $table->add_field('anothernumber', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '4');
1011          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1012          $table->add_index('onenumber', XMLDB_INDEX_NOTUNIQUE, array('onenumber'));
1013          $dbman->create_table($table);
1014  
1015          $record = new \stdClass();
1016          $record->onenumber = 2;
1017          $record->anothernumber = 4;
1018          $recoriginal = $DB->insert_record('test_table_cust0', $record);
1019  
1020          // Change column from integer to varchar. Must return exception because of dependent index.
1021          $field = new xmldb_field('onenumber');
1022          $field->set_attributes(XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, 'test');
1023          try {
1024              $dbman->change_field_type($table, $field);
1025              $this->fail('Exception expected');
1026          } catch (\moodle_exception $e) {
1027              $this->assertInstanceOf('ddl_dependency_exception', $e);
1028          }
1029          // Column continues being integer 10 not null default 2.
1030          $columns = $DB->get_columns('test_table_cust0');
1031          $this->assertSame('I', $columns['onenumber']->meta_type);
1032          // TODO: check the rest of attributes.
1033  
1034          // Change column from integer to varchar. Must work because column has no dependencies.
1035          $field = new xmldb_field('anothernumber');
1036          $field->set_attributes(XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, 'test');
1037          $dbman->change_field_type($table, $field);
1038          // Column is char 30 not null default 'test' now.
1039          $columns = $DB->get_columns('test_table_cust0');
1040          $this->assertSame('C', $columns['anothernumber']->meta_type);
1041          // TODO: check the rest of attributes.
1042  
1043          // Change column back from char to integer.
1044          $field = new xmldb_field('anothernumber');
1045          $field->set_attributes(XMLDB_TYPE_INTEGER, '8', null, XMLDB_NOTNULL, null, '5');
1046          $dbman->change_field_type($table, $field);
1047          // Column is integer 8 not null default 5 now.
1048          $columns = $DB->get_columns('test_table_cust0');
1049          $this->assertSame('I', $columns['anothernumber']->meta_type);
1050          // TODO: check the rest of attributes.
1051  
1052          // Change column once more from integer to char.
1053          $field = new xmldb_field('anothernumber');
1054          $field->set_attributes(XMLDB_TYPE_CHAR, '20', null, XMLDB_NOTNULL, null, "test'n drop");
1055          $dbman->change_field_type($table, $field);
1056          // Column is char 30 not null default "test'n drop" now.
1057          $columns = $DB->get_columns('test_table_cust0');
1058          $this->assertSame('C', $columns['anothernumber']->meta_type);
1059          // TODO: check the rest of attributes.
1060  
1061          // Insert one string value and try to convert to integer. Must throw exception.
1062          $record = new \stdClass();
1063          $record->onenumber = 7;
1064          $record->anothernumber = 'string value';
1065          $rectodrop = $DB->insert_record('test_table_cust0', $record);
1066          $field = new xmldb_field('anothernumber');
1067          $field->set_attributes(XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '5');
1068          try {
1069              $dbman->change_field_type($table, $field);
1070              $this->fail('Exception expected');
1071          } catch (\moodle_exception $e) {
1072              $this->assertInstanceOf('ddl_change_structure_exception', $e);
1073          }
1074          // Column continues being char 30 not null default "test'n drop" now.
1075          $this->assertSame('C', $columns['anothernumber']->meta_type);
1076          // TODO: check the rest of attributes.
1077          $DB->delete_records('test_table_cust0', array('id' => $rectodrop)); // Delete the string record.
1078  
1079          // Change the column from varchar to float.
1080          $field = new xmldb_field('anothernumber');
1081          $field->set_attributes(XMLDB_TYPE_FLOAT, '20,10', null, null, null, null);
1082          $dbman->change_field_type($table, $field);
1083          // Column is float 20,10 null default null.
1084          $columns = $DB->get_columns('test_table_cust0');
1085          $this->assertSame('N', $columns['anothernumber']->meta_type); // Floats are seen as number.
1086          // TODO: check the rest of attributes.
1087  
1088          // Change the column back from float to varchar.
1089          $field = new xmldb_field('anothernumber');
1090          $field->set_attributes(XMLDB_TYPE_CHAR, '20', null, XMLDB_NOTNULL, null, 'test');
1091          $dbman->change_field_type($table, $field);
1092          // Column is char 20 not null default "test" now.
1093          $columns = $DB->get_columns('test_table_cust0');
1094          $this->assertSame('C', $columns['anothernumber']->meta_type);
1095          // TODO: check the rest of attributes.
1096  
1097          // Change the column from varchar to number.
1098          $field = new xmldb_field('anothernumber');
1099          $field->set_attributes(XMLDB_TYPE_NUMBER, '20,10', null, null, null, null);
1100          $dbman->change_field_type($table, $field);
1101          // Column is number 20,10 null default null now.
1102          $columns = $DB->get_columns('test_table_cust0');
1103          $this->assertSame('N', $columns['anothernumber']->meta_type);
1104          // TODO: check the rest of attributes.
1105  
1106          // Change the column from number to integer.
1107          $field = new xmldb_field('anothernumber');
1108          $field->set_attributes(XMLDB_TYPE_INTEGER, '2', null, null, null, null);
1109          $dbman->change_field_type($table, $field);
1110          // Column is integer 2 null default null now.
1111          $columns = $DB->get_columns('test_table_cust0');
1112          $this->assertSame('I', $columns['anothernumber']->meta_type);
1113          // TODO: check the rest of attributes.
1114  
1115          // Change the column from integer to text.
1116          $field = new xmldb_field('anothernumber');
1117          $field->set_attributes(XMLDB_TYPE_TEXT, 'big', null, XMLDB_NOTNULL, null, null);
1118          $dbman->change_field_type($table, $field);
1119          // Column is char text not null default null.
1120          $columns = $DB->get_columns('test_table_cust0');
1121          $this->assertSame('X', $columns['anothernumber']->meta_type);
1122  
1123          // Change the column back from text to number.
1124          $field = new xmldb_field('anothernumber');
1125          $field->set_attributes(XMLDB_TYPE_NUMBER, '20,10', null, null, null, null);
1126          $dbman->change_field_type($table, $field);
1127          // Column is number 20,10 null default null now.
1128          $columns = $DB->get_columns('test_table_cust0');
1129          $this->assertSame('N', $columns['anothernumber']->meta_type);
1130          // TODO: check the rest of attributes.
1131  
1132          // Change the column from number to text.
1133          $field = new xmldb_field('anothernumber');
1134          $field->set_attributes(XMLDB_TYPE_TEXT, 'big', null, XMLDB_NOTNULL, null, null);
1135          $dbman->change_field_type($table, $field);
1136          // Column is char text not null default "test" now.
1137          $columns = $DB->get_columns('test_table_cust0');
1138          $this->assertSame('X', $columns['anothernumber']->meta_type);
1139          // TODO: check the rest of attributes.
1140  
1141          // Change the column back from text to integer.
1142          $field = new xmldb_field('anothernumber');
1143          $field->set_attributes(XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, 10);
1144          $dbman->change_field_type($table, $field);
1145          // Column is integer 10 not null default 10.
1146          $columns = $DB->get_columns('test_table_cust0');
1147          $this->assertSame('I', $columns['anothernumber']->meta_type);
1148          // TODO: check the rest of attributes.
1149  
1150          // Check original value has survived to all the type changes.
1151          $this->assertnotEmpty($rec = $DB->get_record('test_table_cust0', array('id' => $recoriginal)));
1152          $this->assertEquals(4, $rec->anothernumber);
1153  
1154          $dbman->drop_table($table);
1155          $this->assertFalse($dbman->table_exists($table));
1156      }
1157  
1158      /**
1159       * Test behaviour of test_change_field_precision()
1160       */
1161      public function test_change_field_precision() {
1162          $DB = $this->tdb; // Do not use global $DB!
1163          $dbman = $this->tdb->get_manager();
1164  
1165          $table = $this->create_deftable('test_table1');
1166  
1167          // Fill the table with some records before dropping fields.
1168          $this->fill_deftable('test_table1');
1169  
1170          // Change text field from medium to big.
1171          $field = new xmldb_field('intro');
1172          $field->set_attributes(XMLDB_TYPE_TEXT, 'big', null, XMLDB_NOTNULL, null, null);
1173          $dbman->change_field_precision($table, $field);
1174          $columns = $DB->get_columns('test_table1');
1175          // Cannot check the text type, only the metatype.
1176          $this->assertSame('X', $columns['intro']->meta_type);
1177          // TODO: check the rest of attributes.
1178  
1179          // Change char field from 30 to 20.
1180          $field = new xmldb_field('secondname');
1181          $field->set_attributes(XMLDB_TYPE_CHAR, '20', null, XMLDB_NOTNULL, null, null);
1182          $dbman->change_field_precision($table, $field);
1183          $columns = $DB->get_columns('test_table1');
1184          $this->assertSame('C', $columns['secondname']->meta_type);
1185          // TODO: check the rest of attributes.
1186  
1187          // Change char field from 20 to 10, having contents > 10cc. Throw exception.
1188          $field = new xmldb_field('secondname');
1189          $field->set_attributes(XMLDB_TYPE_CHAR, '10', null, XMLDB_NOTNULL, null, null);
1190          try {
1191              $dbman->change_field_precision($table, $field);
1192              $this->fail('Exception expected');
1193          } catch (\moodle_exception $e) {
1194              $this->assertInstanceOf('ddl_change_structure_exception', $e);
1195          }
1196          // No changes in field specs at all.
1197          $columns = $DB->get_columns('test_table1');
1198          $this->assertSame('C', $columns['secondname']->meta_type);
1199          // TODO: check the rest of attributes.
1200  
1201          // Change number field from 20,10 to 10,2.
1202          $field = new xmldb_field('grade');
1203          $field->set_attributes(XMLDB_TYPE_NUMBER, '10,2', null, null, null, null);
1204          $dbman->change_field_precision($table, $field);
1205          $columns = $DB->get_columns('test_table1');
1206          $this->assertSame('N', $columns['grade']->meta_type);
1207          // TODO: check the rest of attributes.
1208  
1209          // Change integer field from 10 to 2.
1210          $field = new xmldb_field('userid');
1211          $field->set_attributes(XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, null, '0');
1212          $dbman->change_field_precision($table, $field);
1213          $columns = $DB->get_columns('test_table1');
1214          $this->assertSame('I', $columns['userid']->meta_type);
1215          // TODO: check the rest of attributes.
1216  
1217          // Change the column from integer (2) to integer (6) (forces change of type in some DBs).
1218          $field = new xmldb_field('userid');
1219          $field->set_attributes(XMLDB_TYPE_INTEGER, '6', null, null, null, null);
1220          $dbman->change_field_precision($table, $field);
1221          // Column is integer 6 null default null now.
1222          $columns = $DB->get_columns('test_table1');
1223          $this->assertSame('I', $columns['userid']->meta_type);
1224          // TODO: check the rest of attributes.
1225  
1226          // Insert one record with 6-digit field.
1227          $record = new \stdClass();
1228          $record->course = 10;
1229          $record->secondname  = 'third record';
1230          $record->intro  = 'third record';
1231          $record->userid = 123456;
1232          $DB->insert_record('test_table1', $record);
1233          // Change integer field from 6 to 2, contents are bigger, must throw exception.
1234          $field = new xmldb_field('userid');
1235          $field->set_attributes(XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, null, '0');
1236          try {
1237              $dbman->change_field_precision($table, $field);
1238              $this->fail('Exception expected');
1239          } catch (\moodle_exception $e) {
1240              $this->assertInstanceOf('ddl_change_structure_exception', $e);
1241          }
1242          // No changes in field specs at all.
1243          $columns = $DB->get_columns('test_table1');
1244          $this->assertSame('I', $columns['userid']->meta_type);
1245          // TODO: check the rest of attributes.
1246  
1247          // Change integer field from 10 to 3, in field used by index. must throw exception.
1248          $field = new xmldb_field('course');
1249          $field->set_attributes(XMLDB_TYPE_INTEGER, '3', null, XMLDB_NOTNULL, null, '0');
1250          try {
1251              $dbman->change_field_precision($table, $field);
1252              $this->fail('Exception expected');
1253          } catch (\moodle_exception $e) {
1254              $this->assertInstanceOf('ddl_dependency_exception', $e);
1255          }
1256          // No changes in field specs at all.
1257          $columns = $DB->get_columns('test_table1');
1258          $this->assertSame('I', $columns['course']->meta_type);
1259          // TODO: check the rest of attributes.
1260      }
1261  
1262      public function testChangeFieldNullability() {
1263          $DB = $this->tdb; // Do not use global $DB!
1264          $dbman = $this->tdb->get_manager();
1265  
1266          $table = new xmldb_table('test_table_cust0');
1267          $table->add_field('id', XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1268          $table->add_field('name', XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, null);
1269          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1270          $dbman->create_table($table);
1271  
1272          $record = new \stdClass();
1273          $record->name = null;
1274  
1275          try {
1276              $result = $DB->insert_record('test_table_cust0', $record, false);
1277          } catch (\dml_exception $e) {
1278              $result = false;
1279          }
1280          $this->resetDebugging();
1281          $this->assertFalse($result);
1282  
1283          $field = new xmldb_field('name');
1284          $field->set_attributes(XMLDB_TYPE_CHAR, '30', null, null, null, null);
1285          $dbman->change_field_notnull($table, $field);
1286  
1287          $this->assertTrue($DB->insert_record('test_table_cust0', $record, false));
1288  
1289          // TODO: add some tests with existing data in table.
1290          $DB->delete_records('test_table_cust0');
1291  
1292          $field = new xmldb_field('name');
1293          $field->set_attributes(XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, null);
1294          $dbman->change_field_notnull($table, $field);
1295  
1296          try {
1297              $result = $DB->insert_record('test_table_cust0', $record, false);
1298          } catch (\dml_exception $e) {
1299              $result = false;
1300          }
1301          $this->resetDebugging();
1302          $this->assertFalse($result);
1303  
1304          $dbman->drop_table($table);
1305      }
1306  
1307      public function testChangeFieldDefault() {
1308          $DB = $this->tdb; // Do not use global $DB!
1309          $dbman = $this->tdb->get_manager();
1310  
1311          $table = new xmldb_table('test_table_cust0');
1312          $table->add_field('id', XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1313          $table->add_field('onenumber', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
1314          $table->add_field('name', XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, 'Moodle');
1315          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1316          $dbman->create_table($table);
1317  
1318          $field = new xmldb_field('name');
1319          $field->set_attributes(XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, 'Moodle2');
1320          $dbman->change_field_default($table, $field);
1321  
1322          $record = new \stdClass();
1323          $record->onenumber = 666;
1324          $id = $DB->insert_record('test_table_cust0', $record);
1325  
1326          $record = $DB->get_record('test_table_cust0', array('id'=>$id));
1327          $this->assertSame('Moodle2', $record->name);
1328  
1329          $field = new xmldb_field('onenumber');
1330          $field->set_attributes(XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, 666);
1331          $dbman->change_field_default($table, $field);
1332  
1333          $record = new \stdClass();
1334          $record->name = 'something';
1335          $id = $DB->insert_record('test_table_cust0', $record);
1336  
1337          $record = $DB->get_record('test_table_cust0', array('id'=>$id));
1338          $this->assertSame('666', $record->onenumber);
1339  
1340          $dbman->drop_table($table);
1341      }
1342  
1343      public function testAddUniqueIndex() {
1344          $DB = $this->tdb; // Do not use global $DB!
1345          $dbman = $this->tdb->get_manager();
1346  
1347          $table = new xmldb_table('test_table_cust0');
1348          $table->add_field('id', XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1349          $table->add_field('onenumber', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
1350          $table->add_field('name', XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, 'Moodle');
1351          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1352          $dbman->create_table($table);
1353  
1354          $record = new \stdClass();
1355          $record->onenumber = 666;
1356          $record->name = 'something';
1357          $DB->insert_record('test_table_cust0', $record, false);
1358  
1359          $index = new xmldb_index('onenumber-name');
1360          $index->set_attributes(XMLDB_INDEX_UNIQUE, array('onenumber', 'name'));
1361          $dbman->add_index($table, $index);
1362  
1363          try {
1364              $result = $DB->insert_record('test_table_cust0', $record, false);
1365          } catch (\dml_exception $e) {
1366              $result = false;
1367          }
1368          $this->resetDebugging();
1369          $this->assertFalse($result);
1370  
1371          $dbman->drop_table($table);
1372      }
1373  
1374      public function testAddNonUniqueIndex() {
1375          $dbman = $this->tdb->get_manager();
1376  
1377          $table = $this->create_deftable('test_table1');
1378          $index = new xmldb_index('secondname');
1379          $index->set_attributes(XMLDB_INDEX_NOTUNIQUE, array('course', 'name'));
1380          $dbman->add_index($table, $index);
1381          $this->assertTrue($dbman->index_exists($table, $index));
1382  
1383          try {
1384              $dbman->add_index($table, $index);
1385              $this->fail('Exception expected for duplicate indexes');
1386          } catch (\moodle_exception $e) {
1387              $this->assertInstanceOf('ddl_exception', $e);
1388          }
1389  
1390          $index = new xmldb_index('third');
1391          $index->set_attributes(XMLDB_INDEX_NOTUNIQUE, array('course'));
1392          try {
1393              $dbman->add_index($table, $index);
1394              $this->fail('Exception expected for duplicate indexes');
1395          } catch (\moodle_exception $e) {
1396              $this->assertInstanceOf('ddl_exception', $e);
1397          }
1398  
1399          $table = new xmldb_table('test_table_cust0');
1400          $table->add_field('id', XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1401          $table->add_field('onenumber', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
1402          $table->add_field('name', XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, 'Moodle');
1403          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1404          $table->add_key('onenumber', XMLDB_KEY_FOREIGN, array('onenumber'));
1405  
1406          try {
1407              $table->add_index('onenumber', XMLDB_INDEX_NOTUNIQUE, array('onenumber'));
1408              $this->fail('Coding exception expected');
1409          } catch (\moodle_exception $e) {
1410              $this->assertInstanceOf('coding_exception', $e);
1411          }
1412  
1413          $table = new xmldb_table('test_table_cust0');
1414          $table->add_field('id', XMLDB_TYPE_INTEGER, '2', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1415          $table->add_field('onenumber', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
1416          $table->add_field('name', XMLDB_TYPE_CHAR, '30', null, XMLDB_NOTNULL, null, 'Moodle');
1417          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1418          $table->add_index('onenumber', XMLDB_INDEX_NOTUNIQUE, array('onenumber'));
1419  
1420          try {
1421              $table->add_key('onenumber', XMLDB_KEY_FOREIGN, array('onenumber'));
1422              $this->fail('Coding exception expected');
1423          } catch (\moodle_exception $e) {
1424              $this->assertInstanceOf('coding_exception', $e);
1425          }
1426  
1427      }
1428  
1429      public function testFindIndexName() {
1430          $dbman = $this->tdb->get_manager();
1431  
1432          $table = $this->create_deftable('test_table1');
1433          $index = new xmldb_index('secondname');
1434          $index->set_attributes(XMLDB_INDEX_NOTUNIQUE, array('course', 'name'));
1435          $dbman->add_index($table, $index);
1436  
1437          // DBM Systems name their indices differently - do not test the actual index name.
1438          $result = $dbman->find_index_name($table, $index);
1439          $this->assertTrue(!empty($result));
1440  
1441          $nonexistentindex = new xmldb_index('nonexistentindex');
1442          $nonexistentindex->set_attributes(XMLDB_INDEX_NOTUNIQUE, array('name'));
1443          $this->assertFalse($dbman->find_index_name($table, $nonexistentindex));
1444      }
1445  
1446      public function testDropIndex() {
1447          $DB = $this->tdb; // Do not use global $DB!
1448  
1449          $dbman = $this->tdb->get_manager();
1450  
1451          $table = $this->create_deftable('test_table1');
1452          $index = new xmldb_index('secondname');
1453          $index->set_attributes(XMLDB_INDEX_NOTUNIQUE, array('course', 'name'));
1454          $dbman->add_index($table, $index);
1455  
1456          $dbman->drop_index($table, $index);
1457          $this->assertFalse($dbman->find_index_name($table, $index));
1458  
1459          // Test we are able to drop indexes having hyphens MDL-22804.
1460          // Create index with hyphens (by hand).
1461          $indexname = 'test-index-with-hyphens';
1462          switch ($DB->get_dbfamily()) {
1463              case 'mysql':
1464                  $indexname = '`' . $indexname . '`';
1465                  break;
1466              default:
1467                  $indexname = '"' . $indexname . '"';
1468          }
1469          $stmt = "CREATE INDEX {$indexname} ON {$DB->get_prefix()}test_table1 (course, name)";
1470          $DB->change_database_structure($stmt);
1471          $this->assertNotEmpty($dbman->find_index_name($table, $index));
1472          // Index created, let's drop it using db manager stuff.
1473          $index = new xmldb_index('indexname', XMLDB_INDEX_NOTUNIQUE, array('course', 'name'));
1474          $dbman->drop_index($table, $index);
1475          $this->assertFalse($dbman->find_index_name($table, $index));
1476      }
1477  
1478      public function testAddUniqueKey() {
1479          $dbman = $this->tdb->get_manager();
1480  
1481          $table = $this->create_deftable('test_table1');
1482          $key = new xmldb_key('id-course-grade');
1483          $key->set_attributes(XMLDB_KEY_UNIQUE, array('id', 'course', 'grade'));
1484          $dbman->add_key($table, $key);
1485  
1486          // No easy way to test it, this just makes sure no errors are encountered.
1487          $this->assertTrue(true);
1488      }
1489  
1490      public function testAddForeignUniqueKey() {
1491          $dbman = $this->tdb->get_manager();
1492  
1493          $table = $this->create_deftable('test_table1');
1494          $this->create_deftable('test_table0');
1495  
1496          $key = new xmldb_key('course');
1497          $key->set_attributes(XMLDB_KEY_FOREIGN_UNIQUE, array('course'), 'test_table0', array('id'));
1498          $dbman->add_key($table, $key);
1499  
1500          // No easy way to test it, this just makes sure no errors are encountered.
1501          $this->assertTrue(true);
1502      }
1503  
1504      public function testDropKey() {
1505          $dbman = $this->tdb->get_manager();
1506  
1507          $table = $this->create_deftable('test_table1');
1508          $this->create_deftable('test_table0');
1509  
1510          $key = new xmldb_key('course');
1511          $key->set_attributes(XMLDB_KEY_FOREIGN_UNIQUE, array('course'), 'test_table0', array('id'));
1512          $dbman->add_key($table, $key);
1513  
1514          $dbman->drop_key($table, $key);
1515  
1516          // No easy way to test it, this just makes sure no errors are encountered.
1517          $this->assertTrue(true);
1518      }
1519  
1520      public function testAddForeignKey() {
1521          $dbman = $this->tdb->get_manager();
1522  
1523          $table = $this->create_deftable('test_table1');
1524          $this->create_deftable('test_table0');
1525  
1526          $key = new xmldb_key('course');
1527          $key->set_attributes(XMLDB_KEY_FOREIGN, array('course'), 'test_table0', array('id'));
1528          $dbman->add_key($table, $key);
1529  
1530          // No easy way to test it, this just makes sure no errors are encountered.
1531          $this->assertTrue(true);
1532      }
1533  
1534      public function testDropForeignKey() {
1535          $dbman = $this->tdb->get_manager();
1536  
1537          $table = $this->create_deftable('test_table1');
1538          $this->create_deftable('test_table0');
1539  
1540          $key = new xmldb_key('course');
1541          $key->set_attributes(XMLDB_KEY_FOREIGN, array('course'), 'test_table0', array('id'));
1542          $dbman->add_key($table, $key);
1543  
1544          $dbman->drop_key($table, $key);
1545  
1546          // No easy way to test it, this just makes sure no errors are encountered.
1547          $this->assertTrue(true);
1548      }
1549  
1550      public function testRenameField() {
1551          $DB = $this->tdb; // Do not use global $DB!
1552          $dbman = $this->tdb->get_manager();
1553  
1554          $table = $this->create_deftable('test_table0');
1555          $field = new xmldb_field('type');
1556          $field->set_attributes(XMLDB_TYPE_CHAR, '20', null, XMLDB_NOTNULL, null, 'general', 'course');
1557  
1558          // 1. Rename the 'type' field into a generic new valid name.
1559          // This represents the standard use case.
1560          $dbman->rename_field($table, $field, 'newfieldname');
1561  
1562          $columns = $DB->get_columns('test_table0');
1563  
1564          $this->assertArrayNotHasKey('type', $columns);
1565          $this->assertArrayHasKey('newfieldname', $columns);
1566          $field->setName('newfieldname');
1567  
1568          // 2. Rename the 'newfieldname' field into a reserved word, for testing purposes.
1569          // This represents a questionable use case: we should support it but discourage the use of it on peer reviewing.
1570          $dbman->rename_field($table, $field, 'where');
1571  
1572          $columns = $DB->get_columns('test_table0');
1573  
1574          $this->assertArrayNotHasKey('newfieldname', $columns);
1575          $this->assertArrayHasKey('where', $columns);
1576  
1577          // 3. Create a table with a column name named w/ a reserved word and get rid of it.
1578          // This represents a "recovering" use case: a field name could be a reserved word in the future, at least for a DB type.
1579          $table = new xmldb_table('test_table_res_word');
1580          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1581          $table->add_field('where', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
1582          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1583          $table->setComment("This is a test'n drop table. You can drop it safely");
1584          $dbman->create_table($table);
1585          $dbman->table_exists('test_table_res_word');
1586  
1587          $columns = $DB->get_columns('test_table_res_word');
1588          $this->assertArrayHasKey('where', $columns);
1589          $field = $table->getField('where');
1590  
1591          $dbman->rename_field($table, $field, 'newfieldname');
1592  
1593          $columns = $DB->get_columns('test_table_res_word');
1594  
1595          $this->assertArrayNotHasKey('where', $columns);
1596          $this->assertArrayHasKey('newfieldname', $columns);
1597      }
1598  
1599      public function testIndexExists() {
1600          // Skipping: this is just a test of find_index_name.
1601      }
1602  
1603      public function testFindKeyName() {
1604          $dbman = $this->tdb->get_manager();
1605  
1606          $table = $this->create_deftable('test_table0');
1607          $key = $table->getKey('primary');
1608  
1609          // With Mysql, the return value is actually "mdl_test_id_pk".
1610          $result = $dbman->find_key_name($table, $key);
1611          $this->assertTrue(!empty($result));
1612      }
1613  
1614      public function testDeleteTablesFromXmldbFile() {
1615          $dbman = $this->tdb->get_manager();
1616  
1617          $this->create_deftable('test_table1');
1618  
1619          $this->assertTrue($dbman->table_exists('test_table1'));
1620  
1621          // Feed nonexistent file.
1622          try {
1623              $dbman->delete_tables_from_xmldb_file('fpsoiudfposui');
1624              $this->fail('Exception expected');
1625          } catch (\moodle_exception $e) {
1626              $this->resetDebugging();
1627              $this->assertInstanceOf('moodle_exception', $e);
1628          }
1629  
1630          try {
1631              $dbman->delete_tables_from_xmldb_file(__DIR__ . '/fixtures/invalid.xml');
1632              $this->fail('Exception expected');
1633          } catch (\moodle_exception $e) {
1634              $this->resetDebugging();
1635              $this->assertInstanceOf('moodle_exception', $e);
1636          }
1637  
1638          // Check that the table has not been deleted from DB.
1639          $this->assertTrue($dbman->table_exists('test_table1'));
1640  
1641          // Real and valid xml file.
1642          // TODO: drop UNSINGED completely in Moodle 2.4.
1643          $dbman->delete_tables_from_xmldb_file(__DIR__ . '/fixtures/xmldb_table.xml');
1644  
1645          // Check that the table has been deleted from DB.
1646          $this->assertFalse($dbman->table_exists('test_table1'));
1647      }
1648  
1649      public function testInstallFromXmldbFile() {
1650          $dbman = $this->tdb->get_manager();
1651  
1652          // Feed nonexistent file.
1653          try {
1654              $dbman->install_from_xmldb_file('fpsoiudfposui');
1655              $this->fail('Exception expected');
1656          } catch (\moodle_exception $e) {
1657              $this->resetDebugging();
1658              $this->assertInstanceOf('moodle_exception', $e);
1659          }
1660  
1661          try {
1662              $dbman->install_from_xmldb_file(__DIR__ . '/fixtures/invalid.xml');
1663              $this->fail('Exception expected');
1664          } catch (\moodle_exception $e) {
1665              $this->resetDebugging();
1666              $this->assertInstanceOf('moodle_exception', $e);
1667          }
1668  
1669          // Check that the table has not yet been created in DB.
1670          $this->assertFalse($dbman->table_exists('test_table1'));
1671  
1672          // Real and valid xml file.
1673          $dbman->install_from_xmldb_file(__DIR__ . '/fixtures/xmldb_table.xml');
1674          $this->assertTrue($dbman->table_exists('test_table1'));
1675      }
1676  
1677      public function test_temp_tables() {
1678          $DB = $this->tdb; // Do not use global $DB!
1679          $dbman = $this->tdb->get_manager();
1680  
1681          // Create temp table0.
1682          $table0 = $this->tables['test_table0'];
1683          $dbman->create_temp_table($table0);
1684          $this->assertTrue($dbman->table_exists('test_table0'));
1685  
1686          // Try to create temp table with same name, must throw exception.
1687          $dupetable = $this->tables['test_table0'];
1688          try {
1689              $dbman->create_temp_table($dupetable);
1690              $this->fail('Exception expected');
1691          } catch (\moodle_exception $e) {
1692              $this->assertInstanceOf('ddl_exception', $e);
1693          }
1694  
1695          // Try to create table with same name, must throw exception.
1696          $dupetable = $this->tables['test_table0'];
1697          try {
1698              $dbman->create_table($dupetable);
1699              $this->fail('Exception expected');
1700          } catch (\moodle_exception $e) {
1701              $this->assertInstanceOf('ddl_exception', $e);
1702          }
1703  
1704          // Create another temp table1.
1705          $table1 = $this->tables['test_table1'];
1706          $dbman->create_temp_table($table1);
1707          $this->assertTrue($dbman->table_exists('test_table1'));
1708  
1709          // Get columns and perform some basic tests.
1710          $columns = $DB->get_columns('test_table1');
1711          $this->assertCount(11, $columns);
1712          $this->assertTrue($columns['name'] instanceof database_column_info);
1713          $this->assertEquals(30, $columns['name']->max_length);
1714          $this->assertTrue($columns['name']->has_default);
1715          $this->assertEquals('Moodle', $columns['name']->default_value);
1716  
1717          // Insert some records.
1718          $inserted = $this->fill_deftable('test_table1');
1719          $records = $DB->get_records('test_table1');
1720          $this->assertCount($inserted, $records);
1721          $this->assertSame($records[1]->course, $this->records['test_table1'][0]->course);
1722          $this->assertSame($records[1]->secondname, $this->records['test_table1'][0]->secondname);
1723          $this->assertSame($records[2]->intro, $this->records['test_table1'][1]->intro);
1724  
1725          // Collect statistics about the data in the temp table.
1726          $DB->update_temp_table_stats();
1727  
1728          // Drop table1.
1729          $dbman->drop_table($table1);
1730          $this->assertFalse($dbman->table_exists('test_table1'));
1731  
1732          // Try to drop non-existing temp table, must throw exception.
1733          $noetable = $this->tables['test_table1'];
1734          try {
1735              $dbman->drop_table($noetable);
1736              $this->fail('Exception expected');
1737          } catch (\moodle_exception $e) {
1738              $this->assertInstanceOf('ddl_table_missing_exception', $e);
1739          }
1740  
1741          // Collect statistics about the data in the temp table with less tables.
1742          $DB->update_temp_table_stats();
1743  
1744          // Fill/modify/delete a few table0 records.
1745  
1746          // Drop table0.
1747          $dbman->drop_table($table0);
1748          $this->assertFalse($dbman->table_exists('test_table0'));
1749  
1750          // Create another temp table1.
1751          $table1 = $this->tables['test_table1'];
1752          $dbman->create_temp_table($table1);
1753          $this->assertTrue($dbman->table_exists('test_table1'));
1754  
1755          // Make sure it can be dropped using deprecated drop_temp_table().
1756          $dbman->drop_temp_table($table1);
1757          $this->assertFalse($dbman->table_exists('test_table1'));
1758          $this->assertDebuggingCalled();
1759  
1760          // Try join with normal tables - MS SQL may use incompatible collation.
1761          $table1 = new xmldb_table('test_table');
1762          $table1->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1763          $table1->add_field('name', XMLDB_TYPE_CHAR, 255, null, XMLDB_NOTNULL, null);
1764          $table1->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1765          $dbman->create_table($table1);
1766  
1767          $table2 = new xmldb_table('test_temp');
1768          $table2->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1769          $table2->add_field('name', XMLDB_TYPE_CHAR, 255, null, XMLDB_NOTNULL, null);
1770          $table2->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1771          $dbman->create_temp_table($table2);
1772  
1773          $record = array('name' => 'a');
1774          $DB->insert_record('test_table', $record);
1775          $DB->insert_record('test_temp', $record);
1776  
1777          $record = array('name' => 'b');
1778          $DB->insert_record('test_table', $record);
1779  
1780          $record = array('name' => 'c');
1781          $DB->insert_record('test_temp', $record);
1782  
1783          $sql = "SELECT *
1784                    FROM {test_table} n
1785                    JOIN {test_temp} t ON t.name = n.name";
1786          $records = $DB->get_records_sql($sql);
1787          $this->assertCount(1, $records);
1788  
1789          // Drop temp table.
1790          $dbman->drop_table($table2);
1791          $this->assertFalse($dbman->table_exists('test_temp'));
1792      }
1793  
1794      public function test_concurrent_temp_tables() {
1795          $DB = $this->tdb; // Do not use global $DB!
1796          $dbman = $this->tdb->get_manager();
1797  
1798          // Define 2 records.
1799          $record1 = (object)array(
1800              'course'     =>  1,
1801              'secondname' => '11 important',
1802              'intro'      => '111 important');
1803          $record2 = (object)array(
1804              'course'     =>  2,
1805              'secondname' => '22 important',
1806              'intro'      => '222 important');
1807  
1808          // Create temp table1 and insert 1 record (in DB).
1809          $table = $this->tables['test_table1'];
1810          $dbman->create_temp_table($table);
1811          $this->assertTrue($dbman->table_exists('test_table1'));
1812          $inserted = $DB->insert_record('test_table1', $record1);
1813  
1814          // Switch to new connection.
1815          $cfg = $DB->export_dbconfig();
1816          if (!isset($cfg->dboptions)) {
1817              $cfg->dboptions = array();
1818          }
1819          $DB2 = moodle_database::get_driver_instance($cfg->dbtype, $cfg->dblibrary);
1820          $DB2->connect($cfg->dbhost, $cfg->dbuser, $cfg->dbpass, $cfg->dbname, $cfg->prefix, $cfg->dboptions);
1821          $dbman2 = $DB2->get_manager();
1822          $this->assertFalse($dbman2->table_exists('test_table1')); // Temp table not exists in DB2.
1823  
1824          // Create temp table1 and insert 1 record (in DB2).
1825          $table = $this->tables['test_table1'];
1826          $dbman2->create_temp_table($table);
1827          $this->assertTrue($dbman2->table_exists('test_table1'));
1828          $inserted = $DB2->insert_record('test_table1', $record2);
1829  
1830          $dbman2->drop_table($table); // Drop temp table before closing DB2.
1831          $this->assertFalse($dbman2->table_exists('test_table1'));
1832          $DB2->dispose(); // Close DB2.
1833  
1834          $this->assertTrue($dbman->table_exists('test_table1')); // Check table continues existing for DB.
1835          $dbman->drop_table($table); // Drop temp table.
1836          $this->assertFalse($dbman->table_exists('test_table1'));
1837      }
1838  
1839      /**
1840       * get_columns should return an empty array for ex-temptables.
1841       */
1842      public function test_leftover_temp_tables_columns() {
1843          $DB = $this->tdb; // Do not use global $DB!
1844          $dbman = $this->tdb->get_manager();
1845  
1846          // Create temp table0.
1847          $table0 = $this->tables['test_table0'];
1848          $dbman->create_temp_table($table0);
1849  
1850          $dbman->drop_table($table0);
1851  
1852          // Get columns and perform some basic tests.
1853          $columns = $DB->get_columns('test_table0');
1854          $this->assertEquals([], $columns);
1855      }
1856  
1857      /**
1858       * Deleting a temp table should not purge the whole cache
1859       */
1860      public function test_leftover_temp_tables_cache() {
1861          $DB = $this->tdb; // Do not use global $DB!
1862          $dbman = $this->tdb->get_manager();
1863  
1864          // Create 2 temp tables.
1865          $table0 = $this->tables['test_table0'];
1866          $dbman->create_temp_table($table0);
1867          $table1 = $this->tables['test_table1'];
1868          $dbman->create_temp_table($table1);
1869  
1870          // Create a normal table.
1871          $table2 = new xmldb_table ('test_table2');
1872          $table2->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1873          $table2->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
1874          $table2->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1875          $table2->setComment("This is a test'n drop table. You can drop it safely");
1876          $this->tables[$table2->getName()] = $table2;
1877          $dbman->create_table($table2);
1878  
1879          // Get columns for the tables, so that relevant caches are populated with their data.
1880          $DB->get_columns('test_table0');
1881          $DB->get_columns('test_table1');
1882          $DB->get_columns('test_table2');
1883  
1884          $dbman->drop_table($table0);
1885  
1886          $rc = new \ReflectionClass('moodle_database');
1887          $rcm = $rc->getMethod('get_temp_tables_cache');
1888          $rcm->setAccessible(true);
1889          $metacachetemp = $rcm->invokeArgs($DB, []);
1890  
1891          // Data of test_table0 should be removed from the cache.
1892          $this->assertEquals(false, $metacachetemp->has('test_table0'));
1893  
1894          // Data of test_table1 should be intact.
1895          $this->assertEquals(true, $metacachetemp->has('test_table1'));
1896  
1897          $rc = new \ReflectionClass('moodle_database');
1898          $rcm = $rc->getMethod('get_metacache');
1899          $rcm->setAccessible(true);
1900          $metacache = $rcm->invokeArgs($DB, []);
1901  
1902          // Data of test_table2 should be intact.
1903          $this->assertEquals(true, $metacache->has('test_table2'));
1904  
1905          // Delete the leftover temp table.
1906          $dbman->drop_table($table1);
1907      }
1908  
1909      public function test_reset_sequence() {
1910          $DB = $this->tdb;
1911          $dbman = $DB->get_manager();
1912  
1913          $table = new xmldb_table('testtable');
1914          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1915          $table->add_field('course', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0');
1916          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1917  
1918          // Drop if exists.
1919          if ($dbman->table_exists($table)) {
1920              $dbman->drop_table($table);
1921          }
1922          $dbman->create_table($table);
1923          $tablename = $table->getName();
1924          $this->tables[$tablename] = $table;
1925  
1926          $record = (object)array('id'=>666, 'course'=>10);
1927          $DB->import_record('testtable', $record);
1928          $DB->delete_records('testtable'); // This delete performs one TRUNCATE.
1929  
1930          $dbman->reset_sequence($table); // Using xmldb object.
1931          $this->assertEquals(1, $DB->insert_record('testtable', (object)array('course'=>13)));
1932  
1933          $record = (object)array('id'=>666, 'course'=>10);
1934          $DB->import_record('testtable', $record);
1935          $DB->delete_records('testtable', array()); // This delete performs one DELETE.
1936  
1937          $dbman->reset_sequence($table); // Using xmldb object.
1938          $this->assertEquals(1, $DB->insert_record('testtable', (object)array('course'=>13)),
1939              'Some versions of MySQL 5.6.x are known to not support lowering of auto-increment numbers.');
1940  
1941          $DB->import_record('testtable', $record);
1942          $dbman->reset_sequence($tablename); // Using string.
1943          $this->assertEquals(667, $DB->insert_record('testtable', (object)array('course'=>13)));
1944  
1945          $dbman->drop_table($table);
1946      }
1947  
1948      public function test_reserved_words() {
1949          $reserved = sql_generator::getAllReservedWords();
1950          $this->assertTrue(count($reserved) > 1);
1951      }
1952  
1953      public function test_index_hints() {
1954          $DB = $this->tdb;
1955          $dbman = $DB->get_manager();
1956  
1957          $table = new xmldb_table('testtable');
1958          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1959          $table->add_field('name', XMLDB_TYPE_CHAR, 255, null, XMLDB_NOTNULL, null);
1960          $table->add_field('path', XMLDB_TYPE_CHAR, 255, null, XMLDB_NOTNULL, null);
1961          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1962          $table->add_index('name', XMLDB_INDEX_NOTUNIQUE, array('name'), array('xxxx,yyyy'));
1963          $table->add_index('path', XMLDB_INDEX_NOTUNIQUE, array('path'), array('varchar_pattern_ops'));
1964  
1965          // Drop if exists.
1966          if ($dbman->table_exists($table)) {
1967              $dbman->drop_table($table);
1968          }
1969          $dbman->create_table($table);
1970          $tablename = $table->getName();
1971          $this->tables[$tablename] = $table;
1972  
1973          $table = new xmldb_table('testtable');
1974          $index = new xmldb_index('name', XMLDB_INDEX_NOTUNIQUE, array('name'), array('xxxx,yyyy'));
1975          $this->assertTrue($dbman->index_exists($table, $index));
1976  
1977          $table = new xmldb_table('testtable');
1978          $index = new xmldb_index('path', XMLDB_INDEX_NOTUNIQUE, array('path'), array('varchar_pattern_ops'));
1979          $this->assertTrue($dbman->index_exists($table, $index));
1980  
1981          // Try unique indexes too.
1982          $dbman->drop_table($this->tables[$tablename]);
1983  
1984          $table = new xmldb_table('testtable');
1985          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
1986          $table->add_field('path', XMLDB_TYPE_CHAR, 255, null, XMLDB_NOTNULL, null);
1987          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
1988          $table->add_index('path', XMLDB_INDEX_UNIQUE, array('path'), array('varchar_pattern_ops'));
1989          $dbman->create_table($table);
1990          $this->tables[$tablename] = $table;
1991  
1992          $table = new xmldb_table('testtable');
1993          $index = new xmldb_index('path', XMLDB_INDEX_UNIQUE, array('path'), array('varchar_pattern_ops'));
1994          $this->assertTrue($dbman->index_exists($table, $index));
1995      }
1996  
1997      public function test_index_max_bytes() {
1998          $DB = $this->tdb;
1999          $dbman = $DB->get_manager();
2000  
2001          $maxstr = '';
2002          for ($i=0; $i<255; $i++) {
2003              $maxstr .= '言'; // Random long string that should fix exactly the limit for one char column.
2004          }
2005  
2006          $table = new xmldb_table('testtable');
2007          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
2008          $table->add_field('name', XMLDB_TYPE_CHAR, 255, null, XMLDB_NOTNULL, null);
2009          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
2010          $table->add_index('name', XMLDB_INDEX_NOTUNIQUE, array('name'));
2011  
2012          // Drop if exists.
2013          if ($dbman->table_exists($table)) {
2014              $dbman->drop_table($table);
2015          }
2016          $dbman->create_table($table);
2017          $tablename = $table->getName();
2018          $this->tables[$tablename] = $table;
2019  
2020          $rec = new \stdClass();
2021          $rec->name = $maxstr;
2022  
2023          $id = $DB->insert_record($tablename, $rec);
2024          $this->assertTrue(!empty($id));
2025  
2026          $rec = $DB->get_record($tablename, array('id'=>$id));
2027          $this->assertSame($maxstr, $rec->name);
2028  
2029          $dbman->drop_table($table);
2030  
2031          $table = new xmldb_table('testtable');
2032          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
2033          $table->add_field('name', XMLDB_TYPE_CHAR, 255+1, null, XMLDB_NOTNULL, null);
2034          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
2035          $table->add_index('name', XMLDB_INDEX_NOTUNIQUE, array('name'));
2036  
2037          try {
2038              $dbman->create_table($table);
2039              $this->fail('Exception expected');
2040          } catch (\moodle_exception $e) {
2041              $this->assertInstanceOf('coding_exception', $e);
2042          }
2043      }
2044  
2045      public function test_index_composed_max_bytes() {
2046          $DB = $this->tdb;
2047          $dbman = $DB->get_manager();
2048  
2049          $maxstr = '';
2050          for ($i=0; $i<200; $i++) {
2051              $maxstr .= '言';
2052          }
2053          $reststr = '';
2054          for ($i=0; $i<133; $i++) {
2055              $reststr .= '言';
2056          }
2057  
2058          $table = new xmldb_table('testtable');
2059          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
2060          $table->add_field('name1', XMLDB_TYPE_CHAR, 200, null, XMLDB_NOTNULL, null);
2061          $table->add_field('name2', XMLDB_TYPE_CHAR, 133, null, XMLDB_NOTNULL, null);
2062          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
2063          $table->add_index('name1-name2', XMLDB_INDEX_NOTUNIQUE, array('name1', 'name2'));
2064  
2065          // Drop if exists.
2066          if ($dbman->table_exists($table)) {
2067              $dbman->drop_table($table);
2068          }
2069          $dbman->create_table($table);
2070          $tablename = $table->getName();
2071          $this->tables[$tablename] = $table;
2072  
2073          $rec = new \stdClass();
2074          $rec->name1 = $maxstr;
2075          $rec->name2 = $reststr;
2076  
2077          $id = $DB->insert_record($tablename, $rec);
2078          $this->assertTrue(!empty($id));
2079  
2080          $rec = $DB->get_record($tablename, array('id'=>$id));
2081          $this->assertSame($maxstr, $rec->name1);
2082          $this->assertSame($reststr, $rec->name2);
2083  
2084          $table = new xmldb_table('testtable');
2085          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
2086          $table->add_field('name1', XMLDB_TYPE_CHAR, 201, null, XMLDB_NOTNULL, null);
2087          $table->add_field('name2', XMLDB_TYPE_CHAR, 133, null, XMLDB_NOTNULL, null);
2088          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
2089          $table->add_index('name1-name2', XMLDB_INDEX_NOTUNIQUE, array('name1', 'name2'));
2090  
2091          // Drop if exists.
2092          if ($dbman->table_exists($table)) {
2093              $dbman->drop_table($table);
2094          }
2095  
2096          try {
2097              $dbman->create_table($table);
2098              $this->fail('Exception expected');
2099          } catch (\moodle_exception $e) {
2100              $this->assertInstanceOf('coding_exception', $e);
2101          }
2102      }
2103  
2104      public function test_char_size_limit() {
2105          $DB = $this->tdb;
2106          $dbman = $DB->get_manager();
2107  
2108          $table = new xmldb_table('testtable');
2109          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
2110          $table->add_field('name', XMLDB_TYPE_CHAR, xmldb_field::CHAR_MAX_LENGTH, null, XMLDB_NOTNULL, null);
2111          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
2112  
2113          // Drop if exists.
2114          if ($dbman->table_exists($table)) {
2115              $dbman->drop_table($table);
2116          }
2117          $dbman->create_table($table);
2118          $tablename = $table->getName();
2119          $this->tables[$tablename] = $table;
2120  
2121          // This has to work in all DBs.
2122          $maxstr = '';
2123          for ($i=0; $i<xmldb_field::CHAR_MAX_LENGTH; $i++) {
2124              $maxstr .= 'a'; // Ascii only.
2125          }
2126  
2127          $rec = new \stdClass();
2128          $rec->name = $maxstr;
2129  
2130          $id = $DB->insert_record($tablename, $rec);
2131          $this->assertTrue(!empty($id));
2132  
2133          $rec = $DB->get_record($tablename, array('id'=>$id));
2134          $this->assertSame($maxstr, $rec->name);
2135  
2136          // Following test is supposed to fail in oracle.
2137          $maxstr = '';
2138          for ($i=0; $i<xmldb_field::CHAR_MAX_LENGTH; $i++) {
2139              $maxstr .= '言'; // Random long string that should fix exactly the limit for one char column.
2140          }
2141  
2142          $rec = new \stdClass();
2143          $rec->name = $maxstr;
2144  
2145          try {
2146              $id = $DB->insert_record($tablename, $rec);
2147              $this->assertTrue(!empty($id));
2148  
2149              $rec = $DB->get_record($tablename, array('id'=>$id));
2150              $this->assertSame($maxstr, $rec->name);
2151          } catch (dml_exception $e) {
2152              if ($DB->get_dbfamily() === 'oracle') {
2153                  $this->fail('Oracle does not support text fields larger than 4000 bytes, this is not a big problem for mostly ascii based languages');
2154              } else {
2155                  throw $e;
2156              }
2157          }
2158  
2159          $table = new xmldb_table('testtable');
2160          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
2161          $table->add_field('name', XMLDB_TYPE_CHAR, xmldb_field::CHAR_MAX_LENGTH+1, null, XMLDB_NOTNULL, null);
2162          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
2163  
2164          // Drop if exists.
2165          if ($dbman->table_exists($table)) {
2166              $dbman->drop_table($table);
2167          }
2168          $tablename = $table->getName();
2169          $this->tables[$tablename] = $table;
2170  
2171          try {
2172              $dbman->create_table($table);
2173              $this->fail('Exception expected');
2174          } catch (\moodle_exception $e) {
2175              $this->assertInstanceOf('coding_exception', $e);
2176          }
2177      }
2178  
2179      public function test_object_name() {
2180          $gen = $this->tdb->get_manager()->generator;
2181  
2182          // This will form short object name and max length should not be exceeded.
2183          $table = 'tablename';
2184          $fields = 'id';
2185          $suffix = 'pk';
2186          for ($i=0; $i<12; $i++) {
2187              $this->assertLessThanOrEqual($gen->names_max_length,
2188                      strlen($gen->getNameForObject($table, $fields, $suffix)),
2189                      'Generated object name is too long. $i = '.$i);
2190          }
2191  
2192          // This will form too long object name always and it must be trimmed to exactly 30 chars.
2193          $table = 'aaaa_bbbb_cccc_dddd_eeee_ffff_gggg';
2194          $fields = 'aaaaa,bbbbb,ccccc,ddddd';
2195          $suffix = 'idx';
2196          for ($i=0; $i<12; $i++) {
2197              $this->assertEquals($gen->names_max_length,
2198                      strlen($gen->getNameForObject($table, $fields, $suffix)),
2199                      'Generated object name is too long. $i = '.$i);
2200          }
2201  
2202          // Same test without suffix.
2203          $table = 'bbbb_cccc_dddd_eeee_ffff_gggg_hhhh';
2204          $fields = 'aaaaa,bbbbb,ccccc,ddddd';
2205          $suffix = '';
2206          for ($i=0; $i<12; $i++) {
2207              $this->assertEquals($gen->names_max_length,
2208                      strlen($gen->getNameForObject($table, $fields, $suffix)),
2209                      'Generated object name is too long. $i = '.$i);
2210          }
2211  
2212          // This must only trim name when counter is 10 or more.
2213          $table = 'cccc_dddd_eeee_ffff_gggg_hhhh_iiii';
2214          $fields = 'id';
2215          $suffix = 'idx';
2216          // Since we don't know how long prefix is, loop to generate tablename that gives exactly maxlengh-1 length.
2217          // Skip this test if prefix is too long.
2218          while (strlen($table) && strlen($gen->prefix.preg_replace('/_/','',$table).'_id_'.$suffix) >= $gen->names_max_length) {
2219              $table = rtrim(substr($table, 0, strlen($table) - 1), '_');
2220          }
2221          if (strlen($table)) {
2222              $this->assertEquals($gen->names_max_length - 1,
2223                          strlen($gen->getNameForObject($table, $fields, $suffix)));
2224              for ($i=0; $i<12; $i++) {
2225                  $this->assertEquals($gen->names_max_length,
2226                          strlen($gen->getNameForObject($table, $fields, $suffix)),
2227                          'Generated object name is too long. $i = '.$i);
2228              }
2229  
2230              // Now test to confirm that a duplicate name isn't issued, even if they come from different root names.
2231              // Move to a new field.
2232              $fields = "fl";
2233  
2234              // Insert twice, moving is to a key with fl2.
2235              $this->assertEquals($gen->names_max_length - 1, strlen($gen->getNameForObject($table, $fields, $suffix)));
2236              $result1 = $gen->getNameForObject($table, $fields, $suffix);
2237  
2238              // Make sure we end up with _fl2_ in the result.
2239              $this->assertMatchesRegularExpression('/_fl2_/', $result1);
2240  
2241              // Now, use a field that would result in the same key if it wasn't already taken.
2242              $fields = "fl2";
2243              // Because we are now at the max key length, it will try:
2244              // - _fl2_ (the natural name)
2245              // - _fl2_ (removing the original 2, and adding a counter 2)
2246              // - then settle on _fl3_.
2247              $result2 = $gen->getNameForObject($table, $fields, $suffix);
2248              $this->assertMatchesRegularExpression('/_fl3_/', $result2);
2249  
2250              // Make sure they don't match.
2251              $this->assertNotEquals($result1, $result2);
2252              // But are only different in the way we expect. This confirms the test is working properly.
2253              $this->assertEquals(str_replace('_fl2_', '', $result1), str_replace('_fl3_', '', $result2));
2254  
2255              // Now go back. We would expect the next result to be fl3 again, but it is taken, so it should move to fl4.
2256              $fields = "fl";
2257              $result3 = $gen->getNameForObject($table, $fields, $suffix);
2258  
2259              $this->assertNotEquals($result2, $result3);
2260              $this->assertMatchesRegularExpression('/_fl4_/', $result3);
2261          }
2262      }
2263  
2264      /**
2265       * Data provider for test_get_enc_quoted().
2266       *
2267       * @return array The type-value pair fixture.
2268       */
2269      public function get_enc_quoted_provider() {
2270          return array(
2271              // Reserved: some examples from SQL-92.
2272              [true, 'from'],
2273              [true, 'table'],
2274              [true, 'where'],
2275              // Not reserved.
2276              [false, 'my_awesome_column_name']
2277          );
2278      }
2279  
2280      /**
2281       * This is a test for sql_generator::getEncQuoted().
2282       *
2283       * @dataProvider get_enc_quoted_provider
2284       * @param bool $reserved Whether the column name is reserved or not.
2285       * @param string $columnname The column name to be quoted, according to the value of $reserved.
2286       **/
2287      public function test_get_enc_quoted($reserved, $columnname) {
2288          $DB = $this->tdb;
2289          $gen = $DB->get_manager()->generator;
2290  
2291          if (!$reserved) {
2292              // No need to quote the column name.
2293              $this->assertSame($columnname, $gen->getEncQuoted($columnname));
2294          } else {
2295              // Column name should be quoted.
2296              $dbfamily = $DB->get_dbfamily();
2297  
2298              switch ($dbfamily) {
2299                  case 'mysql':
2300                      $this->assertSame("`$columnname`", $gen->getEncQuoted($columnname));
2301                      break;
2302                  case 'mssql': // The Moodle connection runs under 'QUOTED_IDENTIFIER ON'.
2303                  case 'oracle':
2304                  case 'postgres':
2305                  case 'sqlite':
2306                  default:
2307                      $this->assertSame('"' . $columnname . '"', $gen->getEncQuoted($columnname));
2308                      break;
2309              }
2310          }
2311      }
2312  
2313      /**
2314       * Data provider for test_sql_generator_get_rename_field_sql().
2315       *
2316       * @return array The type-old-new tuple fixture.
2317       */
2318      public function sql_generator_get_rename_field_sql_provider() {
2319          return array(
2320              // Reserved: an example from SQL-92.
2321              // Both names should be reserved.
2322              [true, 'from', 'where'],
2323              // Not reserved.
2324              [false, 'my_old_column_name', 'my_awesome_column_name']
2325          );
2326      }
2327  
2328      /**
2329       * This is a unit test for sql_generator::getRenameFieldSQL().
2330       *
2331       * @dataProvider sql_generator_get_rename_field_sql_provider
2332       * @param bool $reserved Whether the column name is reserved or not.
2333       * @param string $oldcolumnname The column name to be renamed.
2334       * @param string $newcolumnname The new column name.
2335       **/
2336      public function test_sql_generator_get_rename_field_sql($reserved, $oldcolumnname, $newcolumnname) {
2337          $DB = $this->tdb;
2338          $gen = $DB->get_manager()->generator;
2339          $prefix = $DB->get_prefix();
2340  
2341          $tablename = 'test_get_rename_field_sql';
2342          $table = new xmldb_table($tablename);
2343          $field = new xmldb_field($oldcolumnname, XMLDB_TYPE_INTEGER, '11', null, XMLDB_NOTNULL, null, null, null, '0', 'previous');
2344  
2345          $dbfamily = $DB->get_dbfamily();
2346          if (!$reserved) {
2347              // No need to quote the column name.
2348              switch ($dbfamily) {
2349                  case 'mysql':
2350                      $this->assertSame(
2351                          [ "ALTER TABLE {$prefix}$tablename CHANGE $oldcolumnname $newcolumnname BIGINT(11) NOT NULL" ],
2352                          $gen->getRenameFieldSQL($table, $field, $newcolumnname)
2353                      );
2354                      break;
2355                  case 'sqlite':
2356                      // Skip it, since the DB is not supported yet.
2357                      // BTW renaming a column name is already covered by the integration test 'testRenameField'.
2358                      break;
2359                  case 'mssql': // The Moodle connection runs under 'QUOTED_IDENTIFIER ON'.
2360                      $this->assertSame(
2361                          [ "sp_rename '{$prefix}$tablename.[$oldcolumnname]', '$newcolumnname', 'COLUMN'" ],
2362                          $gen->getRenameFieldSQL($table, $field, $newcolumnname)
2363                      );
2364                      break;
2365                  case 'oracle':
2366                  case 'postgres':
2367                  default:
2368                      $this->assertSame(
2369                          [ "ALTER TABLE {$prefix}$tablename RENAME COLUMN $oldcolumnname TO $newcolumnname" ],
2370                          $gen->getRenameFieldSQL($table, $field, $newcolumnname)
2371                      );
2372                      break;
2373              }
2374          } else {
2375              // Column name should be quoted.
2376              switch ($dbfamily) {
2377                  case 'mysql':
2378                      $this->assertSame(
2379                          [ "ALTER TABLE {$prefix}$tablename CHANGE `$oldcolumnname` `$newcolumnname` BIGINT(11) NOT NULL" ],
2380                          $gen->getRenameFieldSQL($table, $field, $newcolumnname)
2381                      );
2382                      break;
2383                  case 'sqlite':
2384                      // Skip it, since the DB is not supported yet.
2385                      // BTW renaming a column name is already covered by the integration test 'testRenameField'.
2386                  break;
2387                  case 'mssql': // The Moodle connection runs under 'QUOTED_IDENTIFIER ON'.
2388                      $this->assertSame(
2389                          [ "sp_rename '{$prefix}$tablename.[$oldcolumnname]', '$newcolumnname', 'COLUMN'" ],
2390                          $gen->getRenameFieldSQL($table, $field, $newcolumnname)
2391                      );
2392                      break;
2393                  case 'oracle':
2394                  case 'postgres':
2395                  default:
2396                      $this->assertSame(
2397                          [ "ALTER TABLE {$prefix}$tablename RENAME COLUMN \"$oldcolumnname\" TO \"$newcolumnname\"" ],
2398                          $gen->getRenameFieldSQL($table, $field, $newcolumnname)
2399                      );
2400                      break;
2401              }
2402          }
2403      }
2404  
2405      public function test_get_nullable_fields_in_index() {
2406          $DB = $this->tdb;
2407          $gen = $DB->get_manager()->generator;
2408  
2409          $indexwithoutnulls = $this->tables['test_table0']->getIndex('type-name');
2410          $this->assertSame([], $gen->get_nullable_fields_in_index(
2411                  $this->tables['test_table0'], $indexwithoutnulls));
2412  
2413          $indexwithnulls = new xmldb_index('course-grade', XMLDB_INDEX_UNIQUE, ['course', 'grade']);
2414          $this->assertSame(['grade'], $gen->get_nullable_fields_in_index(
2415                  $this->tables['test_table0'], $indexwithnulls));
2416  
2417          $this->create_deftable('test_table0');
2418  
2419          // Now test using a minimal xmldb_table, to ensure we get the data from the DB.
2420          $table = new xmldb_table('test_table0');
2421          $this->assertSame([], $gen->get_nullable_fields_in_index(
2422                  $table, $indexwithoutnulls));
2423          $this->assertSame(['grade'], $gen->get_nullable_fields_in_index(
2424                  $table, $indexwithnulls));
2425      }
2426  
2427      // Following methods are not supported == Do not test.
2428      /*
2429          public function testRenameIndex() {
2430              // Unsupported!
2431              $dbman = $this->tdb->get_manager();
2432  
2433              $table = $this->create_deftable('test_table0');
2434              $index = new xmldb_index('course');
2435              $index->set_attributes(XMLDB_INDEX_UNIQUE, array('course'));
2436  
2437              $this->assertTrue($dbman->rename_index($table, $index, 'newindexname'));
2438          }
2439  
2440          public function testRenameKey() {
2441              // Unsupported!
2442               $dbman = $this->tdb->get_manager();
2443  
2444              $table = $this->create_deftable('test_table0');
2445              $key = new xmldb_key('course');
2446              $key->set_attributes(XMLDB_KEY_UNIQUE, array('course'));
2447  
2448              $this->assertTrue($dbman->rename_key($table, $key, 'newkeyname'));
2449          }
2450      */
2451  
2452      /**
2453       * Tests check_database_schema().
2454       */
2455      public function test_check_database_schema() {
2456          global $CFG, $DB;
2457  
2458          $dbmanager = $DB->get_manager();
2459  
2460          // Create a table in the database we will be using to compare with a schema.
2461          $table = new xmldb_table('test_check_db_schema');
2462          $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null);
2463          $table->add_field('extracolumn', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, null);
2464          $table->add_field('courseid', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, null);
2465          $table->add_field('binaryfield', XMLDB_TYPE_BINARY, null, null, XMLDB_NOTNULL, null, null);
2466          $table->add_key('primary', XMLDB_KEY_PRIMARY, array('id'));
2467          $table->add_key('extraindex', XMLDB_KEY_UNIQUE, array('extracolumn'));
2468          $table->setComment("This is a test table, you can drop it safely.");
2469          $dbmanager->create_table($table);
2470  
2471          // Remove the column so it is not added to the schema and gets reported as an extra column.
2472          $table->deleteField('extracolumn');
2473  
2474          // Change the 'courseid' field to a float in the schema so it gets reported as different.
2475          $table->deleteField('courseid');
2476          $table->add_field('courseid', XMLDB_TYPE_NUMBER, '10, 2', null, XMLDB_NOTNULL, null, null);
2477  
2478          // Add another column to the schema that won't be present in the database and gets reported as missing.
2479          $table->add_field('missingcolumn', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, null);
2480  
2481          // Add another key to the schema that won't be present in the database and gets reported as missing.
2482          $table->add_key('missingkey', XMLDB_KEY_FOREIGN, array('courseid'), 'course', array('id'));
2483  
2484          // Remove the key from the schema which will still be present in the database and reported as extra.
2485          $table->deleteKey('extraindex');
2486  
2487          $schema = new xmldb_structure('testschema');
2488          $schema->addTable($table);
2489  
2490          // Things we want to check for -
2491          // 1. Changed columns.
2492          // 2. Missing columns.
2493          // 3. Missing indexes.
2494          // 4. Unexpected index.
2495          // 5. Extra columns.
2496          $errors = $dbmanager->check_database_schema($schema)['test_check_db_schema'];
2497          // Preprocess $errors to get rid of the non compatible (SQL-dialect dependent) parts.
2498          array_walk($errors, function(&$error) {
2499              $error = trim(strtok($error, PHP_EOL));
2500          });
2501          $this->assertCount(5, $errors);
2502          $this->assertContains("column 'courseid' has incorrect type 'I', expected 'N'", $errors);
2503          $this->assertContains("column 'missingcolumn' is missing", $errors);
2504          $this->assertContains("Missing index 'missingkey' (not unique (courseid)).", $errors);
2505          $this->assertContains("Unexpected index '{$CFG->prefix}testchecdbsche_ext_uix'.", $errors);
2506          $this->assertContains("column 'extracolumn' is not expected (I)", $errors);
2507      }
2508  }