Skip to main content

A robust Python wrapper around SQLite3 providing dict-like API, multi-threading support, and REST service endpoints

Project description

DB86 — the AE‑86 of databases

License Python Version

DB86 is a robust Python 3.10+ wrapper around SQLite3 that provides a powerful, Pythonic interface for database operations. It supports both traditional relational tables and JSON document storage with a dict-like API, comprehensive multi-threading support, and optional REST service endpoints.

Built on lessons from sqlitedict, DB86 modernizes the approach with enhanced features, thread safety, and flexible storage backends.


Table of Contents


Features

Core Capabilities

  • 🗄️ Dict-like Interface: Intuitive Python dict API for all database operations
  • 🔄 Multi-threaded: Thread-safe database access with built-in synchronization
  • 💾 Dual Storage Modes:
    • JSON storage for document-style data
    • Relational tables for structured data
  • ⚙️ Flexible Configuration: Autocommit, journal modes, and timeout controls
  • 🚀 In-Memory & File-Based: Support for both :memory: and persistent databases
  • 🔌 REST API: Built-in FastAPI service for remote database access
  • 🎮 CLI Management: Interactive shell for database administration
  • 📤 Multiple Flags: Read-only ('r'), read-write ('c'), and write-fresh ('w') modes
  • 🔍 Schema Inspection: Database describe(), table metadata, indices, and views
  • 💪 Context Manager Support: Automatic resource cleanup with with statements

Requirements

  • Python: 3.10 or higher
  • Dependencies (installed automatically):
    • click-shell - Interactive CLI framework
    • fastapi - REST service framework
    • uvicorn - ASGI server
    • tabulate - Pretty-print database metadata

Installation

Via Poetry (Recommended)

poetry add db86

Via pip

pip install db86

From Source

git clone https://github.com/anubhav-narayan/db86.git
cd db86
poetry install

Quick Start

Basic Example: Key-Value Storage

from db86 import Database

# Create or open a database
db = Database('./my_db.sqlite', autocommit=True)

# Get a JSON storage table
mydict = db['mytab']

# Store and retrieve data
mydict['some_key'] = {'nested': 'any_picklable_object'}
print(mydict['some_key'])  # {'nested': 'any_picklable_object'}

# Iterate over items
for key, value in mydict.items():
    print(key, value)

# Standard dict operations
print(len(mydict))
if 'some_key' in mydict:
    del mydict['some_key']

# Don't forget to close
mydict.close()

Using Context Manager (Recommended)

from db86 import Database

with Database('./my_db.sqlite', autocommit=True) as db:
    storage = db['data']
    storage['key1'] = 'value1'
    storage['key2'] = {'nested': 'data'}
    # Automatically closes on exit

In-Memory Database

from db86 import Database

db = Database(':memory:')  # No file created
table = db['temp_data', 'json']
table['data'] = [1, 2, 3, 4, 5]
db.close()

Usage Guide

Basic Database Operations

Creating and Opening Databases

from db86 import Database

# File-based database (creates if not exists)
db = Database('./data.sqlite', flag='c', autocommit=True)

# Read-only database
db_read = Database('./data.sqlite', flag='r')

# Fresh database (overwrites existing)
db_new = Database('./fresh.sqlite', flag='w')

# In-memory database
db_mem = Database(':memory:')

Database Configuration Options

db = Database(
    filename='./my_db.sqlite',
    flag='c',                    # 'c'=read/write, 'r'=read-only, 'w'=overwrite
    autocommit=False,            # Auto-save after each operation
    journal_mode='DELETE',       # SQLite journal mode (DELETE, WAL, OFF)
    timeout=5                    # Seconds to wait for thread startup
)

Inspecting Database Structure

with Database('./db.sqlite') as db:
    # List all tables (storages)
    print(db.storages)  # ['table1', 'table2', 'table3']
    
    # List all indices
    print(db.indices)   # ['idx1', 'idx2']
    
    # List all views
    print(db.views)     # ['view1', 'view2']
    
    # Pretty-print database schema
    print(db.describe())
    
    # Check if table exists
    if 'users' in db:
        print("Users table exists")

JSON Storage

