v1.8
⭐ Triggers! No more words (thanks to @f-michaut)
Triggers have to be specified inside make_storage
call just like tables and indexes:
struct Lead {
int id = 0;
std::string firstName;
std::string lastName;
std::string email;
std::string phone;
};
struct LeadLog {
int id = 0;
int oldId = 0;
int newId = 0;
std::string oldPhone;
std::string newPhone;
std::string oldEmail;
std::string newEmail;
std::string userAction;
std::string createdAt;
};
auto storage = make_storage("",
// CREATE TRIGGER validate_email_before_insert_leads
// BEFORE INSERT ON leads
// BEGIN
// SELECT
// CASE
// WHEN NEW.email NOT LIKE '%_@__%.__%' THEN
// RAISE (ABORT,'Invalid email address')
// END;
// END;
make_trigger("validate_email_before_insert_leads",
before()
.insert()
.on<Lead>()
.begin(select(case_<int>()
.when(not like(new_(&Lead::email), "%_@__%.__%"),
then(raise_abort("Invalid email address")))
.end()))
.end()),
// CREATE TRIGGER log_contact_after_update
// AFTER UPDATE ON leads
// WHEN old.phone <> new.phone
// OR old.email <> new.email
// BEGIN
// INSERT INTO lead_logs (
// old_id,
// new_id,
// old_phone,
// new_phone,
// old_email,
// new_email,
// user_action,
// created_at
// )
// VALUES
// (
// old.id,
// new.id,
// old.phone,
// new.phone,
// old.email,
// new.email,
// 'UPDATE',
// DATETIME('NOW')
// ) ;
// END;
make_trigger("log_contact_after_update",
after()
.update()
.on<Lead>()
.when(is_not_equal(old(&Lead::phone), new_(&Lead::phone)) and
is_not_equal(old(&Lead::email), new_(&Lead::email)))
.begin(insert(into<LeadLog>(),
columns(&LeadLog::oldId,
&LeadLog::newId,
&LeadLog::oldPhone,
&LeadLog::newPhone,
&LeadLog::oldEmail,
&LeadLog::newEmail,
&LeadLog::userAction,
&LeadLog::createdAt),
values(std::make_tuple(old(&Lead::id),
new_(&Lead::id),
old(&Lead::phone),
new_(&Lead::phone),
old(&Lead::email),
new_(&Lead::email),
"UPDATE",
datetime("NOW")))))
.end()),
// CREATE TABLE leads (
// id integer PRIMARY KEY,
// first_name text NOT NULL,
// last_name text NOT NULL,
// email text NOT NULL,
// phone text NOT NULL
// );
make_table("leads",
make_column("id", &Lead::id, primary_key()),
make_column("first_name", &Lead::firstName),
make_column("last_name", &Lead::lastName),
make_column("email", &Lead::email),
make_column("phone", &Lead::phone)),
// CREATE TABLE lead_logs (
// id INTEGER PRIMARY KEY,
// old_id int,
// new_id int,
// old_phone text,
// new_phone text,
// old_email text,
// new_email text,
// user_action text,
// created_at text
// );
make_table("lead_logs",
make_column("id", &LeadLog::id, primary_key()),
make_column("old_id", &LeadLog::oldId),
make_column("new_id", &LeadLog::newId),
make_column("old_phone", &LeadLog::oldPhone),
make_column("new_phone", &LeadLog::newPhone),
make_column("old_email", &LeadLog::oldEmail),
make_column("new_email", &LeadLog::newEmail),
make_column("user_action", &LeadLog::userAction),
make_column("created_at", &LeadLog::createdAt)));
Triggers are being created during sync_schema
call if they do not exist.
For more information please check out an example file at examples/triggers.cpp
.
More info about triggers at sqlite.org.
⭐ generated columns support
struct Product {
int id = 0;
std::string name;
int quantity = 0;
float price = 0;
float totalValue = 0;
};
auto storage = make_storage({},
make_table("products",
make_column("id", &Product::id, primary_key()),
make_column("name", &Product::name),
make_column("quantity", &Product::quantity),
make_column("price", &Product::price),
make_column("total_value",
&Product::totalValue,
// GENERATED ALWAYS AS (price * quantity)
generated_always_as(&Product::price * c(&Product::quantity)))));
More info about generated columns on sqlite.org.
⭐ added pointer passing interface API
Leverages the convenient way of communicating non-SQL values between subcomponents or between an extension and the application.
Pointer Passing is superior to transforming a C-language pointer into a BLOB or a 64-bit integer, and allows for writing ad-hoc domain-specific extensions from within an application.
For more information please check the SQLite documentation, as well as the example at examples/pointer_passing_interface.cpp
.
⭐ added `bool` optional argument to `asterisk` and `object` functions which identifies column order
Example:
auto rows = storage.select(object<User>());
// decltype(rows) is std::vector<User>, where the User objects are constructed from columns in implicitly stored order
auto rows = storage.select(object<User>(true));
// decltype(rows) is std::vector<User>, where the User objects are constructed from columns in declared make_table order
⭐ built-in aggregate functions now support `FILTER` clause
avg(&User::id)
-> AVG(users.id)
avg(&User::id).filter(where(length(&User::name) > 5))
-> AVG(users.id) FILTER (WHERE LENGTH(users.name) > 5)
⭐ indexes support `WHERE` clause
make_index("idx_contacts_name", &Contract::firstName, &Contract::lastName, where(length(&Contract::firstName) > 2))
will produce
CREATE INDEX idx_contacts_name ON contacts (first_name, last_name) WHERE LENGTH(first_name) > 2
⭐ added `column_name` member function to `prepared_statement`
Example:
auto statement = storage.prepare(select(columns(&User::id, &User::name)));
auto idColumnName = statement.column_name(0);
auto nameColumnName = statement.column_name(1);
// decltype(idColumnName) and decltype(nameColumnName) is `std::string_view`
This API is available only with C++17 and higher
⭐ added `dbstat` virtual table support
SELECT * FROM dbstat;
can be called with
auto storage =
make_storage("dbstat.sqlite",
make_table("users", make_column("id", &User::id, primary_key()), make_column("name", &User::name)),
make_dbstat_table());
storage.sync_schema();
auto dbstatRows = storage.get_all<dbstat>();
dbstat
struct looks as follows:
struct dbstat {
std::string name;
std::string path;
int pageno = 0;
std::string pagetype;
int ncell = 0;
int payload = 0;
int unused = 0;
int mx_payload = 0;
int pgoffset = 0;
int pgsize = 0;
};
It is available if SQLITE_ENABLE_DBSTAT_VTAB
macro is defined.
More info on sqlite.org
⭐ `order_by` supports column aliases (thanks to @trueqbit)
Example:
// SELECT name, instr(abilities, 'o') i
// FROM marvel
// WHERE i > 0
// ORDER BY i
auto rows = storage.select(columns(&MarvelHero::name, as<colalias_i>(instr(&MarvelHero::abilities, "o"))),
where(greater_than(get<colalias_i>(), 0)),
order_by(get<colalias_i>()));
For more info please check new example in examples/column_aliases.cpp
.
⭐ order by kth column (#950, thanks to @trueqbit)
Example:
// SELECT name, instr(abilities, 'o')
// FROM marvel
// ORDER BY 2
auto rows = storage.select(columns(&MarvelHero::name, as<colalias_i>(instr(&MarvelHero::abilities, "o"))),
order_by(2));
For more info please check new example in examples/column_aliases.cpp
.
⭐ improved `primary_key` API
primary_key().asc().autoincrement()
makesPRIMARY KEY ASC AUTOINCREMENT
primary_key().on_conflict_rollback()
makesPRIMARY KEY ON CONFLICT ROLLBACK
primary_key().desc().on_conflict_abort()
makesPRIMARY KEY DESC ON CONFLICT ABORT
It means that all available SQLite PRIMARY KEY
options now are available in sqlite_orm
!
⭐ enhanced API for `BEGIN TRANSACTION`
Now once can call BEGIN DEFERRED TRANSACTION
, BEGIN IMMEDIATE TRANSACTION
and BEGIN EXCLUSIVE TRANSACTION
queries with this:
storage.begin_deferred_transaction();
storage.begin_immediate_transaction();
storage.begin_exclusive_transaction();
⭐ `asterisk` now supports table aliases as well (thanks to @trueqbit)
auto expression1 = storage.select(asterisk<User>()); // SELECT * FROM users
// will return the same as
using als_u = alias_u<User>;
auto expression2 = storage.select(asterisk<als_u>()); // SELECT * FROM users u
⭐ added explicit columns support in `using_` function
using_(&User::id)
will produce the same as using_(column<User>(&User::id))
.
Why people may need this? To use mapped structs with inheritance.
⭐ added `PRAGMA.application_id`
storage.pragma.application_id(3); // PRAGMA application_id = 3, function accepts int
auto value = storage.pragma.application_id(); // PRAGMA application_id, decltype(value) is int
- ⭐
asterisk
now supports table aliases as well - ⭐
storage.dump
now can accept prepared expressions to serialize them (thanks to @trueqbit) - ⭐ added new example file
chrono_binding.cpp
- ⭐ added scalar versions of
MAX
andMIN
functions (yes we missed them; thanks to @trueqbit) - ⭐ added
NULLIF
andIFNULL
core functions support (thanks to @trueqbit) - ⭐ now you can select
std::optional<T>()
to returnNULL
asT
in raw select - ⭐ also you can select
nullptr
to returnNULL
asstd::nullptr_t
in raw select same way - ⭐
update_all
now allows updating more than one table at once. If no tables specifiedorm_error_code::no_tables_specified
is thrown - ⭐
transaction_guard_t
finally has move constructor. Move it carefully - ⭐ prepared statement can be moved as well. Let's make a moving party!
- ⭐
std::string_view
can be used in queries as read only data not result type (available in C++17 and higher) - ⚙️
sync_schema
function call supportsDROP COLUMN
instead of dropping and recreating the whole table in SQLite 3.35 and higher - ⚙️ Huge inner template code refactoring. Certainly reduces the amount of resources needed by the compiler (thanks to @trueqbit)
- ⚙️
having
function marked as deprecated. Please usegroup_by(...).having(...)
instead. Simplehaving
function will be removed in v1.9 release - ⚙️ Improved unit tests run speed (thanks to @trueqbit)
- ⚙️ Refactored unit tests to make them more brilliant (thanks to @trueqbit)
- ⚙️ Properly quote identifiers (thanks to @trueqbit)
- ⚙️ Refactored complex inner serializing functions and identifiers quoting (thanks to @trueqbit)
- ⚙️ Optimized passing arguments to user defined functions: removed excess arguments copying
- ⚙️ Reduced amount of warnings (thanks to @trueqbit)
- ⚙️ Added new example with blob binding in case you need it
examples/blob_binding.cpp
- ⚙️ Improved multi-threaded use of persistent connections (#1054)
- ⚙️ Identified and encapsulated workarounds for legacy compilers.
Fully works with Visual C++ 2015 Update 3.
🐞 Bug fixes
- Exceptions that occur during transaction rollback or commit are now properly propagated.
#1027 (thanks to @trueqbit) - #969
- #976
- #953
- #922
- Allow explicit columns in USING-clause (as described above).
#955 (thanks to @trueqbit) asterisk
now supports table aliases as well
#945 (thanks to @trueqbit)- #933 (thanks to @trueqbit)
- #893
- fixed CHECK serialization.
#890 - added column pointer support for unique constraint.
#854, #925, #1065 - fixed a crash that occurred when using the backup functionality.
#855 - considered bindable expressions in order-by clauses.
#847 (thanks to @trueqbit) - #841
- fixed updating nullable columns.
#822