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";
61 [[nodiscard]] std::string BuildColumnDefinition(SqlColumnDeclaration
const& column)
const override
63 std::stringstream sqlQueryString;
65 sqlQueryString <<
'"' << column.name <<
"\" ";
70 bool const isAutoIncrementViaDefault = column.defaultValue.contains(
"nextval(");
71 bool const isAutoIncrement = column.primaryKey == SqlPrimaryKeyType::AUTO_INCREMENT || isAutoIncrementViaDefault;
74 sqlQueryString <<
"SERIAL";
76 sqlQueryString << ColumnType(column.type);
79 sqlQueryString <<
" NOT NULL";
84 if (column.primaryKey == SqlPrimaryKeyType::AUTO_INCREMENT)
85 sqlQueryString <<
" PRIMARY KEY";
86 else if (column.primaryKey == SqlPrimaryKeyType::NONE && !column.index && column.unique)
87 sqlQueryString <<
" UNIQUE";
90 if (!column.defaultValue.empty() && !isAutoIncrement)
91 sqlQueryString <<
" DEFAULT " << column.defaultValue;
93 return sqlQueryString.str();
96 [[nodiscard]] std::string ColumnType(SqlColumnTypeDefinition
const& type)
const override
98 using namespace SqlColumnTypeDefinitions;
101 return std::visit(detail::overloaded {
102 [](Bigint
const&) -> std::string {
return "BIGINT"; },
103 [](Binary
const& type) -> std::string {
return std::format(
"BYTEA", type.size); },
104 [](Bool
const&) -> std::string {
return "BOOLEAN"; },
105 [](Char
const& type) -> std::string {
return std::format(
"CHAR({})", type.size); },
106 [](Date
const&) -> std::string {
return "DATE"; },
107 [](DateTime
const&) -> std::string {
return "TIMESTAMP"; },
108 [](Decimal
const& type) -> std::string {
109 return std::format(
"DECIMAL({}, {})", type.precision, type.scale);
111 [](Guid
const&) -> std::string {
return "UUID"; },
112 [](Integer
const&) -> std::string {
return "INTEGER"; },
113 [](NChar
const& type) -> std::string {
return std::format(
"CHAR({})", type.size); },
114 [](NVarchar
const& type) -> std::string {
117 return std::format(
"VARCHAR({})", type.size);
119 [](Real
const&) -> std::string {
return "REAL"; },
120 [](Smallint
const&) -> std::string {
return "SMALLINT"; },
121 [](Text
const&) -> std::string {
return "TEXT"; },
122 [](Time
const&) -> std::string {
return "TIME"; },
123 [](Timestamp
const&) -> std::string {
return "TIMESTAMP"; },
125 [](Tinyint
const&) -> std::string {
return "SMALLINT"; },
126 [](VarBinary
const& ) -> std::string {
return std::format(
"BYTEA"); },
127 [](Varchar
const& type) -> std::string {
130 return std::format(
"VARCHAR({})", type.size);
137 [[nodiscard]]
StringList AlterTable(std::string_view schemaName,
138 std::string_view tableName,
139 std::vector<SqlAlterTableCommand>
const& commands)
const override
141 std::stringstream sqlQueryString;
143 int currentCommand = 0;
146 if (currentCommand > 0)
147 sqlQueryString <<
'\n';
150 using namespace SqlAlterTableCommands;
151 sqlQueryString << std::visit(
153 [schemaName, tableName](RenameTable
const& actualCommand) -> std::string {
154 return std::format(R
"(ALTER TABLE {} RENAME TO "{}";)",
156 actualCommand.newTableName);
158 [schemaName, tableName, this](AddColumn
const& actualCommand) -> std::string {
159 return std::format(R
"(ALTER TABLE {} ADD COLUMN "{}" {} {};)",
161 actualCommand.columnName,
162 ColumnType(actualCommand.columnType),
163 actualCommand.nullable == SqlNullable::NotNull ? "NOT NULL" :
"NULL");
165 [schemaName, tableName,
this](AlterColumn
const& actualCommand) -> std::string {
167 R
"(ALTER TABLE {0} ALTER COLUMN "{1}" TYPE {2}, ALTER COLUMN "{1}" {3} NOT NULL;)",
169 actualCommand.columnName,
170 ColumnType(actualCommand.columnType),
171 actualCommand.nullable == SqlNullable::NotNull ? "SET" :
"DROP");
173 [schemaName, tableName](RenameColumn
const& actualCommand) -> std::string {
174 return std::format(R
"(ALTER TABLE {} RENAME COLUMN "{}" TO "{}";)",
176 actualCommand.oldColumnName,
177 actualCommand.newColumnName);
179 [schemaName, tableName](DropColumn const& actualCommand) -> std::string {
180 return std::format(R
"(ALTER TABLE {} DROP COLUMN "{}";)",
182 actualCommand.columnName);
184 [schemaName, tableName](AddIndex const& actualCommand) -> std::string {
185 using namespace std::string_view_literals;
186 auto const uniqueStr = actualCommand.unique ?
"UNIQUE "sv :
""sv;
187 if (schemaName.empty())
188 return std::format(R
"(CREATE {2}INDEX "{0}_{1}_index" ON "{0}" ("{1}");)",
190 actualCommand.columnName,
193 return std::format(R
"(CREATE {3}INDEX "{0}_{1}_{2}_index" ON "{0}"."{1}" ("{2}");)",
196 actualCommand.columnName,
199 [schemaName, tableName](DropIndex const& actualCommand) -> std::string {
200 if (schemaName.empty())
201 return std::format(R
"(DROP INDEX "{0}_{1}_index";)", tableName, actualCommand.columnName);
204 R
"(DROP INDEX "{0}_{1}_{2}_index";)", schemaName, tableName, actualCommand.columnName);
206 [schemaName, tableName](AddForeignKey const& actualCommand) -> std::string {
211 "DO $$ BEGIN ALTER TABLE {} ADD {}; EXCEPTION WHEN duplicate_object THEN NULL; END $$;",
213 BuildForeignKeyConstraint(tableName, actualCommand.columnName, actualCommand.referencedColumn));
215 [schemaName, tableName](DropForeignKey
const& actualCommand) -> std::string {
216 return std::format(R
"(ALTER TABLE {} DROP CONSTRAINT "{}";)",
219 tableName, std::array { std::string_view { actualCommand.columnName } }));
221 [schemaName, tableName](AddCompositeForeignKey const& actualCommand) -> std::string {
222 std::stringstream ss;
223 ss <<
"ALTER TABLE " <<
FormatTableName(schemaName, tableName) <<
" ADD CONSTRAINT \""
227 for (
auto const& col: actualCommand.columns)
231 ss <<
'"' << col <<
'"';
233 ss <<
") REFERENCES " <<
FormatTableName(schemaName, actualCommand.referencedTableName) <<
" (";
236 for (
auto const& col: actualCommand.referencedColumns)
240 ss <<
'"' << col <<
'"';
245 [schemaName, tableName,
this](AddColumnIfNotExists
const& actualCommand) -> std::string {
247 return std::format(R
"(ALTER TABLE {} ADD COLUMN IF NOT EXISTS "{}" {} {};)",
249 actualCommand.columnName,
250 ColumnType(actualCommand.columnType),
251 actualCommand.nullable == SqlNullable::NotNull ? "NOT NULL" :
"NULL");
253 [schemaName, tableName](DropColumnIfExists
const& actualCommand) -> std::string {
255 return std::format(R
"(ALTER TABLE {} DROP COLUMN IF EXISTS "{}";)",
257 actualCommand.columnName);
259 [schemaName, tableName](DropIndexIfExists const& actualCommand) -> std::string {
261 if (schemaName.empty())
263 R
"(DROP INDEX IF EXISTS "{0}_{1}_index";)", tableName, actualCommand.columnName);
265 return std::format(R
"(DROP INDEX IF EXISTS "{0}_{1}_{2}_index";)",
268 actualCommand.columnName);
274 return { sqlQueryString.str() };
277 [[nodiscard]] std::string QueryServerVersion()
const override
279 return "SELECT version()";
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.