Skip to main content

A schema and toolkit for curating tabular datasets and benchmarking tasks (the data layer behind TabArena).

Project description

Data Foundry: a Schema and Toolkit for Curating Tabular ML Datasets


๐Ÿ“‚ Examples ๐Ÿง‘โ€๐Ÿ”ฌ Contribute a Dataset ๐Ÿ“„ Paper (placeholder โ€” coming soon)

Data Foundry is the data layer behind the next generation of TabArena datasets. It provides:

  • A small, opinionated schema for tabular datasets, tasks (IID / temporal non-IID / grouped non-IID), and outer CV splits โ€” aligned with OpenML where possible, extended where it had to be.
  • A curation toolkit (sanity checks, recommended-split helpers, dtype-preserving save/load) so a curator turns a raw download into a reproducible artifact in one notebook.
  • A collections API that pins datasets (defined by (unique_name, uuid)) to immutable curated containers and resolves them against a local warehouse or directly against the BeyondArena Datasets.

โšก Quickstart

[!TIP] Pull a real curated dataset from BeyondArena and inspect its full metadata + outer CV splits. The first call fetches from Hugging Face; subsequent calls hit your local cache.

pip install data-foundry
python examples/load_curated_container.py
from data_foundry.collections import BEYOND_ARENA

container = BEYOND_ARENA.get_dataset("airfoil_self_noise")
print(container.describe())          # full identity + dtypes + task + splits
print(container.dataset.shape)       # the actual DataFrame
print(container.task_metadata.split_regime)  # "iid", "temporal_non_iid", or "grouped_non_iid"

That's the whole API surface in three lines. See examples/benchmark_on_beyond_arena.py for benchmarking Random Forest on the data!

๐Ÿ•น๏ธ Use Cases

๐Ÿงช Inspect a curated container offline โ€” no Hugging Face download required

The package ships a toy CuratedContainer so you can poke at the full API โ€” schema, dtypes, splits, describe() โ€” without touching the network. Identical interface to a downloaded BeyondArena container.

from data_foundry.curation_container import CuratedContainer
from data_foundry.examples import get_toy_container_path

container = CuratedContainer.load(get_toy_container_path())
print(container.describe())          # full identity + dtypes + task + splits
print(container.dataset.shape)       # the actual DataFrame
print(container.task_metadata.split_regime)  # "iid", "temporal_non_iid", or "grouped_non_iid"

Full inspection script (every metadata field printed): examples/load_curated_container.py.

๐Ÿ“ฆ Use one dataset โ€” IID and non-IID variants

Download a single BeyondArena container by name (or UUID) and iterate its outer CV splits. The collection resolves the container against your local cache; subsequent runs hit disk, not the network.

from data_foundry.collections import BEYOND_ARENA

container = BEYOND_ARENA.get_dataset("airfoil_self_noise")
df = container.dataset
target = container.task_metadata.target_column_name

for repeat_id, folds in container.experiment_metadata.splits.items():
    for fold_id, (train_idx, test_idx) in folds.items():
        X_train, y_train = df.iloc[train_idx].drop(columns=target), df.iloc[train_idx][target]
        X_test,  y_test  = df.iloc[test_idx].drop(columns=target),  df.iloc[test_idx][target]
        # ... fit, evaluate ...

Full worked example (Random Forest, RMSE per fold, full metadata via container.describe()): examples/benchmark_on_beyond_arena.py.

Split regimes. BeyondArena ships datasets from three regimes โ€” which one a dataset is in shows up directly on task_metadata:

Regime Set on PredictiveMLTaskMetadata Meaning
IID neither time_on nor group_on rows are independent; random / stratified splits
temporal non-IID time_on set rows ordered in time; future rows must not leak backwards
grouped non-IID group_on set (+ group_labels) all rows of a group stay together in one fold

Side-by-side regime printout (one IID, two grouped variants โ€” per_group vs per_sample โ€” and one temporal): examples/data_foundry_data_regimes.py.

๐Ÿ—‚๏ธ Use a collection of datasets โ€” pre-download all of BeyondArena

BEYOND_ARENA.prefetch(...) batches every container into a single Hugging Face snapshot_download call (one network round-trip for the whole collection). On a warm cache it skips importing huggingface_hub entirely.

from data_foundry.collections import BEYOND_ARENA

paths = BEYOND_ARENA.prefetch()          # warms the cache once
for container in BEYOND_ARENA.iter_containers():  # now hits disk only
    print(container.dataset_metadata.unique_name, container.dataset.shape)

Cache management:

BEYOND_ARENA.clear_cache()                 # nuke this collection's subdir
BEYOND_ARENA.get_dataset(name, force_download=True)  # re-fetch a single container

