Surreal ORM is an Object-Relational Mapping and query-building library for Rust that provides a high-level API for interacting with SurrealDB, a distributed graph database. This documentation will guide you through the usage and features of the Surreal ORM library.
To use Surreal ORM in your Rust project, you need to add it as a dependency in
your Cargo.toml
file:
[dependencies]
surreal_orm = "https://github.com/Oyelowo/surreal_orm"
After adding the dependency, you can import the necessary modules in your Rust code:
use surreal_orm::*;
Before interacting with SurrealDB, you need to establish a connection to the database. The following example demonstrates how to create a connection to a local SurrealDB instance:
use surrealdb::engine::local::Mem;
use surrealdb::Surreal;
#[tokio::main]
async fn main() {
let db = Surreal::new::<Mem>(()).await.unwrap();
}
In this example, we create a new SurrealDB instance using the Surreal::new
function with the local::Mem
engine. The local::Mem
engine represents a
local in-memory database. You can replace it with other engine types according
to your setup.
A model in Surreal ORM represents a database table. You can define a node by
creating a Rust struct and implementing the Node
or Edge
trait. Here's an
example of defining a SpaceShip
model:
use surreal_orm::*;
#[derive(Node, Serialize, Deserialize, Debug, Clone)]
#[serde(rename_all = "camelCase")]
#[orm(table = space_ship)]
pub struct SpaceShip {
pub id: SurrealSimpleId<Self>,
pub name: String,
pub age: u8,
}
In this example, we define a SpaceShip
struct and annotate it with the Node
derive macro. The table
attribute specifies the name of the corresponding
database table.
Surreal ORM provides a fluent and expressive API for querying data from the
database. You can use the select
function to start a select statement and
chain various methods to build the query. Here's an example:
use surreal_orm::statements::{select, All};
let space_ship::Schema { name, age, .. } = SpaceShip::schema();
let space_ship = SpaceShip::table();
let statement = select(All)
.from(space_ship)
.where_(name.equal("Millennium Falcon"))
.order_by(age.desc())
.limit(10);
In this example, we start a select statement using the select
function and
pass the All
argument to select all fields. We specify the table name using
the from
method and add a condition using the where_
method. We can also use
the order_by
method to specify the sorting order and the limit
method to
limit the number of results.
To insert data into the database, you can use the insert
function and provide
the data as a vector of structs. Here's an example:
use surreal_orm::statements::insert;
let spaceships = vec![
SpaceShip {
id: SpaceShip::create_simple_id(),
name: "Millennium Falcon".to_string(),
age: 79,
},
SpaceShip {
name: "Starship Enterprise".to_string(),
age: 15,
..Default::default()
},
];
insert(spaceships).return_many(db.clone()).await?;
In this example, we define a vector of SpaceShip
structs and pass it to the
insert
function. We then call the run
method to execute the insertion
operation.
We can also use object!
and object_partial!
macros for a much more flexbible
and robust insertion supporting fields and parameter assignment e.g:
let weapon = &Weapon::table();
let weapon::Schema { ref strength, .. } = Weapon::schema();
let created_stats_statement = create::<WeaponStats>().set(object_partial!(WeaponStats {
averageStrength: block! {
let strengths = select_value(strength).from(weapon);
let total = math::sum!(strengths);
let count = array::len!(strengths);
return math::ceil!((((total / count) * (count * total)) / (total + 4)) * 100);
}
}));
To update data in the database, you can use the update
function and provide
the updated data as a struct. Here's an example:
use surreal_orm::statements::update;
let space_ship = SpaceShip::table();
update(space_ship)
.content(SpaceShip {
name: "Oyelowo".to_string(),
age: 90,
..Default::default()
})
.where_(cond(strength.gt(5)).and(strength.lte(15)))
.return_many(db.clone())
.await?;
In this example, we define a SpaceShip
struct with the updated data and pass
it to the update
function to update records within the table based on the
condition. We then call the return_many
method to execute the update
operation.
To delete data from the database, you can use the delete
function and provide
the condition for deletion. Here's an example:
use surreal_orm::{*, statements::{delete}};
let space_ship::Schema { name, age, .. } = SpaceShip::schema();
let space_ship = SpaceShip::table();
delete(space_ship)
.where_(cond(name.eq("Millennium Falcon")).and(age.lt(50)))
.run(db.clone())
.await?;
In this example, we use the delete
function and specify the table name. We add
a condition using the where_
method, and then call the run
method to execute
the deletion operation.
With these macros, you can create extremely complex queries with native-like
syntax. query_turbo!
macro creates a query chain by default which comprises
several statements separated by ;
. In order to create a transaction, you have
to begin the queries with BEGIN TRANSACTION;
and end with
COMMIT TRANSACTION;
or CANCEL TRANSACTION;
. You can also just use the
dedicated transaction!
macro which also enforces beginning and ending with the
necessary transaction statements. Lastly, to create a block, you have to return
an expression typically at the end. You can return a normal value, an if else
expression or a or a variable. You can also enforce a block by using the
dedicated block!
macro. A block is surrounded by curly braces and typically
returns.
let id1 = &Account::create_id("one".to_string());
let id2 = &Account::create_id("two".to_string());
let amount_to_transfer = 300.00;
let acc = Account::schema();
let account::Schema { balance, .. } = Account::schema();
let query_chain = query_turbo! {
begin transaction;
let balance1 = create_only().content(Balance {
id: Balance::create_id("balance1".to_string()),
amount: amount_to_transfer,
});
create_only().content(Balance {
id: Balance::create_id("balance2".to_string()),
amount: amount_to_transfer,
});
if balance.greater_than(100) {
let first_name = "Oyelowo";
let score = 100;
select(All).from(Account::table()).where_(acc.balance.eq(5));
} else if balance.less_than(100) {
let first_name = "Oyelowo";
let score = 100;
select(All).from(Account::table()).where_(acc.balance.eq(5));
} else if balance.gte(100) {
let first_name = "Oyelowo";
let score = 100;
select(All).from(Account::table()).where_(acc.balance.eq(5));
} else {
let first_name = "Oyelowo";
let score = 100;
select(All).from(Account::table()).where_(acc.balance.eq(5));
};
for name in vec!["Oyelowo", "Oyedayo"] {
let first = "Oyelowo";
select(All).from(Account::table()).where_(acc.balance.eq(5));
let good_stmt = select(All).from(Account::table()).where_(acc.balance.eq(64));
if balance.gt(50) {
let first_name = "Oyelowo";
};
select(All).from(Account::table()).where_(acc.balance.eq(34));
let numbers = vec![23, 98];
for age in numbers {
let score = 100;
let first_stmt = select(All).from(Account::table()).where_(acc.balance.eq(5));
let second_stmt = select(All).from(Account::table()).where_(acc.balance.eq(25));
select(All).from(Account::table()).where_(acc.balance.eq(923));
};
};
let balance3 = create().content(Balance {
id: Balance::create_id("balance3".into()),
amount: amount_to_transfer,
});
let accounts = select(All)
.from(id1..=id2);
// You can reference the balance object by using the $balance variable and pass the amount
// as a parameter to the decrement_by function. i.e $balance.amount
let updated1 = update::<Account>(id1).set(acc.balance.increment_by(balance1.with_path::<Balance>(E).amount));
update::<Account>(id1).set(acc.balance.increment_by(balance1.with_path::<Balance>(E).amount));
update::<Account>(id1).set(acc.balance.increment_by(45.3));
// You can also pass the amount directly to the decrement_by function. i.e 300.00
update::<Account>(id2).set(acc.balance.decrement_by(amount_to_transfer));
update::<Account>(id2).set(acc.balance.decrement_by(50));
commit transaction;
};
Surreal ORM supports transactions, which are a series of operations that are treated as a single unit of work. Here's an example of a transaction that involves creating two accounts, updating their balances, and then committing the transaction:
use surreal_orm::{
statements::{begin_transaction, create, update},
*,
};
use surrealdb::{engine::local::Mem, Surreal};
let db = Surreal::new::<Mem>(()).await.unwrap();
db.use_ns("test").use_db("test").await.unwrap();
#[derive(Node, Serialize, Deserialize, Debug, Clone)]
#[serde(rename_all = "camelCase")]
#[orm(table = "account")]
pub struct Account {
pub id: SurrealId<Self, String>,
pub balance: f64,
}
#[derive(Node, Serialize, Deserialize, Debug, Clone)]
#[serde(rename_all = "camelCase")]
#[orm(table = balance)]
pub struct Balance {
pub id: SurrealId<Self, String>,
pub amount: f64,
}
let id1 = &Account::create_id("one".into());
let id2 = &Account::create_id("two".into());
let amount_to_transfer = 300.00;
let acc = Account::schema();
transaction! {
BEGIN TRANSACTION;
let balance = create_only().content(Balance {
id: Balance::create_id("balance1".into()),
amount: amount_to_transfer,
});
create_only().content(Account {
id: id1.clone(),
balance: 135_605.16,
});
create_only().content(Account {
id: id2.clone(),
balance: 91_031.31,
});
// You can reference the balance object by using the $balance variable and pass the amount
// as a parameter to the decrement_by function. i.e $balance.amount
update::<Account>(id1).set(acc.balance.increment_by(balance.with_path::<Balance>(E).amount));
// You can also pass the amount directly to the decrement_by function. i.e 300.00
update::<Account>(id2).set(acc.balance.decrement_by(amount_to_transfer));
COMMIT TRANSACTION;
}
.run(db.clone())
.await?;
let accounts = select(All)
.from(id1..=id2)
.return_many::<Account>(db.clone())
.await?;
In this example, we begin a transaction and then create two accounts with initial balances. We then increment the balance of the first account and decrement the balance of the second account by the same amount. Finally, we commit the transaction and then verify that the balances were updated correctly.
The query!
macro allows for writing SQL queries directly in Rust, providing
automatic handling of placeholders and bindings, ensuring type safety, and
reducing repetitive boilerplate code.
With the query!
macro, you can easily construct and execute SQL queries:
-
Basic Query:
let query = query!(db, "SELECT * FROM users").await; let query = query!(db, "SELECT * FROM users", {}).await;
-
With Placeholders:
let username = "Oyelowo"; let query = query!(db, "SELECT name, age FROM users WHERE age > $age AND name = $ame", { age : 102, ame : username }) .await;
-
Multiple Queries:
let queries = query!( db, [ "SELECT * FROM users WHERE score = $score", "CREATE user:oyelowo SET name = $name, company = $company_name, skills = $skills" ], { score: 100, name: "Oyelowo", skills: vec!["Rust", "python", "typescript"], company_name: "Codebreather" } ) .await;
The query!
macro offers several benefits, including:
- Developer Efficiency: Reduce the time spent writing and debugging SQL queries.
- Code Clarity: Achieve clearer and more maintainable code with direct SQL queries in Rust.
- Error Reduction: Minimize the potential for runtime errors and SQL injection vulnerabilities.
Surreal ORM offers a robust command-line interface (CLI) for automatically diffing and managing database migrations in a SurrealDB environment. This tool supports a variety of tasks including initializing migrations, generating migration files, applying migrations up or down, resetting migrations, listing migrations, pruning unapplied migration files, and more, to efficiently manage your database schema.
It supports both CLI-based and Embedded migrations. Embedded migrations incorporate migration file data into your binary at compile time, making it accessible at runtime.
The process involves gathering codebase resources and setting them up for initializing and generating new migrations.
use surreal_orm::*;
#[derive(Node, Serialize, Deserialize, Debug, Clone, Default)]
#[serde(rename_all = "camelCase")]
#[orm(table = "animal", schemafull)]
pub struct Animal {
pub id: SurrealSimpleId<Self>,
pub species: String,
// #[orm(old_name = "oldField")] // Comment this line out to carry out a renaming operation
pub attributes: Vec<String>,
pub created_at: chrono::DateTime<Utc>,
pub updated_at: chrono::DateTime<Utc>,
pub velocity: u64,
}
impl TableResources for Animal {
fn events_definitions() -> Vec<Raw> {
let animal::Schema { species, velocity, .. } = Self::schema();
let event1 = define_event("event1".to_string())
.on_table("animal".to_string())
.when(cond(species.eq("Homo Erectus")).and(velocity.gt(545)))
.then(select(All).from(Crop::table()))
.to_raw();
vec![event1]
}
fn indexes_definitions() -> Vec<Raw> {
let animal::Schema { species, velocity, .. } = Self::schema();
let idx1 = define_index("species_speed_idx".to_string())
.on_table(Self::table())
.fields(arr![species, velocity])
.unique()
.to_raw();
vec![idx1]
}
}
#[derive(Edge, Serialize, Deserialize, Debug, Clone, Default)]
#[serde(rename_all = "camelCase")]
#[orm(table = "eats", schemafull)]
pub struct Eats<In: Node, Out: Node> {
pub id: SurrealSimpleId<Self>,
#[serde(rename = "in")]
pub in_: LinkOne<In>,
pub out: LinkOne<Out>,
pub place: String,
pub created_at: chrono::DateTime<Utc>,
}
pub type AnimalEatsCrop = Eats<Animal, Crop>;
impl TableResources for AnimalEatsCrop {}
#[derive(Debug, Clone)]
pub struct Resources;
impl DbResources for Resources {
create_table_resources!(
Animal,
Crop,
AnimalEatsCrop,
);
// Define other database resources here. They default to empty vecs
fn analyzers(&self) -> Vec<Raw> {
vec![]
}
fn functions(&self) -> Vec<Raw> {
vec![]
}
fn params(&self) -> Vec<Raw> {
vec![]
}
fn scopes(&self) -> Vec<Raw> {
vec![]
}
fn tokens(&self) -> Vec<Raw> {
vec![]
}
fn users(&self) -> Vec<Raw> {
vec![]
}
}
use surreal_orm::migrator::Migrator;
#[tokio::main]
async fn main() {
Migrator::run(Resources).await;
}
The CLI tool offers a range of commands, each with specific options and flags:
Help
# Check information about all commands
cargo run -- help
# Check information about specific commands
cargo run -- init --help
-
Initialize Migrations:
cargo run -- init --name "initial_migration" -r
Initializes the migrations directory with a reversible migration named "initial_migration". Omit the -r flag for unidirectional migrations.
-
Generate Migrations:
cargo run -- gen --name "add_users_table"
Generates a new migration file named "add_users_table". The migration type (reversible or non-reversible) is automatically detected based on the initial setup.
-
Apply Migrations Up: Various strategies for applying migrations, including applying till latest, by number, or till a specified migration.
# Applies all pending till latest by default cargo run -- up # Applies all pending till latest cargo run -- up -l # Applies by the number specified cargo run -- up -n 5 cargo run -- up --number 5 # Applies till specified migration cargo run -- up -t "20240107015727114_create_first.up.surql" cargo run -- up --till "20240107015727114_create_first.up.surql"
-
Rollback Migrations: Options to rollback migrations to a previous state, by number, till a specified migration, and an optional prune flag.
# Rollback migration to previous by default cargo run -- down # Rollback all pending till previous cargo run -- down --previous # Rollback by the number specified cargo run -- down -n 5 cargo run -- down --number 5 # Rollback till specified migration cargo run -- down -t "20240107015727114_create_first.up.surql" cargo run -- down --till "20240107015727114_create_first.up.surql" # In addition, you can use the --prune flag to delete local migration # files after rolling back. This can be useful in development for rapid changes. cargo run -- down -n 5 --prune
Rolls back the last applied migration.
-
Reset Migrations: Resets and initializes a new reversible migration named "initial_migration". Omit -r for unidirectional migrations.
cargo run -- reset --name "initial_migration" -r
-
Prune Migrations: Removes unapplied migration files.
# List pending migrations by default cargo run -- prune
Prune all pending unapplied migrations.
-
List Migrations: Lists migrations based on their status (all, pending, applied).
# List pending migrations by default cargo run -- ls cargo run -- list # List all migrations cargo run -- list --status all # List pending migrations cargo run -- list --status pending # List applied migrations cargo run -- list --status applied
Detailed instructions for customizing migration processes, including specifying a custom migration directory, enabling verbose output, and configuring database connections.
-
Custom Migration Directory:
cargo run -- init --name "initial_migration" --dir "custom_migrations" -r
Initializes migrations in a custom directory named "custom_migrations".
-
Verbose Output:
cargo run -- up -vvv
Runs migrations with 3 levels verbose output.
-
Database Connection Configuration:
- URL:
ws://localhost:8000
- Database Name:
test
- Namespace:
test
- User:
root
- Password:
root
cargo run -- up --url "ws://localhost:8000" --db "mydb" --ns "myns" --user "username" --pass "password"
Connects to the specified SurrealDB instance with custom credentials and applies migrations.
Other supported urls types include: `bash
- ws://localhost:8000
- wss://cloud.example.com
- http://localhost:8000
- https://cloud.example.com
- mem://
- memory
- file://temp.db
- indxdb://MyDatabase
- tikv://localhost:2379
- fdb://fdb.cluster ` This configuration enables the CLI to connect to different database backends including WebSocket, HTTP(S), In-Memory, File-Backend, and more.
- URL:
Embedding migrations within the binary for runtime access.
use surreal_orm::migrator::{
self, config::DatabaseConnection, EmbeddedMigrationsOneWay, EmbeddedMigrationsTwoWay,
embed_migrations, Mode, UpdateStrategy,
};
// Embed migrations as constant
const MIGRATIONS_ONE_WAY: EmbeddedMigrationsOneWay =
embed_migrations!("tests/migrations-oneway", one_way, strict);
let db = DatabaseConnection::default().setup().await.db().unwrap();
MIGRATIONS_ONE_WAY
.run(db.clone(), UpdateStrategy::Latest, Mode::Strict)
.await
.unwrap();
const MIGRATIONS_TWO_WAY: EmbeddedMigrationsTwoWay =
embed_migrations!("tests/migrations-twoway", two_way, strict);
MIGRATIONS_TWO_WAY
.run(db.clone(), UpdateStrategy::Latest, Mode::Strict)
.await
.unwrap();
This concludes the basic usage and features of the Surreal ORM library. You can explore more advanced features and methods in the API documentation and official book. If you have any further questions or need assistance, feel free to reach out at [email protected] or discord.
© Oyelowo Oyedayo
Email: [email protected]
To carry out certain tasks in any directory, these are the standard commands:
Commands | Purpose |
---|---|
make setup |
To setup the codebase for development |
make install |
install packages |
make upgrade |
upgrade packages |
make sync |
synchronize/generate local code etc |
make dev |
start cluster/app locally in live reloading mode |
make format |
format code |
make check |
check that code aligns with standard |
make test |
run automated tests |