Skip to main content

Simple SQL in Python

Project description

SQL is code. Write it, version control it, comment it, and run it using files. Writing your SQL code in Python programs as strings doesn’t allow you to easily reuse them in SQL GUIs or CLI tools like psql. With aiosql you can organize your SQL statements in .sql files, load them into your python application as methods to call without losing the ability to use them as you would any other SQL file.

This project supports standard PEP 249 and asyncio based drivers for SQLite (sqlite3, aiosqlite, apsw), PostgreSQL (psycopg (3), apsycopg (3), psycopg2, pg8000, pygresql, asyncpg), MySQL (PyMySQL, mysqlclient, mysql-connector, asyncmy with this adapter), MariaDB (mariadb), DuckDB (duckdb) and MS SQL Server (pymssql), However, some detailed feature support may vary depending on the underlying driver and database engine actual capabilities.

Other SQL database drivers which support the pyformat or named PEP 249 paramstyles should work as well by just passing the driver as a parameter when building queries. Thus Oracle Database (oracledb) or Snowflake (snowflake.connector) should work out of the box… Please report with an issue if it actually works for you! Otherwise, extensions to support other database drivers can be written by you! See: Database Driver Adapters. Feel free to pull request!

This module is an implementation of Kris Jenkins’ yesql Clojure library to the Python ecosystem.

Badges

Build status Code Coverage Tests Issues Contributors Pypi Downloads Stars Version Code Size Databases Drivers Language Count Top Language Python Versions Badges BSD 2-Clause License

Usage

Install from pypi, for instance by running pip install aiosql.

Then write parametric SQL queries in a file and execute it from Python methods, eg this greetings.sql file:

-- name: get_all_greetings()
-- Get all the greetings in the database
select greeting_id, greeting
  from greetings
 order by 1;

-- name: get_user_by_username(username)^
-- Get a user from the database using a named parameter
select user_id, username, name
  from users
  where username = :username;

This example has an imaginary SQLite database with greetings and users. It prints greetings in various languages to the user and showcases the basic feature of being able to load queries from a SQL file and call them by name in python code. Query parameter declarations (eg (username)) are optional, and enforced when provided.

You can use aiosql to load the queries in this file for use in your Python application:

import aiosql
import sqlite3

queries = aiosql.from_path("greetings.sql", "sqlite3")

with sqlite3.connect("greetings.db") as conn:
    user = queries.get_user_by_username(conn, username="willvaughn")
    # user: (1, "willvaughn", "William")

    for _, greeting in queries.get_all_greetings(conn):
        # scan: (1, "Hi"), (2, "Aloha"), (3, "Hola"), …
        print(f"{greeting}, {user[2]}!")
    # Hi, William!
    # Aloha, William!
    # …

Or even in an asynchroneous way, with two SQL queries running in parallel using aiosqlite and asyncio:

import asyncio
import aiosql
import aiosqlite

queries = aiosql.from_path("greetings.sql", "aiosqlite")

async def main():
    async with aiosqlite.connect("greetings.db") as conn:
        user = await queries.get_user_by_username(conn, username="willvaughn")

        async for _, greeting in queries.get_all_greetings(conn):
            print(f"{greeting}, {user[2]}!")

asyncio.run(main())

It may seem inconvenient to provide a connection on each call. You may have a look at the AnoDB DB class which wraps both a database connection and queries in one connection-like extended object, including performing automatic reconnection when needed. The wrapper also allows to cache query results.

Why you might want to use this

  • You think SQL is pretty good, and writing SQL is an important part of your applications.

  • You don’t want to write your SQL in strings intermixed with your python code.

  • You’re not using an ORM like SQLAlchemy or Django , with large (100k lines) code imprints vs under 1000 for aiosql and about 300 for anodb, and you don’t need to or don’t want to write SQL-like code with a Python syntax.

  • You want to be able to reuse your SQL in other contexts, eg loading it into psql or other database tools.

Why you might NOT want to use this

  • You’re looking for an ORM.

  • You aren’t comfortable writing SQL code.

  • You don’t have anything in your application that requires complicated SQL beyond basic CRUD operations.

  • Dynamically loaded objects built at runtime really bother you.

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

aiosql-15.0.tar.gz (77.4 kB view details)

Uploaded Source

Built Distribution

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

aiosql-15.0-py3-none-any.whl (26.1 kB view details)

Uploaded Python 3

File details

Details for the file aiosql-15.0.tar.gz.

File metadata

  • Download URL: aiosql-15.0.tar.gz
  • Upload date:
  • Size: 77.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.3

File hashes

Hashes for aiosql-15.0.tar.gz
Algorithm Hash digest
SHA256 744939fdfb3e0c36d88ccaf1f73cb1cf8cc38e7052666b884502db99aff8f3fd
MD5 078c327c56c560bafc779539d5cb227a
BLAKE2b-256 dc3197ebbd15ead5cf9c3951d6e8dfafc5e7b7e8c52148768cb7b95cd443fc8a

See more details on using hashes here.

File details

Details for the file aiosql-15.0-py3-none-any.whl.

File metadata

  • Download URL: aiosql-15.0-py3-none-any.whl
  • Upload date:
  • Size: 26.1 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.3

File hashes

Hashes for aiosql-15.0-py3-none-any.whl
Algorithm Hash digest
SHA256 ba659870914258790da77a999902c0b7712d58754ca2bd335cf2be34a8433b42
MD5 be3985c187db1c28669a932b6f85aa1f
BLAKE2b-256 28a467a07ed3e827a50671d7248624c1d3666243c580b0b0567c62d12e1c6de7

See more details on using hashes here.

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