Seditio Source
Root |
./othercms/croogo-4.0.7/vendor/cakephp/cakephp/src/Database/Schema/SqlserverSchema.php
<?php
/**
 * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
 * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
 *
 * Licensed under The MIT License
 * For full copyright and license information, please see the LICENSE.txt
 * Redistributions of files must retain the above copyright notice.
 *
 * @copyright     Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
 * @link          https://cakephp.org CakePHP(tm) Project
 * @since         3.0.0
 * @license       https://opensource.org/licenses/mit-license.php MIT License
 */
namespace Cake\Database\Schema;

/**
 * Schema management/reflection features for SQLServer.
 */
class SqlserverSchema extends BaseSchema
{
   
/**
     * @var string
     */
   
const DEFAULT_SCHEMA_NAME = 'dbo';

   
/**
     * {@inheritDoc}
     */
   
public function listTablesSql($config)
    {
       
$sql = "SELECT TABLE_NAME
            FROM INFORMATION_SCHEMA.TABLES
            WHERE TABLE_SCHEMA = ?
            AND (TABLE_TYPE = 'BASE TABLE' OR TABLE_TYPE = 'VIEW')
            ORDER BY TABLE_NAME"
;
       
$schema = empty($config['schema']) ? static::DEFAULT_SCHEMA_NAME : $config['schema'];

        return [
$sql, [$schema]];
    }

   
/**
     * {@inheritDoc}
     */
   
public function describeColumnSql($tableName, $config)
    {
       
$sql = 'SELECT DISTINCT
            AC.column_id AS [column_id],
            AC.name AS [name],
            TY.name AS [type],
            AC.max_length AS [char_length],
            AC.precision AS [precision],
            AC.scale AS [scale],
            AC.is_identity AS [autoincrement],
            AC.is_nullable AS [null],
            OBJECT_DEFINITION(AC.default_object_id) AS [default],
            AC.collation_name AS [collation_name]
            FROM sys.[objects] T
            INNER JOIN sys.[schemas] S ON S.[schema_id] = T.[schema_id]
            INNER JOIN sys.[all_columns] AC ON T.[object_id] = AC.[object_id]
            INNER JOIN sys.[types] TY ON TY.[user_type_id] = AC.[user_type_id]
            WHERE T.[name] = ? AND S.[name] = ?
            ORDER BY column_id'
;

       
$schema = empty($config['schema']) ? static::DEFAULT_SCHEMA_NAME : $config['schema'];

        return [
$sql, [$tableName, $schema]];
    }

   
/**
     * Convert a column definition to the abstract types.
     *
     * The returned type will be a type that
     * Cake\Database\Type can handle.
     *
     * @param string $col The column type
     * @param int|null $length the column length
     * @param int|null $precision The column precision
     * @param int|null $scale The column scale
     * @return array Array of column information.
     * @link https://technet.microsoft.com/en-us/library/ms187752.aspx
     */
   
protected function _convertColumn($col, $length = null, $precision = null, $scale = null)
    {
       
$col = strtolower($col);
       
$length = $length !== null ? (int)$length : $length;
       
$precision = $precision !== null ? (int)$precision : $precision;
       
$scale = $scale !== null ? (int)$scale : $scale;

        if (
in_array($col, ['date', 'time'])) {
            return [
'type' => $col, 'length' => null];
        }
        if (
strpos($col, 'datetime') !== false) {
            return [
'type' => TableSchema::TYPE_TIMESTAMP, 'length' => null];
        }

        if (
$col === 'tinyint') {
            return [
'type' => TableSchema::TYPE_TINYINTEGER, 'length' => $precision ?: 3];
        }
        if (
$col === 'smallint') {
            return [
'type' => TableSchema::TYPE_SMALLINTEGER, 'length' => $precision ?: 5];
        }
        if (
$col === 'int' || $col === 'integer') {
            return [
'type' => TableSchema::TYPE_INTEGER, 'length' => $precision ?: 10];
        }
        if (
$col === 'bigint') {
            return [
'type' => TableSchema::TYPE_BIGINTEGER, 'length' => $precision ?: 20];
        }
        if (
$col === 'bit') {
            return [
'type' => TableSchema::TYPE_BOOLEAN, 'length' => null];
        }
        if (
           
strpos($col, 'numeric') !== false ||
           
strpos($col, 'money') !== false ||
           
strpos($col, 'decimal') !== false
       
) {
            return [
'type' => TableSchema::TYPE_DECIMAL, 'length' => $precision, 'precision' => $scale];
        }

        if (
$col === 'real' || $col === 'float') {
            return [
'type' => TableSchema::TYPE_FLOAT, 'length' => null];
        }
       
// SqlServer schema reflection returns double length for unicode
        // columns because internally it uses UTF16/UCS2
       
if ($col === 'nvarchar' || $col === 'nchar' || $col === 'ntext') {
           
$length /= 2;
        }
        if (
strpos($col, 'varchar') !== false && $length < 0) {
            return [
'type' => TableSchema::TYPE_TEXT, 'length' => null];
        }

        if (
strpos($col, 'varchar') !== false) {
            return [
'type' => TableSchema::TYPE_STRING, 'length' => $length ?: 255];
        }

        if (
strpos($col, 'char') !== false) {
            return [
'type' => TableSchema::TYPE_STRING, 'fixed' => true, 'length' => $length];
        }

        if (
strpos($col, 'text') !== false) {
            return [
'type' => TableSchema::TYPE_TEXT, 'length' => null];
        }

        if (
$col === 'image' || strpos($col, 'binary') !== false) {
           
// -1 is the value for MAX which we treat as a 'long' binary
           
if ($length == -1) {
               
$length = TableSchema::LENGTH_LONG;
            }

            return [
'type' => TableSchema::TYPE_BINARY, 'length' => $length];
        }

        if (
$col === 'uniqueidentifier') {
            return [
'type' => TableSchema::TYPE_UUID];
        }

        return [
'type' => TableSchema::TYPE_STRING, 'length' => null];
    }

   
/**
     * {@inheritDoc}
     */
   
public function convertColumnDescription(TableSchema $schema, $row)
    {
       
$field = $this->_convertColumn(
           
$row['type'],
           
$row['char_length'],
           
$row['precision'],
           
$row['scale']
        );
        if (!empty(
$row['default'])) {
           
$row['default'] = trim($row['default'], '()');
        }
        if (!empty(
$row['autoincrement'])) {
           
$field['autoIncrement'] = true;
        }
        if (
$field['type'] === TableSchema::TYPE_BOOLEAN) {
           
$row['default'] = (int)$row['default'];
        }

       
$field += [
           
'null' => $row['null'] === '1',
           
'default' => $this->_defaultValue($row['default']),
           
'collate' => $row['collation_name'],
        ];
       
$schema->addColumn($row['name'], $field);
    }

   
/**
     * Manipulate the default value.
     *
     * Sqlite includes quotes and bared NULLs in default values.
     * We need to remove those.
     *
     * @param string|null $default The default value.
     * @return string|null
     */
   
protected function _defaultValue($default)
    {
        if (
$default === 'NULL') {
            return
null;
        }

       
// Remove quotes
       
if (preg_match("/^N?'(.*)'/", $default, $matches)) {
            return
str_replace("''", "'", $matches[1]);
        }

        return
$default;
    }

   
/**
     * {@inheritDoc}
     */
   
public function describeIndexSql($tableName, $config)
    {
       
$sql = "SELECT
                I.[name] AS [index_name],
                IC.[index_column_id] AS [index_order],
                AC.[name] AS [column_name],
                I.[is_unique], I.[is_primary_key],
                I.[is_unique_constraint]
            FROM sys.[tables] AS T
            INNER JOIN sys.[schemas] S ON S.[schema_id] = T.[schema_id]
            INNER JOIN sys.[indexes] I ON T.[object_id] = I.[object_id]
            INNER JOIN sys.[index_columns] IC ON I.[object_id] = IC.[object_id] AND I.[index_id] = IC.[index_id]
            INNER JOIN sys.[all_columns] AC ON T.[object_id] = AC.[object_id] AND IC.[column_id] = AC.[column_id]
            WHERE T.[is_ms_shipped] = 0 AND I.[type_desc] <> 'HEAP' AND T.[name] = ? AND S.[name] = ?
            ORDER BY I.[index_id], IC.[index_column_id]"
;

       
$schema = empty($config['schema']) ? static::DEFAULT_SCHEMA_NAME : $config['schema'];

        return [
$sql, [$tableName, $schema]];
    }

   
/**
     * {@inheritDoc}
     */
   
public function convertIndexDescription(TableSchema $schema, $row)
    {
       
$type = TableSchema::INDEX_INDEX;
       
$name = $row['index_name'];
        if (
$row['is_primary_key']) {
           
$name = $type = TableSchema::CONSTRAINT_PRIMARY;
        }
        if (
$row['is_unique_constraint'] && $type === TableSchema::INDEX_INDEX) {
           
$type = TableSchema::CONSTRAINT_UNIQUE;
        }

        if (
$type === TableSchema::INDEX_INDEX) {
           
$existing = $schema->getIndex($name);
        } else {
           
$existing = $schema->getConstraint($name);
        }

       
$columns = [$row['column_name']];
        if (!empty(
$existing)) {
           
$columns = array_merge($existing['columns'], $columns);
        }

        if (
$type === TableSchema::CONSTRAINT_PRIMARY || $type === TableSchema::CONSTRAINT_UNIQUE) {
           
$schema->addConstraint($name, [
               
'type' => $type,
               
'columns' => $columns,
            ]);

            return;
        }
       
$schema->addIndex($name, [
           
'type' => $type,
           
'columns' => $columns,
        ]);
    }

   
/**
     * {@inheritDoc}
     */
   
public function describeForeignKeySql($tableName, $config)
    {
       
$sql = 'SELECT FK.[name] AS [foreign_key_name], FK.[delete_referential_action_desc] AS [delete_type],
                FK.[update_referential_action_desc] AS [update_type], C.name AS [column], RT.name AS [reference_table],
                RC.name AS [reference_column]
            FROM sys.foreign_keys FK
            INNER JOIN sys.foreign_key_columns FKC ON FKC.constraint_object_id = FK.object_id
            INNER JOIN sys.tables T ON T.object_id = FKC.parent_object_id
            INNER JOIN sys.tables RT ON RT.object_id = FKC.referenced_object_id
            INNER JOIN sys.schemas S ON S.schema_id = T.schema_id AND S.schema_id = RT.schema_id
            INNER JOIN sys.columns C ON C.column_id = FKC.parent_column_id AND C.object_id = FKC.parent_object_id
            INNER JOIN sys.columns RC ON RC.column_id = FKC.referenced_column_id AND RC.object_id = FKC.referenced_object_id
            WHERE FK.is_ms_shipped = 0 AND T.name = ? AND S.name = ?'
;

       
$schema = empty($config['schema']) ? static::DEFAULT_SCHEMA_NAME : $config['schema'];

        return [
$sql, [$tableName, $schema]];
    }

   
/**
     * {@inheritDoc}
     */
   
public function convertForeignKeyDescription(TableSchema $schema, $row)
    {
       
$data = [
           
'type' => TableSchema::CONSTRAINT_FOREIGN,
           
'columns' => [$row['column']],
           
'references' => [$row['reference_table'], $row['reference_column']],
           
'update' => $this->_convertOnClause($row['update_type']),
           
'delete' => $this->_convertOnClause($row['delete_type']),
        ];
       
$name = $row['foreign_key_name'];
       
$schema->addConstraint($name, $data);
    }

   
/**
     * {@inheritDoc}
     */
   
protected function _foreignOnClause($on)
    {
       
$parent = parent::_foreignOnClause($on);

        return
$parent === 'RESTRICT' ? parent::_foreignOnClause(TableSchema::ACTION_NO_ACTION) : $parent;
    }

   
/**
     * {@inheritDoc}
     */
   
protected function _convertOnClause($clause)
    {
        switch (
$clause) {
            case
'NO_ACTION':
                return
TableSchema::ACTION_NO_ACTION;
            case
'CASCADE':
                return
TableSchema::ACTION_CASCADE;
            case
'SET_NULL':
                return
TableSchema::ACTION_SET_NULL;
            case
'SET_DEFAULT':
                return
TableSchema::ACTION_SET_DEFAULT;
        }

        return
TableSchema::ACTION_SET_NULL;
    }

   
/**
     * {@inheritDoc}
     */
   
public function columnSql(TableSchema $schema, $name)
    {
       
$data = $schema->getColumn($name);
       
$out = $this->_driver->quoteIdentifier($name);
       
$typeMap = [
           
TableSchema::TYPE_TINYINTEGER => ' TINYINT',
           
TableSchema::TYPE_SMALLINTEGER => ' SMALLINT',
           
TableSchema::TYPE_INTEGER => ' INTEGER',
           
TableSchema::TYPE_BIGINTEGER => ' BIGINT',
           
TableSchema::TYPE_BINARY_UUID => ' UNIQUEIDENTIFIER',
           
TableSchema::TYPE_BOOLEAN => ' BIT',
           
TableSchema::TYPE_FLOAT => ' FLOAT',
           
TableSchema::TYPE_DECIMAL => ' DECIMAL',
           
TableSchema::TYPE_DATE => ' DATE',
           
TableSchema::TYPE_TIME => ' TIME',
           
TableSchema::TYPE_DATETIME => ' DATETIME',
           
TableSchema::TYPE_TIMESTAMP => ' DATETIME',
           
TableSchema::TYPE_UUID => ' UNIQUEIDENTIFIER',
           
TableSchema::TYPE_JSON => ' NVARCHAR(MAX)',
        ];

        if (isset(
$typeMap[$data['type']])) {
           
$out .= $typeMap[$data['type']];
        }

        if (
$data['type'] === TableSchema::TYPE_INTEGER || $data['type'] === TableSchema::TYPE_BIGINTEGER) {
            if ([
$name] === $schema->primaryKey() || $data['autoIncrement'] === true) {
                unset(
$data['null'], $data['default']);
               
$out .= ' IDENTITY(1, 1)';
            }
        }

        if (
$data['type'] === TableSchema::TYPE_TEXT && $data['length'] !== TableSchema::LENGTH_TINY) {
           
$out .= ' NVARCHAR(MAX)';
        }

        if (
$data['type'] === TableSchema::TYPE_BINARY) {
            if (
                !isset(
$data['length'])
                ||
in_array($data['length'], [TableSchema::LENGTH_MEDIUM, TableSchema::LENGTH_LONG], true)
            ) {
               
$data['length'] = 'MAX';
            }

            if (
$data['length'] === 1) {
               
$out .= ' BINARY(1)';
            } else {
               
$out .= ' VARBINARY';

               
$out .= sprintf('(%s)', $data['length']);
            }
        }

        if (
           
$data['type'] === TableSchema::TYPE_STRING ||
            (
$data['type'] === TableSchema::TYPE_TEXT && $data['length'] === TableSchema::LENGTH_TINY)
        ) {
           
$type = ' NVARCHAR';

            if (!empty(
$data['fixed'])) {
               
$type = ' NCHAR';
            }

            if (!isset(
$data['length'])) {
               
$data['length'] = 255;
            }

           
$out .= sprintf('%s(%d)', $type, $data['length']);
        }

       
$hasCollate = [TableSchema::TYPE_TEXT, TableSchema::TYPE_STRING];
        if (
in_array($data['type'], $hasCollate, true) && isset($data['collate']) && $data['collate'] !== '') {
           
$out .= ' COLLATE ' . $data['collate'];
        }

        if (
$data['type'] === TableSchema::TYPE_FLOAT && isset($data['precision'])) {
           
$out .= '(' . (int)$data['precision'] . ')';
        }

        if (
           
$data['type'] === TableSchema::TYPE_DECIMAL &&
            (isset(
$data['length']) || isset($data['precision']))
        ) {
           
$out .= '(' . (int)$data['length'] . ',' . (int)$data['precision'] . ')';
        }

        if (isset(
$data['null']) && $data['null'] === false) {
           
$out .= ' NOT NULL';
        }

        if (
            isset(
$data['default']) &&
           
in_array($data['type'], [TableSchema::TYPE_TIMESTAMP, TableSchema::TYPE_DATETIME]) &&
           
strtolower($data['default']) === 'current_timestamp'
       
) {
           
$out .= ' DEFAULT CURRENT_TIMESTAMP';
        } elseif (isset(
$data['default'])) {
           
$default = is_bool($data['default']) ? (int)$data['default'] : $this->_driver->schemaValue($data['default']);
           
$out .= ' DEFAULT ' . $default;
        } elseif (isset(
$data['null']) && $data['null'] !== false) {
           
$out .= ' DEFAULT NULL';
        }

        return
$out;
    }

   
/**
     * {@inheritDoc}
     */
   
public function addConstraintSql(TableSchema $schema)
    {
       
$sqlPattern = 'ALTER TABLE %s ADD %s;';
       
$sql = [];

        foreach (
$schema->constraints() as $name) {
           
$constraint = $schema->getConstraint($name);
            if (
$constraint['type'] === TableSchema::CONSTRAINT_FOREIGN) {
               
$tableName = $this->_driver->quoteIdentifier($schema->name());
               
$sql[] = sprintf($sqlPattern, $tableName, $this->constraintSql($schema, $name));
            }
        }

        return
$sql;
    }

   
/**
     * {@inheritDoc}
     */
   
public function dropConstraintSql(TableSchema $schema)
    {
       
$sqlPattern = 'ALTER TABLE %s DROP CONSTRAINT %s;';
       
$sql = [];

        foreach (
$schema->constraints() as $name) {
           
$constraint = $schema->getConstraint($name);
            if (
$constraint['type'] === TableSchema::CONSTRAINT_FOREIGN) {
               
$tableName = $this->_driver->quoteIdentifier($schema->name());
               
$constraintName = $this->_driver->quoteIdentifier($name);
               
$sql[] = sprintf($sqlPattern, $tableName, $constraintName);
            }
        }

        return
$sql;
    }

   
/**
     * {@inheritDoc}
     */
   
public function indexSql(TableSchema $schema, $name)
    {
       
$data = $schema->getIndex($name);
       
$columns = array_map(
            [
$this->_driver, 'quoteIdentifier'],
           
$data['columns']
        );

        return
sprintf(
           
'CREATE INDEX %s ON %s (%s)',
           
$this->_driver->quoteIdentifier($name),
           
$this->_driver->quoteIdentifier($schema->name()),
           
implode(', ', $columns)
        );
    }

   
/**
     * {@inheritDoc}
     */
   
public function constraintSql(TableSchema $schema, $name)
    {
       
$data = $schema->getConstraint($name);
       
$out = 'CONSTRAINT ' . $this->_driver->quoteIdentifier($name);
        if (
$data['type'] === TableSchema::CONSTRAINT_PRIMARY) {
           
$out = 'PRIMARY KEY';
        }
        if (
$data['type'] === TableSchema::CONSTRAINT_UNIQUE) {
           
$out .= ' UNIQUE';
        }

        return
$this->_keySql($out, $data);
    }

   
/**
     * Helper method for generating key SQL snippets.
     *
     * @param string $prefix The key prefix
     * @param array $data Key data.
     * @return string
     */
   
protected function _keySql($prefix, $data)
    {
       
$columns = array_map(
            [
$this->_driver, 'quoteIdentifier'],
           
$data['columns']
        );
        if (
$data['type'] === TableSchema::CONSTRAINT_FOREIGN) {
            return
$prefix . sprintf(
               
' FOREIGN KEY (%s) REFERENCES %s (%s) ON UPDATE %s ON DELETE %s',
               
implode(', ', $columns),
               
$this->_driver->quoteIdentifier($data['references'][0]),
               
$this->_convertConstraintColumns($data['references'][1]),
               
$this->_foreignOnClause($data['update']),
               
$this->_foreignOnClause($data['delete'])
            );
        }

        return
$prefix . ' (' . implode(', ', $columns) . ')';
    }

   
/**
     * {@inheritDoc}
     */
   
public function createTableSql(TableSchema $schema, $columns, $constraints, $indexes)
    {
       
$content = array_merge($columns, $constraints);
       
$content = implode(",\n", array_filter($content));
       
$tableName = $this->_driver->quoteIdentifier($schema->name());
       
$out = [];
       
$out[] = sprintf("CREATE TABLE %s (\n%s\n)", $tableName, $content);
        foreach (
$indexes as $index) {
           
$out[] = $index;
        }

        return
$out;
    }

   
/**
     * {@inheritDoc}
     */
   
public function truncateTableSql(TableSchema $schema)
    {
       
$name = $this->_driver->quoteIdentifier($schema->name());
       
$queries = [
           
sprintf('DELETE FROM %s', $name),
        ];

       
// Restart identity sequences
       
$pk = $schema->primaryKey();
        if (
count($pk) === 1) {
           
$column = $schema->getColumn($pk[0]);
            if (
in_array($column['type'], ['integer', 'biginteger'])) {
               
$queries[] = sprintf(
                   
"DBCC CHECKIDENT('%s', RESEED, 0)",
                   
$schema->name()
                );
            }
        }

        return
$queries;
    }
}