SQLx: The Rust SQL Toolkit

0
1453

An async, pure Rust SQL crate features compile-time checked queries without a domain specific language. This article will give a basic whirlwind tour of the library by showing CRUD operations and the transactions supported using it.

SQLx is a Rust crate for interacting with your database. Some of its features that distinguish it from other crates in the Rust ecosystem are:

  • Built with async IO in mind
  • Compile-time checked queries
  • Database agnostic
  • Postgres/MySQL drivers are written in pure Rust (as opposed to them being a FFI binding to a C library)
  • Works with various runtimes

Database setup
For our demo, we will start with a simple database with two tables and a simple relationship between them:

CREATe TABLE “person”(“id” SERIAL8  PRIMARY KEY UNIQUE,”name” TEXT NOT NULL,”age” smallint NOT NULL,”email” TEXT NULL)
CREATe TABLE “book”(“id” SERIAL8  PRIMARY KEY UNIQUE,”name” TEXT NOT NULL,”isbn” TEXT NOT NULL,”person” INT8 NOT NULL)
ALTER TABLE “book” ADD CONSTRAINT “book_person_fkey” FOREIGN KEY(“person”) REFERENCES “person”(“id”) ON DELETE RESTRICT  ON UPDATE RESTRICT

We have two tables:

  • Person
  • Book

And there is a relationship between the ‘book’ and the ‘person’ tables. Make sure you have a local (or remote, if you prefer) PostgreSQL running to test it out. You will also need the sqlx cli in your $PATH.

Rust project setup
We will use Cargo to create a new project. Cargo is the package manager for Rust. It can download your Rust package’s dependencies, compile your packages, make distributable packages, and upload them to crates.io. You can refer to Rust documentation for more details.

❯ cargo new sqlx-demo Created binary (application) `sqlx-demo` package

The project is set up as a binary project:

❯ tree
.
├── Cargo.toml
└── src
└── main.rs
1 directory, 2 files

Let’s export the database credentials in an environment variable:

export DATABASE_URL=postgres://postgres:postgres@localhost/sqlx-demo

Next, we will add the SQL file (20211223133946_initial_setup.sql) as part of the migration:

❯ sqlx migrate add initial_setup
Creating migrations/20211223133946_initial_setup.sql

Congratulations on creating your first migration!
Did you know you can embed your migrations in your application binary? On startup, after creating your database connection or pool, add:

sqlx::migrate!().run(<&your_pool OR &mut your_connection>).await?;

