• Stars
    star
    220
  • Rank 180,422 (Top 4 %)
  • Language
    Rust
  • License
    MIT License
  • Created over 1 year ago
  • Updated 2 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

An awesome SurrealDB migration tool, with a user-friendly CLI and a versatile Rust library that enables seamless integration into any project.

Crates.io GitHub Workflow Status GitHub codecov

SurrealDB Migrations

An awesome SurrealDB migration tool, with a user-friendly CLI and a versatile Rust library that enables seamless integration into any project.

Warning This project is not production-ready, use at your own risk.

This project can be used:

  • as a Rust library
cargo add surrealdb-migrations
  • or as a CLI
cargo install surrealdb-migrations

Check surrealdb-migrations GitHub Action in the marketplace

The philosophy

The SurrealDB Migrations project aims to simplify the creation of a SurrealDB database schema and the evolution of the database through migrations. A typical SurrealDB migration project is divided into 3 categories: schema, event and migration.

A schema file represents no more than one SurrealDB table. The list of schemas can be seen as the Query model (in a CQRS pattern). The schemas folder can be seen as a view of the current data model.

An event file represents no more than one SurrealDB event and the underlying table. The list of events can be seen as the Command model (in a CQRS pattern). The events folder can be seen as a view of the different ways to update the data model.

A migration file represents a change in SurrealDB data. It can be a change in the point of time between two schema changes. Examples are: when a column is renamed or dropped, when a table is renamed or dropped, when a new data is required (with default value), etc...

Get started

stateDiagram-v2
    scaffold : Scaffold a project
    changeSchema : Change schema/event
    createMigration: Create migration (data changes)
    apply : Apply to your database

    state fork_state <<fork>>
        [*] --> scaffold
        scaffold --> fork_state
        fork_state --> changeSchema
        fork_state --> createMigration

    state join_state <<join>>
        changeSchema --> join_state
        createMigration --> join_state
        join_state --> apply
        apply --> fork_state

1. Scaffold

You can start a migration project by scaffolding a new project using the following command line:

surrealdb-migrations scaffold template empty

This will create the necessary folders and files in order to perform migrations. The empty template should look like this:

  • /schemas
    • script_migration.surql
  • /events
  • /migrations

There are a number of pre-defined templates so you can play around and get started quickly.

2. Change schema and/or create data change migrations

Once you have created your migration project, you can start writing your own model. Based on the folders you saw earlier, you can create schema files, event files and migration files.

Schemas

You can create strict schema files that represent tables stored in SurrealDB.

surrealdb-migrations create schema post --fields title,content,author,created_at,status

This will create a schemaless table with predefined fields:

DEFINE TABLE post SCHEMALESS;

DEFINE FIELD title ON post;
DEFINE FIELD content ON post;
DEFINE FIELD author ON post;
DEFINE FIELD created_at ON post;
DEFINE FIELD status ON post;

Events

You can also create events in the same way.

surrealdb-migrations create event publish_post --fields post_id,created_at

This will define a table event with predefined fields:

DEFINE TABLE publish_post SCHEMALESS;

DEFINE FIELD post_id ON publish_post;
DEFINE FIELD created_at ON publish_post;

DEFINE EVENT publish_post ON TABLE publish_post WHEN $event == "CREATE" THEN (
    # TODO
);

Migrations

And when updating data, you can create migration files this way:

surrealdb-migrations create AddAdminUser

This will create a new file using the current date & time of the day, like 20230317_153201_AddAdminUser.surql for example. All migrations files should be listed in a temporal order.

3. Apply to your database

Finally, when you are ready, you can apply your schema and migrations to the database using the following command line:

surrealdb-migrations apply

Or directly inside your Rust project using the following code:

use surrealdb_migrations::MigrationRunner;
use surrealdb::engine::any::connect;
use surrealdb::opt::auth::Root;

#[tokio::main]
async fn main() -> Result<()> {
    let db = connect("ws://localhost:8000").await?;

    // Signin as a namespace, database, or root user
    db.signin(Root {
        username: "root",
        password: "root",
    }).await?;

    // Select a specific namespace / database
    db.use_ns("namespace").use_db("database").await?;

    // Apply all migrations
    MigrationRunner::new(&db)
        .up()
        .await
        .expect("Failed to apply migrations");

    Ok(())
}

