Skip to main content

Add your description here

Project description

ysaqml

ysaqml provides a YAML-backed persistence layer for SQLAlchemy. Tables live in memory inside SQLite for fast queries, while their contents are loaded from – and saved back to – .yaml files using the blazing-fast naay parser/dumper.

Why?

  • Keep relational workflows while still committing plain-text fixtures.
  • Edit data in YAML, rely on SQLite for constraints and transactions.
  • Round-trip friendly thanks to naay's strict string-only subset.

Installation

pip install -e .

This pulls in SQLAlchemy and naay. Tests rely on pytest (install with pip install -e .[test]).

Usage

Context-managed workflow

from pathlib import Path
from sqlalchemy import Column, Integer, MetaData, String, Table, select

from ysaqml import YamlSqliteEngine

metadata = MetaData()
users = Table(
		"users",
		metadata,
		Column("id", Integer, primary_key=True),
		Column("name", String, nullable=False),
)

data_dir = Path("./data")

with YamlSqliteEngine(metadata, data_dir) as backend:
		engine = backend.engine
		with engine.begin() as conn:
				conn.execute(users.insert().values(id=1, name="Ada"))
				result = conn.execute(select(users)).mappings().all()
				print(result)
# context manager flushes SQLite rows back into data/users.yaml

Each SQLAlchemy table is materialized to <data_dir>/<table_name>.yaml. When entering the context the engine wipes any existing rows in the in-memory database, rehydrates them from YAML if the file exists, and on successful exit dumps the SQLite contents back to YAML.

Custom dialect / engine helper

ysaqml also registers a SQLAlchemy dialect so you can build engines with ysaqml:/// URLs. The convenience helper wires up the required engine plugin and disposes the engine when you are done:

from contextlib import closing
from pathlib import Path
from sqlalchemy import Column, Integer, MetaData, String, Table, select

from ysaqml import create_yaml_engine

metadata = MetaData()
users = Table(
		"users",
		metadata,
		Column("id", Integer, primary_key=True),
		Column("name", String, nullable=False),
)

data_dir = Path("./data")

with closing(create_yaml_engine(metadata, data_dir)) as engine:
		with engine.begin() as conn:
				conn.execute(users.insert().values(id=1, name="Ada"))
				print(conn.execute(select(users)).mappings().all())
# exiting the context disposes the engine and writes YAML files

Under the hood this builds an in-memory SQLite engine using the ysaqml dialect and attaches an engine plugin that loads YAML files immediately and saves them back when the engine is disposed. If you manage the engine manually, make sure to call engine.dispose() to persist your changes.

File format

Every YAML document uses the following schema:

_naay_version: "1.0"
rows:
	- id: "1"
		name: "Ada"
  • _naay_version is validated but not enforced – it defaults to ysaqml.DEFAULT_NAAY_VERSION and is kept intact when re-saving.
  • rows is an ordered list of dictionaries where every value is a string.

ysaqml serializes Python None values as the sentinel string __NULL__. When loading back, that string becomes None again, so avoid storing it as a literal value.

Testing

pytest

The tests create temporary directories and exercise the full load → mutate → dump loop.

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

ysaqml-2025.12.4.post2.tar.gz (40.4 kB view details)

Uploaded Source

Built Distribution

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

ysaqml-2025.12.4.post2-py3-none-any.whl (22.3 kB view details)

Uploaded Python 3

File details

Details for the file ysaqml-2025.12.4.post2.tar.gz.

File metadata

  • Download URL: ysaqml-2025.12.4.post2.tar.gz
  • Upload date:
  • Size: 40.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for ysaqml-2025.12.4.post2.tar.gz
Algorithm Hash digest
SHA256 9c893b27c7cb7c077c2f14a28be293b49371c2cf36e1bcbc893fe9186943586b
MD5 b788504a68179e8421344d6cd2def001
BLAKE2b-256 43485a54225b14ae46eea0de93272d3724c89360cd08a4ba29d71249fd521363

See more details on using hashes here.

Provenance

The following attestation bundles were made for ysaqml-2025.12.4.post2.tar.gz:

Publisher: release.yml on rbroderi/ysaqml

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

File details

Details for the file ysaqml-2025.12.4.post2-py3-none-any.whl.

File metadata

File hashes

Hashes for ysaqml-2025.12.4.post2-py3-none-any.whl
Algorithm Hash digest
SHA256 937d5b49ddf4c1c03904fcfbd4206cb638969c89113e5cddec79063d63422590
MD5 286e096f2f85905c42804ecdb43b3f85
BLAKE2b-256 54368db100ca87372d202bbf65cc0d7fe90ab5eda783f20b8cc3b6b2865655a0

See more details on using hashes here.

Provenance

The following attestation bundles were made for ysaqml-2025.12.4.post2-py3-none-any.whl:

Publisher: release.yml on rbroderi/ysaqml

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