4#include "../SqlQueryFormatter.hpp"
5#include "SQLiteFormatter.hpp"
7#include <reflection-cpp/reflection.hpp>
14class PostgreSqlFormatter final:
public SQLiteQueryFormatter
17 using SQLiteQueryFormatter::CreateTable;
19 [[nodiscard]]
bool RequiresTableRebuildForForeignKeyChange() const noexcept
override
24 [[nodiscard]]
StringList DropTable(std::string_view schemaName,
25 std::string_view
const& tableName,
26 bool ifExists =
false,
27 bool cascade =
false)
const override
29 std::string sql = ifExists ? std::format(
"DROP TABLE IF EXISTS {}",
FormatTableName(schemaName, tableName))
37 [[nodiscard]] std::string BinaryLiteral(std::span<uint8_t const> data)
const override
40 result.reserve((data.size() * 2) + 4);
42 for (uint8_t
byte: data)
43 result += std::format(
"{:02X}", byte);
48 [[nodiscard]] std::string QueryLastInsertId(std::string_view )
const override
53 return std::format(
"SELECT lastval();");
56 [[nodiscard]] std::string_view DateFunction() const noexcept
override
58 return "CURRENT_DATE";
66 [[nodiscard]] std::string SetDefaultSchemaStatement(std::string_view schema)
const override
70 return std::format(R
"(SET search_path TO "{}", public)", schema);
73 [[nodiscard]] std::string BuildColumnDefinition(SqlColumnDeclaration const& column)
const override
75 std::stringstream sqlQueryString;
77 sqlQueryString <<
'"' << column.name <<
"\" ";
82 bool const isAutoIncrementViaDefault = column.defaultValue.contains(
"nextval(");
83 bool const isAutoIncrement = column.primaryKey == SqlPrimaryKeyType::AUTO_INCREMENT || isAutoIncrementViaDefault;
86 sqlQueryString <<
"SERIAL";
88 sqlQueryString << ColumnType(column.type);
91 sqlQueryString <<
" NOT NULL";
96 if (column.primaryKey == SqlPrimaryKeyType::AUTO_INCREMENT)
97 sqlQueryString <<
" PRIMARY KEY";
98 else if (column.primaryKey == SqlPrimaryKeyType::NONE && !column.index && column.unique)
99 sqlQueryString <<
" UNIQUE";
102 if (!column.defaultValue.empty() && !isAutoIncrement)
103 sqlQueryString <<
" DEFAULT " << column.defaultValue;
105 return sqlQueryString.str();
108 [[nodiscard]] std::string ColumnType(SqlColumnTypeDefinition
const& type)
const override
110 using namespace SqlColumnTypeDefinitions;
113 return std::visit(detail::overloaded {
114 [](Bigint
const&) -> std::string {
return "BIGINT"; },
115 [](Binary
const& type) -> std::string {
return std::format(
"BYTEA", type.size); },
116 [](Bool
const&) -> std::string {
return "BOOLEAN"; },
117 [](Char
const& type) -> std::string {
return std::format(
"CHAR({})", type.size); },
118 [](Date
const&) -> std::string {
return "DATE"; },
119 [](DateTime
const&) -> std::string {
return "TIMESTAMP"; },
120 [](Decimal
const& type) -> std::string {
121 return std::format(
"DECIMAL({}, {})", type.precision, type.scale);
123 [](Guid
const&) -> std::string {
return "UUID"; },
124 [](Integer
const&) -> std::string {
return "INTEGER"; },
125 [](NChar
const& type) -> std::string {
return std::format(
"CHAR({})", type.size); },
126 [](NVarchar
const& type) -> std::string {
129 return std::format(
"VARCHAR({})", type.size);
131 [](Real
const& type) -> std::string {
135 return type.precision > 24 ?
"DOUBLE PRECISION" :
"REAL";
137 [](Smallint
const&) -> std::string {
return "SMALLINT"; },
138 [](Text
const&) -> std::string {
return "TEXT"; },
139 [](Time
const&) -> std::string {
return "TIME"; },
140 [](Timestamp
const&) -> std::string {
return "TIMESTAMP"; },
142 [](Tinyint
const&) -> std::string {
return "SMALLINT"; },
143 [](VarBinary
const& ) -> std::string {
return std::format(
"BYTEA"); },
144 [](Varchar
const& type) -> std::string {
147 return std::format(
"VARCHAR({})", type.size);
154 [[nodiscard]]
StringList AlterTable(std::string_view schemaName,
155 std::string_view tableName,
156 std::vector<SqlAlterTableCommand>
const& commands)
const override
158 std::stringstream sqlQueryString;
160 int currentCommand = 0;
163 if (currentCommand > 0)
164 sqlQueryString <<
'\n';
167 using namespace SqlAlterTableCommands;
168 sqlQueryString << std::visit(
170 [schemaName, tableName](RenameTable
const& actualCommand) -> std::string {
171 return std::format(R
"(ALTER TABLE {} RENAME TO "{}";)",
173 actualCommand.newTableName);
175 [schemaName, tableName, this](AddColumn
const& actualCommand) -> std::string {
176 return std::format(R
"(ALTER TABLE {} ADD COLUMN "{}" {} {};)",
178 actualCommand.columnName,
179 ColumnType(actualCommand.columnType),
180 actualCommand.nullable == SqlNullable::NotNull ? "NOT NULL" :
"NULL");
182 [schemaName, tableName,
this](AlterColumn
const& actualCommand) -> std::string {
184 R
"(ALTER TABLE {0} ALTER COLUMN "{1}" TYPE {2}, ALTER COLUMN "{1}" {3} NOT NULL;)",
186 actualCommand.columnName,
187 ColumnType(actualCommand.columnType),
188 actualCommand.nullable == SqlNullable::NotNull ? "SET" :
"DROP");
190 [schemaName, tableName](RenameColumn
const& actualCommand) -> std::string {
191 return std::format(R
"(ALTER TABLE {} RENAME COLUMN "{}" TO "{}";)",
193 actualCommand.oldColumnName,
194 actualCommand.newColumnName);
196 [schemaName, tableName](DropColumn const& actualCommand) -> std::string {
197 return std::format(R
"(ALTER TABLE {} DROP COLUMN "{}";)",
199 actualCommand.columnName);
201 [schemaName, tableName](AddIndex const& actualCommand) -> std::string {
202 using namespace std::string_view_literals;
203 auto const uniqueStr = actualCommand.unique ?
"UNIQUE "sv :
""sv;
204 if (schemaName.empty())
205 return std::format(R
"(CREATE {2}INDEX "{0}_{1}_index" ON "{0}" ("{1}");)",
207 actualCommand.columnName,
210 return std::format(R
"(CREATE {3}INDEX "{0}_{1}_{2}_index" ON "{0}"."{1}" ("{2}");)",
213 actualCommand.columnName,
216 [schemaName, tableName](DropIndex const& actualCommand) -> std::string {
217 if (schemaName.empty())
218 return std::format(R
"(DROP INDEX "{0}_{1}_index";)", tableName, actualCommand.columnName);
221 R
"(DROP INDEX "{0}_{1}_{2}_index";)", schemaName, tableName, actualCommand.columnName);
223 [schemaName, tableName](AddForeignKey const& actualCommand) -> std::string {
228 "DO $$ BEGIN ALTER TABLE {} ADD {}; EXCEPTION WHEN duplicate_object THEN NULL; END $$;",
230 BuildForeignKeyConstraint(tableName, actualCommand.columnName, actualCommand.referencedColumn));
232 [schemaName, tableName](DropForeignKey
const& actualCommand) -> std::string {
233 return std::format(R
"(ALTER TABLE {} DROP CONSTRAINT "{}";)",
236 tableName, std::array { std::string_view { actualCommand.columnName } }));
238 [schemaName, tableName](AddCompositeForeignKey const& actualCommand) -> std::string {
239 std::stringstream ss;
240 ss <<
"ALTER TABLE " <<
FormatTableName(schemaName, tableName) <<
" ADD CONSTRAINT \""
244 for (
auto const& col: actualCommand.columns)
248 ss <<
'"' << col <<
'"';
250 ss <<
") REFERENCES " <<
FormatTableName(schemaName, actualCommand.referencedTableName) <<
" (";
253 for (
auto const& col: actualCommand.referencedColumns)
257 ss <<
'"' << col <<
'"';
262 [schemaName, tableName,
this](AddColumnIfNotExists
const& actualCommand) -> std::string {
264 return std::format(R
"(ALTER TABLE {} ADD COLUMN IF NOT EXISTS "{}" {} {};)",
266 actualCommand.columnName,
267 ColumnType(actualCommand.columnType),
268 actualCommand.nullable == SqlNullable::NotNull ? "NOT NULL" :
"NULL");
270 [schemaName, tableName](DropColumnIfExists
const& actualCommand) -> std::string {
272 return std::format(R
"(ALTER TABLE {} DROP COLUMN IF EXISTS "{}";)",
274 actualCommand.columnName);
276 [schemaName, tableName](DropIndexIfExists const& actualCommand) -> std::string {
278 if (schemaName.empty())
280 R
"(DROP INDEX IF EXISTS "{0}_{1}_index";)", tableName, actualCommand.columnName);
282 return std::format(R
"(DROP INDEX IF EXISTS "{0}_{1}_{2}_index";)",
285 actualCommand.columnName);
291 return { sqlQueryString.str() };
294 [[nodiscard]] std::string QueryServerVersion()
const override
296 return "SELECT version()";
302 [[nodiscard]] SqlAdvisoryLockHandler
const& AdvisoryLockOps()
const override
304 return PostgreSqlAdvisoryLockOps();
std::variant< SqlAlterTableCommands::RenameTable, SqlAlterTableCommands::AddColumn, SqlAlterTableCommands::AddColumnIfNotExists, SqlAlterTableCommands::AlterColumn, SqlAlterTableCommands::AddIndex, SqlAlterTableCommands::RenameColumn, SqlAlterTableCommands::DropColumn, SqlAlterTableCommands::DropColumnIfExists, SqlAlterTableCommands::DropIndex, SqlAlterTableCommands::DropIndexIfExists, SqlAlterTableCommands::AddForeignKey, SqlAlterTableCommands::AddCompositeForeignKey, SqlAlterTableCommands::DropForeignKey > SqlAlterTableCommand
Represents a single SQL ALTER TABLE command.