4. Repeat

Repeat the process from step 2. Change schema and/or create data change migrations.

Predefined templates

To help you get started quickly, there is a list of predefined templates you can use:

Template Description
empty The smallest migration project you can create.
A clean schema with an already defined script_migration table to store the applied migrations.
blog A blog domain model, with users having the ability to publish/unpublish posts and comments.
ecommerce An ecommerce domain model, with customers having the ability to purchase products.

You can scaffold a project using any of these templates using the following command line:

surrealdb-migrations scaffold template <TEMPLATE>

Configuration

You can create a .surrealdb configuration file at the root of your project. This way you won't have to set the same configuration values every time.

[core]
    path = "./tests-files"
    schema = "less"

[db]
    address = "ws://localhost:8000"
    username = "root"
    password = "root"
    ns = "test"
    db = "test"

In the core section, you can define the path to your schema/migration files, if it is not the current folder.

In the db section, you can define the values used to access your SurrealDB database. It can be the url, username, password, the namespace ns or the name of the database db.

Here is the definition of the .surrealdb configuration file:

[core]
    # Optional
    # Type: String
    # Description: Path to the folder that contains your migration project (root folder by default)
    # Default: "."
    path

    # Optional
    # Type: "less" | "full"
    # Description: Define SCHEMALESS or SCHEMAFULL option by default when creating new table/event file
    # Default: "less"
    schema

[db]
    # Optional
    # Type: String
    # Description: Address of the surrealdb instance
    # Default: "ws://localhost:8000"
    address

    # Optional
    # Type: String
    # Description: Username used to authenticate to the surrealdb instance
    # Default: "root"
    username

    # Optional
    # Type: String
    # Description: Password used to authenticate to the surrealdb instance
    # Default: "root"
    password

    # Optional
    # Type: String
    # Description: Namespace to use inside the surrealdb instance
    # Default: "test"
    ns

    # Optional
    # Type: String
    # Description: Name of the database to use inside the surrealdb instance
    # Default: "test"
    db

Backward migrations

By default, migrations are forward-only. However, it can be interesting to revert a migration in order to undo a mistake. You will find backward migrations in two places:

  1. Inside the /migrations/down folder with the same name as your forward migration
  2. Inside the /migrations but with the .down.surql extension next to the forward migration

So, a migration project with backward migrations might look like this:

  • /schemas
    • script_migration.surql
  • /events
  • /migrations
    • 20231605_205201_AddProduct.surql
    • /down
      • 20231605_205201_AddProduct.surql

Or like this:

  • /schemas
    • script_migration.surql
  • /events
  • /migrations
    • 20231605_205201_AddProduct.surql
    • 20231605_205201_AddProduct.down.surql

If you want to create a DOWN migration file when creating the migration file, use this command:

surrealdb-migrations create AddProduct --down

If you need to, you can revert all migrations back to the one you specified.

surrealdb-migrations apply --down 20231605_205201_AddProduct

And if you need to undo all your migrations, use this command:

surrealdb-migrations apply --down 0

Database branching

Database branching is a similar concept to version control system like Git where you manage code repositories with branches.

With database branching, you can create a separate copy or branch of the main database to perform various tasks such as testing new features, implementing changes, or running experiments. This allows developers or teams to work independently on different branches without interfering with the stability and integrity of the original database.

You can make make schemas changes, apply new migrations and/or change data on a separate branch. These changes are isolated from the main database until they are merged back, allowing for better control and organization of database changes.

Development workflow

In a development workflow, you have a primary/main database that contains the latest features on your project. You often work on multiple features or you want to try the work of your colleagues but it messes up your development database, whether you are using migrations or not. Database branching allows you to create a fork of the main database, work on a new feature, apply schema or data changes and then merge your new changes to the main database when your feature is ready.

