The ultimate full-stack experience.
Warning This project is in early preview.
This project can be used as a CLI:
cargo install ultime
This project includes features that will give you a faster and better development workflow using the following tech stack:
The main feature is the automatic of code generation based on your SurrealDB schemas and queries. An ultime project should look like this:
/schemas
· schemas folder from surrealdb-migrations/events
· events folder from surrealdb-migrations/migrations
· migrations folder from surrealdb-migrations/queries
· a list of .surql files that contains the specific queries for your project (only fetch data)/mutations
· a list of .surql files that contains the specific mutations for your project (change data)/src
/api
· list of API endpoints/components
· list of components that can be used anywhere/db
/crud
· functions for basic CRUD operations, generated from/schemas
files/events
· functions to execute SurrealDB events, generated from/events
files/mutations
· functions to update db, generated from/mutations
files/queries
· functions to query db, generated from/queries
files
/models
· list of structs used in the appqueries.rs
· types of the response of each query from/queries
files (this file is currently not automatically generated)mutations.rs
· types of the response of each mutation from/mutations
files (this file is currently not automatically generated)
/pages
· list of higher order components that can be used as a route
First, you need to create a new project using one of the predefined templates. So, for example:
ultime new my-blog --template blog
A new directory will be created. You can cd
to this new directory and run the following command:
ultime
This command will:
- start a new SurrealDB local instance
- apply schemas and migrations automatically
- generate the
db
module from/schemas
,/events
,/queries
and/mutations
folders - launch the leptos app
As of now, it is not possible to automatically detect the output of a .surql file: queries
or mutations
. However, a type is automatically generated for you so that all you need is to define the properties of this type. All models should be defined in the /src/models
folder.
Here is an example of the model definition from the response of /queries/posts.surql
query:
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
pub struct PostsQueryItem {
pub id: String,
pub title: String,
pub content: String,
pub status: String,
pub number_of_comments: u16,
}
pub type PostsQuery = Vec<PostsQueryItem>;
Another example from the response of /mutations/comment.surql
query:
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
pub struct CommentMutationItem {
pub id: String,
pub content: String,
}
pub type CommentMutation = Vec<CommentMutationItem>;
In order to differentiate internal variables and input variables, we established a pattern to follow in order to successfully extract query and mutation variables.
The following mutation...
// $user_id: String
// $post_id: Option<String>
// $comment_id: Option<String>
// $content: String
LET $user = (SELECT * FROM type::thing("user", $user_id));
LET $post_or_comment = (SELECT * FROM type::thing("post", $post_id), type::thing("comment", $comment_id));
RELATE $user->comment->$post_or_comment
SET content = $content;
...will generate this function:
pub async fn mutate_comment<C: Connection>(
db: &'_ Surreal<C>,
user_id: String,
post_id: Option<String>,
comment_id: Option<String>,
content: String
) -> Result<CommentMutation> {
const QUERY: &str = include_str!("../../../mutations/comment.surql");
let result: CommentMutation = db
.query(QUERY)
.bind(("user_id", user_id))
.bind(("post_id", post_id))
.bind(("comment_id", comment_id))
.bind(("content", content))
.await?
.take(0)?;
Ok(result)
}
As you can see, comments should follow the rules:
- only 1 variable per line
- it should start with the variable name (do not forget the
$
prefix) - then followed by a colon
:
- then it should end with the variable type
- whitespaces are allowed according to your coding convention
To help you get started quickly, there is a list of predefined templates you can use:
Template | Description |
---|---|
empty | The smallest ultime project you can create. A clean schema with an already defined script_migration table to store the applied migrations.A basic leptos app with a Counter example. |
blog | A blog app: create new blog posts, publish/unpublish posts and comments. |
You can create a new ultime project using the following command line:
ultime new <PROJECT_NAME> --template <TEMPLATE>