Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlSchema.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#if defined(_WIN32) || defined(_WIN64)
6 #include <Windows.h>
7#endif
8
9#include "Api.hpp"
10#include "SqlQuery/MigrationPlan.hpp"
11
12#include <format>
13#include <string_view>
14#include <tuple>
15#include <vector>
16
17class SqlStatement;
18
19namespace SqlSchema
20{
21
22namespace detail
23{
24 constexpr std::string_view rtrim(std::string_view value) noexcept
25 {
26 while (!value.empty() && (std::isspace(value.back()) || value.back() == '\0'))
27 value.remove_suffix(1);
28 return value;
29 }
30} // namespace detail
31
32struct FullyQualifiedTableName
33{
34 std::string catalog;
35 std::string schema;
36 std::string table;
37
38 bool operator==(FullyQualifiedTableName const& other) const noexcept
39 {
40 return catalog == other.catalog && schema == other.schema && table == other.table;
41 }
42
43 bool operator!=(FullyQualifiedTableName const& other) const noexcept
44 {
45 return !(*this == other);
46 }
47
48 bool operator<(FullyQualifiedTableName const& other) const noexcept
49 {
50 return std::tie(catalog, schema, table) < std::tie(other.catalog, other.schema, other.table);
51 }
52};
53
54struct FullyQualifiedTableColumn
55{
56 FullyQualifiedTableName table;
57 std::string column;
58
59 bool operator==(FullyQualifiedTableColumn const& other) const noexcept
60 {
61 return table == other.table && column == other.column;
62 }
63
64 bool operator!=(FullyQualifiedTableColumn const& other) const noexcept
65 {
66 return !(*this == other);
67 }
68
69 bool operator<(FullyQualifiedTableColumn const& other) const noexcept
70 {
71 return std::tie(table, column) < std::tie(other.table, other.column);
72 }
73};
74
75struct FullyQualifiedTableColumnSequence
76{
77 FullyQualifiedTableName table;
78 std::vector<std::string> columns;
79};
80
81struct ForeignKeyConstraint
82{
83 FullyQualifiedTableColumn foreignKey;
84 FullyQualifiedTableColumnSequence primaryKey;
85};
86
87/// Holds the definition of a column in a SQL table as read from the database schema.
88struct Column
89{
90 std::string name = {};
91 SqlColumnTypeDefinition type = {};
92 std::string dialectDependantTypeString = {};
93 bool isNullable = true;
94 bool isUnique = false;
95 size_t size = 0;
96 unsigned short decimalDigits = 0;
97 bool isAutoIncrement = false;
98 bool isPrimaryKey = false;
99 bool isForeignKey = false;
100 std::optional<ForeignKeyConstraint> foreignKeyConstraint {};
101 std::string defaultValue = {};
102};
103
104/// Callback interface for handling events while reading a database schema.
106{
107 public:
108 EventHandler() = default;
109 EventHandler(EventHandler&&) = default;
110 EventHandler(EventHandler const&) = default;
111 EventHandler& operator=(EventHandler&&) = default;
112 EventHandler& operator=(EventHandler const&) = default;
113 virtual ~EventHandler() = default;
114
115 virtual bool OnTable(std::string_view table) = 0;
116 virtual void OnPrimaryKeys(std::string_view table, std::vector<std::string> const& columns) = 0;
117 virtual void OnForeignKey(ForeignKeyConstraint const& foreignKeyConstraint) = 0;
118 virtual void OnColumn(Column const& column) = 0;
119 virtual void OnExternalForeignKey(ForeignKeyConstraint const& foreignKeyConstraint) = 0;
120 virtual void OnTableEnd() = 0;
121};
122
123/// Reads all tables in the given database and schema and calls the event handler for each table.
124LIGHTWEIGHT_API void ReadAllTables(std::string_view database, std::string_view schema, EventHandler& eventHandler);
125
126/// Holds the definition of a table in a SQL database as read from the database schema.
127struct Table
128{
129 // FullyQualifiedTableName name;
130
131 /// The name of the table.
132 std::string name;
133
134 /// The columns of the table.
135 std::vector<Column> columns {};
136
137 /// The foreign keys of the table.
138 std::vector<ForeignKeyConstraint> foreignKeys {};
139
140 /// The foreign keys of other tables that reference this table.
141 std::vector<ForeignKeyConstraint> externalForeignKeys {};
142
143 /// The primary keys of the table.
144 std::vector<std::string> primaryKeys {};
145};
146
147/// A list of tables.
148using TableList = std::vector<Table>;
149
150/// Retrieves all tables in the given @p database and @p schema.
151LIGHTWEIGHT_API TableList ReadAllTables(std::string_view database, std::string_view schema = {});
152
153/// Retrieves all tables in the given database and schema that have a foreign key to the given table.
154LIGHTWEIGHT_API std::vector<ForeignKeyConstraint> AllForeignKeysTo(SqlStatement& stmt,
155 FullyQualifiedTableName const& table);
156
157/// Retrieves all tables in the given database and schema that have a foreign key from the given table.
158LIGHTWEIGHT_API std::vector<ForeignKeyConstraint> AllForeignKeysFrom(SqlStatement& stmt,
159 FullyQualifiedTableName const& table);
160
161} // namespace SqlSchema
162
163template <>
164struct LIGHTWEIGHT_API std::formatter<SqlSchema::FullyQualifiedTableName>: std::formatter<std::string>
165{
166 auto format(SqlSchema::FullyQualifiedTableName const& value, format_context& ctx) const -> format_context::iterator
167 {
168 string output = std::string(SqlSchema::detail::rtrim(value.schema));
169 if (!output.empty())
170 output += '.';
171 auto const trimmedSchema = SqlSchema::detail::rtrim(value.catalog);
172 output += trimmedSchema;
173 if (!output.empty() && !trimmedSchema.empty())
174 output += '.';
175 output += SqlSchema::detail::rtrim(value.table);
176 return formatter<string>::format(output, ctx);
177 }
178};
179
180template <>
181struct LIGHTWEIGHT_API std::formatter<SqlSchema::FullyQualifiedTableColumn>: std::formatter<std::string>
182{
183 auto format(SqlSchema::FullyQualifiedTableColumn const& value, format_context& ctx) const
184 -> format_context::iterator
185 {
186 auto const table = std::format("{}", value.table);
187 if (table.empty())
188 return formatter<string>::format(std::format("{}", value.column), ctx);
189 else
190 return formatter<string>::format(std::format("{}.{}", value.table, value.column), ctx);
191 }
192};
193
194template <>
195struct LIGHTWEIGHT_API std::formatter<SqlSchema::FullyQualifiedTableColumnSequence>: std::formatter<std::string>
196{
197 auto format(SqlSchema::FullyQualifiedTableColumnSequence const& value, format_context& ctx) const
198 -> format_context::iterator
199 {
200 auto const resolvedTableName = std::format("{}", value.table);
201 string output;
202
203 for (auto const& column: value.columns)
204 {
205 if (!output.empty())
206 output += ", ";
207 output += resolvedTableName;
208 if (!output.empty() && !resolvedTableName.empty())
209 output += '.';
210 output += column;
211 }
212
213 return formatter<string>::format(output, ctx);
214 }
215};
Callback interface for handling events while reading a database schema.
High level API for (prepared) raw SQL statements.
Holds the definition of a column in a SQL table as read from the database schema.
Definition SqlSchema.hpp:89
Holds the definition of a table in a SQL database as read from the database schema.
std::vector< ForeignKeyConstraint > foreignKeys
The foreign keys of the table.
std::vector< Column > columns
The columns of the table.
std::vector< std::string > primaryKeys
The primary keys of the table.
std::vector< ForeignKeyConstraint > externalForeignKeys
The foreign keys of other tables that reference this table.
std::string name
The name of the table.