6 releases
0.3.1 | Dec 17, 2024 |
---|---|
0.2.0 | Dec 10, 2024 |
0.1.4 | Dec 8, 2024 |
0.1.2 | Nov 29, 2024 |
#721 in Database interfaces
439 downloads per month
40KB
299 lines
Tiny ORM
A minimal ORM for CRUD operations. Built on top of SQLx and with its QueryBuilder. Support Sqlite, Postgres and MySQL right off the bat. It has smart defaults and is also flexible.
This library is the one I wished I had when I built a project in Rust in production and did not want a heavy ORM.
Installation
cargo install tiny-orm -F postgres # sqlite or mysql
Or add this to your Cargo.toml
:
[dependencies]
tiny-orm = {version = "0.3.1", features = ["postgres"] } # Choose between sqlite, mysql and postgres
Principles & advantages of TinyORM
The goals of this library are
- To have the least amount of dependencies
- Tackle all the boilerplate of the CRUD operations
- Add no complexity by default
- Can fit real world problems
Why TinyORM over another one?
-
All the queries are built with the QueryBuilder
-- All the inputs are passed as database arguments
-- All the queries are compatible with Sqlite, Postgres and MySQL -
Minimal set of dependencies (fast compile time)
-- SQLx
-- convert_case
-- regex
-- lazy_static -
Intuitive with smart defaults and flexible
MSRV
MSRV has been tested with cargo-msrv find --features sqlite
. You can learn more about cargo-msrv on their website.
Latest check ran on 0.2.0.
License
This project is licensed under [MIT] - see the LICENSE file for details.
Usage
use sqlx::{FromRow, Row, types::chrono::{DateTime, Utc}};
use tiny_orm::Table;
#[derive(Debug, FromRow, Table, Clone)]
#[tiny_orm(all)]
struct Todo {
id: i32,
created_at: DateTime<Utc>,
updated_at: DateTime<Utc>,
description: String,
done: bool,
}
The code above would generate the following methods on the Todo object
impl Todo {
pub fn get_by_id(pool: &DbPool, id: &i32) -> sqlx::Result<Self> {
// Get a specific record for a given ID
// Use the `id` column by default
}
pub fn list_all(pool: &DbPool) -> sqlx::Result<Vec<Self>> {
// Get all the records
}
pub fn delete(&self, pool: &DbPool) -> sqlx::Result<()> {
// Delete the record in the database
}
pub fn create(&self, pool: &DbPool) -> sqlx::Result<i32> {
// Create the Todo object as a record in
// the database and returns the primary key of the record created.
}
pub fn update(&self, pool: &DbPool) -> sqlx::Result<()> {
// Update the record in the database with the values
// currently part of the Todo object
}
}
Examples
More examples can be found in the examples directory.
Options
The Table macro comes with a few options to give flexibility to the user
At the Struct level
- table_name: The name of the table in the database.
Default being a lower_case version of the Struct name. So
MyStruct
would havemy_struct
as a defaulttable_name
. - only: The methods that will only be available to that struct. Multiple values are comma separated. Default is dependent on the struct name (see below).
- exclude: The methods that will be excluded for that struct. Multiple values are comma separated Default would be nothing.
- all: All the methods will be available to the struct. This will override the default values when none are provided. Default none.
- return_object: A custom object that would be returned instead of
Self
. Useful when creating or updating records with partial information. Default isSelf
which corresponds to the current Strut.
Note: only
and exclude
cannot be used together.
By convention, if a struct name
- Starts with
New
then it will automatically use the following arguments#[tiny_orm(table_name = "table_name_from_return_object", return_object = "NameWithoutNewAsPrefix", only = "create")]
- Starts with
Update
then it will automatically use the following arguments#[tiny_orm(table_name = "table_name_from_return_object", return_object = "NameWithoutUpdateAsPrefix", only = "update")]
- Everything else would be the equivalent of using the following
#[tiny_orm(table_name = "table_name", return_object = "Self", exclude = "create,update")]
Example
#[derive(Debug, FromRow, Table, Clone)]
// #[tiny_orm(table_name = "todo", return_object = "Self", exclude = "create,update")]
struct Todo {
id: i32,
created_at: DateTime<Utc>,
updated_at: DateTime<Utc>,
description: String,
done: bool,
}
#[derive(Debug, FromRow, Table, Clone)]
// Because the struct name starts with "New", it would automatically use the following options
// #[tiny_orm(table_name = "todo", return_object = "Todo", only = "create")]
struct NewTodo {
description: String,
}
#[derive(Debug, FromRow, Table, Clone)]
// Because the struct name starts with "Update", it would automatically use the following options
// #[tiny_orm(table_name = "todo", return_object = "Todo", only = "update")]
struct UpdateTodo {
id: i32,
done: bool
}
The above takes the assumption that some columns can be nullable and others are auto generated (eg an ID column that would be auto increment). Thus it would generate the following
impl NewTodo {
pub fn create(&self, pool: &DbPool) -> sqlx::Result<Todo> {
// Use the NewTodo object to create a record
// in the database and return the record created.
// This would not be supported for MySQL since
// there is a need to know the primary key (auto increment or not)
// to either return the value or use `last_insert_id()`
}
}
impl UpdateTodo {
pub fn update(&self, pool: &DbPool) -> sqlx::Result<Todo> {
// Update the Todo object with partial information based
// on the id (default primary_key).
// Because `return_type` is specified, it will return the whole record in the database.
// If `return_type` was not specified, it would simply return nothing `()`.
// For MySQL, it would always return nothing `()` as it
// cannot return a record after an update.
}
}
At the field level
- primary_key: The field that would be used as a primary key for the queries. For some methods the primary key is mandatory (eg:
get_by_id()
). If not specified, it will default toid
if part of the struct. - primary_key(auto): The field that would be used as a primary key for the queries. When "auto" is set, it will skip the column and returns it during the
create()
method since it will be auto generated by the database itself.
Note: MySQL only supports "auto increment" in that case. It does not support returning the default value of a primary key like a UUID.
Example
#[derive(Debug, FromRow, Table, Clone)]
struct Todo {
#[tiny_orm(primary_key)]
custom_pk: Uuid,
description: String,
done: bool
}
use tiny_orm::{Table, SetOption};
#[derive(Debug, FromRow, Table, Clone)]
struct Todo {
#[tiny_orm(primary_key(auto))]
id: i64,
description: SetOption<String>,
done: SetOption<bool>
}
Thus it would generate the following
impl Todo {
pub fn create(&self, pool: &DbPool) -> sqlx::Result<Uuid> { // or i64 depending on the example
// Create the Todo object as a record in
// the database and returns the created record.
}
}
SetOption
SetOption
is an Enum that behaves similarly to Option
. The main difference is that Tiny ORM will automatically exclude the fields once they are Unset
.
The goal is to not always push all the fields during update or create operations. This would avoid some potential data race condition if the local struct is out of date with what the database has.
You can check tiny-orm-model/src/lib.rs but SetOption
implement most of the Traits needed to automatically be used with Sqlx (Encode, Decode) as well as useful methods like .inner(), .is_set() & .is_not_set()
.
You can check the sqlite-setoption
example to see it in action.
Migrations
From 0.1.* to 0.2.*
In 0.2.0, the API was stabilized with the fact that
- The
update()
anddelete()
methods now always return an empty Ok responsesqlx::Result<()>
- The
create()
method always returns the primary key except when the return type is of another object (eg:NewTodo
has a return_object ofTodo
)
Roadmap
Goal of TinyORM is to stay tiny. That being said there are still a few things I would like to have
- Get all the CRUD methods.
- Ability to have custom fields like a custom primary_key or table_name.
- Ability to choose the operations available on the method through an opt-in strategy or an exclusion one.
- Ability to return a custom object type for more complex setup.
- Support MySQL with non auto increment PK. Won't be able to use the
last_return_id()
function in the case (eg with UUID). - Auto generate the
created_at
/updated_at
column based on some default value when needed.
-- For example, being able to haveUtc.now()
on theupdate()
method for theupdated_at
if the field is of typeDateTime<Utc>
- Ability to skip some fields if not set (especially for the create and update methods).
-- For example if the field is of typeSetOption<DateTime<Utc>>
then it would only put the field as part of the query if the value isSet(value)
. It would skip it if it has the valueNotSet
.
Release
One PR with the following
make release INCREMENT=patch
then add the tag and push it to main
git tag -a v0.2.0 -m "Release v0.2.0"
git push origin main --tags
Dependencies
~9–28MB
~372K SLoC