-
Notifications
You must be signed in to change notification settings - Fork 398
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #1702 from mringler/aggregate_multiple_columns_beh…
…avior Aggregate multiple columns behavior & parameter list support
- Loading branch information
Showing
10 changed files
with
1,061 additions
and
50 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
335 changes: 335 additions & 0 deletions
335
src/Propel/Generator/Behavior/AggregateMultipleColumns/AggregateMultipleColumnsBehavior.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,335 @@ | ||
<?php | ||
|
||
/** | ||
* MIT License. This file is part of the Propel package. | ||
* For the full copyright and license information, please view the LICENSE | ||
* file that was distributed with this source code. | ||
*/ | ||
|
||
namespace Propel\Generator\Behavior\AggregateMultipleColumns; | ||
|
||
use InvalidArgumentException; | ||
use Propel\Generator\Behavior\AggregateColumn\AggregateColumnRelationBehavior; | ||
use Propel\Generator\Builder\Om\ObjectBuilder; | ||
use Propel\Generator\Model\Behavior; | ||
use Propel\Generator\Model\ForeignKey; | ||
use Propel\Generator\Model\Table; | ||
|
||
/** | ||
* Keeps an aggregate column updated with related table | ||
* | ||
* @author François Zaninotto | ||
*/ | ||
class AggregateMultipleColumnsBehavior extends Behavior | ||
{ | ||
public const PARAMETER_KEY_FOREIGN_TABLE = 'foreign_table'; | ||
public const PARAMETER_KEY_FOREIGN_SCHEMA = 'foreign_schema'; | ||
public const PARAMETER_KEY_CONDITION = 'condition'; | ||
public const PARAMETER_KEY_COLUMNS = 'columns'; | ||
public const PARAMETER_KEY_COLUMN_NAME = 'column_name'; | ||
public const PARAMETER_KEY_COLUMN_EXPRESSION = 'expression'; | ||
|
||
/** | ||
* Keeps track of inserted aggregation functions to avoid naming collisions. | ||
* | ||
* @var array | ||
*/ | ||
private static $insertedAggregationNames = []; | ||
|
||
/** | ||
* Default parameters value. | ||
* | ||
* @var array | ||
*/ | ||
protected $parameters = [ | ||
self::PARAMETER_KEY_FOREIGN_TABLE => null, | ||
self::PARAMETER_KEY_FOREIGN_SCHEMA => null, | ||
self::PARAMETER_KEY_CONDITION => null, | ||
self::PARAMETER_KEY_COLUMNS => null, | ||
]; | ||
|
||
/** | ||
* @var string|null | ||
*/ | ||
private $aggregationName; | ||
|
||
/** | ||
* Multiple aggregates on the same table is OK. | ||
* | ||
* @return bool | ||
*/ | ||
public function allowMultiple(): bool | ||
{ | ||
return true; | ||
} | ||
|
||
/** | ||
* @return string | ||
*/ | ||
public function getAggregationName(): string | ||
{ | ||
if ($this->aggregationName === null) { | ||
$this->aggregationName = $this->buildAggregationName(); | ||
} | ||
|
||
return $this->aggregationName; | ||
} | ||
|
||
/** | ||
* @return string | ||
*/ | ||
private function buildAggregationName(): string | ||
{ | ||
$foreignTableName = $this->getForeignTable()->getPhpName(); | ||
$baseAggregationName = 'AggregatedColumnsFrom' . $foreignTableName; | ||
$tableName = $this->getTable()->getPhpName(); | ||
if (!array_key_exists($tableName, static::$insertedAggregationNames)) { | ||
static::$insertedAggregationNames[$tableName] = []; | ||
} | ||
|
||
$existingNames = &static::$insertedAggregationNames[$tableName]; | ||
if (!in_array($baseAggregationName, $existingNames)) { | ||
$existingNames[] = $baseAggregationName; | ||
|
||
return $baseAggregationName; | ||
} | ||
|
||
$duplicateAvoidanceSuffix = 1; | ||
do { | ||
$aggregationName = $baseAggregationName . $duplicateAvoidanceSuffix; | ||
$duplicateAvoidanceSuffix++; | ||
} while (in_array($aggregationName, $existingNames)); | ||
|
||
$existingNames[] = $aggregationName; | ||
|
||
return $aggregationName; | ||
} | ||
|
||
/** | ||
* Add the aggregate key to the current table | ||
* | ||
* @return void | ||
*/ | ||
public function modifyTable(): void | ||
{ | ||
$this->validateColumnParameter(); | ||
$this->addMissingColumnsToTable(); | ||
$this->addAutoupdateBehaviorToForeignTable(); | ||
} | ||
|
||
/** | ||
* @return void | ||
*/ | ||
private function validateColumnParameter(): void | ||
{ | ||
$columnParameters = $this->getParameter(static::PARAMETER_KEY_COLUMNS); | ||
if (empty($columnParameters)) { | ||
$this->throwInvalidArgumentExceptionWithLocation('At least one column is required'); | ||
} | ||
foreach ($columnParameters as $columnDefinition) { | ||
if (empty($columnDefinition[static::PARAMETER_KEY_COLUMN_NAME])) { | ||
$this->throwInvalidArgumentExceptionWithLocation('Parameter \'name\' is missing on a column'); | ||
} | ||
if (empty($columnDefinition[static::PARAMETER_KEY_COLUMN_EXPRESSION])) { | ||
$colName = $columnDefinition[static::PARAMETER_KEY_COLUMN_NAME]; | ||
$this->throwInvalidArgumentExceptionWithLocation('Parameter \'expression\' is missing on column ' . $colName); | ||
} | ||
} | ||
} | ||
|
||
/** | ||
* Add the aggregate column if not present | ||
* | ||
* @return void | ||
*/ | ||
private function addMissingColumnsToTable(): void | ||
{ | ||
$table = $this->getTable(); | ||
$columnParameters = $this->getParameter(static::PARAMETER_KEY_COLUMNS); | ||
foreach ($columnParameters as $columnDefinition) { | ||
$columnName = $columnDefinition[static::PARAMETER_KEY_COLUMN_NAME]; | ||
if ($table->hasColumn($columnName)) { | ||
continue; | ||
} | ||
|
||
$table->addColumn(['name' => $columnName, 'type' => 'INTEGER']); | ||
} | ||
} | ||
|
||
/** | ||
* Add a behavior in the foreign table to autoupdate the aggregate column | ||
* | ||
* @return void | ||
*/ | ||
private function addAutoupdateBehaviorToForeignTable(): void | ||
{ | ||
if (!$this->getParameter(static::PARAMETER_KEY_FOREIGN_TABLE)) { | ||
$this->throwInvalidArgumentExceptionWithLocation('You must define a \'foreign_table\' parameter'); | ||
} | ||
$foreignTable = $this->getForeignTable(); | ||
if ($foreignTable->hasBehavior('concrete_inheritance_parent')) { | ||
return; | ||
} | ||
|
||
$relationBehavior = new AggregateColumnRelationBehavior(); | ||
$relationBehavior->setName('aggregate_multiple_columns_relation'); | ||
$relationBehavior->setId('aggregate_multiple_columns_relation_' . $this->getId()); | ||
$relationBehavior->addParameter(['name' => 'foreign_table', 'value' => $this->getTable()->getName()]); | ||
$relationBehavior->addParameter(['name' => 'aggregate_name', 'value' => $this->getAggregationName()]); | ||
$relationBehavior->addParameter(['name' => 'update_method', 'value' => 'update' . $this->getAggregationName()]); | ||
$foreignTable->addBehavior($relationBehavior); | ||
} | ||
|
||
/** | ||
* @param \Propel\Generator\Builder\Om\ObjectBuilder $builder | ||
* | ||
* @return string | ||
*/ | ||
public function objectMethods(ObjectBuilder $builder): string | ||
{ | ||
$script = ''; | ||
$script .= $this->addObjectCompute($builder); | ||
$script .= $this->addObjectUpdate(); | ||
|
||
return $script; | ||
} | ||
|
||
/** | ||
* @param \Propel\Generator\Builder\Om\ObjectBuilder $builder | ||
* | ||
* @throws \InvalidArgumentException | ||
* | ||
* @return string | ||
*/ | ||
protected function addObjectCompute(ObjectBuilder $builder): string | ||
{ | ||
if ($this->getForeignKey()->isPolymorphic()) { | ||
throw new InvalidArgumentException('AggregateColumnBehavior does not work with polymorphic relations.'); | ||
} | ||
|
||
$conditions = []; | ||
if ($this->getParameter(static::PARAMETER_KEY_CONDITION)) { | ||
$conditions[] = $this->getParameter(static::PARAMETER_KEY_CONDITION); | ||
} | ||
|
||
$bindings = []; | ||
foreach ($this->getForeignKey()->getMapping() as $index => $mapping) { | ||
[$localColumn, $foreignColumn] = $mapping; | ||
$conditions[] = $localColumn->getFullyQualifiedName() . ' = :p' . ($index + 1); | ||
$bindings[$index + 1] = $foreignColumn->getPhpName(); | ||
} | ||
|
||
$foreignTableName = $this->getForeignTableNameFullyQualified(); | ||
|
||
$sql = sprintf( | ||
'SELECT %s FROM %s WHERE %s', | ||
$this->buildSelectionStatement(), | ||
$builder->getTable()->quoteIdentifier($foreignTableName), | ||
implode(' AND ', $conditions) | ||
); | ||
|
||
return $this->renderTemplate('objectCompute', [ | ||
'aggregationName' => $this->getAggregationName(), | ||
'sql' => $sql, | ||
'bindings' => $bindings, | ||
]); | ||
} | ||
|
||
/** | ||
* @return string | ||
*/ | ||
private function buildSelectionStatement(): string | ||
{ | ||
$columnDefinitions = $this->getParameter(static::PARAMETER_KEY_COLUMNS); | ||
$selects = []; | ||
$table = $this->getTable(); | ||
foreach ($columnDefinitions as $columnDefinition) { | ||
$expression = $columnDefinition[static::PARAMETER_KEY_COLUMN_EXPRESSION]; | ||
$columName = $columnDefinition[static::PARAMETER_KEY_COLUMN_NAME]; | ||
$columnPhpName = $table->getColumn($columName)->getPhpName(); | ||
$selects[] = "$expression AS $columnPhpName"; | ||
} | ||
|
||
return implode(', ', $selects); | ||
} | ||
|
||
/** | ||
* @return string | ||
*/ | ||
protected function addObjectUpdate(): string | ||
{ | ||
$table = $this->getTable(); | ||
$columnPhpNames = array_map(function (array $columnParameters) use ($table) { | ||
$columName = $columnParameters[AggregateMultipleColumnsBehavior::PARAMETER_KEY_COLUMN_NAME]; | ||
|
||
return $table->getColumn($columName)->getPhpName(); | ||
}, | ||
$this->getParameter(static::PARAMETER_KEY_COLUMNS)); | ||
|
||
return $this->renderTemplate('objectUpdate', [ | ||
'aggregationName' => $this->getAggregationName(), | ||
'columnPhpNames' => $columnPhpNames, | ||
]); | ||
} | ||
|
||
/** | ||
* @return string | ||
*/ | ||
private function getForeignTableNameFullyQualified(): string | ||
{ | ||
$database = $this->getTable()->getDatabase(); | ||
$foreignTableName = $database->getTablePrefix() . $this->getParameter(static::PARAMETER_KEY_FOREIGN_TABLE); | ||
$platform = $database->getPlatform(); | ||
$foreignSchema = $this->getParameter(static::PARAMETER_KEY_FOREIGN_SCHEMA); | ||
if ($platform->supportsSchemas() && $foreignSchema) { | ||
$foreignTableName = $foreignSchema . $platform->getSchemaDelimiter() . $foreignTableName; | ||
} | ||
|
||
return $foreignTableName; | ||
} | ||
|
||
/** | ||
* @return \Propel\Generator\Model\Table | ||
*/ | ||
protected function getForeignTable(): Table | ||
{ | ||
$database = $this->getTable()->getDatabase(); | ||
$foreignTableName = $this->getForeignTableNameFullyQualified(); | ||
|
||
return $database->getTable($foreignTableName); | ||
} | ||
|
||
/** | ||
* @return \Propel\Generator\Model\ForeignKey | ||
*/ | ||
protected function getForeignKey(): ForeignKey | ||
{ | ||
$foreignTable = $this->getForeignTable(); | ||
// let's infer the relation from the foreign table | ||
$fks = $foreignTable->getForeignKeysReferencingTable($this->getTable()->getName()); | ||
if (!$fks) { | ||
$msg = 'You must define a foreign key from the \'%s\' table to the table witht the aggregated columns'; | ||
$this->throwInvalidArgumentExceptionWithLocation($msg, $foreignTable->getName()); | ||
} | ||
|
||
// FIXME doesn't work when more than one fk to the same table | ||
return array_shift($fks); | ||
} | ||
|
||
/** | ||
* @param string $format | ||
* @param mixed ...$args | ||
* | ||
* @throws \InvalidArgumentException | ||
* | ||
* @return void | ||
*/ | ||
private function throwInvalidArgumentExceptionWithLocation($format, ...$args): void | ||
{ | ||
$format .= ' in the \'aggregate_multiple_columns\' behavior definition in the \'%s\' table definition'; | ||
$args[] = $this->getTable()->getName(); | ||
$message = vsprintf($format, $args); | ||
|
||
throw new InvalidArgumentException($message); | ||
} | ||
} |
18 changes: 18 additions & 0 deletions
18
src/Propel/Generator/Behavior/AggregateMultipleColumns/templates/objectCompute.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
|
||
/** | ||
* Computes the value of the aggregate columns defined as <?=$aggregationName?> ?> | ||
* | ||
* @param ConnectionInterface $con A connection object | ||
* | ||
* @return array The result row from the aggregate query | ||
*/ | ||
public function compute<?=$aggregationName?>(ConnectionInterface $con) | ||
{ | ||
$stmt = $con->prepare('<?=$sql?>'); | ||
<?php foreach ($bindings as $key => $binding):?> | ||
$stmt->bindValue(':p<?=$key?>', $this->get<?=$binding?>()); | ||
<?php endforeach;?> | ||
$stmt->execute(); | ||
|
||
return $stmt->fetch(\PDO::FETCH_NUM); | ||
} |
Oops, something went wrong.