ExTurso
An Elixir library that wraps the turso Rust
crate (v0.5) via Rustler NIFs, exposed
through a DBConnection pool.
It supports local file databases (and ":memory:"), Turso Cloud sync
via embedded replicas, and turso's built-in vector search SQL functions,
with correct ResourceArc lifetime management and a working connection pool.
Migrations and an Ecto adapter are out of scope.
Requirements
A working Rust toolchain (cargo) matching your BEAM's architecture is required
to build the NIF.
Installation
def deps do
[
{:ex_turso, "~> 0.1.0"}
]
end
Usage
Start a pool under your supervision tree:
children = [
{ExTurso, database: "my_app.db", name: MyApp.DB}
]
Supervisor.start_link(children, strategy: :one_for_one)
Then query and execute against the registered name:
{:ok, _} = ExTurso.execute(MyApp.DB, "CREATE TABLE users (id INTEGER, name TEXT)")
{:ok, _} = ExTurso.execute(MyApp.DB, "INSERT INTO users VALUES (?, ?)", [1, "Alice"])
{:ok, %ExTurso.Result{rows: [%{"name" => "Alice"}]}} =
ExTurso.query(MyApp.DB, "SELECT name FROM users WHERE id = ?", [1])
Transactions go through DBConnection:
DBConnection.transaction(MyApp.DB, fn conn ->
{:ok, _} = ExTurso.execute(conn, "UPDATE users SET name = ? WHERE id = ?", ["Bob", 1])
end)
Use database: ":memory:" for an in-memory database (one per pool connection).
Always pass values as bound parameters (?) rather than interpolating them
into the SQL string — statements are logged when a query errors.
Turso Cloud sync
Pass :remote_url and :auth_token to open the local file as an embedded
replica of a Turso Cloud database:
children = [
{ExTurso,
database: "replica.db",
remote_url: "libsql://my-db.turso.io",
auth_token: fn -> System.fetch_env!("TURSO_AUTH_TOKEN") end,
name: MyApp.DB}
]
auth_token accepts a string or a zero-arity function; prefer the function so
the token does not appear in supervisor child specs and crash reports.
Trigger a bidirectional sync (pull then push) with:
:ok = ExTurso.sync(MyApp.DB)
Sync is rejected inside a transaction and on databases not configured with
:remote_url/:auth_token.
Errors
Failures return {:error, %ExTurso.Error{message: message, code: code}}. The
code classifies the failure: :busy (locked, retryable), :constraint,
:invalid_param (unsupported bound parameter type), :misuse, :error, or
:io/:corrupt — the last two mark the connection as broken, so the pool
drops it and opens a fresh one.
Architecture
| Layer | Module / file | Role |
|---|---|---|
| Native | native/ex_turso/src/lib.rs | Rustler NIFs over turso, driven by a global Tokio runtime |
| NIF decls | ExTurso.Native | Loads the compiled NIF |
| Pooling | ExTurso.Connection | DBConnection behaviour implementation |
| Query | ExTurso.Query | Statement struct + DBConnection.Query protocol |
| Public API | ExTurso | start_link/1, child_spec/1, query/3, execute/3 |