JSON Storage is the default mode for flexible, document-style data storage:

from db86 import Database

db = Database('./db.sqlite')

# Create or access JSON storage (default mode)
users = db['users']  # Equivalent to db['users', 'json']

# Store dictionaries and lists
users['user_001'] = {
    'name': 'Alice',
    'email': 'alice@example.com',
    'tags': ['admin', 'developer']
}

users['user_002'] = {
    'name': 'Bob',
    'email': 'bob@example.com',
    'tags': ['user']
}

# Retrieve data
alice = users['user_001']
print(alice['name'])  # 'Alice'

# List all keys
for user_id in users.keys():
    print(user_id)

# Iterate over key-value pairs
for user_id, user_data in users.items():
    print(f"{user_data['name']} ({user_id})")

# Update data
users['user_001']['tags'].append('reviewer')

# Delete entries
del users['user_002']

# Serialize to dict or JSON
all_data = dict(users)  # Convert to Python dict
db.close()

Tables (Structured Storage)

Tables provide relational storage with columns and schema:

from db86 import Database

db = Database('./db.sqlite')

# Create or access a structured table
products = db['products', 'table']

# Tables work like UserDict with additional schema capabilities
products['prod_001'] = {'name': 'Laptop', 'price': 999.99, 'stock': 5}
products['prod_002'] = {'name': 'Mouse', 'price': 29.99, 'stock': 150}

# Inspect table schema
print(products.describe())    # Pretty-print columns and types
print(products.columns)       # List column names
print(products.xschema)       # Get table definition and SQL

# Access like a dictionary
for prod_id, product in products.items():
    print(f"{product['name']}: ${product['price']}")

db.close()

Database Configuration

Autocommit vs Manual Commit

from db86 import Database

# Autocommit (safer, slower)
db_auto = Database('./db.sqlite', autocommit=True)
storage = db_auto['data']
storage['key'] = 'value'  # Automatically saved
db_auto.close()

# Manual commit (faster, requires explicit save)
db_manual = Database('./db.sqlite', autocommit=False)
storage = db_manual['data']
storage['key'] = 'value'
storage['key2'] = 'value2'
db_manual.commit()  # Save all changes at once
db_manual.close()

Journal Modes

# DELETE (default, safest)
db = Database('./db.sqlite', journal_mode='DELETE')

# WAL (Write-Ahead Logging, good for concurrent access)
db = Database('./db.sqlite', journal_mode='WAL')

# OFF (fastest, risky - disables crash recovery)
db = Database('./db.sqlite', journal_mode='OFF')

REST Service

DB86 provides a RESTful API for remote database access.

from db86.service.rest_service import app
import uvicorn

# Run the service
uvicorn.run(app, host="0.0.0.0", port=8000)

Endpoint Overview

  • GET / — health check and list open databases
  • POST /databases — create a new database
  • GET /databases — list open databases
  • GET /databases/{db_name} — get database metadata
  • DELETE /databases/{db_name} — close and remove a database
  • POST /databases/{db_name}/storages — create a storage (json or table)
  • GET /databases/{db_name}/storages — list storages
  • GET /databases/{db_name}/storages/{storage_name} — get storage metadata
  • DELETE /databases/{db_name}/storages/{storage_name} — delete a storage
  • GET /databases/{db_name}/storages/{storage_name}/items — list storage items
  • GET /databases/{db_name}/storages/{storage_name}/items/{item_key} — read an item
  • PUT /databases/{db_name}/storages/{storage_name}/items/{item_key} — create or update an item
  • DELETE /databases/{db_name}/storages/{storage_name}/items/{item_key} — delete an item
  • GET /databases/{db_name}/storages/{storage_name}/{query:path} — query JSON storage by nested path

Example API Calls

# Create a database
curl -X POST http://localhost:8000/databases \
  -H "Content-Type: application/json" \
  -d '{"name": "mydb", "autocommit": true, "journal_mode": "WAL", "flag": "c"}'

# Create JSON storage
curl -X POST http://localhost:8000/databases/mydb/storages \
  -H "Content-Type: application/json" \
  -d '{"name": "items", "storage_type": "json"}'

