$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) { 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_SET_NULL) : $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; } }