4#include "../SqlQueryFormatter.hpp"
5#include "SQLiteFormatter.hpp"
7#include <reflection-cpp/reflection.hpp>
15class SqlServerQueryFormatter final:
public SQLiteQueryFormatter
18 [[nodiscard]]
static std::string FormatFromTable(std::string_view table)
21 if (!table.empty() && (table.front() ==
'[' || table.front() ==
'"'))
22 return std::string(table);
25 return std::format(R
"("{}")", table);
29 [[nodiscard]]
bool RequiresTableRebuildForForeignKeyChange() const noexcept
override
34 [[nodiscard]]
bool SupportsBatchedSchemaIntrospection() const noexcept
override
39 [[nodiscard]]
StringList DropTable(std::string_view schemaName,
40 std::string_view
const& tableName,
41 bool ifExists =
false,
42 bool cascade =
false)
const override
53 std::string
const schemaFilter = schemaName.empty() ?
"SCHEMA_NAME()" : std::format(
"'{}'", schemaName);
55 result.emplace_back(std::format(
56 R
"(DECLARE @sql NVARCHAR(MAX) = N'';
57SELECT @sql = @sql + 'ALTER TABLE ' + QUOTENAME(OBJECT_SCHEMA_NAME(fk.parent_object_id)) + '.' + QUOTENAME(OBJECT_NAME(fk.parent_object_id)) + ' DROP CONSTRAINT ' + QUOTENAME(fk.name) + '; '
58FROM sys.foreign_keys fk
59WHERE OBJECT_NAME(fk.referenced_object_id) = '{}' AND OBJECT_SCHEMA_NAME(fk.referenced_object_id) = {};
60EXEC sp_executesql @sql;)",
67 result.emplace_back(std::format(
"DROP TABLE IF EXISTS {};",
FormatTableName(schemaName, tableName)));
69 result.emplace_back(std::format(
"DROP TABLE {};",
FormatTableName(schemaName, tableName)));
74 [[nodiscard]] std::string BinaryLiteral(std::span<uint8_t const> data)
const override
77 result.reserve((data.size() * 2) + 2);
79 for (uint8_t
byte: data)
80 result += std::format(
"{:02X}", byte);
114 [[nodiscard]] std::string StringLiteral(std::string_view value)
const noexcept override
116 return EncodeUnicodeLiteral(value);
119 [[nodiscard]] std::string StringLiteral(
char value)
const noexcept override
123 char const buf[1] = { value };
124 return EncodeUnicodeLiteral(std::string_view { buf, 1 });
131 static constexpr std::size_t Utf8SequenceLength(
unsigned char c)
noexcept
147 static constexpr std::pair<char32_t, std::size_t> DecodeUtf8(std::string_view s, std::size_t i)
noexcept
149 auto const len = Utf8SequenceLength(
static_cast<unsigned char>(s[i]));
150 if (len == 1 || i + len > s.size())
151 return {
static_cast<char32_t>(
static_cast<unsigned char>(s[i])), 1 };
153 auto const lead =
static_cast<unsigned char>(s[i]);
169 for (std::size_t k = 1; k < len; ++k)
170 cp = (cp << 6) | (
static_cast<unsigned char>(s[i + k]) & 0x3F);
177 static void AppendCodepoint(std::string& out,
bool& inQuotedRun,
char32_t cp)
179 auto const closeRun = [&] {
186 auto const openRun = [&] {
202 out +=
static_cast<char>(cp);
210 out += std::format(
"NCHAR({})",
static_cast<unsigned>(cp));
214 char32_t const adjusted = cp - 0x10000;
215 char32_t const hi = 0xD800 + (adjusted >> 10);
216 char32_t const lo = 0xDC00 + (adjusted & 0x3FF);
217 out += std::format(
"NCHAR({}) + NCHAR({})",
static_cast<unsigned>(hi),
static_cast<unsigned>(lo));
223 static std::string EncodeUnicodeLiteral(std::string_view value)
228 out.reserve(value.size() + 4);
229 bool inQuotedRun =
false;
231 while (i < value.size())
233 auto const [cp, len] = DecodeUtf8(value, i);
234 AppendCodepoint(out, inQuotedRun, cp);
243 [[nodiscard]] std::string QualifiedTableName(std::string_view schema, std::string_view table)
const override
246 return std::format(
"[{}]", table);
247 return std::format(
"[{}].[{}]", schema, table);
250 [[nodiscard]] std::string QueryLastInsertId(std::string_view )
const override
253 return std::format(
"SELECT @@IDENTITY");
256 [[nodiscard]] std::string_view BooleanLiteral(
bool literalValue)
const noexcept override
258 return literalValue ?
"1" :
"0";
261 [[nodiscard]] std::string_view DateFunction() const noexcept
override
266 [[nodiscard]] std::string SelectFirst(
bool distinct,
268 std::string_view fields,
269 std::string_view fromTable,
270 std::string_view fromTableAlias,
271 std::string_view tableJoins,
272 std::string_view whereCondition,
273 std::string_view orderBy,
274 size_t count)
const override
276 std::stringstream sqlQueryString;
277 sqlQueryString <<
"SELECT";
279 sqlQueryString <<
" DISTINCT";
280 sqlQueryString <<
" TOP " << count;
281 sqlQueryString <<
' ' << fields;
282 sqlQueryString <<
" FROM " << FormatFromTable(fromTable);
283 if (!fromTableAlias.empty())
284 sqlQueryString <<
" AS [" << fromTableAlias <<
']';
285 sqlQueryString << tableJoins;
286 sqlQueryString << whereCondition;
287 sqlQueryString << orderBy;
288 return sqlQueryString.str();
291 [[nodiscard]] std::string SelectRange(
bool distinct,
293 std::string_view fields,
294 std::string_view fromTable,
295 std::string_view fromTableAlias,
296 std::string_view tableJoins,
297 std::string_view whereCondition,
298 std::string_view orderBy,
299 std::string_view groupBy,
301 std::size_t limit)
const override
303 assert(!orderBy.empty());
304 std::stringstream sqlQueryString;
305 sqlQueryString <<
"SELECT " << fields;
307 sqlQueryString <<
" DISTINCT";
308 sqlQueryString <<
" FROM " << FormatFromTable(fromTable);
309 if (!fromTableAlias.empty())
310 sqlQueryString <<
" AS [" << fromTableAlias <<
']';
311 sqlQueryString << tableJoins;
312 sqlQueryString << whereCondition;
313 sqlQueryString << groupBy;
314 sqlQueryString << orderBy;
315 sqlQueryString <<
" OFFSET " << offset <<
" ROWS FETCH NEXT " << limit <<
" ROWS ONLY";
316 return sqlQueryString.str();
319 [[nodiscard]] std::string ColumnType(SqlColumnTypeDefinition
const& type)
const override
321 using namespace SqlColumnTypeDefinitions;
322 return std::visit(detail::overloaded {
323 [](Bigint
const&) -> std::string {
return "BIGINT"; },
324 [](Binary
const& type) -> std::string {
325 if (type.size == 0 || type.size > 8000)
326 return "VARBINARY(MAX)";
328 return std::format(
"VARBINARY({})", type.size);
330 [](Bool
const&) -> std::string {
return "BIT"; },
331 [](Char
const& type) -> std::string {
return std::format(
"CHAR({})", type.size); },
332 [](Date
const&) -> std::string {
return "DATE"; },
333 [](DateTime
const&) -> std::string {
return "DATETIME"; },
334 [](Decimal
const& type) -> std::string {
335 return std::format(
"DECIMAL({}, {})", type.precision, type.scale);
337 [](Guid
const&) -> std::string {
return "UNIQUEIDENTIFIER"; },
338 [](Integer
const&) -> std::string {
return "INTEGER"; },
339 [](NChar
const& type) -> std::string {
return std::format(
"NCHAR({})", type.size); },
340 [](NVarchar
const& type) -> std::string {
341 if (type.size == 0 || type.size > SqlOptimalMaxColumnSize)
342 return "NVARCHAR(MAX)";
344 return std::format(
"NVARCHAR({})", type.size);
346 [](Real
const& type) -> std::string {
350 return type.precision > 24 ?
"FLOAT(53)" :
"REAL";
352 [](Smallint
const&) -> std::string {
return "SMALLINT"; },
353 [](Text
const&) -> std::string {
return "VARCHAR(MAX)"; },
354 [](Time
const&) -> std::string {
return "TIME"; },
355 [](Timestamp
const&) -> std::string {
return "TIMESTAMP"; },
356 [](Tinyint
const&) -> std::string {
return "TINYINT"; },
357 [](VarBinary
const& type) -> std::string {
358 if (type.size == 0 || type.size > 8000)
359 return "VARBINARY(MAX)";
361 return std::format(
"VARBINARY({})", type.size);
363 [](Varchar
const& type) -> std::string {
364 if (type.size == 0 || type.size > SqlOptimalMaxColumnSize)
365 return "VARCHAR(MAX)";
367 return std::format(
"VARCHAR({})", type.size);
373 [[nodiscard]] std::string BuildColumnDefinition(SqlColumnDeclaration
const& column)
const override
375 std::stringstream sqlQueryString;
376 sqlQueryString <<
'"' << column.name <<
"\" " << ColumnType(column.type);
379 sqlQueryString <<
" NOT NULL";
381 if (column.primaryKey == SqlPrimaryKeyType::AUTO_INCREMENT)
382 sqlQueryString <<
" IDENTITY(1,1) PRIMARY KEY";
383 else if (column.primaryKey == SqlPrimaryKeyType::NONE && !column.index && column.unique)
384 sqlQueryString <<
" UNIQUE";
386 if (!column.defaultValue.empty())
387 sqlQueryString <<
" DEFAULT " << column.defaultValue;
389 return sqlQueryString.str();
393 [[nodiscard]]
StringList CreateTable(std::string_view schema,
394 std::string_view tableName,
395 std::vector<SqlColumnDeclaration>
const& columns,
396 std::vector<SqlCompositeForeignKeyConstraint>
const& foreignKeys,
397 bool ifNotExists =
false)
const override
399 std::stringstream ss;
404 std::string schemaFilter = schema.empty() ?
"dbo" : std::string(schema);
405 ss << std::format(
"IF NOT EXISTS (SELECT * FROM sys.tables WHERE name = '{}' AND schema_id = SCHEMA_ID('{}'))\n",
410 ss << std::format(
"CREATE TABLE {} (",
FormatTableName(schema, tableName));
413 for (
auto const& column: columns)
418 ss <<
"\n " << BuildColumnDefinition(column);
421 auto const primaryKeys = [&]() -> std::vector<std::string> {
422 std::vector<std::pair<uint16_t, std::string>> indexedPrimaryKeys;
423 for (
auto const& col: columns)
425 indexedPrimaryKeys.emplace_back(col.primaryKeyIndex, col.name);
426 std::ranges::sort(indexedPrimaryKeys, [](
auto const& a,
auto const& b) {
return a.first < b.first; });
428 std::vector<std::string> primaryKeys;
429 primaryKeys.reserve(indexedPrimaryKeys.size());
430 for (
auto const& [index, name]: indexedPrimaryKeys)
431 primaryKeys.push_back(name);
435 if (!primaryKeys.empty())
445 bool hasIdentity =
false;
446 for (
auto const& col: columns)
452 ss <<
",\n PRIMARY KEY (";
454 for (
auto const& pk: primaryKeys)
459 ss <<
'"' << pk <<
'"';
465 if (!foreignKeys.empty())
467 for (
auto const& fk: foreignKeys)
473 for (
auto const& col: fk.columns)
477 ss <<
'"' << col <<
'"';
480 ss <<
") REFERENCES " <<
FormatTableName(schema, fk.referencedTableName) <<
" (";
483 for (
auto const& col: fk.referencedColumns)
487 ss <<
'"' << col <<
'"';
495 for (
auto const& column: columns)
497 if (column.foreignKey)
499 ss <<
",\n " << BuildForeignKeyConstraint(tableName, column.name, *column.foreignKey);
506 result.emplace_back(ss.str());
509 for (SqlColumnDeclaration
const& column: columns)
511 if (column.index && column.primaryKey == SqlPrimaryKeyType::NONE)
517 result.emplace_back(std::format(R
"(CREATE UNIQUE INDEX "{}_{}_index" ON "{}" ("{}");)",
523 result.emplace_back(std::format(R
"(CREATE UNIQUE INDEX "{}_{}_index" ON "{}"."{}" ("{}");)",
533 result.emplace_back(std::format(R
"(CREATE INDEX "{}_{}_index" ON "{}" ("{}");)",
539 result.emplace_back(std::format(R
"(CREATE INDEX "{}_{}_index" ON "{}"."{}" ("{}");)",
553 [[nodiscard]]
StringList AlterTable(std::string_view schemaName,
554 std::string_view tableName,
555 std::vector<SqlAlterTableCommand>
const& commands)
const override
557 std::stringstream sqlQueryString;
559 int currentCommand = 0;
562 if (currentCommand > 0)
563 sqlQueryString <<
'\n';
566 using namespace SqlAlterTableCommands;
567 sqlQueryString << std::visit(
569 [schemaName, tableName](RenameTable
const& actualCommand) -> std::string {
570 return std::format(R
"(ALTER TABLE {} RENAME TO "{}";)",
572 actualCommand.newTableName);
574 [schemaName, tableName, this](AddColumn
const& actualCommand) -> std::string {
575 return std::format(R
"(ALTER TABLE {} ADD "{}" {} {};)",
577 actualCommand.columnName,
578 ColumnType(actualCommand.columnType),
579 actualCommand.nullable == SqlNullable::NotNull ? "NOT NULL" :
"NULL");
581 [schemaName, tableName,
this](AlterColumn
const& actualCommand) -> std::string {
582 return std::format(R
"(ALTER TABLE {} ALTER COLUMN "{}" {} {};)",
584 actualCommand.columnName,
585 ColumnType(actualCommand.columnType),
586 actualCommand.nullable == SqlNullable::NotNull ? "NOT NULL" :
"NULL");
588 [schemaName, tableName](RenameColumn
const& actualCommand) -> std::string {
589 return std::format(R
"(ALTER TABLE {} RENAME COLUMN "{}" TO "{}";)",
591 actualCommand.oldColumnName,
592 actualCommand.newColumnName);
594 [schemaName, tableName](DropColumn const& actualCommand) -> std::string {
595 return std::format(R
"(ALTER TABLE {} DROP COLUMN "{}";)",
597 actualCommand.columnName);
599 [schemaName, tableName](AddIndex const& actualCommand) -> std::string {
600 using namespace std::string_view_literals;
601 auto const uniqueStr = actualCommand.unique ?
"UNIQUE "sv :
""sv;
602 if (schemaName.empty())
603 return std::format(R
"(CREATE {2}INDEX "{0}_{1}_index" ON "{0}" ("{1}");)",
605 actualCommand.columnName,
608 return std::format(R
"(CREATE {3}INDEX "{0}_{1}_{2}_index" ON "{0}"."{1}" ("{2}");)",
611 actualCommand.columnName,
614 [schemaName, tableName](DropIndex const& actualCommand) -> std::string {
615 if (schemaName.empty())
616 return std::format(R
"(DROP INDEX "{0}_{1}_index";)", tableName, actualCommand.columnName);
619 R
"(DROP INDEX "{0}_{1}_{2}_index";)", schemaName, tableName, actualCommand.columnName);
621 [schemaName, tableName](AddForeignKey const& actualCommand) -> std::string {
626 tableName, std::array { std::string_view { actualCommand.columnName } });
628 R
"(IF NOT EXISTS (SELECT 1 FROM sys.foreign_keys WHERE name = '{0}') ALTER TABLE {1} ADD {2};)",
631 BuildForeignKeyConstraint(tableName, actualCommand.columnName, actualCommand.referencedColumn));
633 [schemaName, tableName](DropForeignKey const& actualCommand) -> std::string {
634 return std::format(R
"(ALTER TABLE {} DROP CONSTRAINT "{}";)",
637 tableName, std::array { std::string_view { actualCommand.columnName } }));
639 [schemaName, tableName](AddCompositeForeignKey const& actualCommand) -> std::string {
640 std::stringstream ss;
641 ss <<
"ALTER TABLE " <<
FormatTableName(schemaName, tableName) <<
" ADD CONSTRAINT \""
645 for (
auto const& col: actualCommand.columns)
649 ss <<
'"' << col <<
'"';
651 ss <<
") REFERENCES " <<
FormatTableName(schemaName, actualCommand.referencedTableName) <<
" (";
654 for (
auto const& col: actualCommand.referencedColumns)
658 ss <<
'"' << col <<
'"';
663 [schemaName, tableName,
this](AddColumnIfNotExists
const& actualCommand) -> std::string {
666 R
"(IF NOT EXISTS (SELECT * FROM sys.columns WHERE object_id = OBJECT_ID('{}') AND name = '{}')
667ALTER TABLE {} ADD "{}" {} {};)",
669 actualCommand.columnName,
671 actualCommand.columnName,
672 ColumnType(actualCommand.columnType),
673 actualCommand.nullable == SqlNullable::NotNull ? "NOT NULL" :
"NULL");
675 [schemaName, tableName](DropColumnIfExists
const& actualCommand) -> std::string {
678 R
"(IF EXISTS (SELECT * FROM sys.columns WHERE object_id = OBJECT_ID('{}') AND name = '{}')
679ALTER TABLE {} DROP COLUMN "{}";)",
681 actualCommand.columnName,
683 actualCommand.columnName);
685 [schemaName, tableName](DropIndexIfExists const& actualCommand) -> std::string {
686 if (schemaName.empty())
688 R
"(IF EXISTS (SELECT * FROM sys.indexes WHERE name = '{0}_{1}_index' AND object_id = OBJECT_ID('{0}'))
689DROP INDEX "{0}_{1}_index" ON "{0}";)",
691 actualCommand.columnName);
694 R
"(IF EXISTS (SELECT * FROM sys.indexes WHERE name = '{0}_{1}_{2}_index')
695DROP INDEX "{0}_{1}_{2}_index" ON "{0}"."{1}";)",
698 actualCommand.columnName);
704 return { sqlQueryString.str() };
707 [[nodiscard]] std::string QueryServerVersion()
const override
709 return "SELECT @@VERSION";
715 [[nodiscard]] SqlAdvisoryLockHandler
const& AdvisoryLockOps()
const override
717 return SqlServerAdvisoryLockOps();
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.
SqlPrimaryKeyType
Represents a primary key type.