#database #orm #sql

butane

An ORM with a focus on simplicity and on writing Rust, not SQL

7 unstable releases (3 breaking)

0.4.1 Jul 20, 2021
0.4.0 Jul 20, 2021
0.3.1 Jun 24, 2021
0.2.1 Feb 7, 2021
0.1.0 Dec 29, 2020

#99 in Database interfaces

45 downloads per month
Used in butane_cli

MIT/Apache

220KB
5.5K SLoC

Latest Version docs Build Status

Butane

An experimental ORM for Rust with a focus on simplicity and on writing Rust, not SQL

Butane takes an object-oriented approach to database operations. It may be thought of as much as an object-persistence system as an ORM -- the fact that it is backed by a SQL database is mostly an implementation detail to the API consumer.

Features

  • Relational queries using Rust-like syntax (via proc-macros)
  • Automatic migrations without writing SQL (although the generated SQL may be hand-tuned if necessary)
  • Ability to embed migrations in Rust code (so that a library may easily bundle its migrations)
  • SQLite and PostgreSQL backends
  • Write entirely or nearly entirely the same code regardless of database backend

Getting Started

Models, declared with struct attributes define the database schema. For example the Post model for a blog might look like this:

#[model]
#[derive(Default)]
struct Post {
    #[auto]
    id: i64,
    title: String,
    body: String,
    published: bool,
    likes: i32,
    tags: Many<Tag>,
    blog: ForeignKey<Blog>,
    byline: Option<String>,
}

An object is an instance of a model. An object is created like a normal struct instance, but must be saved in order to be persisted.

let mut post = Post::new(blog, title, body);
post.save(conn)?;

Changes to the instance are only applied to the database when saved:

post.published = true;
post.save(conn)?;

Queries are performed ergonmically with the query! macro.

let posts = query!(Post, published == true).limit(5).load(&conn)?;

For a detailed tutorial, see the Getting Started Guide.

Cargo Features

Butane exposes several featues to Cargo. By default, no backends are enabled: you will want to enabled either sqlite or pg:

  • default: Turns on datetime and uuid.
  • debug: Used in developing Butane, not expected to be enabled by consumers.
  • datetime: Support for timestamps (using chrono::NaiveDateTime).
  • log: Log certain warnings to the log crate facade (target "butane").
  • pg: Support for PostgreSQL.
  • r2d2: R2D2 support (propane::db::ConnectionManager).
  • sqlite: Support for SQLite.
  • sqlite-bundled: Bundles sqlite instead of using the system version.
  • tls: Support for TLS when using PostgreSQL.
  • uuid: Support for UUIDs (using the uuid crate).

Limitations

  • Butane, and its migration system especially, expects to own the database. It can be used with an existing database accessed also by other consumers, but it is not a design goal and there is no facility to infer butane models from an existing database schema.
  • API ergonomics are prioritized above performance. This does not mean Butane is slow, but that when given a choice between a simple, straightforward API and ekeing out the smallest possible overhead, the API will win.

Roadmap

Butane is young. The following features are currently missing, but planned

  • Foreign key constraints
  • Incremental object save
  • Backreferences for ForeignKey and Many.
  • Field/column rename support in migrations
  • Prepared/reusable queries
  • Benchmarking and performance tuning
  • Support for other databases such as MySQL or SQL Server are not explicitly planned, but contributions are welcome.

Comparison to Diesel

Butane is inspired by Diesel and by Django's ORM. If you're looking for a mature, performant, and flexible ORM, go use Diesel. Butane doesn't aim to be better than Diesel, but makes some different decisions, including:

  1. It is more object-oriented, at the cost of flexibility.

  2. Automatic migrations are prioritized.

  3. Rust code is the source of truth. The schema is understood from the definition of Models in Rust code, rather than inferred from the database.

  4. Queries are constructed using a DSL inside a proc-macro invocation rather than by importing dsl methods/names to use into the current scope. For Diesel, you might write

    use diesel_demo::schema::posts::dsl::*;
    let posts = posts.filter(published.eq(true))
         .limit(5)
         .load::<Post>(&conn)?
    

    whereas for Butane, you would instead write

    let posts = query!(Post, published == true).limit(5).load(&conn)?;
    

    Which form is preferable is primarily an aesthetic judgement.

  5. Differences between database backends are largely hidden.

  6. Diesel is overall significantly more mature and full-featured.

For a detailed tutorial, see the getting started guide.

License

Butane is licensed under either of the MIT license or the Apache License, Version 2.0 at your option.

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in Butane by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.

Dependencies

~2.3–6.5MB
~148K SLoC