*/ protected $_constraintsIdMap = []; /** * Whether there is any table in this connection to SQLite containing sequences. * * @var bool */ protected $_hasSequences; /** * 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 unable to parse column type * @return array Array of column information. */ protected function _convertColumn(string $column): array { if ($column === '') { return ['type' => TableSchema::TYPE_TEXT, 'length' => null]; } preg_match('/(unsigned)?\s*([a-z]+)(?:\(([0-9,]+)\))?/i', $column, $matches); if (empty($matches)) { throw new DatabaseException(sprintf('Unable to parse column type from "%s"', $column)); } $unsigned = false; if (strtolower($matches[1]) === 'unsigned') { $unsigned = true; } $col = strtolower($matches[2]); $length = $precision = $scale = null; if (isset($matches[3])) { $length = $matches[3]; if (strpos($length, ',') !== false) { [$length, $precision] = explode(',', $length); } $length = (int)$length; $precision = (int)$precision; } $type = $this->_applyTypeSpecificColumnConversion( $col, compact('length', 'precision', 'scale') ); if ($type !== null) { return $type; } if ($col === 'bigint') { return ['type' => TableSchema::TYPE_BIGINTEGER, 'length' => $length, 'unsigned' => $unsigned]; } if ($col === 'smallint') { return ['type' => TableSchema::TYPE_SMALLINTEGER, 'length' => $length, 'unsigned' => $unsigned]; } if ($col === 'tinyint') { return ['type' => TableSchema::TYPE_TINYINTEGER, 'length' => $length, 'unsigned' => $unsigned]; } if (strpos($col, 'int') !== false) { return ['type' => TableSchema::TYPE_INTEGER, 'length' => $length, 'unsigned' => $unsigned]; } if (strpos($col, 'decimal') !== false) { return [ 'type' => TableSchema::TYPE_DECIMAL, 'length' => $length, 'precision' => $precision, 'unsigned' => $unsigned, ]; } if (in_array($col, ['float', 'real', 'double'])) { return [ 'type' => TableSchema::TYPE_FLOAT, 'length' => $length, 'precision' => $precision, 'unsigned' => $unsigned, ]; } if (strpos($col, 'boolean') !== false) { return ['type' => TableSchema::TYPE_BOOLEAN, 'length' => null]; } if (($col === 'char' && $length === 36) || $col === 'uuid') { return ['type' => TableSchema::TYPE_UUID, 'length' => null]; } if ($col === 'char') { return ['type' => TableSchema::TYPE_CHAR, 'length' => $length]; } if (strpos($col, 'char') !== false) { return ['type' => TableSchema::TYPE_STRING, 'length' => $length]; } if ($col === 'binary' && $length === 16) { return ['type' => TableSchema::TYPE_BINARY_UUID, 'length' => null]; } if (in_array($col, ['blob', 'clob', 'binary', 'varbinary'])) { return ['type' => TableSchema::TYPE_BINARY, 'length' => $length]; } $datetimeTypes = [ 'date', 'time', 'timestamp', 'timestampfractional', 'timestamptimezone', 'datetime', 'datetimefractional', ]; if (in_array($col, $datetimeTypes)) { return ['type' => $col, 'length' => null]; } return ['type' => TableSchema::TYPE_TEXT, 'length' => null]; } /** * Generate the SQL to list the tables and views. * * @param array $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 { return [ 'SELECT name FROM sqlite_master ' . 'WHERE (type="table" OR type="view") ' . 'AND name != "sqlite_sequence" ORDER BY name', [], ]; } /** * 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 { return [ 'SELECT name FROM sqlite_master WHERE type="table" ' . 'AND name != "sqlite_sequence" ORDER BY name', [], ]; } /** * @inheritDoc */ public function describeColumnSql(string $tableName, array $config): array { $sql = sprintf( 'PRAGMA table_info(%s)', $this->_driver->quoteIdentifier($tableName) ); return [$sql, []]; } /** * @inheritDoc */ public function convertColumnDescription(TableSchema $schema, array $row): void { $field = $this->_convertColumn($row['type']); $field += [ 'null' => !$row['notnull'], 'default' => $this->_defaultValue($row['dflt_value']), ]; $primary = $schema->getConstraint('primary'); if ($row['pk'] && empty($primary)) { $field['null'] = false; $field['autoIncrement'] = true; } // SQLite does not support autoincrement on composite keys. if ($row['pk'] && !empty($primary)) { $existingColumn = $primary['columns'][0]; /** @psalm-suppress PossiblyNullOperand */ $schema->addColumn($existingColumn, ['autoIncrement' => null] + $schema->getColumn($existingColumn)); } $schema->addColumn($row['name'], $field); if ($row['pk']) { $constraint = (array)$schema->getConstraint('primary') + [ 'type' => TableSchema::CONSTRAINT_PRIMARY, 'columns' => [], ]; $constraint['columns'] = array_merge($constraint['columns'], [$row['name']]); $schema->addConstraint('primary', $constraint); } } /** * Manipulate the default value. * * Sqlite includes quotes and bared NULLs in default values. * We need to remove those. * * @param string|int|null $default The default value. * @return string|int|null */ protected function _defaultValue($default) { if ($default === 'NULL' || $default === null) { return null; } // Remove quotes if (is_string($default) && preg_match("/^'(.*)'$/", $default, $matches)) { return str_replace("''", "'", $matches[1]); } return $default; } /** * @inheritDoc */ public function describeIndexSql(string $tableName, array $config): array { $sql = sprintf( 'PRAGMA index_list(%s)', $this->_driver->quoteIdentifier($tableName) ); return [$sql, []]; } /** * {@inheritDoc} * * Since SQLite does not have a way to get metadata about all indexes at once, * additional queries are done here. Sqlite constraint names are not * stable, and the names for constraints will not match those used to create * the table. This is a limitation in Sqlite's metadata features. * * @param \Cake\Database\Schema\TableSchema $schema The table object to append * an index or constraint to. * @param array $row The row data from `describeIndexSql`. * @return void */ public function convertIndexDescription(TableSchema $schema, array $row): void { $sql = sprintf( 'PRAGMA index_info(%s)', $this->_driver->quoteIdentifier($row['name']) ); $statement = $this->_driver->prepare($sql); $statement->execute(); $columns = []; /** @psalm-suppress PossiblyFalseIterator */ foreach ($statement->fetchAll('assoc') as $column) { $columns[] = $column['name']; } $statement->closeCursor(); if ($row['unique']) { $schema->addConstraint($row['name'], [ 'type' => TableSchema::CONSTRAINT_UNIQUE, 'columns' => $columns, ]); } else { $schema->addIndex($row['name'], [ 'type' => TableSchema::INDEX_INDEX, 'columns' => $columns, ]); } } /** * @inheritDoc */ public function describeForeignKeySql(string $tableName, array $config): array { $sql = sprintf('PRAGMA foreign_key_list(%s)', $this->_driver->quoteIdentifier($tableName)); return [$sql, []]; } /** * @inheritDoc */ public function convertForeignKeyDescription(TableSchema $schema, array $row): void { $name = $row['from'] . '_fk'; $update = $row['on_update'] ?? ''; $delete = $row['on_delete'] ?? ''; $data = [ 'type' => TableSchema::CONSTRAINT_FOREIGN, 'columns' => [$row['from']], 'references' => [$row['table'], $row['to']], 'update' => $this->_convertOnClause($update), 'delete' => $this->_convertOnClause($delete), ]; if (isset($this->_constraintsIdMap[$schema->name()][$row['id']])) { $name = $this->_constraintsIdMap[$schema->name()][$row['id']]; } else { $this->_constraintsIdMap[$schema->name()][$row['id']] = $name; } $schema->addConstraint($name, $data); } /** * {@inheritDoc} * * @param \Cake\Database\Schema\TableSchema $schema The table instance the column is in. * @param string $name The name of the column. * @return string SQL fragment. * @throws \Cake\Database\Exception\DatabaseException when the column type is unknown */ public function columnSql(TableSchema $schema, string $name): string { /** @var array $data */ $data = $schema->getColumn($name); $sql = $this->_getTypeSpecificColumnSql($data['type'], $schema, $name); if ($sql !== null) { return $sql; } $typeMap = [ TableSchema::TYPE_BINARY_UUID => ' BINARY(16)', TableSchema::TYPE_UUID => ' CHAR(36)', TableSchema::TYPE_CHAR => ' CHAR', TableSchema::TYPE_TINYINTEGER => ' TINYINT', TableSchema::TYPE_SMALLINTEGER => ' SMALLINT', TableSchema::TYPE_INTEGER => ' INTEGER', TableSchema::TYPE_BIGINTEGER => ' BIGINT', TableSchema::TYPE_BOOLEAN => ' BOOLEAN', TableSchema::TYPE_FLOAT => ' FLOAT', TableSchema::TYPE_DECIMAL => ' DECIMAL', TableSchema::TYPE_DATE => ' DATE', TableSchema::TYPE_TIME => ' TIME', TableSchema::TYPE_DATETIME => ' DATETIME', TableSchema::TYPE_DATETIME_FRACTIONAL => ' DATETIMEFRACTIONAL', TableSchema::TYPE_TIMESTAMP => ' TIMESTAMP', TableSchema::TYPE_TIMESTAMP_FRACTIONAL => ' TIMESTAMPFRACTIONAL', TableSchema::TYPE_TIMESTAMP_TIMEZONE => ' TIMESTAMPTIMEZONE', TableSchema::TYPE_JSON => ' TEXT', ]; $out = $this->_driver->quoteIdentifier($name); $hasUnsigned = [ TableSchema::TYPE_TINYINTEGER, TableSchema::TYPE_SMALLINTEGER, TableSchema::TYPE_INTEGER, TableSchema::TYPE_BIGINTEGER, TableSchema::TYPE_FLOAT, TableSchema::TYPE_DECIMAL, ]; if ( in_array($data['type'], $hasUnsigned, true) && isset($data['unsigned']) && $data['unsigned'] === true ) { if ($data['type'] !== TableSchema::TYPE_INTEGER || $schema->getPrimaryKey() !== [$name]) { $out .= ' UNSIGNED'; } } if (isset($typeMap[$data['type']])) { $out .= $typeMap[$data['type']]; } if ($data['type'] === TableSchema::TYPE_TEXT && $data['length'] !== TableSchema::LENGTH_TINY) { $out .= ' TEXT'; } if ($data['type'] === TableSchema::TYPE_CHAR) { $out .= '(' . $data['length'] . ')'; } if ( $data['type'] === TableSchema::TYPE_STRING || ( $data['type'] === TableSchema::TYPE_TEXT && $data['length'] === TableSchema::LENGTH_TINY ) ) { $out .= ' VARCHAR'; if (isset($data['length'])) { $out .= '(' . $data['length'] . ')'; } } if ($data['type'] === TableSchema::TYPE_BINARY) { if (isset($data['length'])) { $out .= ' BLOB(' . $data['length'] . ')'; } else { $out .= ' BLOB'; } } $integerTypes = [ TableSchema::TYPE_TINYINTEGER, TableSchema::TYPE_SMALLINTEGER, TableSchema::TYPE_INTEGER, ]; if ( in_array($data['type'], $integerTypes, true) && isset($data['length']) && $schema->getPrimaryKey() !== [$name] ) { $out .= '(' . (int)$data['length'] . ')'; } $hasPrecision = [TableSchema::TYPE_FLOAT, TableSchema::TYPE_DECIMAL]; if ( in_array($data['type'], $hasPrecision, true) && ( isset($data['length']) || isset($data['precision']) ) ) { $out .= '(' . (int)$data['length'] . ',' . (int)$data['precision'] . ')'; } if (isset($data['null']) && $data['null'] === false) { $out .= ' NOT NULL'; } if ($data['type'] === TableSchema::TYPE_INTEGER && $schema->getPrimaryKey() === [$name]) { $out .= ' PRIMARY KEY AUTOINCREMENT'; } $timestampTypes = [ TableSchema::TYPE_DATETIME, TableSchema::TYPE_DATETIME_FRACTIONAL, TableSchema::TYPE_TIMESTAMP, TableSchema::TYPE_TIMESTAMP_FRACTIONAL, TableSchema::TYPE_TIMESTAMP_TIMEZONE, ]; if (isset($data['null']) && $data['null'] === true && in_array($data['type'], $timestampTypes, true)) { $out .= ' DEFAULT NULL'; } if (isset($data['default'])) { $out .= ' DEFAULT ' . $this->_driver->schemaValue($data['default']); } return $out; } /** * {@inheritDoc} * * Note integer primary keys will return ''. This is intentional as Sqlite requires * that integer primary keys be defined in the column definition. * * @param \Cake\Database\Schema\TableSchema $schema The table instance the column is in. * @param string $name The name of the column. * @return string SQL fragment. */ public function constraintSql(TableSchema $schema, string $name): string { /** @var array $data */ $data = $schema->getConstraint($name); /** @psalm-suppress PossiblyNullArrayAccess */ if ( $data['type'] === TableSchema::CONSTRAINT_PRIMARY && count($data['columns']) === 1 && $schema->getColumn($data['columns'][0])['type'] === TableSchema::TYPE_INTEGER ) { return ''; } $clause = ''; $type = ''; if ($data['type'] === TableSchema::CONSTRAINT_PRIMARY) { $type = 'PRIMARY KEY'; } if ($data['type'] === TableSchema::CONSTRAINT_UNIQUE) { $type = 'UNIQUE'; } if ($data['type'] === TableSchema::CONSTRAINT_FOREIGN) { $type = 'FOREIGN KEY'; $clause = sprintf( ' REFERENCES %s (%s) ON UPDATE %s ON DELETE %s', $this->_driver->quoteIdentifier($data['references'][0]), $this->_convertConstraintColumns($data['references'][1]), $this->_foreignOnClause($data['update']), $this->_foreignOnClause($data['delete']) ); } $columns = array_map( [$this->_driver, 'quoteIdentifier'], $data['columns'] ); return sprintf( 'CONSTRAINT %s %s (%s)%s', $this->_driver->quoteIdentifier($name), $type, implode(', ', $columns), $clause ); } /** * {@inheritDoc} * * SQLite can not properly handle adding a constraint to an existing table. * This method is no-op * * @param \Cake\Database\Schema\TableSchema $schema The table instance the foreign key constraints are. * @return array SQL fragment. */ public function addConstraintSql(TableSchema $schema): array { return []; } /** * {@inheritDoc} * * SQLite can not properly handle dropping a constraint to an existing table. * This method is no-op * * @param \Cake\Database\Schema\TableSchema $schema The table instance the foreign key constraints are. * @return array SQL fragment. */ public function dropConstraintSql(TableSchema $schema): array { return []; } /** * @inheritDoc */ public function indexSql(TableSchema $schema, string $name): string { /** @var array $data */ $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 createTableSql(TableSchema $schema, array $columns, array $constraints, array $indexes): array { $lines = array_merge($columns, $constraints); $content = implode(",\n", array_filter($lines)); $temporary = $schema->isTemporary() ? ' TEMPORARY ' : ' '; $table = sprintf("CREATE%sTABLE \"%s\" (\n%s\n)", $temporary, $schema->name(), $content); $out = [$table]; foreach ($indexes as $index) { $out[] = $index; } return $out; } /** * @inheritDoc */ public function truncateTableSql(TableSchema $schema): array { $name = $schema->name(); $sql = []; if ($this->hasSequences()) { $sql[] = sprintf('DELETE FROM sqlite_sequence WHERE name="%s"', $name); } $sql[] = sprintf('DELETE FROM "%s"', $name); return $sql; } /** * Returns whether there is any table in this connection to SQLite containing * sequences * * @return bool */ public function hasSequences(): bool { $result = $this->_driver->prepare( 'SELECT 1 FROM sqlite_master WHERE name = "sqlite_sequence"' ); $result->execute(); $this->_hasSequences = (bool)$result->rowCount(); $result->closeCursor(); return $this->_hasSequences; } } // phpcs:disable class_alias( 'Cake\Database\Schema\SqliteSchemaDialect', 'Cake\Database\Schema\SqliteSchema' ); // phpcs:enable