Skip to main content

SQL Toolkit for Python.

Project description

version license python versions pipeline status coverage report

SQLs

SQLs is a collection of libraries to interact with SQL databases. SQLs is not an object-relational mapper (ORM), but offers useful low-level primitives for handling transactions, defining a data model, and formulating queries in Python. Its main design goal is helping developers write SQL in idiomatic, type-checked Python, while being fast and efficient on runtime.

Getting started

For more, see the documentation.

SQLs is published on PyPI and does not have any special dependencies. Simply install with pip:

pip install sqls

Dependencies on database interface libraries are strictly optional. If you want to speak to a specific SQL implementation use any of:

pip install sqls[mysql]
pip install sqls[postgresql]
pip install sqls[sqlite]
pip install sqls[mysql,postgre,sqlite]

Connect to a database

Asynchronous transaction managers from sqls.transactions handle SQL database connections:

from sqls.transactions import get_manager

async def application() -> list[tuple[int]]:
    # Create a transaction manager.
    manager = get_manager('file:///path/to/sqlite.db')

    # Initialize database connections.
    await manager.init()

    # Open transaction.
    async with manager.txn() as txn:

       # Execute query.
       return await txn.execute('SELECT 1')

    # Close database connections.
    await manager.close()

All SQL statements inside the asynchronous context manager are executed in one single transaction. Uncaught exceptions in the context will cause the transaction to be automatically rolled back, on regular exit the transaction will automatically be commited.

Define a data model

The data model is defined through annotated Python dataclasses and the Model base class from sqls.models.

Basic usage

The syntax uses builtin Python primitives to express the rich details of SQL types:

from dataclasses import dataclass
from typing import Annotated

from sqls.models import CharField, Fieldmeta, IntegerField, Model

@dataclass
class User(Model):
    """A user model."""

    # Names are unique.
    name: Annotated[CharField, Fieldmeta(max_length=32, unique=True)]

    # Passwords are nullable.
    password: Annotated[CharField, Fieldmeta(max_length=128)] | None

    # Use just a plain integer.
    time_created: IntegerField

The Model base class automatically adds an integer primary key id field.

Relationships

Relationships are expressed through annotations on fields that store the actual information:

@dataclass
class User(Model):
    """Same as above, add some relationships."""

    # Table name and field are inferred from the attribute name.
    company_id: ForeignKeyField

    # Table name and field are explicitly set though Fieldmeta.
    team_id: Annotated[ForeignKeyField, Fieldmeta(foreign_key=('department', 'id'))

Many-to-many relationships cannot be expressed on the related models themselves, the table needs to be defined explicitly:

@dataclass
class UserGroup(Model):
    """User group relationship."""

    # Disable automatic injection of id field.
    id: None
    user_id: ForeignKeyField
    group_id: ForeignKeyField

Create database tables

The sqls.models package can generate CREATE TABLE queries from model definitions:

from sqls.models import get_create_queries

# Inside a transaction context (txn) create tables for User and Group.
for query in get_create_queries([User, UserGroup, Group]):
    # Execute generated query with txn.exq.
    await txn.exq(query)

Build queries

The sqls.queries package helps writing queries in idiomatic python:

from sqls.queries import Query, Table

# Create Table object from sql table name.
user_t = Table('user')

# Create query for id and password of one specific user.
query = (
    Query
    .select(
        user_t.id.typed(int),
        user_t.password.typed(str | None),
     )
    .from_(user_t)
    .where(user_t.name == 'Ringo')
)

As SQLs is not an ORM, Query knows nothing about the data model. By expressing the expected return type of the id field with .typed(int) static typed checkers like mypy are able to infer the return types when the query is executed.

Development

Clone the repository and setup your local checkout:

git clone https://gitlab.com/durko/sqls.git

cd sqls
python -m venv venv
. venv/bin/activate

pip install -r requirements-dev.txt
pip install -e .

This creates a new virtual environment with the necessary python dependencies and installs SQLs in editable mode. The SQLs code base uses pytest as its test runner, run the test suite by simply invoking:

pytest

To build the documentation from its source run sphinx-build:

sphinx-build -a docs build/public

The entry point to the local documentation build should be available under build/public/index.html.

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

sqls-0.0.100.tar.gz (69.2 kB view details)

Uploaded Source

Built Distribution

sqls-0.0.100-py3-none-any.whl (48.2 kB view details)

Uploaded Python 3

File details

Details for the file sqls-0.0.100.tar.gz.

File metadata

  • Download URL: sqls-0.0.100.tar.gz
  • Upload date:
  • Size: 69.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.11.3

File hashes

Hashes for sqls-0.0.100.tar.gz
Algorithm Hash digest
SHA256 70ebff44d9c9940f0ccce607525ae93bc5a2c54761fd6861dc46737d072096c9
MD5 9f96406aa93c730988a0cf17c7c748b7
BLAKE2b-256 afc7f3b7a6dd6ab368b4ce57898cfc9f7da1ae5c93e06a94b89c000775aa231e

See more details on using hashes here.

Provenance

File details

Details for the file sqls-0.0.100-py3-none-any.whl.

File metadata

  • Download URL: sqls-0.0.100-py3-none-any.whl
  • Upload date:
  • Size: 48.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.11.3

File hashes

Hashes for sqls-0.0.100-py3-none-any.whl
Algorithm Hash digest
SHA256 254d1b0cbbee5cdb7e9ce374a3eb58e40811fb8d509f7986145531f33a424f70
MD5 1c2982e61eb76f7f0c12b29e9053909d
BLAKE2b-256 e165ccc4717459e9d2d062e01600c4e84b9bdfd9e68f0d6d7b1d5a3149856d7e

See more details on using hashes here.

Provenance

Supported by

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