Guide
Learn how to build a minimal Evolu application. You can use Next.js (opens in a new tab) or any other React (opens in a new tab) meta-framework.
Define Data
First, we have to define the database schema. Evolu uses Schema (opens in a new tab)
for data modeling. Instead of plain JavaScript types like string or number,
we recommend using branded types (opens in a new tab).
With branded types, we can define and enforce domain rules like
NonEmptyString1000
or PositiveInt
.
Evolu.create
ensures actual SQLite schema and returns React Hooks.
import * as Schema from "@effect/schema/Schema";
import * as Evolu from "@evolu/react";
const TodoId = Evolu.id("Todo");
type TodoId = Schema.Schema.To<typeof TodoId>;
const TodoTable = Schema.struct({
id: TodoId,
title: Evolu.NonEmptyString1000,
isCompleted: Evolu.SqliteBoolean,
});
type TodoTable = Schema.Schema.To<typeof TodoTable>;
const Database = Schema.struct({
todo: TodoTable,
});
export const {
useQuery,
useMutation,
useOwner,
useOwnerActions,
useEvoluError,
} = Evolu.create(Database);
Validate Data
Learn more about Schema (opens in a new tab).
import * as Schema from "@effect/schema/Schema";
import * as Evolu from "@evolu/react";
Schema.parse(Evolu.String1000)(title);
Mutate Data
Mutation API is designed for local-first apps to ensure changes are always merged without conflicts and to limit the possibility that a developer accidentally makes unnecessary changes.
It's easy to change thousands of rows with traditional SQL, and that could be a problem when all those changes must be propagated to other devices. Evolu mutation API forces developers to think about the mutations they make.
const { create, update } = useMutation();
const { id } = create("todo", { title, isCompleted: false });
update("todo", { id, isCompleted: true });
Query Data
Evolu uses type-safe TypeScript SQL query builder kysely (opens in a new tab), so autocompletion works out-of-the-box.
const { rows } = useQuery(
(db) => db.selectFrom("todo").select(["id", "title"]).orderBy("updatedAt"),
// (row) => row
({ title, ...rest }) => title && { title, ...rest },
);
Protect Data
Evolu encrypts data with Mnemonic, a safe autogenerated password based on bip39 (opens in a new tab).
const owner = useOwner();
alert(owner.mnemonic);
Delete Data
Leave no traces on a device.
const ownerActions = useOwnerActions();
if (confirm("Are you sure? It will delete all your local data."))
ownerActions.reset();
Restore Data
Restore data elsewhere. Encrypted data can only be restored with a Mnemonic.
const ownerActions = useOwnerActions();
ownerActions.restore(mnemonic).then((either) => {
if (either._tag === "Left") alert(JSON.stringify(either.left, null, 2));
});
Handle Errors
Evolu useQuery
and useMutation
never fail, it's the advantage of local first apps, but Evolu, in rare cases, can.
const evoluError = useEvoluError();
useEffect(() => {
// eslint-disable-next-line no-console
if (evoluError) console.log(evoluError);
}, [evoluError]);