| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591 |
- <?php
- /**
- * CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
- * Copyright (c) Cake Software Foundation, Inc. (http://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. (http://cakefoundation.org)
- * @link http://cakephp.org CakePHP(tm) Project
- * @since 3.0.0
- * @license http://www.opensource.org/licenses/mit-license.php MIT License
- */
- namespace Cake\Database\Schema;
- use Cake\Database\Exception;
- /**
- * Schema management/reflection features for Postgres.
- */
- class PostgresSchema extends BaseSchema
- {
- /**
- * {@inheritDoc}
- */
- public function listTablesSql($config)
- {
- $sql = 'SELECT table_name as name FROM information_schema.tables WHERE table_schema = ? ORDER BY name';
- $schema = empty($config['schema']) ? 'public' : $config['schema'];
- return [$sql, [$schema]];
- }
- /**
- * {@inheritDoc}
- */
- public function describeColumnSql($tableName, $config)
- {
- $sql = 'SELECT DISTINCT table_schema AS schema,
- column_name AS name,
- data_type AS type,
- is_nullable AS null, column_default AS default,
- character_maximum_length AS char_length,
- c.collation_name,
- d.description as comment,
- ordinal_position,
- c.numeric_precision as column_precision,
- c.numeric_scale as column_scale,
- pg_get_serial_sequence(attr.attrelid::regclass::text, attr.attname) IS NOT NULL AS has_serial
- FROM information_schema.columns c
- INNER JOIN pg_catalog.pg_namespace ns ON (ns.nspname = table_schema)
- INNER JOIN pg_catalog.pg_class cl ON (cl.relnamespace = ns.oid AND cl.relname = table_name)
- LEFT JOIN pg_catalog.pg_index i ON (i.indrelid = cl.oid AND i.indkey[0] = c.ordinal_position)
- LEFT JOIN pg_catalog.pg_description d on (cl.oid = d.objoid AND d.objsubid = c.ordinal_position)
- LEFT JOIN pg_catalog.pg_attribute attr ON (cl.oid = attr.attrelid AND column_name = attr.attname)
- WHERE table_name = ? AND table_schema = ? AND table_catalog = ?
- ORDER BY ordinal_position';
- $schema = empty($config['schema']) ? 'public' : $config['schema'];
- return [$sql, [$tableName, $schema, $config['database']]];
- }
- /**
- * Convert a column definition to the abstract types.
- *
- * The returned type will be a type that
- * Cake\Database\Type can handle.
- *
- * @param string $column The column type + length
- * @throws \Cake\Database\Exception when column cannot be parsed.
- * @return array Array of column information.
- */
- protected function _convertColumn($column)
- {
- preg_match('/([a-z\s]+)(?:\(([0-9,]+)\))?/i', $column, $matches);
- if (empty($matches)) {
- throw new Exception(sprintf('Unable to parse column type from "%s"', $column));
- }
- $col = strtolower($matches[1]);
- $length = null;
- if (isset($matches[2])) {
- $length = (int)$matches[2];
- }
- if (in_array($col, ['date', 'time', 'boolean'])) {
- return ['type' => $col, 'length' => null];
- }
- if (strpos($col, 'timestamp') !== false) {
- return ['type' => 'timestamp', 'length' => null];
- }
- if (strpos($col, 'time') !== false) {
- return ['type' => 'time', 'length' => null];
- }
- if ($col === 'serial' || $col === 'integer') {
- return ['type' => 'integer', 'length' => 10];
- }
- if ($col === 'bigserial' || $col === 'bigint') {
- return ['type' => 'biginteger', 'length' => 20];
- }
- if ($col === 'smallint') {
- return ['type' => 'integer', 'length' => 5];
- }
- if ($col === 'inet') {
- return ['type' => 'string', 'length' => 39];
- }
- if ($col === 'uuid') {
- return ['type' => 'uuid', 'length' => null];
- }
- if ($col === 'char' || $col === 'character') {
- return ['type' => 'string', 'fixed' => true, 'length' => $length];
- }
- // money is 'string' as it includes arbitrary text content
- // before the number value.
- if (strpos($col, 'char') !== false ||
- strpos($col, 'money') !== false
- ) {
- return ['type' => 'string', 'length' => $length];
- }
- if (strpos($col, 'text') !== false) {
- return ['type' => 'text', 'length' => null];
- }
- if ($col === 'bytea') {
- return ['type' => 'binary', 'length' => null];
- }
- if ($col === 'real' || strpos($col, 'double') !== false) {
- return ['type' => 'float', 'length' => null];
- }
- if (strpos($col, 'numeric') !== false ||
- strpos($col, 'decimal') !== false
- ) {
- return ['type' => 'decimal', 'length' => null];
- }
- if (strpos($col, 'json') !== false) {
- return ['type' => 'json', 'length' => null];
- }
- return ['type' => 'text', 'length' => null];
- }
- /**
- * {@inheritDoc}
- */
- public function convertColumnDescription(Table $table, $row)
- {
- $field = $this->_convertColumn($row['type']);
- if ($field['type'] === 'boolean') {
- if ($row['default'] === 'true') {
- $row['default'] = 1;
- }
- if ($row['default'] === 'false') {
- $row['default'] = 0;
- }
- }
- if (!empty($row['has_serial'])) {
- $field['autoIncrement'] = true;
- }
- $field += [
- 'default' => $this->_defaultValue($row['default']),
- 'null' => $row['null'] === 'YES' ? true : false,
- 'collate' => $row['collation_name'],
- 'comment' => $row['comment']
- ];
- $field['length'] = $row['char_length'] ?: $field['length'];
- if ($field['type'] === 'numeric' || $field['type'] === 'decimal') {
- $field['length'] = $row['column_precision'];
- $field['precision'] = $row['column_scale'] ? $row['column_scale'] : null;
- }
- $table->addColumn($row['name'], $field);
- }
- /**
- * Manipulate the default value.
- *
- * Postgres includes sequence data and casting information in default values.
- * We need to remove those.
- *
- * @param string|null $default The default value.
- * @return string|null
- */
- protected function _defaultValue($default)
- {
- if (is_numeric($default) || $default === null) {
- return $default;
- }
- // Sequences
- if (strpos($default, 'nextval') === 0) {
- return null;
- }
- // Remove quotes and postgres casts
- return preg_replace(
- "/^'(.*)'(?:::.*)$/",
- "$1",
- $default
- );
- }
- /**
- * {@inheritDoc}
- */
- public function describeIndexSql($tableName, $config)
- {
- $sql = "SELECT
- c2.relname,
- a.attname,
- i.indisprimary,
- i.indisunique
- FROM pg_catalog.pg_namespace n
- INNER JOIN pg_catalog.pg_class c ON (n.oid = c.relnamespace)
- INNER JOIN pg_catalog.pg_index i ON (c.oid = i.indrelid)
- INNER JOIN pg_catalog.pg_class c2 ON (c2.oid = i.indexrelid)
- INNER JOIN pg_catalog.pg_attribute a ON (a.attrelid = c.oid AND i.indrelid::regclass = a.attrelid::regclass)
- WHERE n.nspname = ?
- AND a.attnum = ANY(i.indkey)
- AND c.relname = ?
- ORDER BY i.indisprimary DESC, i.indisunique DESC, c.relname, a.attnum";
- $schema = 'public';
- if (!empty($config['schema'])) {
- $schema = $config['schema'];
- }
- return [$sql, [$schema, $tableName]];
- }
- /**
- * {@inheritDoc}
- */
- public function convertIndexDescription(Table $table, $row)
- {
- $type = Table::INDEX_INDEX;
- $name = $row['relname'];
- if ($row['indisprimary']) {
- $name = $type = Table::CONSTRAINT_PRIMARY;
- }
- if ($row['indisunique'] && $type === Table::INDEX_INDEX) {
- $type = Table::CONSTRAINT_UNIQUE;
- }
- if ($type === Table::CONSTRAINT_PRIMARY || $type === Table::CONSTRAINT_UNIQUE) {
- $this->_convertConstraint($table, $name, $type, $row);
- return;
- }
- $index = $table->index($name);
- if (!$index) {
- $index = [
- 'type' => $type,
- 'columns' => []
- ];
- }
- $index['columns'][] = $row['attname'];
- $table->addIndex($name, $index);
- }
- /**
- * Add/update a constraint into the schema object.
- *
- * @param \Cake\Database\Schema\Table $table The table to update.
- * @param string $name The index name.
- * @param string $type The index type.
- * @param array $row The metadata record to update with.
- * @return void
- */
- protected function _convertConstraint($table, $name, $type, $row)
- {
- $constraint = $table->constraint($name);
- if (!$constraint) {
- $constraint = [
- 'type' => $type,
- 'columns' => []
- ];
- }
- $constraint['columns'][] = $row['attname'];
- $table->addConstraint($name, $constraint);
- }
- /**
- * {@inheritDoc}
- */
- public function describeForeignKeySql($tableName, $config)
- {
- $sql = "SELECT
- c.conname AS name,
- c.contype AS type,
- a.attname AS column_name,
- c.confmatchtype AS match_type,
- c.confupdtype AS on_update,
- c.confdeltype AS on_delete,
- c.confrelid::regclass AS references_table,
- ab.attname AS references_field
- FROM pg_catalog.pg_namespace n
- INNER JOIN pg_catalog.pg_class cl ON (n.oid = cl.relnamespace)
- INNER JOIN pg_catalog.pg_constraint c ON (n.oid = c.connamespace)
- INNER JOIN pg_catalog.pg_attribute a ON (a.attrelid = cl.oid AND c.conrelid = a.attrelid AND a.attnum = ANY(c.conkey))
- INNER JOIN pg_catalog.pg_attribute ab ON (a.attrelid = cl.oid AND c.confrelid = ab.attrelid AND ab.attnum = ANY(c.confkey))
- WHERE n.nspname = ?
- AND cl.relname = ?
- ORDER BY name, a.attnum, ab.attnum DESC";
- $schema = empty($config['schema']) ? 'public' : $config['schema'];
- return [$sql, [$schema, $tableName]];
- }
- /**
- * {@inheritDoc}
- */
- public function convertForeignKeyDescription(Table $table, $row)
- {
- $data = [
- 'type' => Table::CONSTRAINT_FOREIGN,
- 'columns' => $row['column_name'],
- 'references' => [$row['references_table'], $row['references_field']],
- 'update' => $this->_convertOnClause($row['on_update']),
- 'delete' => $this->_convertOnClause($row['on_delete']),
- ];
- $table->addConstraint($row['name'], $data);
- }
- /**
- * {@inheritDoc}
- */
- protected function _convertOnClause($clause)
- {
- if ($clause === 'r') {
- return Table::ACTION_RESTRICT;
- }
- if ($clause === 'a') {
- return Table::ACTION_NO_ACTION;
- }
- if ($clause === 'c') {
- return Table::ACTION_CASCADE;
- }
- return Table::ACTION_SET_NULL;
- }
- /**
- * {@inheritDoc}
- */
- public function columnSql(Table $table, $name)
- {
- $data = $table->column($name);
- $out = $this->_driver->quoteIdentifier($name);
- $typeMap = [
- 'boolean' => ' BOOLEAN',
- 'binary' => ' BYTEA',
- 'float' => ' FLOAT',
- 'decimal' => ' DECIMAL',
- 'date' => ' DATE',
- 'time' => ' TIME',
- 'datetime' => ' TIMESTAMP',
- 'timestamp' => ' TIMESTAMP',
- 'uuid' => ' UUID',
- 'json' => ' JSONB'
- ];
- if (isset($typeMap[$data['type']])) {
- $out .= $typeMap[$data['type']];
- }
- if ($data['type'] === 'integer' || $data['type'] === 'biginteger') {
- $type = $data['type'] === 'integer' ? ' INTEGER' : ' BIGINT';
- if ([$name] === $table->primaryKey() || $data['autoIncrement'] === true) {
- $type = $data['type'] === 'integer' ? ' SERIAL' : ' BIGSERIAL';
- unset($data['null'], $data['default']);
- }
- $out .= $type;
- }
- if ($data['type'] === 'text' && $data['length'] !== Table::LENGTH_TINY) {
- $out .= ' TEXT';
- }
- if ($data['type'] === 'string' || ($data['type'] === 'text' && $data['length'] === Table::LENGTH_TINY)) {
- $isFixed = !empty($data['fixed']);
- $type = ' VARCHAR';
- if ($isFixed) {
- $type = ' CHAR';
- }
- $out .= $type;
- if (isset($data['length']) && $data['length'] != 36) {
- $out .= '(' . (int)$data['length'] . ')';
- }
- }
- $hasCollate = ['text', 'string'];
- if (in_array($data['type'], $hasCollate, true) && isset($data['collate']) && $data['collate'] !== '') {
- $out .= ' COLLATE "' . $data['collate'] . '"';
- }
- if ($data['type'] === 'float' && isset($data['precision'])) {
- $out .= '(' . (int)$data['precision'] . ')';
- }
- if ($data['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';
- } elseif (isset($data['null']) && $data['null'] === true) {
- $out .= ' NULL';
- }
- if (isset($data['default']) &&
- in_array($data['type'], ['timestamp', 'datetime']) &&
- strtolower($data['default']) === 'current_timestamp') {
- $out .= ' DEFAULT CURRENT_TIMESTAMP';
- } elseif (isset($data['default'])) {
- $defaultValue = $data['default'];
- if ($data['type'] === 'boolean') {
- $defaultValue = (bool)$defaultValue;
- }
- $out .= ' DEFAULT ' . $this->_driver->schemaValue($defaultValue);
- } elseif (isset($data['null']) && $data['null'] === true) {
- $out .= ' DEFAULT NULL';
- }
- return $out;
- }
- /**
- * {@inheritDoc}
- */
- public function addConstraintSql(Table $table)
- {
- $sqlPattern = 'ALTER TABLE %s ADD %s;';
- $sql = [];
- foreach ($table->constraints() as $name) {
- $constraint = $table->constraint($name);
- if ($constraint['type'] === Table::CONSTRAINT_FOREIGN) {
- $tableName = $this->_driver->quoteIdentifier($table->name());
- $sql[] = sprintf($sqlPattern, $tableName, $this->constraintSql($table, $name));
- }
- }
- return $sql;
- }
- /**
- * {@inheritDoc}
- */
- public function dropConstraintSql(Table $table)
- {
- $sqlPattern = 'ALTER TABLE %s DROP CONSTRAINT %s;';
- $sql = [];
- foreach ($table->constraints() as $name) {
- $constraint = $table->constraint($name);
- if ($constraint['type'] === Table::CONSTRAINT_FOREIGN) {
- $tableName = $this->_driver->quoteIdentifier($table->name());
- $constraintName = $this->_driver->quoteIdentifier($name);
- $sql[] = sprintf($sqlPattern, $tableName, $constraintName);
- }
- }
- return $sql;
- }
- /**
- * {@inheritDoc}
- */
- public function indexSql(Table $table, $name)
- {
- $data = $table->index($name);
- $columns = array_map(
- [$this->_driver, 'quoteIdentifier'],
- $data['columns']
- );
- return sprintf(
- 'CREATE INDEX %s ON %s (%s)',
- $this->_driver->quoteIdentifier($name),
- $this->_driver->quoteIdentifier($table->name()),
- implode(', ', $columns)
- );
- }
- /**
- * {@inheritDoc}
- */
- public function constraintSql(Table $table, $name)
- {
- $data = $table->constraint($name);
- $out = 'CONSTRAINT ' . $this->_driver->quoteIdentifier($name);
- if ($data['type'] === Table::CONSTRAINT_PRIMARY) {
- $out = 'PRIMARY KEY';
- }
- if ($data['type'] === Table::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'] === Table::CONSTRAINT_FOREIGN) {
- return $prefix . sprintf(
- ' FOREIGN KEY (%s) REFERENCES %s (%s) ON UPDATE %s ON DELETE %s DEFERRABLE INITIALLY IMMEDIATE',
- 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(Table $table, $columns, $constraints, $indexes)
- {
- $content = array_merge($columns, $constraints);
- $content = implode(",\n", array_filter($content));
- $tableName = $this->_driver->quoteIdentifier($table->name());
- $temporary = $table->temporary() ? ' TEMPORARY ' : ' ';
- $out = [];
- $out[] = sprintf("CREATE%sTABLE %s (\n%s\n)", $temporary, $tableName, $content);
- foreach ($indexes as $index) {
- $out[] = $index;
- }
- foreach ($table->columns() as $column) {
- $columnData = $table->column($column);
- if (isset($columnData['comment'])) {
- $out[] = sprintf(
- 'COMMENT ON COLUMN %s.%s IS %s',
- $tableName,
- $this->_driver->quoteIdentifier($column),
- $this->_driver->schemaValue($columnData['comment'])
- );
- }
- }
- return $out;
- }
- /**
- * {@inheritDoc}
- */
- public function truncateTableSql(Table $table)
- {
- $name = $this->_driver->quoteIdentifier($table->name());
- return [
- sprintf('TRUNCATE %s RESTART IDENTITY CASCADE', $name)
- ];
- }
- /**
- * Generate the SQL to drop a table.
- *
- * @param \Cake\Database\Schema\Table $table Table instance
- * @return array SQL statements to drop a table.
- */
- public function dropTableSql(Table $table)
- {
- $sql = sprintf(
- 'DROP TABLE %s CASCADE',
- $this->_driver->quoteIdentifier($table->name())
- );
- return [$sql];
- }
- }
|