Skip to main content

Slayer of Hydra

Project description

Heracls - Slayer of Hydra

heracls is a tiny utility package to instantiate typed dataclasses from flexible config sources (dictionary, OmegaConf, YAML, dotlist, CLI, ...). It is designed for projects that want strict, typed config objects while supporting the dynamic overrides commonly used in scripts and experiments.

Installation

The heracls package is available on PyPi and can be installed with pip.

pip install heracls

Alternatively, if you need the latest features, you can install it from source.

pip install git+https://github.com/francois-rozet/heracls

Getting started

The following example demonstrates how to declare a nested dataclass config, instantiate it from command line arguments, serialize it to YAML, and use it in a script. The heracls.ArgumentParser parser infers arguments from the dataclass structure and types.

from dataclasses import dataclass, field
from typing import Literal

import heracls

@dataclass
class ModelConfig:
    name: str = "mlp"
    depth: int = 3
    width: int = 256
    dropout: float | None = None

@dataclass
class AdamConfig:
    name: Literal["adam"] = "adam"
    betas: tuple[float, float] = (0.95, 0.95)
    learning_rate: float = 1e-3
    weight_decay: float = 0.0

@dataclass
class SGDConfig:
    name: Literal["sgd"] = "sgd"
    momentum: float = 0.0
    learning_rate: float = 1e-3
    weight_decay: float = 0.0
    nesterov: bool = False

@dataclass
class TrainConfig:
    output: str
    model: ModelConfig = field(default_factory=ModelConfig)
    optimizer: AdamConfig | SGDConfig = heracls.field(
        choices={"adam": AdamConfig, "sgd": SGDConfig},
        default_factory=AdamConfig,
    )
    dataset: str = "mnist"
    data_splits: tuple[float, ...] = (0.8, 0.1)
    n_epochs: int = 1024
    n_steps_per_epoch: int = 256
    tasks: list[str] = field(default_factory=list)

def main() -> None:
    parser = heracls.ArgumentParser()
    parser.add_argument("--dry", action="store_true", help="dry run")
    parser.add_arguments(TrainConfig, dest="train", root=True)

    args = parser.parse_args()

    print(heracls.to_yaml(args.train))

    if args.dry:
        return

    trainset, validset, testset = load_dataset(args.train.data_splits)
    model = init_model(args.train.model)

    for epoch in range(args.train.n_epochs):
        ...

if __name__ == "__main__":
    main()
$ python examples/train.py --dry --output ./out --model.dropout 0.1 --optimizer sgd --data_splits '[0.7, 0.2]'
output: ./out
model:
  name: mlp
  depth: 3
  width: 256
  dropout: 0.1
optimizer:
  name: sgd
  momentum: 0.0
  learning_rate: 0.001
  weight_decay: 0.0
  nesterov: false
dataset: mnist
data_splits:
- 0.7
- 0.2
n_epochs: 1024
n_steps_per_epoch: 256
tasks: []
$ python examples/train.py --help
usage: train.py [-h] [--dry] --output str [--model.name str] [--model.depth int]
                [--model.width int] [--model.dropout float] [--optimizer {adam,sgd}]
                [--dataset str] [--data_splits tuple[float, ...]] [--n_epochs int]
                [--n_steps_per_epoch int] [--tasks list[str]] [--optimizer.name {adam}]
                [--optimizer.betas tuple[float, float]] [--optimizer.learning_rate float]
                [--optimizer.weight_decay float]

options:
  -h, --help                             show this help message and exit
  --dry                                  dry run (default: False)

  --output str
  --optimizer {adam,sgd}                 (default: adam)
  --dataset str                          (default: mnist)
  --data_splits tuple[float, ...]        (default: (0.8, 0.1))
  --n_epochs int                         (default: 1024)
  --n_steps_per_epoch int                (default: 256)
  --tasks list[str]                      (default: [])

model:
  --model.name str                       (default: mlp)
  --model.depth int                      (default: 3)
  --model.width int                      (default: 256)
  --model.dropout float                  (default: None)

optimizer:
  --optimizer.name {adam}                (default: adam)
  --optimizer.betas tuple[float, float]  (default: (0.95, 0.95))
  --optimizer.learning_rate float        (default: 0.001)
  --optimizer.weight_decay float         (default: 0.0)

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

heracls-0.6.0.tar.gz (9.0 kB view details)

Uploaded Source

Built Distribution

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

heracls-0.6.0-py3-none-any.whl (8.7 kB view details)

Uploaded Python 3

File details

Details for the file heracls-0.6.0.tar.gz.

File metadata

  • Download URL: heracls-0.6.0.tar.gz
  • Upload date:
  • Size: 9.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.22 {"installer":{"name":"uv","version":"0.9.22","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for heracls-0.6.0.tar.gz
Algorithm Hash digest
SHA256 1e2629b735b51978b17e02f9a179bfb1500d3cb0d57dc1ebf3731665bcbe4f22
MD5 f9826fb399244f6068ecef93368879a4
BLAKE2b-256 4568a34ef186a1f95d1c4c35183ec7334a78a082c7caab5291d87615a4b01bcf

See more details on using hashes here.

File details

Details for the file heracls-0.6.0-py3-none-any.whl.

File metadata

  • Download URL: heracls-0.6.0-py3-none-any.whl
  • Upload date:
  • Size: 8.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.22 {"installer":{"name":"uv","version":"0.9.22","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for heracls-0.6.0-py3-none-any.whl
Algorithm Hash digest
SHA256 283b76436888af40609d22fb472ff07bc9c325a80c2f7858609939bc0423de46
MD5 1d1bcd7b6f867de48fcff4c2c34203b6
BLAKE2b-256 f331d9920b4a47324fc2ed21ee4b64488ef36e12b06524a6173948652f5a92b3

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