# Store an item
curl -X PUT http://localhost:8000/databases/mydb/storages/items/items/123 \
  -H "Content-Type: application/json" \
  -d '{"value": {"name": "Alice", "email": "alice@example.com"}}'

# Read an item
curl http://localhost:8000/databases/mydb/storages/items/items/123

# Delete an item
curl -X DELETE http://localhost:8000/databases/mydb/storages/items/items/123

Access the interactive API documentation at http://localhost:8000/docs

Advanced Examples

Transactions

from db86 import Database, Transaction

db = Database('./db.sqlite')

# Create a transaction context
with Transaction(db) as txn:
    storage = txn['data']
    storage['key1'] = 'value1'
    storage['key2'] = 'value2'
    # Auto-commits on success, rolls back on error

Working with Multiple Storages

db = Database('./db.sqlite', autocommit=True)

users = db['users', 'json']
orders = db['orders', 'table']
logs = db['logs', 'json']

users['alice'] = {'name': 'Alice', 'join_date': '2024-01-15'}
orders['order_001'] = {'user': 'alice', 'total': 99.99}
logs['entry_001'] = {'action': 'user_created', 'user_id': 'alice'}

db.close()

Large Dataset Handling

db = Database('./large.sqlite', autocommit=False, journal_mode='WAL')
data = db['large_dataset']

# Batch inserts without autocommit (faster)
for i in range(100000):
    data[f'key_{i}'] = {'index': i, 'value': f'value_{i}'}
    
    # Commit every 1000 records
    if i % 1000 == 0:
        db.commit()

db.commit()  # Final commit
db.close()

Read-Only Database Access

# Multiple readers can access simultaneously
db_read1 = Database('./data.sqlite', flag='r')
db_read2 = Database('./data.sqlite', flag='r')

# Read operations
storage1 = db_read1['data']
print(storage1['key'])

storage2 = db_read2['data']
print(storage2['key'])

db_read1.close()
db_read2.close()

Testing

Run the test suite:

poetry run pytest

Run specific test categories:

poetry run pytest tests/ -m unit -v

Contributing

Contributions are welcome! Please feel free to submit pull requests or open issues for bugs and feature requests.

Development Setup

git clone https://github.com/anubhav-narayan/db86.git
cd db86
poetry install
poetry run pytest tests/ -v

License

DB86 is released under the MIT License. See LICENSE.md for full details.

# MIT License

Copyright (c) 2021-2026 Anubhav Mattoo.

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

Acknowledgments


Questions? Open an issue on GitHub

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

db86-0.6.2.tar.gz (40.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

db86-0.6.2-py3-none-any.whl (39.6 kB view details)

Uploaded Python 3

File details

Details for the file db86-0.6.2.tar.gz.

File metadata

  • Download URL: db86-0.6.2.tar.gz
  • Upload date:
  • Size: 40.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for db86-0.6.2.tar.gz
Algorithm Hash digest
SHA256 66274dbfdf5ba26a67658d9eb07796b81e1db6c2fee0c5d6b054f24632ed483f
MD5 277e5b7eb89cc434d76c566ad06a46db
BLAKE2b-256 41be31f7e19cbb29a30d9953e17206a4ab00edb8102d387f0bb52c20c2dd50be

See more details on using hashes here.

Provenance

The following attestation bundles were made for db86-0.6.2.tar.gz:

Publisher: publish-workflow.yml on anubhav-narayan/db86

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file db86-0.6.2-py3-none-any.whl.

File metadata

  • Download URL: db86-0.6.2-py3-none-any.whl
  • Upload date:
  • Size: 39.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for db86-0.6.2-py3-none-any.whl
Algorithm Hash digest
SHA256 144d410a280b0e202740d7ef1b0a8ff7fd32dac13e164df7407b193de71d1c1b
MD5 62f8fe08d6c7a2de96607c0605aad4c5
BLAKE2b-256 beb2b2422d05cc1daa1b349c55f85a363a946d149fc5a47f3a1c8a265e57b4b9

See more details on using hashes here.

Provenance

The following attestation bundles were made for db86-0.6.2-py3-none-any.whl:

Publisher: publish-workflow.yml on anubhav-narayan/db86

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page