ayb
With ayb
, all your base can finally belong to you. Move SQL for great justice.
ayb
makes it easy to create databases, share them with collaborators, and query them from a web application or the command line. An ayb
server allows users or organizations to create SQLite databases (other databases to come), and then exposes those databases through an HTTP API. In database terms, ayb
is a multi-tenant RDBMS that relies on embedded databases like SQLite for persistence and query processing while exposing those databases over HTTP.
alpha warning: ayb
is neither feature complete nor production-ready. Functionality like authentication, permissions, isolation, high availability, transaction support, collaboration, or DuckDB hosting are on the Roadmap but not available today. I work on ayb
as a hobbyist side project.
ayb
is written in Rust, and is available as the ayb
crate. Assuming you have installed Rust on your machine, installing ayb
takes a single command:
bash
cargo install ayb
An ayb
server stores its metadata in SQLite or PostgreSQL, and its embedded databases on a local disk. First, create an ayb.toml
configuration file to tell the server what host/port to listen for connections on, how to connect to the database, and the data path for the embedded databases:
```bash
cat <
datapath = "./aybdata" EOF ```
Running the server then requires one command
bash
$ ayb server
Once the server is running, you can set its URL as an environment variable called AYB_SERVER_URL
, register a user (in this case, marcua
), create a database marcua/test.sqlite
, and issue SQL as your heart pleases. Here's how to do that at the command line:
```bash $ export AYBSERVERURL=http://127.0.0.1:5433
$ ayb client register marcua Successfully registered marcua
$ ayb client create_database marcua/test.sqlite Successfully created marcua/test.sqlite
$ ayb client query marcua/test.sqlite "CREATE TABLE favorite_databases(name varchar, score integer);"
Rows: 0
$ ayb client query marcua/test.sqlite "INSERT INTO favorite_databases (name, score) VALUES (\"PostgreSQL\", 10);"
Rows: 0
$ ayb client query marcua/test.sqlite "INSERT INTO favorite_databases (name, score) VALUES (\"SQLite\", 9);"
Rows: 0
$ ayb client query marcua/test.sqlite "INSERT INTO favorite_databases (name, score) VALUES (\"DuckDB\", 9);"
Rows: 0
$ ayb client query marcua/test.sqlite "SELECT * FROM favorite_databases;" name | score ------------+------- PostgreSQL | 10 SQLite | 9 DuckDB | 9
Rows: 3 ```
The command line invocations above are a thin wrapper around ayb
's HTTP API. Here are the same commands as above, but with curl
:
```bash
$ curl -w "\n" -X POST http://127.0.0.1:5433/v1/marcua -H "entity-type: user"
{"entity":"marcua","entity_type":"user"}
$ curl -w "\n" -X POST http://127.0.0.1:5433/v1/marcua/test.sqlite -H "db-type: sqlite"
{"entity":"marcua","database":"test.sqlite","database_type":"sqlite"}
$ curl -w "\n" -X POST http://127.0.0.1:5433/v1/marcua/test.sqlite/query -d 'CREATE TABLE favorite_databases(name varchar, score integer);'
{"fields":[],"rows":[]}
$ curl -w "\n" -X POST http://127.0.0.1:5433/v1/marcua/test.sqlite/query -d "INSERT INTO favorite_databases (name, score) VALUES (\"PostgreSQL\", 10);"
{"fields":[],"rows":[]}
$ curl -w "\n" -X POST http://127.0.0.1:5433/v1/marcua/test.sqlite/query -d "INSERT INTO favorite_databases (name, score) VALUES (\"SQLite\", 9);"
{"fields":[],"rows":[]}
$ curl -w "\n" -X POST http://127.0.0.1:5433/v1/marcua/test.sqlite/query -d "INSERT INTO favorite_databases (name, score) VALUES (\"DuckDB\", 9);"
{"fields":[],"rows":[]}
$ curl -w "\n" -X POST http://127.0.0.1:5433/v1/marcua/test.sqlite/query -d "SELECT * FROM favorite_databases;"
{"fields":["name","score"],"rows":[["PostgreSQL","10"],["SQLite","9"],["DuckDB","9"]]} ```
Thank you for asking. I hope the answer elicits some nostalgia! Shout out to Meelap Shah and Eugene Wu for convincing me to not call this project stacks
, to Andrew Lange-Abramowitz for making the connection to the storied meme, and to Meredith Blumenstock for listening to me fret over it all.
Here's a rough roadmap for the project, with items near the top of the list more likely to be completed first. The nitty-gritty list of prioritized issues can be found on this project board, with the most-likely-to-be-completed issues near the top of the to-do list.
ayb
experience excellent
ayb
is to make it easier to create, share, and query databases, it's frustrating that running ayb
requires you to pay the nontrivial cost of operationalizing PostgreSQL. While Postgres will be helpful for eventually coordinating between multiple ayb
nodes, a single-node version should be able to store its metadata in SQLite with little setup costs.ayb
nodes to serve databases and requests. Whereas a single database will not span multiple machines, parallelism/distribution will happen across users and databases.ayb
node disappears.ayb
instance can have multiple tenants/databases, we want to use one of the many container/isolate/microVM projects to ensure that one tenant isn't able to access another tenant's data.ayb
's query API is a stateless request/response API, making it impossible to start a database transaction or issue multiple queries in a session. Exposing sessions in the API will allow multiple statements per session, and by extension, transactions.ayb
already uses existing well-established file formats (e.g., SQLite). There should be endpoints to import existing databases into ayb
in those formats or export the underlying files so you're not locked in.ayb
to more people and software
ayb
over the PostgreSQL wire protocol will allow existing tools and libraries to connect to and query an ayb
database.ayb
database.ayb
-hosted databases.(This section is inspired by the LiteFS project, and is just one of the many things to love about that project.)
ayb
contributions work a little different than most GitHub projects:
* If you have a small bug fix or typo fix, please PR directly to this repository.
* If you want to contribute documentation, please PR directly to this repository.
* If you would like to contribute a feature, create and discuss the feature in an issue on this GitHub repository first. Once the feature and some of its finer details are hashed out in the issue and potentially a design document, submit a pull request. I might politely decline pull requests that haven't first been discussed/designed.
This project has a roadmap and features are added and tested in a certain order. I'm adding a little friction in requiring a discussion/design document for features before submitting a pull request to ensure that I can focus my attention on well-motivated, well-sequenced, and well-understood functionality.