mirror of
https://github.com/zoe-may/TDoG-Skin.git
synced 2025-01-19 20:47:24 +08:00
377 lines
8.7 KiB
PHP
377 lines
8.7 KiB
PHP
|
<?php
|
||
|
|
||
|
namespace Doctrine\DBAL\Schema;
|
||
|
|
||
|
use Doctrine\DBAL\Platforms\AbstractPlatform;
|
||
|
use InvalidArgumentException;
|
||
|
|
||
|
use function array_filter;
|
||
|
use function array_keys;
|
||
|
use function array_map;
|
||
|
use function array_search;
|
||
|
use function array_shift;
|
||
|
use function count;
|
||
|
use function is_string;
|
||
|
use function strtolower;
|
||
|
|
||
|
class Index extends AbstractAsset implements Constraint
|
||
|
{
|
||
|
/**
|
||
|
* Asset identifier instances of the column names the index is associated with.
|
||
|
* array($columnName => Identifier)
|
||
|
*
|
||
|
* @var Identifier[]
|
||
|
*/
|
||
|
protected $_columns = [];
|
||
|
|
||
|
/** @var bool */
|
||
|
protected $_isUnique = false;
|
||
|
|
||
|
/** @var bool */
|
||
|
protected $_isPrimary = false;
|
||
|
|
||
|
/**
|
||
|
* Platform specific flags for indexes.
|
||
|
* array($flagName => true)
|
||
|
*
|
||
|
* @var true[]
|
||
|
*/
|
||
|
protected $_flags = [];
|
||
|
|
||
|
/**
|
||
|
* Platform specific options
|
||
|
*
|
||
|
* @todo $_flags should eventually be refactored into options
|
||
|
* @var mixed[]
|
||
|
*/
|
||
|
private $options = [];
|
||
|
|
||
|
/**
|
||
|
* @param string $name
|
||
|
* @param string[] $columns
|
||
|
* @param bool $isUnique
|
||
|
* @param bool $isPrimary
|
||
|
* @param string[] $flags
|
||
|
* @param mixed[] $options
|
||
|
*/
|
||
|
public function __construct(
|
||
|
$name,
|
||
|
array $columns,
|
||
|
$isUnique = false,
|
||
|
$isPrimary = false,
|
||
|
array $flags = [],
|
||
|
array $options = []
|
||
|
) {
|
||
|
$isUnique = $isUnique || $isPrimary;
|
||
|
|
||
|
$this->_setName($name);
|
||
|
$this->_isUnique = $isUnique;
|
||
|
$this->_isPrimary = $isPrimary;
|
||
|
$this->options = $options;
|
||
|
|
||
|
foreach ($columns as $column) {
|
||
|
$this->_addColumn($column);
|
||
|
}
|
||
|
|
||
|
foreach ($flags as $flag) {
|
||
|
$this->addFlag($flag);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $column
|
||
|
*
|
||
|
* @return void
|
||
|
*
|
||
|
* @throws InvalidArgumentException
|
||
|
*/
|
||
|
protected function _addColumn($column)
|
||
|
{
|
||
|
if (! is_string($column)) {
|
||
|
throw new InvalidArgumentException('Expecting a string as Index Column');
|
||
|
}
|
||
|
|
||
|
$this->_columns[$column] = new Identifier($column);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* {@inheritdoc}
|
||
|
*/
|
||
|
public function getColumns()
|
||
|
{
|
||
|
return array_keys($this->_columns);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* {@inheritdoc}
|
||
|
*/
|
||
|
public function getQuotedColumns(AbstractPlatform $platform)
|
||
|
{
|
||
|
$subParts = $platform->supportsColumnLengthIndexes() && $this->hasOption('lengths')
|
||
|
? $this->getOption('lengths') : [];
|
||
|
|
||
|
$columns = [];
|
||
|
|
||
|
foreach ($this->_columns as $column) {
|
||
|
$length = array_shift($subParts);
|
||
|
|
||
|
$quotedColumn = $column->getQuotedName($platform);
|
||
|
|
||
|
if ($length !== null) {
|
||
|
$quotedColumn .= '(' . $length . ')';
|
||
|
}
|
||
|
|
||
|
$columns[] = $quotedColumn;
|
||
|
}
|
||
|
|
||
|
return $columns;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return string[]
|
||
|
*/
|
||
|
public function getUnquotedColumns()
|
||
|
{
|
||
|
return array_map([$this, 'trimQuotes'], $this->getColumns());
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Is the index neither unique nor primary key?
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function isSimpleIndex()
|
||
|
{
|
||
|
return ! $this->_isPrimary && ! $this->_isUnique;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function isUnique()
|
||
|
{
|
||
|
return $this->_isUnique;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function isPrimary()
|
||
|
{
|
||
|
return $this->_isPrimary;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $name
|
||
|
* @param int $pos
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function hasColumnAtPosition($name, $pos = 0)
|
||
|
{
|
||
|
$name = $this->trimQuotes(strtolower($name));
|
||
|
$indexColumns = array_map('strtolower', $this->getUnquotedColumns());
|
||
|
|
||
|
return array_search($name, $indexColumns) === $pos;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Checks if this index exactly spans the given column names in the correct order.
|
||
|
*
|
||
|
* @param string[] $columnNames
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function spansColumns(array $columnNames)
|
||
|
{
|
||
|
$columns = $this->getColumns();
|
||
|
$numberOfColumns = count($columns);
|
||
|
$sameColumns = true;
|
||
|
|
||
|
for ($i = 0; $i < $numberOfColumns; $i++) {
|
||
|
if (
|
||
|
isset($columnNames[$i])
|
||
|
&& $this->trimQuotes(strtolower($columns[$i])) === $this->trimQuotes(strtolower($columnNames[$i]))
|
||
|
) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$sameColumns = false;
|
||
|
}
|
||
|
|
||
|
return $sameColumns;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Checks if the other index already fulfills all the indexing and constraint needs of the current one.
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function isFullfilledBy(Index $other)
|
||
|
{
|
||
|
// allow the other index to be equally large only. It being larger is an option
|
||
|
// but it creates a problem with scenarios of the kind PRIMARY KEY(foo,bar) UNIQUE(foo)
|
||
|
if (count($other->getColumns()) !== count($this->getColumns())) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Check if columns are the same, and even in the same order
|
||
|
$sameColumns = $this->spansColumns($other->getColumns());
|
||
|
|
||
|
if ($sameColumns) {
|
||
|
if (! $this->samePartialIndex($other)) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if (! $this->hasSameColumnLengths($other)) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if (! $this->isUnique() && ! $this->isPrimary()) {
|
||
|
// this is a special case: If the current key is neither primary or unique, any unique or
|
||
|
// primary key will always have the same effect for the index and there cannot be any constraint
|
||
|
// overlaps. This means a primary or unique index can always fulfill the requirements of just an
|
||
|
// index that has no constraints.
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if ($other->isPrimary() !== $this->isPrimary()) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return $other->isUnique() === $this->isUnique();
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Detects if the other index is a non-unique, non primary index that can be overwritten by this one.
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function overrules(Index $other)
|
||
|
{
|
||
|
if ($other->isPrimary()) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if ($this->isSimpleIndex() && $other->isUnique()) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return $this->spansColumns($other->getColumns())
|
||
|
&& ($this->isPrimary() || $this->isUnique())
|
||
|
&& $this->samePartialIndex($other);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns platform specific flags for indexes.
|
||
|
*
|
||
|
* @return string[]
|
||
|
*/
|
||
|
public function getFlags()
|
||
|
{
|
||
|
return array_keys($this->_flags);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Adds Flag for an index that translates to platform specific handling.
|
||
|
*
|
||
|
* @param string $flag
|
||
|
*
|
||
|
* @return Index
|
||
|
*
|
||
|
* @example $index->addFlag('CLUSTERED')
|
||
|
*/
|
||
|
public function addFlag($flag)
|
||
|
{
|
||
|
$this->_flags[strtolower($flag)] = true;
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Does this index have a specific flag?
|
||
|
*
|
||
|
* @param string $flag
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function hasFlag($flag)
|
||
|
{
|
||
|
return isset($this->_flags[strtolower($flag)]);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Removes a flag.
|
||
|
*
|
||
|
* @param string $flag
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
public function removeFlag($flag)
|
||
|
{
|
||
|
unset($this->_flags[strtolower($flag)]);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $name
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function hasOption($name)
|
||
|
{
|
||
|
return isset($this->options[strtolower($name)]);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $name
|
||
|
*
|
||
|
* @return mixed
|
||
|
*/
|
||
|
public function getOption($name)
|
||
|
{
|
||
|
return $this->options[strtolower($name)];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return mixed[]
|
||
|
*/
|
||
|
public function getOptions()
|
||
|
{
|
||
|
return $this->options;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return whether the two indexes have the same partial index
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
private function samePartialIndex(Index $other)
|
||
|
{
|
||
|
if (
|
||
|
$this->hasOption('where')
|
||
|
&& $other->hasOption('where')
|
||
|
&& $this->getOption('where') === $other->getOption('where')
|
||
|
) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
return ! $this->hasOption('where') && ! $other->hasOption('where');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns whether the index has the same column lengths as the other
|
||
|
*/
|
||
|
private function hasSameColumnLengths(self $other): bool
|
||
|
{
|
||
|
$filter = static function (?int $length): bool {
|
||
|
return $length !== null;
|
||
|
};
|
||
|
|
||
|
return array_filter($this->options['lengths'] ?? [], $filter)
|
||
|
=== array_filter($other->options['lengths'] ?? [], $filter);
|
||
|
}
|
||
|
}
|