stateDiagram-v2
    main : Main branch
    createBranch : Create branch "feature-1"
    makeChanges : Make changes on "feature-1"
    merge : Merge "feature-1" on main
    remove : Remove branch "feature-1"

    [*] --> main
    main --> [*]
    main --> createBranch
    createBranch --> makeChanges
    makeChanges --> remove
    makeChanges --> merge
    remove --> [*]
    merge --> [*]

You start by creating a new branch using the following command line:

surrealdb-migrations branch new --address http://localhost:8000

You will then receive a message like this:

You can now use the branch with the following configuration:

ns: branches
db: bright-fold-1617

You can now make your changes on the newly generated database using ns and db properties. When you are done with your changes, you can merge your branch to the origin branch using the following command line:

surrealdb-migrations branch merge bright-fold-1617 --mode all --address http://localhost:8000

There are 3 merge modes, each with its own interest:

Mode Description Status
schema-only A diff of schema will be applied between the branch and the origin branch at the moment of the branch creation.

If possible, the merge will operate schema changes on the origin branch:
* defining new tables, fields, etc...
* removing tables, fields, etc...
Planned
all As an extension to the schema-only mode, a diff of schema and data will be applied between the branch and the origin branch at the moment of the branch creation.

If possible, the merge will operate schema and data changes on the origin branch:
* defining new tables, fields, etc...
* removing tables, fields, etc...
* adding, updating or removing table rows/columns
Planned
overwrite Merging the branch will completely destroy the origin branch and replace it with the new one.
The main branch will have the schema and the data set in the merged branch.
In progress

Production workflow

TBD

Database restrictions

This feature requires 3 namespaces:

  • features
  • branches
  • branches/origin

It is strongly recommended to avoid using one of these namespaces in your SurrealDB instance.

Documentation

# create new branch from current branch with a random name, from default ns/db
surrealdb-migrations branch new
# create new branch from current branch with a name, from default ns/db
surrealdb-migrations branch new <BRANCH_NAME>
# create new branch, from a specific ns/db
surrealdb-migrations branch new --ns <NS> --db <DB>

# review diffs between original branch and the new branch
surrealdb-migrations branch diff <BRANCH_NAME>

# commit and merge branch changes to the original branch
surrealdb-migrations branch merge <BRANCH_NAME>

# remove branch (ie. rollback)
surrealdb-migrations branch remove <BRANCH_NAME>

# list all existing branches
surrealdb-migrations branch list

# display infos of a branch
surrealdb-migrations branch status <BRANCH_NAME>
surrealdb-migrations branch <BRANCH_NAME>

Samples

This project contains sample apps that demontrates how to use the surrealdb-migrations given certain contexts. Here is a list of existing samples:

Name Description Languages/Frameworks
wasm This project shows how to use the surrealdb-migrations crate with embedded migrations files in a WASM context.
The app entry point is powered by SvelteKit and the vite-plugin-rsw plugin.
The SurrealDB data is stored locally in IndexedDb.
SvelteKit/Rust (WASM)

Let's see Paul Allen's contributions

Thanks goes to these wonderful people (emoji key):

Amar Sood
Amar Sood

💻
Rom's
Rom's

💻 🤔 🚇
Pranay Pratyush
Pranay Pratyush

🐛
Zafar Ansari
Zafar Ansari

🤔
Tim
Tim

🤔
Matt Jackson
Matt Jackson

🐛
Lucas
Lucas

🤔

This project follows the all-contributors specification. Contributions of any kind welcome!

Credits

Inspired by awesome projects:

More Repositories

1

ReduxSimple

Simple Stupid Redux Store using Reactive Extensions
C#
143
star
2

cqrs-dotnet-core-example

A naive introduction to CQRS in C#
C#
63
star
3

FeatureManagement.UI

Perfectly designed UI for Configuration management/Feature Flags in ASP.NET Web API
C#
29
star
4

ngrx-signalr-core

A library to handle realtime SignalR (.NET Core) events using @angular, rxjs and the @ngrx library
TypeScript
28
star
5

gitter-bot-how-to

A tutorial on how to make a Gitter Bot
27
star
6

Modern-Gitter

