niwid-db is a toy database management system built from scratch, designed to explore fundamental database concepts, including memory and disk management, ACID properties, basic transaction handling, indexing, and query execution.
-
ACID Compliance via Shadow Paging: Ensures Atomicity, Consistency, Isolation (read-committed), and Durability by implementing shadow paging, maintaining a stable, crash-resistant state. Shadow paging allows a table to only have single-writer for simplified concurrency control.
-
Transaction Management: Supports commits and rollbacks to execute or discard changes within a transaction block to ensure transactional atomicity and isolation.
-
Custom Execution Engine: A simple query engine that processes SQL statements, validates the query and datatypes, and performs basic DDL, TCL, and DML operations. Also supports
EXPLAIN
andEXPLAIN ANALYZE
. Unoptimized logical plans are executed directly, no physical plan generation or optimizations yet. -
SQL Parsing: The only part not written from scratch. Leverages the
sqlparser-rs
crate for SQL syntax parsing. -
B+ Tree Indexing: Implements a B+ Tree index to ensure uniqueness, enabling efficient lookups and range queries for unique columns. Since there is no optimizer yet, you can force an index lookup using the
PREWHERE
clause. Check outindex.slt
for more examples. -
Constraints: Supports
NOT NULL
andUNIQUE
constraints to enforce data integrity, although primary keys and foreign keys (referential integrity) are not supported. -
Joins: Performs basic nested loop join operations between tables, allowing for relational queries.
- Rust: niwid-db is written in Rust.
-
Clone the repository:
git clone https://github.com/MohamedAbdeen21/niwid-db.git cd niwid-db
-
Run the server:
cargo run --release --bin idk-server
-
Connect to the server from a different terminal:
nc localhost 8080
niwid-db can execute SQL queries for managing tables, running transactions, and executing basic SQL commands.
Example:
CREATE TABLE users (
id INT PRIMARY KEY,
name TEXT NOT NULL,
email TEXT,
);
INSERT INTO users VALUES (1, 'Alice', '[email protected]');
You can also use transactions to batch operations:
BEGIN;
UPDATE users SET email = '[email protected]' WHERE id = 1;
ROLLBACK; -- Discards the update
For more examples, check out:
- the sqllogictests. If you're not familiar with slt files, check out how to read sqllogictests
- Examples panel in the demo website
- Slotted pages, with string indirection and a B+ Tree Index.
- LRU page eviction policy
- Shadow-Paging for ACID. No MVCC, OCC, 2PL, or WAL.
- Simple query engine, directly execute the raw logicial plan. No optimizer, or physical plan builder. The project focused on exploring the internals of the database, not the query engine.
- Catalog as a read-only Table, check it out using
SELECT * FROM __CATALOG__
; - Using
sqlparser.rs
, this is the only part not written from scratch. I wrote my fair share of parsers (and I did some contributing to sqlparser-rs), but again, this was not the main focus of the project. - Demo website made with HTMX and Askama.
Contributions are always welcome! Feel free to submit issues or pull requests.