Note that the compiler won’t pick up new migrations if no Rust source files have changed. You can create a Cargo build script to work around this with `sqlx migrate build-script` (see https://docs.rs/sqlx/0.5/sqlx/macro.migrate.html).

Now you have to go to the above file and add your SQL statements. Once you have done that, you can set up the database:

❯ sqlx database setup
Applied 20211223133946/migrate initial setup (27.149609ms)

Next, let’s add the required dependencies to the Cargo.toml file:

[dependencies]
anyhow = “1.0.51”
sqlx = { version = “0.5.9”, features = [  “runtime-tokio-rustls”, “postgres” ]}
tokio = { version = “1”, features = [“full”] }

And you can modify the main.rs code to do the following things:

  • Set up a connection pool
  • Pass credentials to get a connection from the pool
use sqlx::{PgPool};
use anyhow::*;
use sqlx::postgres::PgPoolOptions;

#[derive(Clone)]
pub struct DBApplication {
    pool: PgPool
}

impl DBApplication {
    pub async fn new(config: String) -> Result<DBApplication> {
        let pool = PgPoolOptions::new()
            .max_connections(5)
            .connect(&config)
            .await?;
        Ok(DBApplication { pool })
    }
}

#[tokio::main]
async fn main() -> Result<()> {
    let db = DBApplication::new(“postgres://postgres:postgres@localhost/sqlx-demo”.into()).await?;
    println!(“Connection acquired!”);
    Ok(())
}

You can confirm if this works by running the command cargo run:

❯ cargo run
    Finished dev [unoptimized + debuginfo] target(s) in 0.04s
     Running `target/debug/sqlx-demo`
Connection acquired!

Domain types
We have two tables and, usually, we would want to model them into Rust structs. The easy way of doing this is to use derive macros:

#[derive(sqlx::FromRow, Debug)]
pub struct Person {
    id: i64,
    name: String,
    age: i16,
    email: String
}

The usual convention is to have two sets of records — one for inserting and the other for fetching. I use the New prefix to distinguish them:

pub struct NewPerson {
    pub name: String,
    pub age: i16,
    pub email: String
}

Another nice option is to use the query_as! macro, as it doesn’t need the above derive macro. I will be using that later in the demo.

Transactions
Let’s write a function to create a person, and let’s use a transaction to do the same:

pub async fn create_person(&self, person: NewPerson) -> Result<Person> {
    let mut transaction = self.pool.begin().await?;
    let person = sqlx::query_as(“INSERT INTO person(name, age, email) VALUES ($1, $2, $3) RETURNING *;”)
        .bind(person.name)
        .bind(person.age)
        .bind(person.email)
        .fetch_one(&mut transaction).await?;
    transaction.commit().await?;
    Ok(person)
}

You can modify your main fetch to use it:

#[tokio::main]
async fn main() -> Result<()> {
    let db = DBApplication::new(“postgres://postgres:postgres@localhost/sqlx-demo”.into()).await?;
    let sibi = db.create_person(NewPerson{ name: “Sibi”.into(), age: 20, email: “test@psibi.in”.into()}).await?;
    println!(“Sibi: {:?}”, sibi);
    Ok(())
}

Now let’s check if the function returns early because of an error, if the transaction rolls back. We will simulate a failure in the above function by an early return after insert:

pub async fn create_person(&self, person: NewPerson) -> Result<Person> {
    let mut transaction = self.pool.begin().await?;
    let person = sqlx::query_as(“INSERT INTO person(name, age, email) VALUES ($1, $2, $3) RETURNING *;”)
        .bind(person.name)
        .bind(person.age)
        .bind(person.email)
        .fetch_one(&mut transaction).await?;
    bail!(“Simulate error”);
    transaction.commit().await?;
    Ok(person)
}

And the main function will be like this:

#[tokio::main]
async fn main() -> Result<()> {
    let db = DBApplication::new(“postgres://postgres:postgres@localhost/sqlx-demo”.into()).await?;
    let sibi = db.create_person(NewPerson{ name: “Sibi”.into(), age: 20, email: “test@psibi.in”.into()}).await?;
    println!(“Sibi: {:?}”, sibi);
    Ok(())
}

Before running the executable, you can check the current rows in your table:

sqlx-demo=# select * from person;
 id | name | age | email
----+------+-----+-------
(0 rows)

Now let’s execute the program:

❯ cargo run     Finished dev [unoptimized + debuginfo] target(s) in 0.05s
     Running `target/debug/sqlx-demo`
Error: Simulate error

You can see that it did return an error. You can also verify that the transaction has rolled back and has not inserted the row, by inspecting the rows:

sqlx-demo=# select * from person;
 id | name | age | email
----+------+-----+-------
(0 rows)

Other CRUD operations
Similarly, you can implement other CRUD (create, read, update, delete) operations too:

pub async fn get_person(&self, id: i64) -> Result<Person> {
    let mut transaction = self.pool.begin().await?;
    let person = sqlx::query_as(“SELECT * from person where id = $1”)
        .bind(id)
        .fetch_one(&mut transaction).await?;
    transaction.commit().await?;
    Ok(person)
}

pub async fn update_person_name(&self, id: i64, name: String) -> Result<Person> {
    let mut transaction = self.pool.begin().await?;
    let person = sqlx::query_as(“UPDATE person SET name = $1 WHERE id = $2 RETURNING *”)
        .bind(name)
        .bind(id)
        .fetch_one(&mut transaction).await?;
    transaction.commit().await?;
    Ok(person)
}

pub async fn delete_book(&self, id: i64) -> Result<()> {
    let mut transaction = self.pool.begin().await?;
    sqlx::query(“DELETE FROM book WHERE id = $1”)
        .bind(id)
        .fetch_optional(&mut transaction).await?;
    transaction.commit().await?;
    Ok(())
}

Compile time queries
One nice thing about SQLx is that it can perform static checks against your database server. Using the query_as macro will achieve this. Rewriting the above get_person function to use the macro will look like this:

pub async fn get_person_macro(&self, id: i64) -> Result<Person> {
    let mut transaction = self.pool.begin().await?;
    let person = sqlx::query_as!(Person, “SELECT * from person where id = $1”, id)
        .fetch_one(&mut transaction)
        .await?;
    transaction.commit().await?;
    Ok(person)
}

The real power of it shines when you make mistakes in your SQL query. Let’s change the above code to use id_wrong instead of id:

pub async fn get_person_macro(&self, id: i64) -> Result<Person> {
    let mut transaction = self.pool.begin().await?;
    let person = sqlx::query_as!(Person, “SELECT * from person where id_wrong = $1”, id)
        .fetch_one(&mut transaction)
        .await?;
    transaction.commit().await?;
    Ok(person)
}


*Compiling* the above code will give this error:

error: error returned from database: column “id_wrong” does not exist
  --> src/main.rs:99:22
   |
99 |         let person = sqlx::query_as!(Person, “SELECT * from person where id_wrong = $1”, id)
   |                      ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   |
   = note: this error originates in the macro `$crate::sqlx_macros::expand_query` (in Nightly builds, run with -Z macro-backtrace for more info)

As the error message indicates, there is no column named id_wrong in your table.

Migrations
You could see above that we ran the migrations using the sqlx-cli tool. But in our application code, it will often be handy to run the migrations on a fresh database. The sqlx::migration macro is specifically meant for that. It embeds the migration into your binary and can be used to run it before the application is started. This is how your new main function will look like with migrations integration:

#[tokio::main]
async fn main() -> Result<()> {
    let db = DBApplication::new(“postgres://postgres:postgres@localhost/sqlx-demo”.into()).await?;
    sqlx::migrate!().run(&db.pool).await?;
    ...
    ...
    Ok(())
}

Hopefully, this article gave you a good idea of how to use SQLx for interacting with your database. If you prefer an object-relational mapping (ORM) based solution, these are some of the popular options:

  • Diesel: https://diesel.rs/
  • sea_orm: https://docs.rs/sea-orm/latest/sea_orm/
  • ormx: https://github.com/NyxCode/ormx

Both sea_orm and ormx are built on top of SQLx. The major differentiating factor between Diesel and the other ORM based solutions is that the former isn’t async IO ready yet.

LEAVE A REPLY

Please enter your comment!
Please enter your name here