Skip to main content

Polars-like lazy access to a database.

Project description

LazyBear

LazyBear is a polars-flavored lazy SQL query builder on top of sqlalchemy. It lets you compose familiar transformations like select, filter, with_columns, join, group_by().agg(...), and sort/order_by against a SQL database, and then materialize results to polars or arrow (or stream them in batches).

The purpose of this library is to provide lazy, polars-like access to a single sql database server, providing multi-site (and multi-server) stable code while pushing most memory-intensive operations to the remote server.

  • Familiar, chainable API similar to polars
  • Backed by sqlalchemy for broad database support
  • Zero data is loaded until you call collect()/to_arrow()/writers
  • Convenient I/O helpers for CSV and Parquet (using polars)

Full API documentation:

Installation

  • LazyBear targets Python 3.10+.
  • Required dependencies:
    • sqlalchemy
    • polars
  • Optional:
    • pyarrow for to_arrow()

Install from PyPI:

pip install lazybear-polars

Install from source using pip:

# after git clone
pip install -e .
# straight from repo
pip install git+https://github.com/kpwhri/lazybear.git@master

Usage

Quickstart

Below is an end-to-end walkthrough using an in-memory sqlite database. The same api should work for other databases supported by sqlalchemy.

Create Playgruond

First, let's setup the backend play data:

import sqlalchemy as sa
import polars as pl

# create a sqlalchemy engine
eng = sa.create_engine('sqlite:///:memory:')

# prepare some tables for the demo
meta = sa.MetaData()
users = sa.Table(
    'users', meta,
    sa.Column('id', sa.Integer, primary_key=True),
    sa.Column('name', sa.String),
    sa.Column('age', sa.Integer),
)
orders = sa.Table(
    'orders', meta,
    sa.Column('id', sa.Integer, primary_key=True),
    sa.Column('user_id', sa.Integer),
    sa.Column('amount', sa.Float),
)
meta.create_all(eng)
with eng.begin() as conn:
    conn.execute(users.insert(), [
        {'id': 1, 'name': 'Ahti', 'age': 30},
        {'id': 2, 'name': 'Kalma', 'age': 28},
        {'id': 3, 'name': 'Tellervo', 'age': 41},
        {'id': 4, 'name': 'Ukko', 'age': 41},
    ])
    conn.execute(orders.insert(), [
        {'id': 10, 'user_id': 1, 'amount': 12.5},
        {'id': 11, 'user_id': 1, 'amount': 7.5},
        {'id': 12, 'user_id': 2, 'amount': 99.0},
    ])

Implemented SQL Operations

Now, let's see what we can do:

from lazybear import scan_table, scan_sql_query, col

# scan tables lazily
lf_users = scan_table('users', eng)  # columns exposed lowercase by default
lf_orders = scan_table('orders', eng)

# basic select / filter / collect → returns a polars DataFrame
out_df = (
    lf_users
    .filter(col('age') > 30)
    .select('id', 'name', 'age')
    .collect()
)
print(out_df)

# with_columns — add or replace columns
with_cols_df = (
    lf_users
    .with_columns(('age2', col('age') * 2), decade=col('age') / 10 * 10)
    .select('id', 'age2', 'decade')
    .collect()
)
print(with_cols_df)

# order and limit (keeps ordering stable even across subqueries)
ordered_df = lf_users.order_by('-age', 'name').limit(2).collect()
print(ordered_df)

# polars-style sort api
sorted_df = lf_users.sort(by='age', descending=True).collect()
print(sorted_df)

# joins (left columns keep names; right overlaps get suffixed with `_y` by default)
joined_df = (
    lf_users
    .join(lf_orders, on={'id': 'user_id'}, how='left')
    .select('id', 'name', 'age', ('amount_y', col('amount')))
    .collect()
)
print(joined_df)

# group_by + aggregations
agg_df = (
    lf_users
    .group_by('age')
    .agg(n=('id', 'count'), min_id=('id', 'min'))
    .collect()
)
print(agg_df)

# expressions: membership, null checks, and string helpers
expr_df = (
    lf_users
    .filter(
        (col('name').str.contains('k', literal=True)) |
        (col('name').startswith('A')) |
        (col('age').is_in([28, 41]))
    )
    .collect()
)
print(expr_df)

Uploading Temp Tables (Beta)

You can use a local polars DataFrame in lazy sql operations by creating a temporary table. This is useful for joining local data with database tables.

Be aware that the table will be inserted when collect is called. After the collection is complete, a best effort attempt to delete the temp table will be completed.

import polars as pl
from lazybear import scan_df, scan_table, col

# local polars dataframe
df_local = pl.DataFrame({
    'user_id': [1, 3],
    'status': ['active', 'inactive']
})

# create a temp table frame
lf_temp = scan_df(df_local, eng, table_name='user_status')

# join with a database table
lf_users = scan_table('users', eng)
result = (
    lf_users
    .join(lf_temp, on={'id': 'user_id'})
    .select('name', 'status')
    .collect()
)

