Skip to content

Commit

Permalink
Merge pull request #1702 from mringler/aggregate_multiple_columns_beh…
Browse files Browse the repository at this point in the history
…avior

Aggregate multiple columns behavior & parameter list support
  • Loading branch information
dereuromark authored Mar 4, 2021
2 parents b5821e3 + f076d50 commit 399c82f
Show file tree
Hide file tree
Showing 10 changed files with 1,061 additions and 50 deletions.
1 change: 1 addition & 0 deletions phpstan.neon
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ parameters:
excludes_analyse:
- '%rootDir%/../../../src/Propel/Generator/Command/templates/*'
- '%rootDir%/../../../src/Propel/Generator/Behavior/AggregateColumn/templates/*'
- '%rootDir%/../../../src/Propel/Generator/Behavior/AggregateMultipleColumns/templates/*'
- '%rootDir%/../../../src/Propel/Generator/Behavior/I18n/templates/*'
- '%rootDir%/../../../src/Propel/Generator/Builder/Om/templates/*'
- '%rootDir%/../../../src/Propel/Generator/Behavior/Validate/templates/*'
Expand Down
16 changes: 16 additions & 0 deletions psalm-baseline.xml
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,22 @@
<code>$variableName</code>
</UndefinedGlobalVariable>
</file>
<file src="src/Propel/Generator/Behavior/AggregateMultipleColumns/templates/objectCompute.php">
<UndefinedGlobalVariable occurrences="4">
<code>$aggregationName</code>
<code>$aggregationName</code>
<code>$bindings</code>
<code>$sql</code>
</UndefinedGlobalVariable>
</file>
<file src="src/Propel/Generator/Behavior/AggregateMultipleColumns/templates/objectUpdate.php">
<UndefinedGlobalVariable occurrences="4">
<code>$aggregationName</code>
<code>$aggregationName</code>
<code>$aggregationName</code>
<code>$columnPhpNames</code>
</UndefinedGlobalVariable>
</file>
<file src="src/Propel/Generator/Behavior/Archivable/ArchivableBehavior.php">
<UndefinedPropertyAssignment occurrences="1">
<code>$archiveTable-&gt;isArchiveTable</code>
Expand Down
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);
}
}
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);
}
Loading

0 comments on commit 399c82f

Please sign in to comment.