Thank you for considering a contribution to sqlit! This guide walks you through setting up your environment, running the test suite, and understanding the CI expectations.
-
Clone the repository:
git clone https://github.com/Maxteabag/sqlit.git cd sqlit -
Install in development mode with test dependencies:
pip install -e ".[dev]"For the full integration suite (all database drivers):
pip install -e ".[dev,all]"
CLI end-to-end tests run the entrypoint in a subprocess:
pytest tests/cli/ -vSQLite tests can run without any external dependencies:
pytest tests/ -v -k sqliteTo run the complete test suite including SQL Server, PostgreSQL, MySQL, MariaDB, FirebirdSQL, Oracle, ClickHouse, Turso (libsql), D1 (miniflare), SSH tunnel, DuckDB, CockroachDB, and Flight SQL tests:
-
Start the test database containers:
docker compose -f infra/docker/docker-compose.test.yml up -d
To include the enterprise test containers (Db2, Trino, Presto, Oracle 11g):
docker compose -f infra/docker/docker-compose.test.yml --profile enterprise up -d
-
Wait for the databases to be ready (about 30-45 seconds), then run tests:
pytest tests/ -v
To include Docker detection tests that spin up temporary containers:
pytest tests/integration/docker_detect/ -v --run-docker-container
You can leave the containers running between test runs - the test fixtures handle database setup/teardown automatically. Stop them when you're done developing:
docker compose -f infra/docker/docker-compose.test.yml downpytest tests/ -v -k sqlite
pytest tests/ -v -k mssql
pytest tests/ -v -k PostgreSQL
pytest tests/ -v -k MySQL
pytest tests/ -v -k cockroach
pytest tests/ -v -k firebird
pytest tests/ -v -k flightThe database tests can be configured with these environment variables:
SQL Server:
| Variable | Default | Description |
|---|---|---|
MSSQL_HOST |
localhost |
SQL Server hostname |
MSSQL_PORT |
1434 |
SQL Server port |
MSSQL_USER |
sa |
SQL Server username |
MSSQL_PASSWORD |
TestPassword123! |
SQL Server password |
PostgreSQL:
| Variable | Default | Description |
|---|---|---|
POSTGRES_HOST |
localhost |
PostgreSQL hostname |
POSTGRES_PORT |
5432 |
PostgreSQL port |
POSTGRES_USER |
testuser |
PostgreSQL username |
POSTGRES_PASSWORD |
TestPassword123! |
PostgreSQL password |
POSTGRES_DATABASE |
test_sqlit |
PostgreSQL database |
MySQL:
| Variable | Default | Description |
|---|---|---|
MYSQL_HOST |
localhost |
MySQL hostname |
MYSQL_PORT |
3306 |
MySQL port |
MYSQL_USER |
root |
MySQL username |
MYSQL_PASSWORD |
TestPassword123! |
MySQL password |
MYSQL_DATABASE |
test_sqlit |
MySQL database |
CockroachDB:
| Variable | Default | Description |
|---|---|---|
COCKROACHDB_HOST |
localhost |
CockroachDB hostname |
COCKROACHDB_PORT |
26257 |
CockroachDB port |
COCKROACHDB_USER |
root |
CockroachDB username |
COCKROACHDB_PASSWORD |
`` | CockroachDB password (empty for the included Docker container) |
COCKROACHDB_DATABASE |
test_sqlit |
CockroachDB database |
FirebirdSQL:
| Variable | Default | Description |
|---|---|---|
FIREBIRD_HOST |
localhost |
Firebird hostname |
FIREBIRD_PORT |
3050 |
Firebird port |
FIREBIRD_USER |
testuser |
Firebird username |
FIREBIRD_PASSWORD |
TestPassword123! |
Firebird password |
FIREBIRD_DATABASE |
/var/lib/firebird/data/test_sqlit.fdb |
Firebird database path or alias |
AWS Athena:
| Variable | Default | Description |
|---|---|---|
AWS_PROFILE |
default |
AWS CLI profile to use (must be configured in ~/.aws/credentials) |
AWS_REGION |
us-east-1 |
AWS Region |
IBM Db2:
| Variable | Default | Description |
|---|---|---|
DB2_HOST |
localhost |
Db2 hostname |
DB2_PORT |
50000 |
Db2 port |
DB2_USER |
db2inst1 |
Db2 username |
DB2_PASSWORD |
TestPassword123! |
Db2 password |
DB2_DATABASE |
testdb |
Db2 database name |
Trino:
| Variable | Default | Description |
|---|---|---|
TRINO_HOST |
localhost |
Trino hostname |
TRINO_PORT |
8082 |
Trino port |
TRINO_USER |
testuser |
Trino username |
TRINO_PASSWORD |
`` | Trino password |
TRINO_CATALOG |
memory |
Trino catalog |
TRINO_SCHEMA |
default |
Trino schema |
TRINO_HTTP_SCHEME |
http |
Trino HTTP scheme |
Presto:
| Variable | Default | Description |
|---|---|---|
PRESTO_HOST |
localhost |
Presto hostname |
PRESTO_PORT |
8083 |
Presto port |
PRESTO_USER |
testuser |
Presto username |
PRESTO_PASSWORD |
`` | Presto password |
PRESTO_CATALOG |
memory |
Presto catalog |
PRESTO_SCHEMA |
default |
Presto schema |
PRESTO_HTTP_SCHEME |
http |
Presto HTTP scheme |
Oracle 11g (Legacy):
| Variable | Default | Description |
|---|---|---|
ORACLE11G_RUN_TESTS |
`` | Enable Oracle 11g tests when set to 1 |
ORACLE11G_HOST |
localhost |
Oracle 11g hostname |
ORACLE11G_PORT |
1522 |
Oracle 11g port |
ORACLE11G_USER |
system |
Oracle 11g username |
ORACLE11G_PASSWORD |
oracle |
Oracle 11g password |
ORACLE11G_SERVICE |
XE |
Oracle 11g service name |
ORACLE11G_CLIENT_MODE |
thick |
Oracle client mode |
ORACLE11G_CLIENT_LIB_DIR |
`` | Oracle Instant Client library directory |
Flight SQL:
| Variable | Default | Description |
|---|---|---|
FLIGHT_HOST |
localhost |
Flight SQL server hostname |
FLIGHT_PORT |
31337 |
Flight SQL server port |
FLIGHT_USER |
`` | Flight SQL username (optional) |
FLIGHT_PASSWORD |
`` | Flight SQL password (optional) |
FLIGHT_DATABASE |
`` | Flight SQL database/catalog (optional) |
The core purpose of this application is to read Read&Write to a SQL database. The core elements to achieve this purpose is: CEQR:
- C: Connecting
- E: Exploring
- Q: Querying
- R: Viewing results
Connecting: Connecting to a server Exploring: Understanding the structure and content of the databases Querying: Executing SQL queries D: Viewing the results of SQL queries
Additionally, we have requirements 'EAFF':
- E: Easy
- A: Aesthetically pleasing
- F: Fun
- F: Fast
If an idea or feature does not achieve any of the 'CBQV' elements adhering to all of the 'EAFF' requirements. It does not belong to sqlit.
[E]asy: Sqlit should not require any external documentation at all. It must prioritize intuitiveness above all.
[A]esthetically pleasing: Sqlit should not render one unnecessary pixel. It should prioritize beauty above anything. Minimalism over bloat.
Fun: Sqlit aims make fulfilling its core purpose be an enjoyable experience for the user, even a source of pleasure.
Fast: Sqlit aims to fulfill its core purpose for the user, with intention to giving the user the results they want with as few actions as possible.
Essentially, sqlit aims to do CRUD on SQL really well.
This implies this tool is more suited for developer's daily use than an database administrator. Every feature in sqlit should have a target audience in which they will use it every time they use sqlit. If nobody is going to a feature every day. It does not belong to sqlit. E.g.
- advanced query performance debugging -> rarely used -> does not belong in sqlit
- edit cell key-binding -> a audience who will use this every day -> belongs to sqlit
On complexity: Minimalism: sqlit aims to abstract away as much complexity as possible from the user, while giving them enough control to achieve CBQV. sqlit should never do anything under the hood that the user might have interest in understanding. Universal state problems deem for magical fixes. Conditional state problems, explicit user awareness. User should never ask "wait, how did it know?" "why is this here?" "why did it work then, but not now?"
Voluntary advanced usage: Anything beyond most essentials to achieve CEQR should be voluntary to be exposed to. Minimal cogntive overhead. Always assume by default our user just wants to perform CRUD with SQL.
The idea is the user is exposed to an interface that's minimalistic and easy, but if they want to become 'power users' they may dig into command menu or see help and memorize.
Advances features should not be advertised on the main tool bar or anywhere else where the user has no say in whether it's rendered, as they take up space and distract from the most essential features for crud.
One state: There should be no settings or preferences with important exception of interface (aesthetics, keyboard bindings). No settings to enable or disable features for conditional behaviour. Do not include a feature that a user finds annoying. Settings to disable a feature is a symptom of this. Anything beyond essential must be sought after if needed, not disabled if unwanted.
Keybindings philosophy: To make sqlit as fast' as possible, sqlit has a large focus on keybindings. To make sqlit as 'easy' as possible, all necessary keybindings to do 'CEQR' well, must be visible at all times. To make sqlit as 'aesthetically pleasing' any keybinding not strictly necessary to perform 'CEQR' in a 'easy' and 'fast' way will be hidden behind help <?> or command menu Keybindings will favour 'vim' traditions as the core audience is developers who enjoy working in terminals. We shy away from ^+ commands and will only use them where it is not natural to have a "insert/normal" mode and where input is crucial. (Typical is pop up modals)
Ideal: It should be easy to use for someone who just started using sqlit. sqlit should provide fun and a feeling of mastery and satisfaction for those who want to achieve it, by becoming a sql-manipulating wizard with creative keybinding combos.
Designing keybindings decision hierarchy:
- Intuitive to learn
- Harmony (we should think about which keybindings are used in sequence, in typical to flow and maximize user mastery satisfaction and opportunity to combine them fast)
- Traditions (vim, specifically)
Example:
= explorer pane, = query pane, = results pane.
Rationale: E;Q;R satisfies both intuitiveness (each binding is the first letter of the pane), harmony (proximity: qwerty speaks for itself)