Skip to main content

Dataclass driven argparse handling

Project description

ddargparse - dataclass driven argparse

test coverage: 100% GitHub Workflow Status PyPI

A small Python library that simplifies command-line argument parsing by leveraging dataclasses. It allows developers to define their command-line interfaces using dataclass fields, making the code more concise and easier to maintain. With ddargparse, you can easily create complex command-line applications with minimal boilerplate code. It puts a particular focus on properly handling type annotations and defaults, minimizing additional required annotation and maintaining interoperability with the standard argparse.

Installation

pip install ddargparse

Usage

Mode a: argparse + dataclasses

Subclass OptionsBase and annotate fields with standard argparse metadata via the dataclasses.field function (help, required, positional, metavar). Then call register_cli_args to populate an ArgumentParser and from_cli_args to instantiate your options from the parsed result.

Step 1: Option dataclass definition

from argparse import ArgumentParser
from dataclasses import dataclass, field
import ddargparse

@dataclass
class Options(ddargparse.OptionsBase):
    inputfile: str = field(
        metadata={"help": "Input file", "positional": True, "metavar": "FILE"},
    )
    verbose: bool = field(
        metadata={"help": "Enable verbose output"},
    )
    tags: list[str] = field(
        default_factory=list,
        metadata={"help": "One or more tags"},
    )

@dataclass
class DoSomethingOptions(ddargparse.OptionsBase):
    threshold: str = field(metadata={"help": "Some threshold"})
    mode: SomeMode = field(default=SomeMode.DEFAULT, metadata={"help": "Mode to be used"})

Step 2: CLI argument parser declaration

parser = ArgumentParser()
# register dataclass options as argparse parser arguments
Options.register_cli_args(parser)
subparsers = parser.add_subparsers(dest="subcommand")
subparser = subparsers.add_parser("do-something")
# register dataclass options as argparse subparser arguments
DoSomethingOptions.register_cli_args(subparser)

Step 3: CLI argument parsing and option dataclass instantiation

args = parser.parse_args()
# obtain instance of dataclass with global options
options = Options.from_cli_args(args)
match args.subcommand:
    case "do-something":
        # obtain instance of dataclass with subcommand options
        do_something_options = DoSomethingOptions.from_cli_args(args)

Mode b: dataclass only

In this mode, control over argparse is handed over to ddargparse entirely and happening under the hood. The advantage is that subcommands can be expressed implicitly in the dataclass hierarchy.

Step 1: Option and subcommand dataclass definition

from argparse import ArgumentParser
from dataclasses import dataclass, field
import ddargparse

@dataclass
class Options(ddargparse.OptionsBase):
    "The docstring is used as description in the CLI interface"

    inputfile: str = field(
        metadata={"help": "Input file", "positional": True, "metavar": "FILE"},
    )
    verbose: bool = field(
        metadata={"help": "Enable verbose output"},
    )
    tags: list[str] = field(
        default_factory=list,
        metadata={"help": "One or more tags"},
    )
    subcommand_do_something: DoSomethingOptions | None


class SomeMode(Enum):
    DEFAULT = 0
    FAST = 1


@dataclass
class DoSomethingOptions(ddargparse.OptionsBase):
    """This is the subcommand description
    
    The first line of the subcommand is used as short help, while the full description
    is displayed upon "mytool do-something --help"
    """

    threshold: str = field(metadata={"help": "Some threshold"})
    mode: SomeMode = field(default=SomeMode.DEFAULT, metadata={"help": "Mode to be used"})

Step 2: CLI argument parsing and option dataclass instantiation

Then, the entire hierarchy of options, including automatic determination of the selected subcommand and its options can be obtained via calling parse_args() on the top-level Options class.

options = Options.parse_args()

Features

  • Declare CLI arguments as typed dataclass fields — no repetitive add_argument calls, no need for the type argument. Automatic convertion of field names into kebab-case CLI arguments.
  • Booleans (store_true / store_false), and list arguments (nargs="+") are inferred automatically from the dataclass field definitions.
  • Custom parse methods: define a parse_<field_name> classmethod to override the argument type converter.
  • Mark options as positional ("positional": True).
  • Automatic and natural inference whether option is required (no field(default=...) and no | None in type annotation).
  • Choose between append-style (--arg item1 --arg item2 --arg item3) and nargs-style (default, --arg item1 item2 item3) list arguments via register_cli_args(..., list_append=True|False) or parse_args(list_append=True|False).
  • Proper support for enums: simply specify an enum type and ddargparse handles ensures that proper (lower, kebab-cased) choices are inferred from the enum item names and the correct type is returned.
  • Seamless integration with standard argparse API or dataclass only modes that hides all technical details of the argument parsing.
  • No additional dependencies.

Requirements

  • Python ≥ 3.11

License

See LICENSE.

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

ddargparse-0.3.2.tar.gz (10.3 kB view details)

Uploaded Source

Built Distribution

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

ddargparse-0.3.2-py3-none-any.whl (8.1 kB view details)

Uploaded Python 3

File details

Details for the file ddargparse-0.3.2.tar.gz.

File metadata

  • Download URL: ddargparse-0.3.2.tar.gz
  • Upload date:
  • Size: 10.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for ddargparse-0.3.2.tar.gz
Algorithm Hash digest
SHA256 4dec862f4a41de7252ab3517101a632561600aae095715b6ab6b56fb83a8524b
MD5 95165aad61dfb2712c106c2ef9e88987
BLAKE2b-256 03251b9418c194a2beed209e7f3af371861c53fd2fc630fc4ee64a3deed0ef12

See more details on using hashes here.

Provenance

The following attestation bundles were made for ddargparse-0.3.2.tar.gz:

Publisher: release-please.yml on koesterlab/ddargparse

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

File details

Details for the file ddargparse-0.3.2-py3-none-any.whl.

File metadata

  • Download URL: ddargparse-0.3.2-py3-none-any.whl
  • Upload date:
  • Size: 8.1 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for ddargparse-0.3.2-py3-none-any.whl
Algorithm Hash digest
SHA256 5fdc63325da703f5d81c2821a317903f5078df7d104a63e72b53a0beefbf6d89
MD5 85c1bc9db624ace55d669d539fce5f98
BLAKE2b-256 1ab16241b45ceedbc9729cb2426c102e671418c19d99cac18c31a77793f67524

See more details on using hashes here.

Provenance

The following attestation bundles were made for ddargparse-0.3.2-py3-none-any.whl:

Publisher: release-please.yml on koesterlab/ddargparse

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