Limitations & Behavior:

  • Temp tables are created and data is inserted only when collect() (or another execution method) is called.
  • Best-effort cleanup (DROP TABLE) is performed after the data is fetched.
  • Currently offers beta support for sqlite, PostgreSQL, SQL Server (MSSQL), Oracle, and Teradata. A warning is issued for other dialects.
  • For certain dialects, will attempt bulk insert to speed up processing

Exporting Data

Finally, we'll

# iterating rows
for row in lf_users.order_by('id').iter_rows():
    # row is a tuple by default
    print(row)
for row in lf_users.order_by('id').iter_rows(named=True):
    # named=True yields dicts
    print(row)

# streaming in batches
for batch_df in lf_users.order_by('id').collect_batches(chunk_size=2):
    print('Batch:', batch_df)

# scan a raw SQL query (must be SELECT)
q_df = (
    scan_sql_query('select name, age from users where age > 30', eng)
    .filter(col('age') < 60)
    .select('name')
    .collect()
)
print(q_df)

# materialize to arrow (requires optional pyarrow)
arrow_tbl = lf_users.to_arrow()

# explain shows the composed SQL (with literal binds where possible)
print(lf_users.filter(col('age') > 30).explain())

I/O helpers

  • CSV
# single file
lf_users.order_by('id').write_csv('users.csv', include_header=True)

# append in chunks to one file
lf_users.order_by('id').write_csv('users_chunked.csv', chunk_size=2, include_header=True)
  • Parquet
# single file
lf_users.collect().write_parquet('users.parquet')

# chunked parts with a common prefix like users-00000.parquet, users-00001.parquet, ...
lf_users.order_by('id').write_parquet('users.parquet', chunk_size=2)

Notes:

  • write_csv/write_parquet use polars under the hood. For chunked Parquet, files are created with a numeric suffix.
  • to_arrow() requires pyarrow to be installed.

Properties

  • columns: Returns a list of column names in the frame.
  • engine: Returns the SQLAlchemy Engine that the frame is bound to.

Materialization & Execution

  • collect(limit=None, infer_schema_length=200): Materializes the lazy query into a polars DataFrame.
  • to_arrow(limit=None): Materializes the query into a pyarrow.Table.
  • collect_batches(chunk_size=10_000): Returns an iterator of polars DataFrame chunks.
  • iter_rows(named=False, chunk_size=10_000): Returns an iterator of row tuples (or dicts if named=True).
  • explain(): Returns the rendered SQL string for the query.

I/O Helpers

  • write_parquet(file, chunk_size=None, start_index=0, **kwargs): Writes the result to one or more Parquet files.
  • write_csv(file, chunk_size=None, **kwargs): Writes the result to a CSV file.

Advanced

  • immutability: every transform (select, filter, with_columns, join, sort, limit, group_by) returns a new LazyBearFrame.
  • to_select() returns the current SQLAlchemy Select if you need to interop with SQLAlchemy APIs directly.
  • case sensitivity: scan_table(..., lowercase=True) exposes columns as lowercase labels by default. Set lowercase=False to preserve database-reflected casing.
  • explain() returns the rendered SQL string; if supported, literal binds are inlined.

Minimal example

import sqlalchemy as sa
from lazybear import scan_table, col

engine = sa.create_engine('sqlite:///:memory:')
# ... create a table `users` with columns id, name, age ...
lf = scan_table('users', engine)
df = lf.filter(col('age') >= 30).select('id', 'name').collect()
print(df)

License

This project is licensed under the MIT License. See LICENSE.txt or https://kpwhri.mit-license.org for the full text.

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

lazybear_polars-0.2.1.tar.gz (25.8 kB view details)

Uploaded Source

Built Distribution

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

lazybear_polars-0.2.1-py3-none-any.whl (23.9 kB view details)

Uploaded Python 3

File details

Details for the file lazybear_polars-0.2.1.tar.gz.

File metadata

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

File hashes

Hashes for lazybear_polars-0.2.1.tar.gz
Algorithm Hash digest
SHA256 7bcd04b7427a1a76813587d1a2c2468d03edb6d49b56cd166a3ef4398282478b
MD5 05ace4c591e75caa7247467b6cca2c3d
BLAKE2b-256 0a11345baef3d9ee7ee1fa671b08f93fc7e15f4768cd7985cbc219f48acad899

See more details on using hashes here.

Provenance

The following attestation bundles were made for lazybear_polars-0.2.1.tar.gz:

Publisher: release.yml on kpwhri/lazybear

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

File details

Details for the file lazybear_polars-0.2.1-py3-none-any.whl.

File metadata

File hashes

Hashes for lazybear_polars-0.2.1-py3-none-any.whl
Algorithm Hash digest
SHA256 b5f24019a4a4d0482a9cee98ec4e4a29448dd0ca223d979af0c02003678b2a07
MD5 69b90d9a10234ee0dd9aa84f37123d34
BLAKE2b-256 889a67bbfa0e08dcae68b4ed25d570925f811431cd402eca775f0c4e361db13a

See more details on using hashes here.

Provenance

The following attestation bundles were made for lazybear_polars-0.2.1-py3-none-any.whl:

Publisher: release.yml on kpwhri/lazybear

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