Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlMigration.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "Api.hpp"
6#include "DataMapper/DataMapper.hpp"
7#include "SqlQuery.hpp"
8#include "SqlTransaction.hpp"
9
10#include <functional>
11#include <list>
12#include <optional>
13
14class SqlConnection;
15
16namespace SqlMigration
17{
18
19class MigrationBase;
20
21struct MigrationTimestamp
22{
23 uint64_t value {};
24
25 constexpr bool operator==(MigrationTimestamp const& other) const noexcept
26 {
27 return value == other.value;
28 }
29
30 constexpr bool operator!=(MigrationTimestamp const& other) const noexcept
31 {
32 return !(*this == other);
33 }
34};
35
36// Interface to be implemented by the user to execute SQL migrations
37class SqlMigrationExecutor
38{
39 public:
40 virtual ~SqlMigrationExecutor() = default;
41
42 virtual void OnCreateTable(SqlCreateTablePlan const& createTable) = 0;
43 virtual void OnAlterTable(SqlAlterTablePlan const& alterTable) = 0;
44
45 virtual void OnDropTable(std::string_view tableName) = 0;
46};
47
48// Main API to use for managing SQL migrations
49class MigrationManager
50{
51 public:
52 using MigrationList = std::list<MigrationBase const*>;
53
54 LIGHTWEIGHT_API static MigrationManager& GetInstance()
55 {
56 static MigrationManager instance;
57 return instance;
58 }
59
60 LIGHTWEIGHT_API void AddMigration(MigrationBase const* migration);
61
62 [[nodiscard]] LIGHTWEIGHT_API MigrationList const& GetAllMigrations() const noexcept;
63
64 [[nodiscard]] LIGHTWEIGHT_API MigrationBase const* GetMigration(MigrationTimestamp timestamp) const;
65
66 LIGHTWEIGHT_API void RemoveAllMigrations();
67
68 [[nodiscard]] LIGHTWEIGHT_API std::list<MigrationBase const*> GetPending() const noexcept;
69
70 using ExecuteCallback =
71 std::function<void(MigrationBase const& /*migration*/, size_t /*current*/, size_t /*total*/)>;
72
73 LIGHTWEIGHT_API void ApplySingleMigration(MigrationTimestamp timestamp);
74 LIGHTWEIGHT_API void ApplySingleMigration(MigrationBase const& migration);
75 LIGHTWEIGHT_API size_t ApplyPendingMigrations(ExecuteCallback const& feedbackCallback = {});
76
77 LIGHTWEIGHT_API void CreateMigrationHistory();
78 [[nodiscard]] LIGHTWEIGHT_API std::vector<MigrationTimestamp> GetAppliedMigrationIds() const;
79
80 [[nodiscard]] LIGHTWEIGHT_API DataMapper& GetDataMapper();
81
82 [[nodiscard]] LIGHTWEIGHT_API DataMapper& GetDataMapper() const
83 {
84 return const_cast<MigrationManager*>(this)->GetDataMapper();
85 }
86
87 LIGHTWEIGHT_API void CloseDataMapper();
88
89 LIGHTWEIGHT_API SqlTransaction Transaction();
90
91 private:
92 MigrationList _migrations;
93 mutable std::optional<DataMapper> _mapper;
94};
95
96// Represents a single unique SQL migration
97class MigrationBase
98{
99 public:
100 MigrationBase(MigrationTimestamp timestamp, std::string_view title):
101 _timestamp { timestamp },
102 _title { title }
103 {
104 MigrationManager::GetInstance().AddMigration(this);
105 }
106
107 virtual ~MigrationBase() = default;
108
109 virtual void Execute(SqlMigrationQueryBuilder& planer) const = 0;
110
111 [[nodiscard]] MigrationTimestamp GetTimestamp() const noexcept
112 {
113 return _timestamp;
114 }
115
116 [[nodiscard]] std::string_view GetTitle() const noexcept
117 {
118 return _title;
119 }
120
121 private:
122 MigrationTimestamp _timestamp;
123 std::string_view _title;
124};
125
126class Migration: public MigrationBase
127{
128 public:
129 Migration(MigrationTimestamp timestamp,
130 std::string_view title,
131 std::function<void(SqlMigrationQueryBuilder&)> const& plan):
132 MigrationBase(timestamp, title),
133 _plan { plan }
134 {
135 }
136
137 void Execute(SqlMigrationQueryBuilder& planer) const override
138 {
139 _plan(planer);
140 }
141
142 private:
143 std::function<void(SqlMigrationQueryBuilder&)> _plan;
144};
145
146} // namespace SqlMigration
147
148#define LIGHTWEIGHT_CONCATENATE(s1, s2) s1##s2
149
150#define LIGHTWEIGHT_MIGRATION_INSTANCE(timestamp) migration_##timestamp
151
152// Example use:
153//
154// LIGHTWEIGHT_SQL_MIGRATION(std::chrono::system_clock(...), "Create table 'MyTable'")
155// {
156// // ...
157// }
158#define LIGHTWEIGHT_SQL_MIGRATION(timestamp, description) \
159 struct Migration_##timestamp: public SqlMigration::MigrationBase \
160 { \
161 explicit Migration_##timestamp(): \
162 MigrationBase(SqlMigration::MigrationTimestamp { timestamp }, description) \
163 { \
164 } \
165 \
166 void Execute(SqlMigrationQueryBuilder& planer) const override; \
167 }; \
168 \
169 static Migration_##timestamp LIGHTWEIGHT_CONCATENATE(migration_, timestamp); \
170 \
171 void Migration_##timestamp::Execute(SqlMigrationQueryBuilder& plan) const
Main API for mapping records to and from the database using high level C++ syntax.
Represents a connection to a SQL database.
Query builder for building SQL migration queries.
Definition Migrate.hpp:184
Represents a SQL ALTER TABLE plan on a given table.