2008-11-10 00:56:24 +01:00
|
|
|
|
<?php
|
2016-06-15 06:03:16 +02:00
|
|
|
|
|
2016-10-14 03:30:05 +02:00
|
|
|
|
namespace SilverStripe\ORM\Tests;
|
|
|
|
|
|
2016-06-15 06:03:16 +02:00
|
|
|
|
use SilverStripe\ORM\DB;
|
|
|
|
|
use SilverStripe\ORM\Connect\MySQLDatabase;
|
2016-06-23 01:37:22 +02:00
|
|
|
|
use SilverStripe\MSSQL\MSSQLDatabase;
|
2016-08-19 00:51:35 +02:00
|
|
|
|
use SilverStripe\Dev\SapphireTest;
|
2016-10-14 03:30:05 +02:00
|
|
|
|
use Exception;
|
|
|
|
|
use SilverStripe\ORM\Tests\DatabaseTest\MyObject;
|
2016-06-23 01:37:22 +02:00
|
|
|
|
|
2016-12-16 05:34:21 +01:00
|
|
|
|
/**
|
|
|
|
|
* @skipUpgrade
|
|
|
|
|
*/
|
|
|
|
|
class DatabaseTest extends SapphireTest
|
|
|
|
|
{
|
|
|
|
|
|
2020-04-20 19:58:09 +02:00
|
|
|
|
protected static $extra_dataobjects = [
|
2016-12-16 05:34:21 +01:00
|
|
|
|
MyObject::class,
|
2020-04-20 19:58:09 +02:00
|
|
|
|
];
|
2016-12-16 05:34:21 +01:00
|
|
|
|
|
|
|
|
|
protected $usesDatabase = true;
|
|
|
|
|
|
|
|
|
|
public function testDontRequireField()
|
|
|
|
|
{
|
|
|
|
|
$schema = DB::get_schema();
|
|
|
|
|
$this->assertArrayHasKey(
|
|
|
|
|
'MyField',
|
|
|
|
|
$schema->fieldList('DatabaseTest_MyObject')
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$schema->dontRequireField('DatabaseTest_MyObject', 'MyField');
|
|
|
|
|
|
|
|
|
|
$this->assertArrayHasKey(
|
|
|
|
|
'_obsolete_MyField',
|
|
|
|
|
$schema->fieldList('DatabaseTest_MyObject'),
|
|
|
|
|
'Field is renamed to _obsolete_<fieldname> through dontRequireField()'
|
|
|
|
|
);
|
|
|
|
|
|
2017-03-24 12:17:26 +01:00
|
|
|
|
static::resetDBSchema(true);
|
2016-12-16 05:34:21 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testRenameField()
|
|
|
|
|
{
|
|
|
|
|
$schema = DB::get_schema();
|
|
|
|
|
|
|
|
|
|
$schema->clearCachedFieldlist();
|
|
|
|
|
|
|
|
|
|
$schema->renameField('DatabaseTest_MyObject', 'MyField', 'MyRenamedField');
|
|
|
|
|
|
|
|
|
|
$this->assertArrayHasKey(
|
|
|
|
|
'MyRenamedField',
|
|
|
|
|
$schema->fieldList('DatabaseTest_MyObject'),
|
|
|
|
|
'New fieldname is set through renameField()'
|
|
|
|
|
);
|
|
|
|
|
$this->assertArrayNotHasKey(
|
|
|
|
|
'MyField',
|
|
|
|
|
$schema->fieldList('DatabaseTest_MyObject'),
|
|
|
|
|
'Old fieldname isnt preserved through renameField()'
|
|
|
|
|
);
|
|
|
|
|
|
2017-03-24 12:17:26 +01:00
|
|
|
|
static::resetDBSchema(true);
|
2016-12-16 05:34:21 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testMySQLCreateTableOptions()
|
|
|
|
|
{
|
|
|
|
|
if (!(DB::get_conn() instanceof MySQLDatabase)) {
|
|
|
|
|
$this->markTestSkipped('MySQL only');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$ret = DB::query(
|
|
|
|
|
sprintf(
|
|
|
|
|
'SHOW TABLE STATUS WHERE "Name" = \'%s\'',
|
|
|
|
|
'DatabaseTest_MyObject'
|
|
|
|
|
)
|
|
|
|
|
)->first();
|
|
|
|
|
$this->assertEquals(
|
|
|
|
|
$ret['Engine'],
|
|
|
|
|
'InnoDB',
|
|
|
|
|
"MySQLDatabase tables can be changed to InnoDB through DataObject::\$create_table_options"
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function testIsSchemaUpdating()
|
|
|
|
|
{
|
|
|
|
|
$schema = DB::get_schema();
|
|
|
|
|
|
|
|
|
|
$this->assertFalse($schema->isSchemaUpdating(), 'Before the transaction the flag is false.');
|
|
|
|
|
|
|
|
|
|
// Test complete schema update
|
|
|
|
|
$test = $this;
|
|
|
|
|
$schema->schemaUpdate(
|
|
|
|
|
function () use ($test, $schema) {
|
|
|
|
|
$test->assertTrue($schema->isSchemaUpdating(), 'During the transaction the flag is true.');
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
$this->assertFalse($schema->isSchemaUpdating(), 'After the transaction the flag is false.');
|
|
|
|
|
|
|
|
|
|
// Test cancelled schema update
|
|
|
|
|
$schema->schemaUpdate(
|
|
|
|
|
function () use ($test, $schema) {
|
|
|
|
|
$schema->cancelSchemaUpdate();
|
|
|
|
|
$test->assertFalse($schema->doesSchemaNeedUpdating(), 'After cancelling the transaction the flag is false');
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testSchemaUpdateChecking()
|
|
|
|
|
{
|
|
|
|
|
$schema = DB::get_schema();
|
|
|
|
|
|
|
|
|
|
// Initially, no schema changes necessary
|
|
|
|
|
$test = $this;
|
|
|
|
|
$schema->schemaUpdate(
|
|
|
|
|
function () use ($test, $schema) {
|
|
|
|
|
$test->assertFalse($schema->doesSchemaNeedUpdating());
|
|
|
|
|
|
|
|
|
|
// If we make a change, then the schema will need updating
|
|
|
|
|
$schema->transCreateTable("TestTable");
|
|
|
|
|
$test->assertTrue($schema->doesSchemaNeedUpdating());
|
|
|
|
|
|
|
|
|
|
// If we make cancel the change, then schema updates are no longer necessary
|
|
|
|
|
$schema->cancelSchemaUpdate();
|
|
|
|
|
$test->assertFalse($schema->doesSchemaNeedUpdating());
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testHasTable()
|
|
|
|
|
{
|
|
|
|
|
$this->assertTrue(DB::get_schema()->hasTable('DatabaseTest_MyObject'));
|
|
|
|
|
$this->assertFalse(DB::get_schema()->hasTable('asdfasdfasdf'));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testGetAndReleaseLock()
|
|
|
|
|
{
|
|
|
|
|
$db = DB::get_conn();
|
|
|
|
|
|
|
|
|
|
if (!$db->supportsLocks()) {
|
|
|
|
|
return $this->markTestSkipped('Tested database doesn\'t support application locks');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
|
$db->getLock('DatabaseTest'),
|
2021-12-13 09:05:33 +01:00
|
|
|
|
'Can acquire lock'
|
2016-12-16 05:34:21 +01:00
|
|
|
|
);
|
2021-12-13 09:05:33 +01:00
|
|
|
|
// $this->assertFalse($db->getLock('DatabaseTest'), 'Can\'t repeatedly acquire the same lock');
|
2016-12-16 05:34:21 +01:00
|
|
|
|
$this->assertTrue(
|
|
|
|
|
$db->getLock('DatabaseTest'),
|
2021-12-13 09:05:33 +01:00
|
|
|
|
'The same lock can be acquired multiple times in the same connection'
|
2016-12-16 05:34:21 +01:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
|
$db->getLock('DatabaseTestOtherLock'),
|
2021-12-13 09:05:33 +01:00
|
|
|
|
'Can acquire different lock'
|
2016-12-16 05:34:21 +01:00
|
|
|
|
);
|
|
|
|
|
$db->releaseLock('DatabaseTestOtherLock');
|
|
|
|
|
|
|
|
|
|
// Release potentially stacked locks from previous getLock() invocations
|
|
|
|
|
$db->releaseLock('DatabaseTest');
|
|
|
|
|
$db->releaseLock('DatabaseTest');
|
|
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
|
$db->getLock('DatabaseTest'),
|
2021-12-13 09:05:33 +01:00
|
|
|
|
'Can acquire lock after releasing it'
|
2016-12-16 05:34:21 +01:00
|
|
|
|
);
|
|
|
|
|
$db->releaseLock('DatabaseTest');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testCanLock()
|
|
|
|
|
{
|
|
|
|
|
$db = DB::get_conn();
|
|
|
|
|
|
|
|
|
|
if (!$db->supportsLocks()) {
|
|
|
|
|
return $this->markTestSkipped('Database doesn\'t support locks');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($db instanceof MSSQLDatabase) {
|
|
|
|
|
return $this->markTestSkipped('MSSQLDatabase doesn\'t support inspecting locks');
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-13 09:05:33 +01:00
|
|
|
|
$this->assertTrue($db->canLock('DatabaseTest'), 'Can lock before first acquiring one');
|
2016-12-16 05:34:21 +01:00
|
|
|
|
$db->getLock('DatabaseTest');
|
2021-12-13 09:05:33 +01:00
|
|
|
|
$this->assertFalse($db->canLock('DatabaseTest'), 'Can\'t lock after acquiring one');
|
2016-12-16 05:34:21 +01:00
|
|
|
|
$db->releaseLock('DatabaseTest');
|
|
|
|
|
$this->assertTrue($db->canLock('DatabaseTest'), 'Can lock again after releasing it');
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-04 05:13:32 +02:00
|
|
|
|
public function testFieldTypes()
|
|
|
|
|
{
|
|
|
|
|
// Scaffold some data
|
|
|
|
|
$obj = new MyObject();
|
|
|
|
|
$obj->MyField = "value";
|
|
|
|
|
$obj->MyInt = 5;
|
|
|
|
|
$obj->MyFloat = 6.0;
|
2018-11-10 00:25:16 +01:00
|
|
|
|
|
|
|
|
|
// Note: in non-PDO SQLite, whole numbers of a decimal field will be returned as integers rather than floats
|
|
|
|
|
$obj->MyDecimal = 7.1;
|
2018-10-04 05:13:32 +02:00
|
|
|
|
$obj->MyBoolean = true;
|
|
|
|
|
$obj->write();
|
|
|
|
|
|
|
|
|
|
$record = DB::prepared_query(
|
|
|
|
|
'SELECT * FROM "DatabaseTest_MyObject" WHERE "ID" = ?',
|
|
|
|
|
[ $obj->ID ]
|
|
|
|
|
)->record();
|
|
|
|
|
|
|
|
|
|
// IDs and ints are returned as ints
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsInt($record['ID'], 'Primary key should be integer');
|
|
|
|
|
$this->assertIsInt($record['MyInt'], 'DBInt fields should be integer');
|
2018-11-10 00:25:16 +01:00
|
|
|
|
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsFloat($record['MyFloat'], 'DBFloat fields should be float');
|
|
|
|
|
$this->assertIsFloat($record['MyDecimal'], 'DBDecimal fields should be float');
|
2018-11-10 00:25:16 +01:00
|
|
|
|
|
|
|
|
|
// Booleans are returned as ints – we follow MySQL's lead
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsInt($record['MyBoolean'], 'DBBoolean fields should be int');
|
2018-11-10 00:25:16 +01:00
|
|
|
|
|
|
|
|
|
// Strings and enums are returned as strings
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsString($record['MyField'], 'DBVarchar fields should be string');
|
|
|
|
|
$this->assertIsString($record['ClassName'], 'DBEnum fields should be string');
|
2018-11-10 00:25:16 +01:00
|
|
|
|
|
|
|
|
|
// Dates are returned as strings
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsString($record['Created'], 'DBDatetime fields should be string');
|
2018-11-10 00:25:16 +01:00
|
|
|
|
|
2019-03-08 04:15:49 +01:00
|
|
|
|
|
|
|
|
|
// Ensure that the same is true when calling a query a second time (cached prepared statement)
|
|
|
|
|
|
|
|
|
|
$record = DB::prepared_query(
|
|
|
|
|
'SELECT * FROM "DatabaseTest_MyObject" WHERE "ID" = ?',
|
|
|
|
|
[ $obj->ID ]
|
|
|
|
|
)->record();
|
|
|
|
|
|
|
|
|
|
// IDs and ints are returned as ints
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsInt($record['ID'], 'Primary key should be integer (2nd call)');
|
|
|
|
|
$this->assertIsInt($record['MyInt'], 'DBInt fields should be integer (2nd call)');
|
2019-03-08 04:15:49 +01:00
|
|
|
|
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsFloat($record['MyFloat'], 'DBFloat fields should be float (2nd call)');
|
|
|
|
|
$this->assertIsFloat($record['MyDecimal'], 'DBDecimal fields should be float (2nd call)');
|
2019-03-08 04:15:49 +01:00
|
|
|
|
|
|
|
|
|
// Booleans are returned as ints – we follow MySQL's lead
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsInt($record['MyBoolean'], 'DBBoolean fields should be int (2nd call)');
|
2019-03-08 04:15:49 +01:00
|
|
|
|
|
|
|
|
|
// Strings and enums are returned as strings
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsString($record['MyField'], 'DBVarchar fields should be string (2nd call)');
|
|
|
|
|
$this->assertIsString($record['ClassName'], 'DBEnum fields should be string (2nd call)');
|
2019-03-08 04:15:49 +01:00
|
|
|
|
|
|
|
|
|
// Dates are returned as strings
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsString($record['Created'], 'DBDatetime fields should be string (2nd call)');
|
2019-03-08 04:15:49 +01:00
|
|
|
|
|
|
|
|
|
|
2018-11-10 00:25:16 +01:00
|
|
|
|
// Ensure that the same is true when using non-prepared statements
|
|
|
|
|
$record = DB::query('SELECT * FROM "DatabaseTest_MyObject" WHERE "ID" = ' . (int)$obj->ID)->record();
|
|
|
|
|
|
|
|
|
|
// IDs and ints are returned as ints
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsInt($record['ID'], 'Primary key should be integer (non-prepared)');
|
|
|
|
|
$this->assertIsInt($record['MyInt'], 'DBInt fields should be integer (non-prepared)');
|
2018-10-04 05:13:32 +02:00
|
|
|
|
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsFloat($record['MyFloat'], 'DBFloat fields should be float (non-prepared)');
|
|
|
|
|
$this->assertIsFloat($record['MyDecimal'], 'DBDecimal fields should be float (non-prepared)');
|
2018-10-04 05:13:32 +02:00
|
|
|
|
|
|
|
|
|
// Booleans are returned as ints – we follow MySQL's lead
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsInt($record['MyBoolean'], 'DBBoolean fields should be int (non-prepared)');
|
2018-10-04 05:13:32 +02:00
|
|
|
|
|
|
|
|
|
// Strings and enums are returned as strings
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsString($record['MyField'], 'DBVarchar fields should be string (non-prepared)');
|
|
|
|
|
$this->assertIsString($record['ClassName'], 'DBEnum fields should be string (non-prepared)');
|
2018-10-04 05:13:32 +02:00
|
|
|
|
|
|
|
|
|
// Dates are returned as strings
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsString($record['Created'], 'DBDatetime fields should be string (non-prepared)');
|
2019-04-17 05:15:17 +02:00
|
|
|
|
|
|
|
|
|
// Booleans selected directly are ints
|
|
|
|
|
$result = DB::query('SELECT TRUE')->first();
|
2021-10-27 04:39:47 +02:00
|
|
|
|
$this->assertIsInt(reset($result));
|
2018-10-04 05:13:32 +02:00
|
|
|
|
}
|
2019-06-28 00:57:51 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Test that repeated iteration of a query returns all records.
|
|
|
|
|
* See https://github.com/silverstripe/silverstripe-framework/issues/9097
|
|
|
|
|
*/
|
|
|
|
|
public function testRepeatedIteration()
|
|
|
|
|
{
|
|
|
|
|
$inputData = ['one', 'two', 'three', 'four'];
|
|
|
|
|
|
|
|
|
|
foreach ($inputData as $i => $text) {
|
|
|
|
|
$x = new MyObject();
|
|
|
|
|
$x->MyField = $text;
|
|
|
|
|
$x->MyInt = $i;
|
|
|
|
|
$x->write();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$query = DB::query('SELECT "MyInt", "MyField" FROM "DatabaseTest_MyObject" ORDER BY "MyInt"');
|
|
|
|
|
|
|
|
|
|
$this->assertEquals($inputData, $query->map());
|
|
|
|
|
$this->assertEquals($inputData, $query->map());
|
|
|
|
|
}
|
2008-11-10 00:56:24 +01:00
|
|
|
|
}
|