$config The connection configuration to use for * getting tables from. * @return array An array of (sql, params) to execute. */ public function listTablesSql(array $config): array { $sql = 'SELECT table_name as name FROM information_schema.tables WHERE table_schema = ? ORDER BY name'; $schema = $config['schema'] ?? 'public'; return [$sql, [$schema]]; } /** * Generate the SQL to list the tables, excluding all views. * * @param array $config The connection configuration to use for * getting tables from. * @return array An array of (sql, params) to execute. */ public function listTablesWithoutViewsSql(array $config): array { $sql = 'SELECT table_name as name FROM information_schema.tables WHERE table_schema = ? AND table_type = \'BASE TABLE\' ORDER BY name'; $schema = $config['schema'] ?? 'public'; return [$sql, [$schema]]; } /** * @inheritDoc */ public function describeColumnSql(string $tableName, array $config): array { $sql = $this->describeColumnQuery(); $schema = $config['schema'] ?? 'public'; return [$sql, [$tableName, $schema, $config['database']]]; } /** * Helper method for creating SQL to describe columns in a table. * * @return string SQL to reflect columns */ private function describeColumnQuery(): string { return 'SELECT DISTINCT table_schema AS schema, column_name AS name, data_type AS type, udt_name, is_identity, is_nullable AS null, column_default AS default, character_maximum_length AS char_length, c.collation_name, d.description as comment, ordinal_position, c.datetime_precision, c.numeric_precision as column_precision, c.numeric_scale as column_scale, c.identity_generation, 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'; } /** * Convert a column definition to the abstract types. * * The returned type will be a type that * Cake\Database\TypeFactory can handle. * * @param string $column The column type + length * @throws \Cake\Database\Exception\DatabaseException when column cannot be parsed. * @return array Array of column information. */ protected function _convertColumn(string $column): array { preg_match('/([a-z\s]+)(?:\(([0-9,]+)\))?/i', $column, $matches); if (!$matches) { throw new DatabaseException(sprintf('Unable to parse column type from `%s`', $column)); } $col = strtolower($matches[1]); $length = null; $precision = null; $scale = null; if (isset($matches[2])) { $length = (int)$matches[2]; } $type = $this->_applyTypeSpecificColumnConversion( $col, compact('length', 'precision', 'scale'), ); if ($type !== null) { return $type; } if (in_array($col, ['date', 'time', 'boolean', 'inet', 'cidr', 'macaddr', 'citext', 'interval'], true)) { return ['type' => $col, 'length' => null]; } if (in_array($col, ['timestamptz', 'timestamp with time zone'], true)) { return ['type' => TableSchemaInterface::TYPE_TIMESTAMP_TIMEZONE, 'length' => null]; } if (str_contains($col, 'timestamp')) { return ['type' => TableSchemaInterface::TYPE_TIMESTAMP_FRACTIONAL, 'length' => null]; } if (str_contains($col, 'time')) { return ['type' => TableSchemaInterface::TYPE_TIME, 'length' => null]; } if ($col === 'serial' || $col === 'integer') { return ['type' => TableSchemaInterface::TYPE_INTEGER, 'length' => 10]; } if ($col === 'bigserial' || $col === 'bigint') { return ['type' => TableSchemaInterface::TYPE_BIGINTEGER, 'length' => 20]; } if ($col === 'smallint') { return ['type' => TableSchemaInterface::TYPE_SMALLINTEGER, 'length' => 5]; } if ($col === 'uuid') { return ['type' => TableSchemaInterface::TYPE_UUID, 'length' => null]; } if ($col === 'char') { return ['type' => TableSchemaInterface::TYPE_CHAR, 'length' => $length]; } if (str_contains($col, 'character')) { return ['type' => TableSchemaInterface::TYPE_STRING, 'length' => $length]; } // money is 'string' as it includes arbitrary text content // before the number value. if (str_contains($col, 'money') || $col === 'string') { return ['type' => TableSchemaInterface::TYPE_STRING, 'length' => $length]; } if (str_contains($col, 'text')) { return ['type' => TableSchemaInterface::TYPE_TEXT, 'length' => null]; } if ($col === 'bytea') { return ['type' => TableSchemaInterface::TYPE_BINARY, 'length' => null]; } if ($col === 'real' || str_contains($col, 'double')) { return ['type' => TableSchemaInterface::TYPE_FLOAT, 'length' => null]; } if (str_contains($col, 'numeric') || str_contains($col, 'decimal')) { return ['type' => TableSchemaInterface::TYPE_DECIMAL, 'length' => null]; } if (str_contains($col, 'json')) { return ['type' => TableSchemaInterface::TYPE_JSON, 'length' => null]; } if (in_array($col, ['geometry', 'geography'])) { return ['type' => $col, 'length' => null]; } $length = is_numeric($length) ? $length : null; return ['type' => TableSchemaInterface::TYPE_STRING, 'length' => $length]; } /** * @inheritDoc */ public function convertColumnDescription(TableSchema $schema, array $row): void { $field = $this->_convertColumn($row['type']); if ($field['type'] === TableSchemaInterface::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', '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'] ?: null; } if ($field['type'] === TableSchemaInterface::TYPE_TIMESTAMP_FRACTIONAL) { $field['precision'] = $row['datetime_precision']; if ($field['precision'] === 0) { $field['type'] = TableSchemaInterface::TYPE_TIMESTAMP; } } if ($field['type'] === TableSchemaInterface::TYPE_TIMESTAMP_TIMEZONE) { $field['precision'] = $row['datetime_precision']; } $schema->addColumn($row['name'], $field); } /** * Split a tablename into a tuple of schema, table * If the table does not have a schema name included, the connection * schema will be used. * * @param string $tableName The table name to split * @param array $config Additional configuration data * @return array A tuple of [schema, tablename] */ private function splitTablename(string $tableName, array $config = []): array { if (str_contains($tableName, '.')) { return explode('.', $tableName); } $driverConfig = $this->_driver->config(); $schema = $config['schema'] ?? $driverConfig['schema'] ?? 'public'; return [$schema, $tableName]; } /** * @inheritDoc */ public function describeColumns(string $tableName): array { $config = $this->_driver->config(); [$schema, $name] = $this->splitTablename($tableName); $sql = $this->describeColumnQuery(); $statement = $this->_driver->execute($sql, [$name, $schema, $config['database']]); $columns = []; foreach ($statement->fetchAll('assoc') as $row) { $type = $row['type']; if ($type === 'USER-DEFINED') { $type = $row['udt_name']; } $field = $this->_convertColumn($type); if ($field['type'] === TableSchemaInterface::TYPE_BOOLEAN) { if ($row['default'] === 'true') { $row['default'] = 1; } elseif ($row['default'] === 'false') { $row['default'] = 0; } } if (!empty($row['has_serial'])) { $field['autoIncrement'] = true; } $field += [ 'name' => $row['name'], 'default' => $this->_defaultValue($row['default']), 'null' => $row['null'] === 'YES', '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'] ?: null; } if ($field['type'] === TableSchemaInterface::TYPE_TIMESTAMP_FRACTIONAL) { $field['precision'] = $row['datetime_precision']; if ($field['precision'] === 0) { $field['type'] = TableSchemaInterface::TYPE_TIMESTAMP; } } if ($field['type'] === TableSchemaInterface::TYPE_TIMESTAMP_TIMEZONE) { $field['precision'] = $row['datetime_precision']; } if (isset($row['identity_generation']) && $row['identity_generation']) { $field['generated'] = $row['identity_generation']; } $columns[] = $field; } return $columns; } /** * Manipulate the default value. * * Postgres includes sequence data and casting information in default values. * We need to remove those. * * @param string|int|null $default The default value. * @return string|int|null */ protected function _defaultValue(string|int|null $default): string|int|null { if (is_numeric($default) || $default === null) { return $default; } // Sequences if (str_starts_with($default, 'nextval')) { return null; } if (str_starts_with($default, 'NULL::')) { return null; } // Remove quotes and postgres casts return preg_replace( "/^'(.*)'(?:::.*)$/", '$1', $default, ); } /** * Get the query to describe indexes * * @return string */ private function describeIndexQuery(): string { return 'SELECT c2.relname, a.attname, i.indisprimary, i.indisunique, i.indnkeyatts 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'; } /** * @inheritDoc */ public function describeIndexSql(string $tableName, array $config): array { $sql = $this->describeIndexQuery(); [$schema, $name] = $this->splitTablename($tableName, $config); return [$sql, [$schema, $name]]; } /** * @inheritDoc */ public function convertIndexDescription(TableSchema $schema, array $row): void { $type = TableSchema::INDEX_INDEX; $name = $row['relname']; if ($row['indisprimary']) { $name = TableSchema::CONSTRAINT_PRIMARY; $type = TableSchema::CONSTRAINT_PRIMARY; } if ($row['indisunique'] && $type === TableSchema::INDEX_INDEX) { $type = TableSchema::CONSTRAINT_UNIQUE; } if ($type === TableSchema::CONSTRAINT_PRIMARY || $type === TableSchema::CONSTRAINT_UNIQUE) { $this->_convertConstraint($schema, $name, $type, $row); return; } $index = $schema->getIndex($name); if (!$index) { $index = [ 'type' => $type, 'columns' => [], ]; } $index['columns'][] = $row['attname']; $schema->addIndex($name, $index); } /** * @inheritDoc */ public function describeIndexes(string $tableName): array { [$schema, $name] = $this->splitTablename($tableName); $sql = $this->describeIndexQuery(); $indexes = []; $statement = $this->_driver->execute($sql, [$schema, $name]); foreach ($statement->fetchAll('assoc') as $row) { $type = TableSchema::INDEX_INDEX; $name = $row['relname']; $constraint = null; $includeColumnIndex = $row['indnkeyatts']; if ($row['indisprimary']) { $constraint = $name; $name = TableSchema::CONSTRAINT_PRIMARY; $type = TableSchema::CONSTRAINT_PRIMARY; } if ($row['indisunique'] && $type === TableSchema::INDEX_INDEX) { $type = TableSchema::CONSTRAINT_UNIQUE; } if (!isset($indexes[$name])) { $indexes[$name] = [ 'name' => $name, 'type' => $type, 'columns' => [], 'length' => [], ]; } if ($constraint) { $indexes[$name]['constraint'] = $constraint; } if (count($indexes[$name]['columns']) < $includeColumnIndex) { $indexes[$name]['columns'][] = $row['attname']; } else { $indexes[$name]['include'][] = $row['attname']; } } return array_values($indexes); } /** * Add/update a constraint into the schema object. * * @param \Cake\Database\Schema\TableSchema $schema 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(TableSchema $schema, string $name, string $type, array $row): void { $constraint = $schema->getConstraint($name); if (!$constraint) { $constraint = [ 'type' => $type, 'columns' => [], ]; } $constraint['columns'][] = $row['attname']; $schema->addConstraint($name, $constraint); } /** * @inheritDoc */ public function describeForeignKeySql(string $tableName, array $config): array { $sql = $this->describeForeignKeyQuery(); [$schema, $name] = $this->splitTablename($tableName, $config); return [$sql, [$schema, $name]]; } /** * @inheritDoc */ public function convertForeignKeyDescription(TableSchema $schema, array $row): void { $data = [ 'type' => TableSchema::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']), 'deferrable' => $this->convertDeferrable($row), ]; $schema->addConstraint($row['name'], $data); } /** * @inheritDoc */ public function describeForeignKeys(string $tableName): array { [$schema, $name] = $this->splitTablename($tableName); $sql = $this->describeForeignKeyQuery(); $keys = []; $statement = $this->_driver->execute($sql, [$schema, $name]); foreach ($statement->fetchAll('assoc') as $row) { $name = $row['name']; if (!isset($keys[$name])) { $keys[$name] = [ 'name' => $name, 'type' => TableSchema::CONSTRAINT_FOREIGN, 'columns' => [], 'references' => [$row['references_table'], []], 'update' => $this->_convertOnClause($row['on_update']), 'delete' => $this->_convertOnClause($row['on_delete']), 'deferrable' => $this->convertDeferrable($row), ]; } // column indexes start at 1 $columnOrder = $row['column_order'] - 1; $referencedColumnOrder = $row['references_field_order'] - 1; $keys[$name]['columns'][$columnOrder] = $row['column_name']; $keys[$name]['references'][1][$referencedColumnOrder] = $row['references_field']; } foreach ($keys as $id => $key) { // references.1 is the referenced columns. Backwards compat // requires a single column to be a string, but multiple to be an array. if (count($key['references'][1]) === 1) { $keys[$id]['references'][1] = $key['references'][1][0]; } } return array_values($keys); } /** * Get the query to describe foreign keys * * @return string */ private function describeForeignKeyQuery(): string { // phpcs:disable Generic.Files.LineLength $sql = 'SELECT c.conname AS name, c.contype AS type, a.attname AS column_name, array_position(c.conkey, a.attnum) AS column_order, 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, array_position(c.confkey, ab.attnum) AS references_field_order, c.condeferrable AS deferrable, c.condeferred AS initially_deferred 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, column_order ASC, references_field_order ASC'; // phpcs:enable Generic.Files.LineLength return $sql; } /** * @inheritDoc */ public function describeCheckConstraints(string $tableName): array { [$schema, $name] = $this->splitTablename($tableName); $sql = 'SELECT con.conname AS name, pg_get_constraintdef(con.oid) AS expression FROM pg_catalog.pg_constraint AS con INNER JOIN pg_catalog.pg_namespace AS ns ON (ns.oid = con.connamespace) INNER JOIN pg_catalog.pg_class AS cls ON (cls.oid = con.conrelid) WHERE ns.nspname = ? AND cls.relname = ? AND con.contype = \'c\''; $results = []; $statement = $this->_driver->execute($sql, [$schema, $name]); foreach ($statement->fetchAll('assoc') as $row) { $expression = preg_replace('/^CHECK \(\((.*)\)\)$/i', '$1', $row['expression']); $results[] = [ 'name' => $row['name'], 'type' => TableSchema::CONSTRAINT_CHECK, 'expression' => $expression, ]; } return $results; } /** * @inheritDoc */ public function describeOptions(string $tableName): array { return []; } /** * @inheritDoc */ protected function _convertOnClause(string $clause): string { if ($clause === 'r') { return TableSchema::ACTION_RESTRICT; } if ($clause === 'a') { return TableSchema::ACTION_NO_ACTION; } if ($clause === 'c') { return TableSchema::ACTION_CASCADE; } return TableSchema::ACTION_SET_NULL; } /** * Convert deferrable option from the postgres metadata into a string * * @param array $row The row to convert. * @return string|null The deferrable value or null if not deferrable. */ protected function convertDeferrable(array $row): ?string { if (!isset($row['deferrable'])) { return null; } if (!$row['deferrable']) { return ForeignKey::NOT_DEFERRED; } if (isset($row['initially_deferred']) && $row['initially_deferred']) { return ForeignKey::DEFERRED; } return ForeignKey::IMMEDIATE; } /** * @inheritDoc */ public function columnSql(TableSchema $schema, string $name): string { $data = $schema->getColumn($name); assert($data !== null); $data['name'] = $name; $sql = $this->_getTypeSpecificColumnSql($data['type'], $schema, $name); if ($sql !== null) { return $sql; } $autoIncrementTypes = [ TableSchemaInterface::TYPE_TINYINTEGER, TableSchemaInterface::TYPE_SMALLINTEGER, TableSchemaInterface::TYPE_INTEGER, TableSchemaInterface::TYPE_BIGINTEGER, ]; $primaryKey = $schema->getPrimaryKey(); if ( in_array($data['type'], $autoIncrementTypes, true) && $primaryKey === [$name] && $name === 'id' ) { $data['autoIncrement'] = true; } return $this->columnDefinitionSql($data); } /** * @inheritDoc */ public function columnDefinitionSql(array $column): string { $name = $column['name']; $column += [ 'length' => null, 'precision' => null, ]; $out = $this->_driver->quoteIdentifier($name); $typeMap = [ TableSchemaInterface::TYPE_TINYINTEGER => ' SMALLINT', TableSchemaInterface::TYPE_SMALLINTEGER => ' SMALLINT', TableSchemaInterface::TYPE_INTEGER => ' INT', TableSchemaInterface::TYPE_BIGINTEGER => ' BIGINT', TableSchemaInterface::TYPE_BINARY => ' BYTEA', TableSchemaInterface::TYPE_BINARY_UUID => ' UUID', TableSchemaInterface::TYPE_BOOLEAN => ' BOOLEAN', TableSchemaInterface::TYPE_FLOAT => ' FLOAT', TableSchemaInterface::TYPE_DECIMAL => ' DECIMAL', TableSchemaInterface::TYPE_DATE => ' DATE', TableSchemaInterface::TYPE_TIME => ' TIME', TableSchemaInterface::TYPE_DATETIME => ' TIMESTAMP', TableSchemaInterface::TYPE_DATETIME_FRACTIONAL => ' TIMESTAMP', TableSchemaInterface::TYPE_TIMESTAMP => ' TIMESTAMP', TableSchemaInterface::TYPE_TIMESTAMP_FRACTIONAL => ' TIMESTAMP', TableSchemaInterface::TYPE_TIMESTAMP_TIMEZONE => ' TIMESTAMPTZ', TableSchemaInterface::TYPE_UUID => ' UUID', TableSchemaInterface::TYPE_NATIVE_UUID => ' UUID', TableSchemaInterface::TYPE_CHAR => ' CHAR', TableSchemaInterface::TYPE_CITEXT => ' CITEXT', TableSchemaInterface::TYPE_JSON => ' JSONB', TableSchemaInterface::TYPE_INTERVAL => ' INTERVAL', TableSchemaInterface::TYPE_GEOMETRY => ' GEOGRAPHY(GEOMETRY, %s)', TableSchemaInterface::TYPE_POINT => ' GEOGRAPHY(POINT, %s)', TableSchemaInterface::TYPE_LINESTRING => ' GEOGRAPHY(LINESTRING, %s)', TableSchemaInterface::TYPE_POLYGON => ' GEOGRAPHY(POLYGON, %s)', TableSchemaInterface::TYPE_CIDR => ' CIDR', TableSchemaInterface::TYPE_INET => ' INET', TableSchemaInterface::TYPE_MACADDR => ' MACADDR', ]; $autoIncrementTypes = [ TableSchemaInterface::TYPE_TINYINTEGER, TableSchemaInterface::TYPE_SMALLINTEGER, TableSchemaInterface::TYPE_INTEGER, TableSchemaInterface::TYPE_BIGINTEGER, ]; $autoIncrement = (bool)($column['autoIncrement'] ?? false); $isAutoincrement = ( in_array($column['type'], $autoIncrementTypes, true) && $autoIncrement ); $version = $this->_driver->version(); $identityVersion = version_compare($version, '10.0', '>='); if ($isAutoincrement && !$identityVersion) { $typeMap[$column['type']] = str_replace('INT', 'SERIAL', $typeMap[$column['type']]); unset($column['default']); } $foundType = false; if (isset($typeMap[$column['type']])) { $out .= $typeMap[$column['type']]; $foundType = true; } $hasLength = [ TableSchemaInterface::TYPE_CHAR, TableSchemaInterface::TYPE_STRING, ]; if ($column['type'] === TableSchemaInterface::TYPE_TEXT && $column['length'] !== TableSchema::LENGTH_TINY) { $out .= ' TEXT'; $foundType = true; } elseif ( $column['type'] === TableSchemaInterface::TYPE_STRING || ( $column['type'] === TableSchemaInterface::TYPE_TEXT && $column['length'] === TableSchema::LENGTH_TINY ) ) { $out .= ' VARCHAR'; $hasLength[] = $column['type']; $foundType = true; } if (!$foundType) { $out .= ' ' . strtoupper($column['type']); $hasLength[] = $column['type']; } if (in_array($column['type'], $hasLength, true) && !empty($column['length'])) { $out .= '(' . $column['length'] . ')'; } $hasCollate = [ TableSchemaInterface::TYPE_TEXT, TableSchemaInterface::TYPE_STRING, TableSchemaInterface::TYPE_CHAR, ]; if (in_array($column['type'], $hasCollate, true) && isset($column['collate']) && $column['collate'] !== '') { $out .= ' COLLATE "' . $column['collate'] . '"'; } $hasPrecision = [ TableSchemaInterface::TYPE_FLOAT, TableSchemaInterface::TYPE_DATETIME, TableSchemaInterface::TYPE_DATETIME_FRACTIONAL, TableSchemaInterface::TYPE_TIMESTAMP, TableSchemaInterface::TYPE_TIMESTAMP_FRACTIONAL, TableSchemaInterface::TYPE_TIMESTAMP_TIMEZONE, ]; if (in_array($column['type'], $hasPrecision) && isset($column['precision'])) { $out .= '(' . $column['precision'] . ')'; } if ( $column['type'] === TableSchemaInterface::TYPE_DECIMAL && ( isset($column['length']) || isset($column['precision']) ) ) { $out .= '(' . $column['length'] . ',' . (int)$column['precision'] . ')'; } if (in_array($column['type'], TableSchemaInterface::GEOSPATIAL_TYPES)) { $out = sprintf($out, $column['srid'] ?? self::DEFAULT_SRID); } if (isset($column['null']) && $column['null'] === false) { $out .= ' NOT NULL'; } if ($isAutoincrement && $identityVersion) { $generated = $column['generated'] ?? static::GENERATED_BY_DEFAULT; $out .= ' GENERATED ' . $generated . ' AS IDENTITY'; } $datetimeTypes = [ TableSchemaInterface::TYPE_DATETIME, TableSchemaInterface::TYPE_DATETIME_FRACTIONAL, TableSchemaInterface::TYPE_TIMESTAMP, TableSchemaInterface::TYPE_TIMESTAMP_FRACTIONAL, TableSchemaInterface::TYPE_TIMESTAMP_TIMEZONE, ]; if ( isset($column['default']) && in_array($column['type'], $datetimeTypes) && is_string($column['default']) && strtolower($column['default']) === 'current_timestamp' ) { $out .= ' DEFAULT CURRENT_TIMESTAMP'; } elseif (isset($column['default'])) { $defaultValue = $column['default']; if ($column['type'] === 'boolean') { $defaultValue = (bool)$defaultValue; } $out .= ' DEFAULT ' . $this->_driver->schemaValue($defaultValue); } elseif (isset($column['null']) && $column['null'] !== false) { $out .= ' DEFAULT NULL'; } return $out; } /** * @inheritDoc */ public function addConstraintSql(TableSchema $schema): array { $sqlPattern = 'ALTER TABLE %s ADD %s;'; $sql = []; foreach ($schema->constraints() as $name) { $constraint = $schema->getConstraint($name); assert($constraint !== null); 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): array { $sqlPattern = 'ALTER TABLE %s DROP CONSTRAINT %s;'; $sql = []; foreach ($schema->constraints() as $name) { $constraint = $schema->getConstraint($name); assert($constraint !== null); 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, string $name): string { $index = $schema->index($name); $columns = array_map( $this->_driver->quoteIdentifier(...), (array)$index->getColumns(), ); $include = ''; if ($index->getInclude()) { $included = array_map( $this->_driver->quoteIdentifier(...), $index->getInclude(), ); $include = sprintf(' INCLUDE (%s)', implode(', ', $included)); } return sprintf( 'CREATE INDEX %s ON %s (%s)%s', $this->_driver->quoteIdentifier($name), $this->_driver->quoteIdentifier($schema->name()), implode(', ', $columns), $include, ); } /** * @inheritDoc */ public function constraintSql(TableSchema $schema, string $name): string { $data = $schema->getConstraint($name); assert($data !== null); $out = 'CONSTRAINT ' . $this->_driver->quoteIdentifier($name); if ($data['type'] === TableSchema::CONSTRAINT_PRIMARY) { $out = 'PRIMARY KEY'; } elseif ($data['type'] === TableSchema::CONSTRAINT_UNIQUE) { $out .= ' UNIQUE'; } elseif ($data['type'] === TableSchema::CONSTRAINT_CHECK) { return $out . ' CHECK (' . $data['expression'] . ')'; } 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(string $prefix, array $data): string { $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 %s', implode(', ', $columns), $this->_driver->quoteIdentifier($data['references'][0]), $this->_convertConstraintColumns($data['references'][1]), $this->_foreignOnClause($data['update']), $this->_foreignOnClause($data['delete']), // Historically CakePHP used 'DEFERRABLE INITIALLY IMEDIATE, and this maintains backwards compat. $data['deferrable'] ?? ForeignKey::IMMEDIATE, ); } return $prefix . ' (' . implode(', ', $columns) . ')'; } /** * @inheritDoc */ public function createTableSql(TableSchema $schema, array $columns, array $constraints, array $indexes): array { $content = array_merge($columns, $constraints); $content = implode(",\n", array_filter($content)); $tableName = $this->_driver->quoteIdentifier($schema->name()); $dbSchema = $this->_driver->schema(); if ($dbSchema !== 'public') { $tableName = $this->_driver->quoteIdentifier($dbSchema) . '.' . $tableName; } $temporary = $schema->isTemporary() ? ' TEMPORARY ' : ' '; $out = []; $out[] = sprintf("CREATE%sTABLE %s (\n%s\n)", $temporary, $tableName, $content); foreach ($indexes as $index) { $out[] = $index; } foreach ($schema->columns() as $column) { $columnData = $schema->getColumn($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(TableSchema $schema): array { $name = $this->_driver->quoteIdentifier($schema->name()); return [ sprintf('TRUNCATE %s RESTART IDENTITY CASCADE', $name), ]; } /** * Generate the SQL to drop a table. * * @param \Cake\Database\Schema\TableSchema $schema Table instance * @return array SQL statements to drop a table. */ public function dropTableSql(TableSchema $schema): array { $sql = sprintf( 'DROP TABLE %s CASCADE', $this->_driver->quoteIdentifier($schema->name()), ); return [$sql]; } }