Unified tabular + vector storage in a single Iceberg-compatible file
Project description
ailake — AI-Lake Format Python SDK
Unified storage for tabular data, embeddings, and HNSW vector index in a single Parquet-compatible file. 100% Apache Iceberg Spec v2 compatible.
Install
pip install ailake
Requires Python ≥ 3.9. Dependencies: pyarrow >= 14.0, numpy >= 1.24.
Quickstart
Write + search — fluent API (recommended)
import ailake
import numpy as np
# Open or create a table
table = ailake.open_table(
"./my_table",
dim=1536,
metric="cosine", # cosine | euclidean | dot_product | normalized_cosine
pre_normalize=True, # normalize at write time; enables fast 1-dot(a,b) path
hnsw_m=16, # HNSW connections per node (default 16)
hnsw_ef_construction=150,
embedding_model="text-embedding-3-small", # tracked in Iceberg metadata
embedding_model_version="v1",
)
texts = ["Document about AI", "Another document"]
embeddings = np.random.rand(2, 1536).astype(np.float32)
table.insert(texts, embeddings) # accepts list or numpy array
snapshot_id = table.commit()
# Pattern B — auto-embed without passing embeddings explicitly
def my_embed(texts: list[str]) -> list[list[float]]:
return np.random.rand(len(texts), 1536).tolist() # replace with real model
table2 = ailake.open_table("./my_table2", dim=1536, embed_fn=my_embed)
table2.insert(["Document about AI", "Another document"]) # embed_fn called automatically
table2.commit()
# Pointer-only search (default — backward-compatible)
df = table.search(embeddings[0], top_k=10).to_pandas() # row_id, distance, file
lf = table.search(embeddings[0]).limit(5).to_polars()
results = table.search(embeddings[0]).to_list() # list[dict]
# Full row data — all Parquet columns + _distance
df_full = table.search(embeddings[0], top_k=10, fetch_data=True).to_pandas()
Async API
import ailake, asyncio
import numpy as np
async def main():
table = ailake.open_table("./my_table", dim=1536)
await table.insert_async(texts, embeddings)
await table.commit_async()
# fluent async chain
df = await table.search(query_vec).limit(10).to_pandas_async()
# parallel searches via asyncio.gather
r1, r2 = await asyncio.gather(
table.search(q1).to_list_async(),
table.search(q2).to_list_async(),
)
asyncio.run(main())
Module-level search
import ailake
import numpy as np
query = np.random.rand(1536).astype(np.float32)
df = ailake.search("./my_table", query, top_k=10).to_pandas()
lf = ailake.search("./my_table", query).limit(5).to_polars()
items = ailake.search("./my_table", query).to_list()
Assemble context for LLMs
import ailake
chunks = [
{
"document_id": "doc-1",
"chunk_index": 0,
"chunk_text": "AI-Lake stores vectors and tabular data together.",
"document_title": "AI-Lake Overview",
"section_path": "Introduction",
"source_uri": "s3://my-lake/docs/overview.pdf",
"distance": 0.12,
},
]
context_xml = ailake.assemble_context(
chunks=chunks,
max_tokens=4096, # token budget (4 chars ≈ 1 token)
dedup_threshold=0.05, # drop near-duplicate chunks
)
# Pass context_xml directly to Claude / GPT-4 as a user message
API reference
open_table(path, *, ...) → Table
Opens or creates an AI-Lake table at path.
| Parameter | Default | Description |
|---|---|---|
path |
required | Table root (local, s3://, gs://, az://) |
vector_column |
"embedding" |
Vector column name |
dim |
1536 |
Embedding dimension |
metric |
"cosine" |
cosine, euclidean, dot_product, normalized_cosine |
pre_normalize |
False |
Normalize to unit L2 at write; enables 1-dot(a,b) fast path (~12-20 % speedup) |
hnsw_m |
None (=16) |
HNSW connections per node |
hnsw_ef_construction |
None (=150) |
HNSW build pool size |
pq_only |
False |
Discard raw F16 vectors after index build — only PQ codes stored. ~98 % storage reduction; reranking disabled; recall@10 ~93-95 %. |
ivf_residual |
False |
Encode vec − cluster_centroid per IVF cell (residual PQ). Same storage as standard PQ; ~2-4 pp better recall@10. |
embedding_model |
None |
Embedding model name stored in Iceberg properties (ailake.embedding-model). Used for mismatch detection and migration tracking. |
embedding_model_version |
None |
Optional model version. Stored as "<name>@<version>" in Iceberg properties. |
embed_fn |
None |
Auto-embed callable list[str] → list[list[float]]. When set, insert(texts) and write_batch(texts) can be called without passing embeddings — the callable is invoked automatically. |
Table
| Method | Description |
|---|---|
insert(texts, embeddings=None) → Table |
Buffer a batch. embeddings: list[list[float]] or numpy array. When embed_fn was set on open_table(), embeddings may be omitted — the callable is invoked automatically. |
write_batch_auto_deferred(texts, embeddings=None) → Table |
Deferred write — Parquet persisted immediately (~200k vec/s); index (HNSW or IVF-PQ, auto-selected) built in a background thread. Shard served via flat scan until index ready. |
commit() → int |
Persist as a new Iceberg snapshot; returns snapshot ID. |
search(query, top_k=10, fetch_data=False) → SearchQuery |
Lazy, chainable search. query: list[float] or numpy array. Set fetch_data=True to return full row data. Raises ModelMismatch if query dim ≠ table dim. |
insert_async(...) |
Async variant of insert. |
write_batch_auto_deferred_async(...) |
Async variant of write_batch_auto_deferred. |
commit_async() → int |
Async variant of commit. |
Table is a context manager: with ailake.open_table(...) as t: ...
In Jupyter, table renders a styled HTML card showing path and vector config.
SearchQuery
Lazy result set — no I/O until materialised.
| Method | Description |
|---|---|
limit(n) → SearchQuery |
Cap to n nearest neighbours (chainable). |
to_list() → list[dict] |
Always pointer-only: [{"row_id": int, "distance": float, "file": str}, ...] |
to_arrow() → pyarrow.Table |
Full row data (all columns + _distance) when fetch_data=True; pointer-only pyarrow.Table with columns row_id, distance, file otherwise. |
to_pandas() → pd.DataFrame |
Full row DataFrame when fetch_data=True; pointer-only otherwise. |
to_polars() → pl.DataFrame |
Full row DataFrame when fetch_data=True; pointer-only otherwise. |
to_list_async() |
Async variant. |
to_arrow_async() |
Async variant. |
to_pandas_async() |
Async variant. |
to_polars_async() |
Async variant. |
In Jupyter, results renders as an HTML table when executed, pending state otherwise.
When fetch_data=True, the HTML table shows all Parquet columns.
Full-read mode
# Pointer-only (default — backward-compatible)
df = ailake.search("./my_table", query, top_k=10).to_pandas()
# columns: row_id, distance, file
# Full row data — all Parquet columns + _distance
df = ailake.search("./my_table", query, top_k=10, fetch_data=True).to_pandas()
# columns: text, embedding, ..., _distance
# Same via Table handle
df = table.search(query, top_k=10, fetch_data=True).to_pandas()
fetch_data=True reads each matching Parquet file once and uses arrow_select::take to extract only the matched rows — no full table scan.
search(path, query, top_k=10, fetch_data=False) → SearchQuery
Module-level search returning the same chainable SearchQuery.
VectorColSpec(column, dim, metric="cosine", modality=None)
Declares one vector column for multi-column writes or searches.
| Arg | Description | Example |
|---|---|---|
column |
Parquet column name | "image_embedding" |
dim |
Embedding dimension | 512 |
metric |
Distance metric | "cosine" |
modality |
Optional tag — stored as ailake.modality-<column> |
"text" / "image" / "audio" / "video" |
TableWriter.write_batch_multi(texts, columns)
Write a batch with N independent vector columns in one call. Each column gets its own HNSW index in the AILK section of the file footer.
from ailake import TableWriter, VectorColSpec
text_spec = VectorColSpec("embedding", 1536, "cosine", "text")
image_spec = VectorColSpec("image_embedding", 512, "cosine", "image")
writer = TableWriter("s3://my-lake/media/", dim=1536, metric="cosine")
writer.write_batch_multi(
texts,
[(text_spec, text_embeddings), (image_spec, image_embeddings)],
)
snapshot_id = writer.commit()
search_multimodal(path, queries, top_k=10) → list[dict]
Cross-modal search: fuse results from N vector columns via Reciprocal Rank Fusion.
rrf_score = Σ weight_i / (60 + rank_i) — higher is better.
results = ailake.search_multimodal(
"s3://my-lake/media/",
queries=[
("embedding", text_vec, 0.7), # 70% weight on text similarity
("image_embedding", image_vec, 0.3), # 30% weight on image similarity
],
top_k=20,
)
# Returns: [{"row_id": int, "rrf_score": float, "file": str}, ...]
# Ordered by descending rrf_score
Each column is searched by its own HNSW. Per-column dimensions are auto-detected
from ailake.dim-<col> Iceberg properties written at commit() time — no dim
argument needed when reading tables written with write_batch_multi.
migrate_embeddings(path, old_column, new_column, embed_fn, *, ...)
Re-embeds all chunks in a table with a new model, committing the result as a new Iceberg snapshot.
ailake.migrate_embeddings(
path = "s3://my-lake/docs/",
old_column = "embedding", # existing vector column
new_column = "embedding_v2", # destination column (may be same name)
embed_fn = my_embed_fn, # callable: list[str] → list[list[float]]
text_column = "chunk_text", # source text column
strategy = "dual_write_then_cutover", # or "atomic_replace"
batch_size = 512,
new_model = "text-embedding-3-large",
new_model_version = "v1",
on_progress = lambda *, files_done, files_total, rows_migrated: print(
f"{files_done}/{files_total} files, {rows_migrated} rows"
),
)
| Parameter | Default | Description |
|---|---|---|
path |
required | Table root URI |
old_column |
required | Existing vector column to migrate from |
new_column |
required | Destination vector column |
embed_fn |
required | list[str] → list[list[float]] callable |
text_column |
"chunk_text" |
Parquet column containing the source text |
strategy |
"dual_write_then_cutover" |
"dual_write_then_cutover" (zero downtime, 2× peak storage) or "atomic_replace" (lower storage, brief mixed-model window) |
batch_size |
512 |
Rows passed to embed_fn per call |
new_model |
None |
Model name written to ailake.embedding-model after migration |
new_model_version |
None |
Optional version suffix |
on_progress |
None |
Callable invoked after each file with keyword args files_done, files_total, rows_migrated |
TableWriter (low-level — use open_table() for most cases)
# Standard HNSW write with model tracking
writer = ailake.TableWriter(
path, dim=1536, metric="cosine",
embedding_model="text-embedding-3-small",
embedding_model_version="v1",
)
writer.write_batch(texts, embeddings)
snapshot_id = writer.commit()
# Pattern B — auto-embed: omit embeddings, SDK calls embed_fn
writer = ailake.TableWriter(
path, dim=1536,
embed_fn=lambda texts: my_model.encode(texts).tolist(),
)
writer.write_batch(texts) # no embeddings arg needed
writer.commit()
# PQ-only — raw vectors discarded after index build (~98 % storage reduction)
writer = ailake.TableWriter(path, dim=1536, metric="cosine", pq_only=True)
writer.write_batch(texts, embeddings)
writer.commit()
# Residual PQ — per-cluster encoding for better recall
writer = ailake.TableWriter(path, dim=1536, metric="cosine", ivf_residual=True)
writer.write_batch(texts, embeddings)
writer.commit()
# Deferred write — Parquet immediate, index background (~200k vec/s)
writer = ailake.TableWriter(path, dim=1536, metric="cosine")
writer.write_batch_auto_deferred(texts, embeddings)
writer.commit()
TableWriter parameters: same as open_table() (includes pq_only, ivf_residual, pre_normalize, hnsw_m, hnsw_ef_construction, embedding_model, embedding_model_version, embed_fn).
assemble_context(chunks, max_tokens=4096, dedup_threshold=0.05) → str
Assembles chunk dicts into structured XML for LLM input. Deduplicates near-identical chunks within the token budget.
Storage modes and index types
| Mode | pq_only |
ivf_residual |
Storage (dim=1536, 1M rows) | Reranking | Recall@10 |
|---|---|---|---|---|---|
| HNSW + F16 raw (default) | False |
False |
~300 GB vectors + ~30 GB HNSW | Yes (exact) | ~97 % |
| IVF-PQ + F16 raw | False |
False |
~300 GB + ~5 GB PQ codes | Yes (exact) | ~93 % inline |
| IVF-PQ residual + raw | False |
True |
~300 GB + ~5 GB | Yes (exact) | ~96 % |
| PQ-only | True |
False |
~5 GB total | No | ~93-95 % |
| PQ-only residual | True |
True |
~5 GB total | No | ~94-96 % |
# Deferred write — all modes, instant Parquet commit, index in background
writer = ailake.TableWriter(path, dim=1536, pq_only=True, ivf_residual=True)
writer.write_batch_auto_deferred(texts, embeddings)
writer.commit()
HNSW tuning guide
| Goal | hnsw_m |
hnsw_ef_construction |
|---|---|---|
| Low latency / high QPS | 8 | 100 |
| General purpose (default) | 16 | 150 |
| High recall (RAG) | 24 | 200 |
| Max recall (medical, legal) | 32 | 400 |
Type checking
Ships py.typed (PEP 561) and ailake/_ailake.pyi stubs. mypy and pyright work out of the box with no configuration.
Iceberg compatibility
Tables are valid Apache Iceberg Spec v2. Spark, Trino, DuckDB, and PyIceberg read tabular columns normally; the HNSW index lives in an extension section that standard Parquet readers silently ignore.
License
MIT OR Apache-2.0
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distributions
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file ailake-0.0.19.tar.gz.
File metadata
- Download URL: ailake-0.0.19.tar.gz
- Upload date:
- Size: 187.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.9.25
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
17ea957604050edd7e8ddcb4cf5b20fbfa0208180e6a379ca4195c0a7cedb722
|
|
| MD5 |
004593e63b183d43bf613667945f4e68
|
|
| BLAKE2b-256 |
b3ce32ea9d4a954dc6906d00c4064697847e2186200ff29aa039b3b7ea793c6e
|
File details
Details for the file ailake-0.0.19-cp39-abi3-win_amd64.whl.
File metadata
- Download URL: ailake-0.0.19-cp39-abi3-win_amd64.whl
- Upload date:
- Size: 3.4 MB
- Tags: CPython 3.9+, Windows x86-64
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.9.25
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
dba54d401d512a2e3094234b69948a26b2d85e2c3a5348f21fe9519cecdc8768
|
|
| MD5 |
cae4ff9380d753dfde7bd36eb7b34026
|
|
| BLAKE2b-256 |
9fda00db0f5d51f8420d9f9afb534d65abe044328e8ca341b0d7e8f869e39377
|
File details
Details for the file ailake-0.0.19-cp39-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.
File metadata
- Download URL: ailake-0.0.19-cp39-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
- Upload date:
- Size: 3.9 MB
- Tags: CPython 3.9+, manylinux: glibc 2.17+ x86-64
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.9.25
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5be1c440566a2930e3db4065877b1208ece352b1231d860ad3f58fd4ab7755a2
|
|
| MD5 |
dd86cda648535d69e9eb4b46a2cabd0a
|
|
| BLAKE2b-256 |
d2a3dc387f1afb5e154f2872e773c79790be66ba09af5437cf94f6bf1844bf8e
|
File details
Details for the file ailake-0.0.19-cp39-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl.
File metadata
- Download URL: ailake-0.0.19-cp39-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl
- Upload date:
- Size: 3.6 MB
- Tags: CPython 3.9+, manylinux: glibc 2.17+ ARM64
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.9.25
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
bb662ae60c76975fcb2a2bf73cf7779f6f6bb94ded3c0c84912df5c5c6976a68
|
|
| MD5 |
54f99a1851db6e9de8b10b5fdfb960e3
|
|
| BLAKE2b-256 |
f656b3bcab9d61e0296ba652f0224eabcdd2e751721e30002e4d409e753264e8
|