A Gitter client application for the Windows platform
C#
16
star
7

surrealdb-presence-demo

Demo project on how to create a realtime presence web application using SurrealDB Live Queries
TypeScript
16
star
8

ultime

The ultimate full-stack experience
Rust
15
star
9

azuread-msgraph-dotnet-core-example

How to consume Microsoft Graph API using Azure AD authentication in .NET Core
C#
15
star
10

react-redux-axios-middleware-netcore-example

Example of a react-redux frontend using redux-axios-middleware that calls a .NET Core Web API
TypeScript
12
star
11

azuread-react-dotnet-core-example

A memo on how to implement Azure AD authentication using React and .NET Core
TypeScript
12
star
12

signalr-core-rx-streaming-example

Example on how to use SignalR Core streams with Rx (both client-side and server-side)
TypeScript
10
star
13

tagada

For those who dream to make an ASP.NET Core Web API in one line of code
C#
9
star
14

modern-gitter-winjs

Windows 10 client of gitter, written in WinJS
JavaScript
9
star
15

RxSignalrStreams.Extensions

Extensions to use Reactive observable in SignalR streams
C#
9
star
16

pomodoro-code

A pomodoro timer inside your Visual Studio Code IDE
TypeScript
9
star
17

typescript-vanilla-starter

Starter project for vanilla js project written in TypeScript with webpack
JavaScript
6
star
18

surrealdb-migrations-action

Apply migrations to a SurrealDB instance
TypeScript
6
star
19

react-state-management-comparison

Comparison of different React state management libraries (hooks, mobx, recoiljs)
TypeScript
6
star
20

Converto

A C# library which gives you basic functions for type conversion and object transformation
C#
5
star
21

Modern-GitHub

A Universal Windows companion app (Windows 8 & Windows Phone 8) for GitHub users
C#
4
star
22

ngrx-signalr

A library to handle realtime SignalR (.NET Framework) events using @angular, rxjs and the @ngrx library
TypeScript
3
star
23

magic-json-vscode

Get meaningful insights from your JSON files
TypeScript
3
star
24

colorrrs-winjs

A HEX to RGB converter on Windows 10 - based on the work of @jamiebrittain
JavaScript
2
star
25

EnableNullableReferenceTypes

Example on how to enable C# 8.0 and Nullable Reference Types in any .NET project
C#
2
star
26

new-cmd

Launch a new command line that target the current workspace
TypeScript
2
star
27

iron-maiden-albums

A list of Iron Maiden albums & songs
TypeScript
2
star
28

SimpleEventSourcing

Event Sourcing made simple using Reactive Extensions
C#
2
star
29

Ntiers-dotNet-webservices

An ASP.NET MVC project with WCF webservices using the Northwind database with Entity Framework.
C#
1
star
30

sveltekit-wasm-game-of-life

Conway's game of life using SvelteKit and Rust wasm
Rust
1
star
31

AspNetCoreAzureTemplates

A list of templates used to generate ASP.NET Core web apps using ready-to-use Azure features
C#
1
star
32

Tamazombie

Mixing the famous game Tamagotchi with the idea of zombie, Java game using libgdx
Java
1
star
33

Modern-Wordreference

A Wordreference client application for Windows 8 & Windows Phone 8
C#
1
star
34

Vigenere

A Vigenere application (encryption/decryption) in C#/WPF using the Vigenere table
C#
1
star
35

Arduino-Alarm-CSharp

Manage Alarm with Arduino using a GUI in WPF (C#/XAML)
C#
1
star
36

photo-album

C#
1
star
37

uwp-togglebutton-oneway

UWP ToggleButton with OneWay binding and command sample
C#
1
star
38

modern-wordreference-uwp

The non official Windows 10 app based on wordreference.com
C#
1
star
39

Prolog-Sudoku-Resolver

A sudoku resolver in C#/WPF using prolog implementation...
C#
1
star
40

ngrx-watch-component-store

Simplify debugging of ngrx's ComponentStore
TypeScript
1
star
41

monorepo-migration

Example repository on how to migrate a monorepo from Next.js to Astro
TypeScript
1
star