Full worked example with tqdm progress + checksum verification: examples/download_all_beyond_arena_datasets.py. For a single dataset round-trip with checksum verification, see examples/download_beyond_arena_dataset.py.

๐Ÿง‘โ€๐Ÿ”ฌ Curate a dataset โ€” turn a raw download into a CuratedContainer

End-to-end pipeline, condensed (the full runnable version is examples/curate_a_dataset.py):

from data_foundry.schema import DatasetMetadata, PredictiveMLTaskMetadata

# --- Basic metadata
dataset_mold = DatasetMetadata(
    unique_name="blood_transfusion",
    dataset_year="2008",
    domain_str="medical & healthcare",
    dataset_source="UCI",
    original_dataset_source_download_link="https://doi.org/10.24432/C5GS39",
    download_description="""
We download the data from the UCI repository and unzip it to a predefined folder.

mkdir -p local-data-warehouse/blood_transfusion/ \\
  && wget -P local-data-warehouse/blood_transfusion/ \\
       https://archive.ics.uci.edu/static/public/176/blood+transfusion+service+center.zip \\
  && unzip local-data-warehouse/blood_transfusion/blood+transfusion+service+center.zip \\
       -d local-data-warehouse/blood_transfusion/
""",
    academic_reference_bibtex="""@article{yeh2009knowledge,
  title={Knowledge discovery on RFM model using Bernoulli sequence},
  author={Yeh, I-Cheng and Yang, King-Jang and Ting, Tao-Ming},
  journal={Expert Systems with applications},
  volume={36}, number={3}, pages={5866--5871},
  year={2009}, publisher={Elsevier},
}
""",
    academic_reference_bibtex_key="yeh2009knowledge",
    license="CC BY 4.0",
    data_tags=["IID"],
    curation_comments="Renamed features for clarity; mapped target 0/1 โ†’ No/Yes; ~29% duplicate rows kept.",
)
task_mold = PredictiveMLTaskMetadata(
    target_column_name="DonatedBloodInMarch2007",
    problem_type="binary_classification",
    objective_metric_name="roc_auc",
    stratify_on="DonatedBloodInMarch2007",
)

# --- Preprocessing
import pandas as pd
df = pd.read_csv(f"{dataset_mold.path}/transfusion.data")
df.columns = [
    "MonthsSinceLastDonation", "NumberOfDonations", "TotalBloodDonated",
    "MonthsSinceFirstDonation", "DonatedBloodInMarch2007",
]
df["DonatedBloodInMarch2007"] = df["DonatedBloodInMarch2007"].map({1: "Yes", 0: "No"})
df["DonatedBloodInMarch2007"] = df["DonatedBloodInMarch2007"].astype("category")
df = df.sample(frac=1, random_state=42).reset_index(drop=True)

# --- Sanity checks
from data_foundry import dataset_checks
df_head, summary, numeric_stats, cat_stats, target_df = dataset_checks.run_all_checks(
    data=df,
    target_feature=task_mold.target_column_name,
    problem_type=task_mold.problem_type,
)

# --- Outer CV splits
from data_foundry.curation_recommendations import (
    get_recommended_iid_splits,
    get_recommended_splits_dimensions,
)

n_repeats, n_splits, test_size = get_recommended_splits_dimensions(dataset=df)
splits = get_recommended_iid_splits(
    dataset=df,
    n_repeats=n_repeats,
    n_splits=n_splits,
    test_size=test_size,
    stratify_on=task_mold.stratify_on,
)

# --- Split metadata + container
from data_foundry.schema import PredictiveMLSplitsMetadata
from data_foundry.curation_container import CuratedContainer

splits_mold = PredictiveMLSplitsMetadata(
    splits_comment="Default splits for IID data.",
    splits=splits,
)
curated_data = CuratedContainer(
    dataset=df,
    dataset_metadata=dataset_mold,
    task_metadata=task_mold,
    experiment_metadata=splits_mold,
)
curated_data.save()
print(curated_data.uuid, curated_data.checksum)

For the contributor flow (where to put the notebook, how to open the PR, the /new-dataset Claude Code skill, best practices around versioning, anomaly tracking, and dtype handling), see CONTRIBUTING_DATASETS.md.

๐Ÿช„ Installation

[!IMPORTANT] Requires Python 3.10+.

๐Ÿ“ฆ From PyPI โ€” use Data Foundry as a library
pip install data-foundry
๐ŸŒฑ From source โ€” clone and install editable
git clone https://github.com/TabArena/data-foundry.git
cd data-foundry
uv pip install -e .
๐Ÿ› ๏ธ Developer setup โ€” extras for curation, tests, and tooling
git clone https://github.com/TabArena/data-foundry.git
cd data-foundry
uv pip install -e ".[dev,tests]"
pytest                                 # run the test suite
ruff check . && ruff format --check .  # lint + format

The dev extra adds curation-time deps (openml, kaggle, seaborn, polars, etc.); tests adds pytest and scikit-learn (needed for the recommended-split helpers and examples).

๐Ÿ—‚๏ธ Repository Structure

data-foundry/
โ”œโ”€โ”€ src/data_foundry/         # the package โ€” schema, container, collections, checks, splits
โ”‚   โ”œโ”€โ”€ schema.py             # DatasetMetadata, PredictiveMLTaskMetadata, PredictiveMLSplitsMetadata
โ”‚   โ”œโ”€โ”€ curation_container.py # CuratedContainer (save/load + describe + checksum)
โ”‚   โ”œโ”€โ”€ collections/          # BEYOND_ARENA, DatasetCollection, HuggingFaceSource, cache helpers
โ”‚   โ”œโ”€โ”€ curation_recommendations.py  # recommended split helpers (IID, grouped, temporal)
โ”‚   โ”œโ”€โ”€ dataset_checks.py     # run_all_checks(...) โ€” sanity stats for the curation notebook
โ”‚   โ””โ”€โ”€ examples/toy_container/  # tiny ready-to-load CuratedContainer shipped in-package
โ”œโ”€โ”€ datasets/                 # curation notebooks
โ”‚   โ”œโ”€โ”€ _template/            # canonical notebook skeleton
โ”‚   โ”œโ”€โ”€ _dev/                 # contributions land here first
โ”‚   โ”œโ”€โ”€ _maintenance/         # re-runs / fixes for already-released datasets
โ”‚   โ””โ”€โ”€ beyond_iid/           # promoted datasets โ€” pinned by `final_uuid_list.py`
โ”œโ”€โ”€ examples/                 # runnable demos (covers the use-cases above)
โ”œโ”€โ”€ scripts/                  # one-off tooling (toy container builder)
โ”‚   โ””โ”€โ”€ beyond_arena/         # BeyondArena-specific scripts and outputs (warehouse stats, plots)
โ”œโ”€โ”€ tests/                    # pytest test suite
โ””โ”€โ”€ local-data-warehouse/     # gitignored โ€” curators write raw + saved containers here

๐Ÿง‘โ€๐Ÿ”ฌ Contributing a Dataset

The short version:

  1. Copy datasets/_template/_template.ipynb to datasets/_dev/<topic>/<unique_name>/<unique_name>.ipynb.
  2. Run the notebook end-to-end so the saved cells contain populated check tables and the final uuid / checksum.
  3. Open a PR โ€” reviewers will move the notebook into the right beyond_iid/ subfolder and append the UUID to datasets/beyond_iid/final_uuid_list.py.

The long version (field-by-field walkthrough, split-helper choice, dtype gotchas, the /new-dataset Claude Code scaffolding skill): see CONTRIBUTING_DATASETS.md.

๐Ÿ“„ Citation

PLACEHOLDER

PLACEHOLDER

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

data_foundry-0.0.4.tar.gz (47.6 kB view details)

Uploaded Source

Built Distribution

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

data_foundry-0.0.4-py3-none-any.whl (53.3 kB view details)

Uploaded Python 3

File details

Details for the file data_foundry-0.0.4.tar.gz.

File metadata

  • Download URL: data_foundry-0.0.4.tar.gz
  • Upload date:
  • Size: 47.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.22 {"installer":{"name":"uv","version":"0.11.22","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for data_foundry-0.0.4.tar.gz
Algorithm Hash digest
SHA256 70d2e8eab18d273c828d5b385b517f1d1aaef658d1644e4b08bcdaaccddb4fcf
MD5 6f11b23356758285ab1836f26a77e20d
BLAKE2b-256 076c562b39e6ae749816a4dd6e8eb63c2b7a18b37d8364b7787402af80c552d9

See more details on using hashes here.

File details

Details for the file data_foundry-0.0.4-py3-none-any.whl.

File metadata

  • Download URL: data_foundry-0.0.4-py3-none-any.whl
  • Upload date:
  • Size: 53.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.22 {"installer":{"name":"uv","version":"0.11.22","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for data_foundry-0.0.4-py3-none-any.whl
Algorithm Hash digest
SHA256 c2fd000c13db9ab260778b10fe9060844b4b51144f330b97db71a56da2b99765
MD5 ad7459b5f1229dfb3e2462f6fb8e4278
BLAKE2b-256 10964e5983026ad285e06751ae08471db16ce493ab8551bd2d157d0c8552b46f

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