Commit 063c536d authored by Benjamin Eberlei's avatar Benjamin Eberlei

DBAL-91 - Fix auto-generation of asset names (foreign keys and indexes) which...

DBAL-91 - Fix auto-generation of asset names (foreign keys and indexes) which could lead to exceptions due to auto-quoting of PostgreSQL for example. Now hashing the names with crc32+dechex
parent 1f09b3fe
...@@ -51,19 +51,35 @@ abstract class AbstractAsset ...@@ -51,19 +51,35 @@ abstract class AbstractAsset
*/ */
protected function _setName($name) protected function _setName($name)
{ {
if (strlen($name)) { if ($this->isQuoted($name)) {
// TODO: find more elegant way to solve this issue. $this->_quoted = true;
if ($name[0] == '`') { $name = $this->trimQuotes($name);
$this->_quoted = true;
$name = trim($name, '`');
} else if ($name[0] == '"') {
$this->_quoted = true;
$name = trim($name, '"');
}
} }
$this->_name = $name; $this->_name = $name;
} }
/**
* Check if this identifier is quoted.
*
* @param string $identifier
* @return bool
*/
protected function isQuoted($identifier)
{
return (isset($identifier[0]) && ($identifier[0] == '`' || $identifier[0] == '"'));
}
/**
* Trim quotes from the identifier.
*
* @param string $identifier
* @return string
*/
protected function trimQuotes($identifier)
{
return trim($identifier, '`"');
}
/** /**
* Return name of this schema asset. * Return name of this schema asset.
* *
...@@ -94,13 +110,13 @@ abstract class AbstractAsset ...@@ -94,13 +110,13 @@ abstract class AbstractAsset
* very long names. * very long names.
* *
* @param array $columnNames * @param array $columnNames
* @param string $postfix * @param string $prefix
* @param int $maxSize * @param int $maxSize
* @return string * @return string
*/ */
protected function _generateIdentifierName($columnNames, $postfix='', $maxSize=30) protected function _generateIdentifierName($columnNames, $prefix='', $maxSize=30)
{ {
$columnCount = count($columnNames); /*$columnCount = count($columnNames);
$postfixLen = strlen($postfix); $postfixLen = strlen($postfix);
$parts = array_map(function($columnName) use($columnCount, $postfixLen, $maxSize) { $parts = array_map(function($columnName) use($columnCount, $postfixLen, $maxSize) {
return substr($columnName, -floor(($maxSize-$postfixLen)/$columnCount - 1)); return substr($columnName, -floor(($maxSize-$postfixLen)/$columnCount - 1));
...@@ -116,6 +132,12 @@ abstract class AbstractAsset ...@@ -116,6 +132,12 @@ abstract class AbstractAsset
$identifier = "i" . substr($identifier, 0, strlen($identifier)-1); $identifier = "i" . substr($identifier, 0, strlen($identifier)-1);
} }
return $identifier; return $identifier;*/
$hash = implode("", array_map(function($column) {
return dechex(crc32($column));
}, $columnNames));
return substr(strtoupper($prefix . "_" . $hash), 0, $maxSize);
} }
} }
\ No newline at end of file
...@@ -46,6 +46,8 @@ class PostgreSqlSchemaManager extends AbstractSchemaManager ...@@ -46,6 +46,8 @@ class PostgreSqlSchemaManager extends AbstractSchemaManager
} }
if (preg_match('/FOREIGN KEY \((.+)\) REFERENCES (.+)\((.+)\)/', $tableForeignKey['condef'], $values)) { if (preg_match('/FOREIGN KEY \((.+)\) REFERENCES (.+)\((.+)\)/', $tableForeignKey['condef'], $values)) {
// PostgreSQL returns identifiers that are keywords with quotes, we need them later, don't get
// the idea to trim them here.
$localColumns = array_map('trim', explode(",", $values[1])); $localColumns = array_map('trim', explode(",", $values[1]));
$foreignColumns = array_map('trim', explode(",", $values[3])); $foreignColumns = array_map('trim', explode(",", $values[3]));
$foreignTable = $values[2]; $foreignTable = $values[2];
......
...@@ -496,7 +496,7 @@ class Table extends AbstractAsset ...@@ -496,7 +496,7 @@ class Table extends AbstractAsset
*/ */
public function hasColumn($columnName) public function hasColumn($columnName)
{ {
$columnName = strtolower($columnName); $columnName = $this->trimQuotes(strtolower($columnName));
return isset($this->_columns[$columnName]); return isset($this->_columns[$columnName]);
} }
...@@ -508,7 +508,7 @@ class Table extends AbstractAsset ...@@ -508,7 +508,7 @@ class Table extends AbstractAsset
*/ */
public function getColumn($columnName) public function getColumn($columnName)
{ {
$columnName = strtolower($columnName); $columnName = strtolower($this->trimQuotes($columnName));
if (!$this->hasColumn($columnName)) { if (!$this->hasColumn($columnName)) {
throw SchemaException::columnDoesNotExist($columnName, $this->_name); throw SchemaException::columnDoesNotExist($columnName, $this->_name);
} }
......
...@@ -132,6 +132,29 @@ class PostgreSqlSchemaManagerTest extends SchemaManagerFunctionalTestCase ...@@ -132,6 +132,29 @@ class PostgreSqlSchemaManagerTest extends SchemaManagerFunctionalTestCase
$relatedFk = array_pop($relatedFks); $relatedFk = array_pop($relatedFks);
$this->assertEquals("nested.schemarelated", $relatedFk->getForeignTableName()); $this->assertEquals("nested.schemarelated", $relatedFk->getForeignTableName());
} }
/**
* @group DBAL-91
* @group DBAL-88
*/
public function testReturnQuotedAssets()
{
$sql = 'create table dbal91_something ( id integer CONSTRAINT id_something PRIMARY KEY NOT NULL ,"table" integer );';
$this->_conn->exec($sql);
$sql = 'ALTER TABLE dbal91_something ADD CONSTRAINT something_input FOREIGN KEY( "table" ) REFERENCES dbal91_something ON UPDATE CASCADE;';
$this->_conn->exec($sql);
$table = $this->_sm->listTableDetails('dbal91_something');
$this->assertEquals(
array(
"CREATE TABLE dbal91_something (id INT NOT NULL, table INT DEFAULT NULL, PRIMARY KEY(id))",
"CREATE INDEX IDX_A9401304ECA7352B ON dbal91_something (\"table\")",
),
$this->_conn->getDatabasePlatform()->getCreateTableSQL($table)
);
}
} }
class MoneyType extends Type class MoneyType extends Type
......
...@@ -24,7 +24,7 @@ class MsSqlPlatformTest extends AbstractPlatformTestCase ...@@ -24,7 +24,7 @@ class MsSqlPlatformTest extends AbstractPlatformTestCase
{ {
return array( return array(
'CREATE TABLE test (foo NVARCHAR(255) DEFAULT NULL, bar NVARCHAR(255) DEFAULT NULL)', 'CREATE TABLE test (foo NVARCHAR(255) DEFAULT NULL, bar NVARCHAR(255) DEFAULT NULL)',
'CREATE UNIQUE INDEX test_foo_bar_uniq ON test (foo, bar) WHERE foo IS NOT NULL AND bar IS NOT NULL' 'CREATE UNIQUE INDEX UNIQ_D87F7E0C8C73652176FF8CAA ON test (foo, bar) WHERE foo IS NOT NULL AND bar IS NOT NULL'
); );
} }
......
...@@ -31,7 +31,7 @@ class MySqlPlatformTest extends AbstractPlatformTestCase ...@@ -31,7 +31,7 @@ class MySqlPlatformTest extends AbstractPlatformTestCase
public function getGenerateTableWithMultiColumnUniqueIndexSql() public function getGenerateTableWithMultiColumnUniqueIndexSql()
{ {
return array( return array(
'CREATE TABLE test (foo VARCHAR(255) DEFAULT NULL, bar VARCHAR(255) DEFAULT NULL, UNIQUE INDEX test_foo_bar_uniq (foo, bar)) ENGINE = InnoDB' 'CREATE TABLE test (foo VARCHAR(255) DEFAULT NULL, bar VARCHAR(255) DEFAULT NULL, UNIQUE INDEX UNIQ_D87F7E0C8C73652176FF8CAA (foo, bar)) ENGINE = InnoDB'
); );
} }
......
...@@ -23,7 +23,7 @@ class OraclePlatformTest extends AbstractPlatformTestCase ...@@ -23,7 +23,7 @@ class OraclePlatformTest extends AbstractPlatformTestCase
{ {
return array( return array(
'CREATE TABLE test (foo VARCHAR2(255) DEFAULT NULL, bar VARCHAR2(255) DEFAULT NULL)', 'CREATE TABLE test (foo VARCHAR2(255) DEFAULT NULL, bar VARCHAR2(255) DEFAULT NULL)',
'CREATE UNIQUE INDEX test_foo_bar_uniq ON test (foo, bar)', 'CREATE UNIQUE INDEX UNIQ_D87F7E0C8C73652176FF8CAA ON test (foo, bar)',
); );
} }
......
...@@ -23,7 +23,7 @@ class PostgreSqlPlatformTest extends AbstractPlatformTestCase ...@@ -23,7 +23,7 @@ class PostgreSqlPlatformTest extends AbstractPlatformTestCase
{ {
return array( return array(
'CREATE TABLE test (foo VARCHAR(255) DEFAULT NULL, bar VARCHAR(255) DEFAULT NULL)', 'CREATE TABLE test (foo VARCHAR(255) DEFAULT NULL, bar VARCHAR(255) DEFAULT NULL)',
'CREATE UNIQUE INDEX test_foo_bar_uniq ON test (foo, bar)' 'CREATE UNIQUE INDEX UNIQ_D87F7E0C8C73652176FF8CAA ON test (foo, bar)'
); );
} }
......
...@@ -23,7 +23,7 @@ class SqlitePlatformTest extends AbstractPlatformTestCase ...@@ -23,7 +23,7 @@ class SqlitePlatformTest extends AbstractPlatformTestCase
{ {
return array( return array(
'CREATE TABLE test (foo VARCHAR(255) DEFAULT NULL, bar VARCHAR(255) DEFAULT NULL)', 'CREATE TABLE test (foo VARCHAR(255) DEFAULT NULL, bar VARCHAR(255) DEFAULT NULL)',
'CREATE UNIQUE INDEX test_foo_bar_uniq ON test (foo, bar)', 'CREATE UNIQUE INDEX UNIQ_D87F7E0C8C73652176FF8CAA ON test (foo, bar)',
); );
} }
......
...@@ -170,14 +170,15 @@ class SchemaTest extends \PHPUnit_Framework_TestCase ...@@ -170,14 +170,15 @@ class SchemaTest extends \PHPUnit_Framework_TestCase
public function testConfigMaxIdentifierLength() public function testConfigMaxIdentifierLength()
{ {
$schemaConfig = new \Doctrine\DBAL\Schema\SchemaConfig(); $schemaConfig = new \Doctrine\DBAL\Schema\SchemaConfig();
$schemaConfig->setMaxIdentifierLength(10); $schemaConfig->setMaxIdentifierLength(5);
$schema = new Schema(array(), array(), $schemaConfig); $schema = new Schema(array(), array(), $schemaConfig);
$table = $schema->createTable("smalltable"); $table = $schema->createTable("smalltable");
$table->addColumn('long_id', 'integer'); $table->addColumn('long_id', 'integer');
$table->addIndex(array('long_id')); $table->addIndex(array('long_id'));
$this->assertTrue($table->hasIndex('le_id_idx')); $index = current($table->getIndexes());
$this->assertEquals(5, strlen($index->getName()));
} }
public function testDeepClone() public function testDeepClone()
......
...@@ -112,8 +112,8 @@ class TableTest extends \PHPUnit_Framework_TestCase ...@@ -112,8 +112,8 @@ class TableTest extends \PHPUnit_Framework_TestCase
$columns = array(new Column("foo", $type), new Column("bar", $type), new Column("baz", $type)); $columns = array(new Column("foo", $type), new Column("bar", $type), new Column("baz", $type));
$table = new Table("foo", $columns); $table = new Table("foo", $columns);
$table->addIndex(array("foo", "bar")); $table->addIndex(array("foo", "bar"), "foo_foo_bar_idx");
$table->addUniqueIndex(array("bar", "baz")); $table->addUniqueIndex(array("bar", "baz"), "foo_bar_baz_uniq");
$this->assertTrue($table->hasIndex("foo_foo_bar_idx")); $this->assertTrue($table->hasIndex("foo_foo_bar_idx"));
$this->assertTrue($table->hasIndex("foo_bar_baz_uniq")); $this->assertTrue($table->hasIndex("foo_bar_baz_uniq"));
...@@ -311,11 +311,12 @@ class TableTest extends \PHPUnit_Framework_TestCase ...@@ -311,11 +311,12 @@ class TableTest extends \PHPUnit_Framework_TestCase
$constraints = $table->getForeignKeys(); $constraints = $table->getForeignKeys();
$this->assertEquals(1, count($constraints)); $this->assertEquals(1, count($constraints));
$this->assertType('Doctrine\DBAL\Schema\ForeignKeyConstraint', $constraints["foo_id_fk"]); $constraint = current($constraints);
$this->assertInstanceOf('Doctrine\DBAL\Schema\ForeignKeyConstraint', $constraint);
$this->assertEquals("foo_id_fk", $constraints["foo_id_fk"]->getName()); $this->assertTrue($constraint->hasOption("foo"));
$this->assertTrue($constraints["foo_id_fk"]->hasOption("foo")); $this->assertEquals("bar", $constraint->getOption("foo"));
$this->assertEquals("bar", $constraints["foo_id_fk"]->getOption("foo"));
} }
public function testAddIndexWithCaseSensitiveColumnProblem() public function testAddIndexWithCaseSensitiveColumnProblem()
...@@ -351,7 +352,7 @@ class TableTest extends \PHPUnit_Framework_TestCase ...@@ -351,7 +352,7 @@ class TableTest extends \PHPUnit_Framework_TestCase
$table->addColumn('baz', 'integer', array()); $table->addColumn('baz', 'integer', array());
$table->addIndex(array('baz')); $table->addIndex(array('baz'));
$this->assertTrue($table->hasIndex('foo_bar_baz_idx')); $this->assertEquals(1, count($table->getIndexes()));
} }
/** /**
...@@ -380,9 +381,12 @@ class TableTest extends \PHPUnit_Framework_TestCase ...@@ -380,9 +381,12 @@ class TableTest extends \PHPUnit_Framework_TestCase
$table->addForeignKeyConstraint($foreignTable, array("id"), array("id"), array("foo" => "bar")); $table->addForeignKeyConstraint($foreignTable, array("id"), array("id"), array("foo" => "bar"));
$this->assertEquals(1, count($table->getIndexes())); $indexes = $table->getIndexes();
$this->assertTrue($table->hasIndex("foo_id_idx")); $this->assertEquals(1, count($indexes));
$this->assertEquals(array('id'), $table->getIndex('foo_id_idx')->getColumns()); $index = current($indexes);
$this->assertTrue($table->hasIndex($index->getName()));
$this->assertEquals(array('id'), $index->getColumns());
} }
/** /**
...@@ -393,13 +397,14 @@ class TableTest extends \PHPUnit_Framework_TestCase ...@@ -393,13 +397,14 @@ class TableTest extends \PHPUnit_Framework_TestCase
$table = new Table("bar"); $table = new Table("bar");
$table->addColumn('baz', 'integer', array()); $table->addColumn('baz', 'integer', array());
$table->addIndex(array('baz')); $table->addIndex(array('baz'));
$this->assertEquals(1, count($table->getIndexes()));
$this->assertTrue($table->hasIndex('bar_baz_idx')); $indexes = $table->getIndexes();
$this->assertEquals(1, count($indexes));
$index = current($indexes);
$table->addUniqueIndex(array('baz')); $table->addUniqueIndex(array('baz'));
$this->assertEquals(1, count($table->getIndexes())); $this->assertEquals(1, count($table->getIndexes()));
$this->assertFalse($table->hasIndex('bar_baz_idx')); $this->assertFalse($table->hasIndex($index->getName()));
$this->assertTrue($table->hasIndex('bar_baz_uniq'));
} }
/** /**
...@@ -431,4 +436,26 @@ class TableTest extends \PHPUnit_Framework_TestCase ...@@ -431,4 +436,26 @@ class TableTest extends \PHPUnit_Framework_TestCase
$this->assertTrue($table->hasPrimaryKey()); $this->assertTrue($table->hasPrimaryKey());
} }
/**
* @group DBAL-91
*/
public function testAddIndexWithQuotedColumns()
{
$table = new Table("test");
$table->addColumn('"foo"', 'integer');
$table->addColumn('bar', 'integer');
$table->addIndex(array('"foo"', '"bar"'));
}
/**
* @group DBAL-91
*/
public function testAddForeignKeyWithQuotedColumnsAndTable()
{
$table = new Table("test");
$table->addColumn('"foo"', 'integer');
$table->addColumn('bar', 'integer');
$table->addForeignKeyConstraint('"boing"', array('"foo"', '"bar"'), array("id"));
}
} }
\ No newline at end of file
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment