Reindeer π¦ lifts your sled
!
Reindeer is a small embedded entity store built on top of sled
, using serde
and bincode
for serialization, written entirely in rust.
It serves as a convenient middle ground to store, retreive and update structs in an embedded database with a bare-minimum relationnal model.
Reindeer aims to provide a very small and very fast embedded database, suitable for deployment on very low-end devices. It uses a binary tree to search for keys, and is designed to handle parent-child relationships very efficiently, while other types of relations are slower.
Let's say you have to write a small utility for a network of schools. Each school has a set of promotions, and each promotion a list of students. Each School has a set of modules as well, on which they can register one or several promotions. You will have entities for School
, Promotion
, Module
and Student
. Most of the relationships are "parent-child" relationships, apart from Promotions registered to a given module (with each module having 1 or two promotions).
For such a case, full-fledged SQL databases are not necessary, and will use computing resources. SQLite has a relatioin model that makes making it unecessarily complex.
Here, Reindeer is organised such that each school has an ID (let's say, 1), each promotion has an ID beginning with its school ID (let's say : (1,9)) and each student has an ID begining with its promotion ID (so ((1,9),32) for instance). This makes searching all students in a school or all students in a promotion very efficient using a binary tree.
Plus, Reindeer uses bincode
for its data representation on disk, making it super fast at serializing and deserializing data, even on low-end hardware. Its schema and data layout are solely dependent on the code that uses it, being serialized versions of Rust struct instances.
The goal of Reindeer is providing a super-fast, super-efficient embedded solution for this sort of case, while running on low-end hardware.
Reindeer is meant to be used in scenarios where you need very good performance and a minimal footprint while still needing a basic relationnal model. If you have a data structure that can be summarized as mainly parent-child relations, then Reindeer will do the trick.
Reindeer does not provide data redundancy or integrity checks other than deletion behaviours out of the box, so keep that in mind. It does provide a JSON import and export function if you want to secure data.
A migration system is on the way.
use reindeer::Db;
let db = reindeer::open("./my-db")?;
π‘ Since this is just a sled
DB, this object can be copied and sent accross threads safely.
From there, you have two options :
- Derive the
Entity
trait - Implement the trait manually.
Entities need to implement the Serialize
and Deserialize
traits from serde
, which are conveniently re-exported from reindeer
use reindeer::{Serialize,Deserialize,Entity}
#[derive(Serialize,Deserialize,Entity)]
pub struct MyStruct {
pub id : u32,
pub prop1 : String,
pub prop2 : u64,
}
If your struct already has an id
field, then it will be used as the key for your store. Its type must either be an integral type, a String
or a Vec<u8>
, or a tuple of those types.
The name of your store in the database will be the name of the entity, with its original case. Make sure, in this case, that it's the only entity with this name.
Otherwise, you can use the entity
helper attribute to specify a different key field and name :
#[derive(Serialize,Deserialize,Entity)]
#[entity(name = "user", id = "email")]
pub struct User {
pub email : String,
pub prop1 : String,
pub prop2 : u64,
}
Entities need to implement the Serialize
and Deserialize
traits from serde
, which are conveniently re-exported from reindeer
:
use reindeer::{Serialize,Deserialize,Entity}
#[derive(Serialize,Deserialize)]
pub struct MyStruct {
pub id : u32,
pub prop1 : String,
pub prop2 : u64,
}
Then you need to implement the Entity
trait and implement three methods : get_key
, set_key
and store_name
, as well as define an associated type, Key
-
Key
is the type of the identifier for each instance of your entity ("primary key"). It must implement theAsBytes
trait. πβ It's already implemented forString
,u32
,i32
,u64
,i64
andVec<u8>
, as well as for any 2-elements tuple of those types, so you should not need to implement it yourself. -
The key represents the unique key that will be used to identify each instance of your struct in the database, to retreive and update them, it is of type
Key
-
The
store_name
is the name of the entity store. It should be unique for each Entity type (see it as the table name).
use reindeer::{Entity, Serialize,Deserialize};
#[derive(Serialize,Deserialize)]
struct MyStruct { key : u32, prop1 : String }
impl Entity for MyStruct{
type Key = u32;
fn store_name() -> &'static str {
"my-struct"
}
fn get_key(&self) -> &Self::Key {
&self.key
}
fn set_key(&mut self, key : &Self::Key) {
self.key = key.clone();
}
}
Register the entity once, when you launch your application.
let db = reindeer::open("./my-db")?;
MyStruct::register(db)?;
π‘ Registering the entity will make it possible for Reindeer to handle safe deletion of entity entries. Without this, trying to delete an unregistered entity entry will result in an error.
You can now save an instance of your struct MyStruct
to the database :
let db = reindeer::open("./")?;
let instance = MyStruct {
id : 0,
prop1 : String::from("Hello"),
prop2 : 2335,
}
instance.save(&db)?;
π‘ If id
0 already exists in the database, it will be overwritten!
let instance = MyStruct::get(0,&db)?;
let instances = MyStruct::get_all(&db)?;
let instances = MyStruct::get_with_filter(|m_struct| {mstruct.prop1.len > 20},&db)?;
MyStruct::remove(0,&db)?;
You can combine conditions easily with the QueryBuilder
helper object :
let students = QueryBuilder::new()
.with_parent(&school_id)
.with_named_relation_to::<Club>(&club_id, "member")
.get_with_filter(|s : &Student| s.age > 18,&data.db)?;
Refer to the documentation for more information.
reindeer
has three types of relations :
sibling
: An entity that has the same key in another store (one to one relation)parent-child
: An entity which key is composed of its parent's key and au32
(as a two-element tuple) for efficient one-to-many relationsfree-relation
you freely connect two instances of two separate Entities together. This can be used to achieve many-to-many relationships, but is less efficient than sibling and parent-child relationships in regard to querying the database. Use when sibling and parent-child are not possible.
To create a sibling Entity, you need to link the Entity structs together by overriding the specifying sibling stores and what happens when we delete one of them.
Sibling stores must share the same key type (and thus matching entities will have the same id).
π‘ DeletionBehaviour determines what happens to the sibbling when the current entity is removed :
DeletionBehaviour::Cascade
also deletes sibling entityDeletionBehaviour::Error
causes an Error if a sibling still exists and does not delete the source elementDeletionBehaviour::BreakLink
just removes the entity without removing its sibling.
You can specify siblings using the siblings
helper attribute :
#[derive(Serialize,Deserialize,Entity)]
#[entity(name = "user", id = "email")]
#[siblings(("user_data",Cascade),("user_data2",Cascade))]
pub struct User {
pub email : String,
pub prop1 : String,
pub prop2 : u64,
}
#[derive(Serialize,Deserialize,Entity)]
#[entity(name = "user_data", id = "email")]
#[siblings(("user",Error),("user_data2",Cascade))]
pub struct UserData {
pub email : String,
pub prop3 : String,
pub prop4 : String,
pub prop5 : i64
}
In the above example, deleting a User
instance also deletes its sibling UserData
instance, but deleting the UserData
instance causes an error and deletes neither.
use reindeer::{Entity,DeletionBehaviour};
impl Entity for MyStruct1{
/* ... */
fn store_name() -> &'static str {
"my_struct_1"
}
fn get_sibling_stores() -> Vec<(&'static str,DeletionBehaviour)> {
return vec![("my_struct_2",DeletionBehaviour::Cascade)]
}
}
impl Entity for MyStruct2{
/* ... */
fn store_name() -> &'static str {
"my_struct_2"
}
fn get_sibling_stores() -> Vec<(&'static str,DeletionBehaviour)> {
return vec![("my_struct_1",DeletionBehaviour::BreakLink)]
}
}
π‘ if sibling stores are defined, an entity instance might or might not have a sibling of the other Sibling store! Siblings are optionnal by default.
In the above example, deleting a MyStruct1
instance also deletes its sibling MyStruct2
instance, but deleting the MyStruct2
instance leaves its sibling MyStruct1
instance intact.
π‘ Sibling Entities must have the same Key
type.
let m_struct_1 = MyStruct1 {
/* ... */
};
let mut m_struct_2 = MyStruct2 {
/* ... */
};
m_struct_1.save(&db)?;
m_struct_1.save_sibling(m_struct_2,&db)?;
π‘ this will update m_struct_2
's key to m_struct_1
's key using the set_key
method, so it does not matter which key you initially provide before calling save_child
.
MyStruct2
's store with the same key as an entity in MyStruct1
's store without using save_sibling
, the result is the same, and the two entities will be considered siblings all the same.
if let Some(sibling) = m_struct_1.get_sibling::<MyStruct2>(&db)? {
/* ... */
}
π‘ Note that a sibling may or may not be present, thus the Option
type.
For a parent-child relationship between entities to exist, the child entity must have a Key
type being a tuple of :
- The parent
Key
type u32
π‘ Children entities will be auto-incremeted and easily retreived through their parent key.
You can define child stores the same way you define sibling stores, but using the children
helper attribute:
#[derive(Serialize,Deserialize,Entity)]
#[entity(name = "user", id = "email")]
#[children(("document",Cascade))]
pub struct User {
pub email : String,
pub prop1 : String,
pub prop2 : u64,
}
#[derive(Serialize,Deserialize,Entity)]
#[entity(name = "document")]
pub struct Document {
pub id : (String,u32),
pub prop3 : String,
pub prop4 : String,
pub prop5 : i64
}
impl Entity for Parent{
type Key = String;
/* ... */
fn store_name() -> &'static str {
"parent"
}
fn get_child_stores() -> Vec<(&'static str)> {
return vec![("child", DeletionBehaviour::Cascade)]
}
}
impl Entity for Child{
type Key = (String, u32);
/* ... */
fn store_name() -> &'static str {
"child"
}
}
In the above example, deleting the parent entity will remove all child entities automatically (thanks to the Cascade
deletion behaviour).
For database integrity, it is strongly advised not to use DeletionBehaviour::BreakLink
on parent/child relations, and instead use either Error
of Cascade
let parent = Parent {
/* ... */
};
let mut child = Child {
/* ... */
}
parent.save_child(child,&db)?;
π‘ this will update child
's key to parent
's key and an auto-incremented index using the set_key
method, so it does not matter which key you initially provide before calling save_child
.
let children = parent.get_children::<Child>(&db)?;
Free relations follow the same pattern as other relation types, except they are freely created between any two entities. This can be used to achieve many to many relationships.
π‘ Creating a free relation will automatically create its opposite relation, making it two-way.
let e1 = Entity1 {
/* ... */
};
let mut e2 = Entity2 {
/* ... */
}
e1.create_relation(e2,DeletionBehaviour::Cascade, DeletionBehaviour::BreakLink,None,&db)?;
In the above example, deletion behaviour in both ways are provided : deleting e1
will automatically delete e2
, but deleting e2
will leave e1
untouched and break the link between them.
DeletionBehaviour::Error
is also an option here.
let related_entities = e1.get_related::<Entity2>(db)?;
To get only the first related entity from the other store, use
let related_entity = e1.get_single_related::<Entity2>(db)?;
A name must have been supplied when creating the relation :
e1.create_relation(e2,DeletionBehaviour::Cascade, DeletionBehaviour::BreakLink,Some("main"),&db)?;
let related_entities = e1.get_related_with_name::<Entity2>("secondary",db)?;
To get only the first related entity from the other store, use
let related_entity = e1.get_single_related_with_name::<Entity2>("main",db)?;
If needed, you can remove an existing link between entities:
e1.remove_relation(other,db)?;
or
e1.remove_relation_with_key::<OtherEntity>(otherKey,db)?;
When defining DeletionBehaviour
for your relations, be careful not to create deadlocks.
For instance, if two siblings mutually define a DeletionBehaviour::Error
link, then none of them can ever be removed...
Also, be aware of the cycles you create in databases. While you can create relation cycles safely, the same deadlock rules as above apply, and the library will not detect them until you try to delete something.
While Sibling and Parent-child relations are performant by default, Free relations are less performant and rely on hidden object stores to work, forcing reads and writes to the database on relation creation and entity deletion. Be aware of this pitfall.
Also, defining cascading relations will run through relations reccursively when deleting entities, making the operation heavier than relation-less entities.
If your entity Key
type is u32
, you can auto-increment new entities using
use reindeer::AutoIncrementEntity;
let mut new_entity = Entity {
id : 0 // if you setup id with any key, saving will update it
/* ... */
};
new_entity.save_next(db)?;
// new_entity's key is now the auto-incremente value
You entitie's key will be automatically updated with set_key
to match the last found entry's ID, incremented by 1.
π‘ Note that the AutoIncrementEntity
trait needs to be in scope.