1
0
Fork 0
mirror of synced 2025-04-03 13:23:37 +03:00

Added Doctrine_Hydrate_Alias for more flexible alias handling, small fixes for many classes

This commit is contained in:
zYne 2006-11-27 22:22:31 +00:00
parent 83d68f5d43
commit be7931ed3a
10 changed files with 172 additions and 71 deletions

View file

@ -50,8 +50,6 @@ abstract class Doctrine_Connection extends Doctrine_Configurable implements Coun
* one of the following (true, false, 'emulated') * one of the following (true, false, 'emulated')
*/ */
protected $supported = array(); protected $supported = array();
protected $options = array();
/** /**
* @var array $modules an array containing all modules * @var array $modules an array containing all modules
* transaction Doctrine_Transaction driver, handles savepoint and transaction isolation abstraction * transaction Doctrine_Transaction driver, handles savepoint and transaction isolation abstraction
@ -75,6 +73,10 @@ abstract class Doctrine_Connection extends Doctrine_Configurable implements Coun
'export' => false, 'export' => false,
'unitOfWork' => false, 'unitOfWork' => false,
); );
/**
* @var array $properties an array of connection properties
*/
protected $properties = array();
/** /**
* @var array $availibleDrivers an array containing all availible drivers * @var array $availibleDrivers an array containing all availible drivers
*/ */
@ -145,7 +147,10 @@ abstract class Doctrine_Connection extends Doctrine_Configurable implements Coun
$class = 'Doctrine_' . ucwords($name) . '_' . $this->getName(); $class = 'Doctrine_' . ucwords($name) . '_' . $this->getName();
$this->modules[$name] = new $class($this); $this->modules[$name] = new $class($this);
} }
} }
if(isset($this->properties[$name]))
return $this->properties[$name];
return $this->modules[$name]; return $this->modules[$name];
} }
@ -206,9 +211,12 @@ abstract class Doctrine_Connection extends Doctrine_Configurable implements Coun
if ($checkOption && ! $this->getAttribute(Doctrine::ATTR_QUOTE_IDENTIFIER)) { if ($checkOption && ! $this->getAttribute(Doctrine::ATTR_QUOTE_IDENTIFIER)) {
return $str; return $str;
} }
return $str; $str = str_replace($this->properties['identifier_quoting']['end'],
//$str = str_replace($this->identifier_quoting['end'], $this->identifier_quoting['escape'] . $this->identifier_quoting['end'], $str); $this->properties['identifier_quoting']['escape'] .
//return $this->identifier_quoting['start'] . $str . $this->identifier_quoting['end']; $this->properties['identifier_quoting']['end'], $str);
return $this->properties['identifier_quoting']['start']
. $str . $this->properties['identifier_quoting']['end'];
} }
/** /**
* returns the manager that created this connection * returns the manager that created this connection
@ -291,7 +299,7 @@ abstract class Doctrine_Connection extends Doctrine_Configurable implements Coun
*/ */
public function getIndexName($idx) { public function getIndexName($idx) {
return sprintf($this->getAttribute(Doctrine::ATTR_IDXNAME_FORMAT), return sprintf($this->getAttribute(Doctrine::ATTR_IDXNAME_FORMAT),
preg_replace('/[^a-z0-9_\$]/i', '_', $idx)); preg_replace('/[^a-z0-9_\$]/i', '_', $idx));
} }
/** /**

View file

@ -65,9 +65,26 @@ class Doctrine_Connection_Mysql extends Doctrine_Connection_Common {
'identifier_quoting' => true, 'identifier_quoting' => true,
'pattern_escaping' => true 'pattern_escaping' => true
); );
$this->properties['string_quoting'] = array('start' => "'",
'end' => "'",
'escape' => '\\',
'escape_pattern' => '\\');
$this->properties['identifier_quoting'] = array('start' => '`',
'end' => '`',
'escape' => '`');
$this->properties['sql_comments'] = array(
array('start' => '-- ', 'end' => "\n", 'escape' => false),
array('start' => '#', 'end' => "\n", 'escape' => false),
array('start' => '/*', 'end' => '*/', 'escape' => false),
);
$this->properties['varchar_max_length'] = 255;
parent::__construct($manager, $adapter); parent::__construct($manager, $adapter);
} }
/** /**
* Returns the next free id of a sequence * Returns the next free id of a sequence
* *
@ -103,7 +120,7 @@ class Doctrine_Connection_Mysql extends Doctrine_Connection_Common {
$sequenceName = $this->quoteIdentifier($this->getSequenceName($seqName), true); $sequenceName = $this->quoteIdentifier($this->getSequenceName($seqName), true);
$seqcolName = $this->quoteIdentifier($this->options['seqcol_name'], true); $seqcolName = $this->quoteIdentifier($this->options['seqcol_name'], true);
$query = 'SELECT MAX(' . $seqcolName . ') FROM ' . $sequenceName; $query = 'SELECT MAX(' . $seqcolName . ') FROM ' . $sequenceName;
return $this->queryOne($query, 'integer'); return $this->fetchOne($query);
} }
/** /**
* Execute a SQL REPLACE query. A REPLACE query is identical to a INSERT * Execute a SQL REPLACE query. A REPLACE query is identical to a INSERT

View file

@ -65,8 +65,6 @@ class Doctrine_Connection_Pgsql extends Doctrine_Connection_Common {
'pattern_escaping' => true, 'pattern_escaping' => true,
); );
$this->options['multi_query'] = false;
parent::__construct($manager, $adapter); parent::__construct($manager, $adapter);
} }
/** /**

View file

@ -21,19 +21,20 @@
/** /**
* Doctrine_DataDict * Doctrine_DataDict
* *
* @package Doctrine * @package Doctrine
* @license http://www.opensource.org/licenses/lgpl-license.php LGPL * @license http://www.opensource.org/licenses/lgpl-license.php LGPL
* @category Object Relational Mapping * @category Object Relational Mapping
* @link www.phpdoctrine.com * @link www.phpdoctrine.com
* @since 1.0 * @since 1.0
* @version $Revision$ * @version $Revision$
* @author Konsta Vesterinen <kvesteri@cc.hut.fi> * @author Konsta Vesterinen <kvesteri@cc.hut.fi>
*/ */
class Doctrine_DataDict { class Doctrine_DataDict {
protected $dbh; protected $dbh;
public function __construct($dbh = null) { public function __construct($dbh = null) {
$file = Doctrine::getPath().DIRECTORY_SEPARATOR."Doctrine".DIRECTORY_SEPARATOR."adodb-hack".DIRECTORY_SEPARATOR."adodb.inc.php"; $file = Doctrine::getPath().DIRECTORY_SEPARATOR."Doctrine".DIRECTORY_SEPARATOR."adodb-hack".DIRECTORY_SEPARATOR."adodb.inc.php";
if( ! file_exists($file)) if( ! file_exists($file))

View file

@ -29,7 +29,7 @@ Doctrine::autoload('Doctrine_DataDict');
* @link www.phpdoctrine.com * @link www.phpdoctrine.com
* @since 1.0 * @since 1.0
*/ */
class Doctrine_DataDict_Mysql extends Doctrine_DataDict { class Doctrine_DataDict_Mysql extends Doctrine_Connection_Module {
/** /**
* Obtain DBMS specific SQL code portion needed to declare an text type * Obtain DBMS specific SQL code portion needed to declare an text type
* field to be used in statements like CREATE TABLE. * field to be used in statements like CREATE TABLE.
@ -55,6 +55,11 @@ class Doctrine_DataDict_Mysql extends Doctrine_DataDict {
*/ */
public function getNativeDeclaration($field) { public function getNativeDeclaration($field) {
switch ($field['type']) { switch ($field['type']) {
case 'char':
$length = (! empty($field['length'])) ? $field['length'] : false;
return $length ? 'CHAR('.$length.')' : 'CHAR(255)';
case 'varchar':
case 'array': case 'array':
case 'object': case 'object':
case 'string': case 'string':
@ -321,12 +326,15 @@ class Doctrine_DataDict_Mysql extends Doctrine_DataDict {
$field['default'] = empty($field['notnull']) ? null : 0; $field['default'] = empty($field['notnull']) ? null : 0;
} }
$default = ' DEFAULT '.$this->conn->getDbh()->quote($field['default']); $default = ' DEFAULT '.$this->conn->getDbh()->quote($field['default']);
} elseif (empty($field['notnull'])) { }
/**
elseif (empty($field['notnull'])) {
$default = ' DEFAULT NULL'; $default = ' DEFAULT NULL';
} }
*/
$notnull = empty($field['notnull']) ? '' : ' NOT NULL'; $notnull = (isset($field['notnull']) && $field['notnull']) ? ' NOT NULL' : '';
$unsigned = empty($field['unsigned']) ? '' : ' UNSIGNED'; $unsigned = (isset($field['unsigned']) && $field['unsigned']) ? ' UNSIGNED' : '';
$name = $this->conn->quoteIdentifier($name, true); $name = $this->conn->quoteIdentifier($name, true);

View file

@ -357,15 +357,15 @@ class Doctrine_DataDict_Pgsql extends Doctrine_Connection_Module {
*/ */
public function getNativeDeclaration(array $field) { public function getNativeDeclaration(array $field) {
switch ($field['type']) { switch ($field['type']) {
case 'char':
case 'string': case 'string':
case 'array': case 'array':
case 'object': case 'object':
case 'varchar': case 'varchar':
case 'char': $length = (isset($field['length']) && $field['length']) ? $field['length'] : null;
$length = !empty($field['length']) // TODO: $db->options['default_text_field_length'];
? $field['length'] : $db->options['default_text_field_length'];
$fixed = !empty($field['fixed']) ? $field['fixed'] : false; $fixed = ((isset($field['fixed']) && $field['fixed']) || $field['type'] == 'char') ? true : false;
return $fixed ? ($length ? 'CHAR('.$length.')' : 'CHAR('.$db->options['default_text_field_length'].')') return $fixed ? ($length ? 'CHAR('.$length.')' : 'CHAR('.$db->options['default_text_field_length'].')')
: ($length ? 'VARCHAR('.$length.')' : 'TEXT'); : ($length ? 'VARCHAR('.$length.')' : 'TEXT');
@ -413,16 +413,16 @@ class Doctrine_DataDict_Pgsql extends Doctrine_Connection_Module {
} }
} }
/** /**
* Maps a native array description of a field to a MDB2 datatype and length * Maps a native array description of a field to a portable Doctrine datatype and length
* *
* @param array $field native field description * @param array $field native field description
* *
* @return array containing the various possible types, length, sign, fixed * @return array containing the various possible types, length, sign, fixed
*/ */
public function getDoctrineDeclaration(array $field) { public function getPortableDeclaration(array $field) {
$length = $field['length']; $length = $field['length'];
if ($length == '-1' && !empty($field['atttypmod'])) { if ($length == '-1' && isset($field['atttypmod'])) {
$length = $field['atttypmod'] - 4; $length = $field['atttypmod'] - 4;
} }
if ((int)$length <= 0) { if ((int)$length <= 0) {

View file

@ -347,11 +347,102 @@ class Doctrine_Export extends Doctrine_Connection_Module {
* @return string * @return string
*/ */
public function getFieldDeclarationList(array $fields) { public function getFieldDeclarationList(array $fields) {
foreach ($fields as $field_name => $field) { foreach ($fields as $fieldName => $field) {
$query = $this->conn->dataDict->getNativeDeclaration($field['type'], $field_name, $field); $query = $this->getDeclaration($fieldName, $field);
$query_fields[] = $query;
$queryFields[] = $query;
} }
return implode(', ', $query_fields); return implode(', ', $queryFields);
}
/**
* Obtain DBMS specific SQL code portion needed to declare a generic type
* field to be used in statements like CREATE TABLE.
*
* @param string $name name the field to be declared.
* @param array $field associative array with the name of the properties
* of the field being declared as array indexes. Currently, the types
* of supported field properties are as follows:
*
* length
* Integer value that determines the maximum length of the text
* field. If this argument is missing the field should be
* declared to have the longest length allowed by the DBMS.
*
* default
* Text value to be used as default for this field.
*
* notnull
* Boolean flag that indicates whether this field is constrained
* to not be set to null.
* charset
* Text value with the default CHARACTER SET for this field.
* collation
* Text value with the default COLLATION for this field.
* @return string DBMS specific SQL code portion that should be used to
* declare the specified field.
*/
public function getDeclaration($name, $field) {
$default = '';
if(isset($field['default'])) {
if ($field['default'] === '') {
$field['default'] = empty($field['notnull'])
? null : $this->valid_default_values[$field['type']];
if ($field['default'] === ''
&& ($db->options['portability'] & Doctrine::PORTABILITY_EMPTY_TO_NULL)
) {
$field['default'] = ' ';
}
}
$default = ' DEFAULT ' . $this->conn->getDbh()->quote($field['default']);
}
/**
TODO: is this really needed for portability?
elseif(empty($field['notnull'])) {
$default = ' DEFAULT NULL';
}
*/
$charset = empty($field['charset']) ? '' :
' '.$this->getCharsetFieldDeclaration($field['charset']);
$collation = empty($field['collation']) ? '' :
' '.$this->getCollationFieldDeclaration($field['collation']);
$notnull = empty($field['notnull']) ? '' : ' NOT NULL';
$name = $this->conn->quoteIdentifier($name, true);
$method = 'get' . $field['type'] . 'Declaration';
if(method_exists($this->conn->dataDict, $method))
return $this->conn->dataDict->$method($name, $field);
else
$dec = $this->conn->dataDict->getNativeDeclaration($field);
return $name . ' ' . $dec . $charset . $default . $notnull . $collation;
}
/**
* Obtain DBMS specific SQL code portion needed to set the CHARACTER SET
* of a field declaration to be used in statements like CREATE TABLE.
*
* @param string $charset name of the charset
* @return string DBMS specific SQL code portion needed to set the CHARACTER SET
* of a field declaration.
*/
public function getCharsetFieldDeclaration($charset) {
return '';
}
/**
* Obtain DBMS specific SQL code portion needed to set the COLLATION
* of a field declaration to be used in statements like CREATE TABLE.
*
* @param string $collation name of the collation
* @return string DBMS specific SQL code portion needed to set the COLLATION
* of a field declaration.
*/
public function getCollationFieldDeclaration($collation) {
return '';
} }
/** /**
* export * export

View file

@ -92,10 +92,8 @@ abstract class Doctrine_Hydrate extends Doctrine_Access {
protected $pendingAggregates = array(); protected $pendingAggregates = array();
protected $aggregateMap = array(); protected $aggregateMap = array();
protected $shortAliases = array(); protected $aliasHandler;
protected $shortAliasIndexes = array();
/** /**
* @var array $parts SQL query string parts * @var array $parts SQL query string parts
*/ */
@ -120,6 +118,7 @@ abstract class Doctrine_Hydrate extends Doctrine_Access {
$connection = Doctrine_Manager::getInstance()->getCurrentConnection(); $connection = Doctrine_Manager::getInstance()->getCurrentConnection();
$this->connection = $connection; $this->connection = $connection;
$this->aliasHandler = new Doctrine_Hydrate_Alias();
} }
/** /**
* getComponentAliases * getComponentAliases
@ -235,8 +234,7 @@ abstract class Doctrine_Hydrate extends Doctrine_Access {
$this->joins = array(); $this->joins = array();
$this->tableIndexes = array(); $this->tableIndexes = array();
$this->tableAliases = array(); $this->tableAliases = array();
$this->shortAliases = array(); $this->aliasHandler->clear();
$this->shortAliasIndexes = array();
} }
/** /**
* getConnection * getConnection
@ -544,34 +542,14 @@ abstract class Doctrine_Hydrate extends Doctrine_Access {
return false; return false;
} }
public function getShortAliasIndex($alias) { public function getShortAliasIndex($alias) {
if( ! isset($this->shortAliasIndexes[$alias])) return $this->aliasHandler->getShortAliasIndex($alias);
return 0;
return $this->shortAliasIndexes[$alias];
} }
public function generateShortAlias($tableName) { public function generateShortAlias($tableName) {
$char = strtolower(substr($tableName, 0, 1)); return $this->aliasHandler->generateShortAlias($tableName);
$alias = $char;
if( ! isset($this->shortAliasIndexes[$alias]))
$this->shortAliasIndexes[$alias] = 1;
while(isset($this->shortAliases[$alias])) {
$alias = $char . ++$this->shortAliasIndexes[$alias];
}
$this->shortAliases[$alias] = $tableName;
return $alias;
} }
public function getShortAlias($tableName) { public function getShortAlias($tableName) {
$alias = array_search($tableName, $this->shortAliases); return $this->aliasHandler->getShortAlias($tableName);
if($alias !== false)
return $alias;
return $this->generateShortAlias($tableName);
} }
/** /**
* applyInheritance * applyInheritance

View file

@ -98,8 +98,8 @@ class Doctrine_Manager extends Doctrine_Configurable implements Countable, Itera
Doctrine::ATTR_AUTO_TYPE_VLD => true, Doctrine::ATTR_AUTO_TYPE_VLD => true,
Doctrine::ATTR_CREATE_TABLES => true, Doctrine::ATTR_CREATE_TABLES => true,
Doctrine::ATTR_QUERY_LIMIT => Doctrine::LIMIT_RECORDS, Doctrine::ATTR_QUERY_LIMIT => Doctrine::LIMIT_RECORDS,
Doctrine::ATTR_IDXNAME_FORMAT => '%_idx', Doctrine::ATTR_IDXNAME_FORMAT => "%s_idx",
Doctrine::ATTR_SEQNAME_FORMAT => '%_seq', Doctrine::ATTR_SEQNAME_FORMAT => "%s_seq",
Doctrine::ATTR_QUOTE_IDENTIFIER => false, Doctrine::ATTR_QUOTE_IDENTIFIER => false,
Doctrine::ATTR_SEQCOL_NAME => 'id', Doctrine::ATTR_SEQCOL_NAME => 'id',
Doctrine::ATTR_PORTABILITY => Doctrine::PORTABILITY_ALL, Doctrine::ATTR_PORTABILITY => Doctrine::PORTABILITY_ALL,

View file

@ -343,8 +343,7 @@ class Doctrine_Query extends Doctrine_Hydrate implements Countable {
$this->fetchModes = array(); $this->fetchModes = array();
$this->tableIndexes = array(); $this->tableIndexes = array();
$this->tableAliases = array(); $this->tableAliases = array();
$this->shortAliases = array(); $this->aliasHandler->clear();
$this->shortAliasIndexes = array();
$class = "Doctrine_Query_".ucwords($name); $class = "Doctrine_Query_".ucwords($name);
$parser = new $class($this); $parser = new $class($this);
@ -621,8 +620,8 @@ class Doctrine_Query extends Doctrine_Hydrate implements Countable {
if(strpos($part, "'") !== false) if(strpos($part, "'") !== false)
continue; continue;
if(isset($this->shortAliases[$part])) { if($this->aliasHandler->hasAliasFor($part)) {
$parts[$k] = $this->generateNewAlias($part); $parts[$k] = $this->aliasHandler->generateNewAlias($part);
} }
if(strpos($part, '.') !== false) { if(strpos($part, '.') !== false) {
@ -631,7 +630,7 @@ class Doctrine_Query extends Doctrine_Hydrate implements Countable {
$trimmed = ltrim($e[0], '( '); $trimmed = ltrim($e[0], '( ');
$pos = strpos($e[0], $trimmed); $pos = strpos($e[0], $trimmed);
$e[0] = substr($e[0], 0, $pos) . $this->generateNewAlias($trimmed); $e[0] = substr($e[0], 0, $pos) . $this->aliasHandler->generateNewAlias($trimmed);
$parts[$k] = implode('.', $e); $parts[$k] = implode('.', $e);
} }
} }
@ -639,7 +638,7 @@ class Doctrine_Query extends Doctrine_Hydrate implements Countable {
return $subquery; return $subquery;
} }
/**
public function generateNewAlias($alias) { public function generateNewAlias($alias) {
if(isset($this->shortAliases[$alias])) { if(isset($this->shortAliases[$alias])) {
// generate a new alias // generate a new alias
@ -656,6 +655,7 @@ class Doctrine_Query extends Doctrine_Hydrate implements Countable {
return $alias; return $alias;
} }
*/