Tool runnerยถ
repomatic run is a unified entry point for running external linters, formatters, and security scanners. It installs each tool at a pinned version, resolves configuration through a strict precedence chain, and invokes the tool: no manual setup, no dotfile sprawl.
Why not run the tools directly?ยถ
Installing a tool and running yamllint . yourself is fine for one tool on one machine. Once a project leans on a dozen, the same three chores repeat for each, and repomatic run takes care of all of them:
Configuration stays in
pyproject.toml, one reviewed file rather than a dotfile per tool. Even tools that canโt readpyproject.tomlthemselves get their[tool.X]table translated to a temporary native config at run time, following the precedence chain below.Installation is automatic: binaries come from GitHub Releases and are checksum-verified, PyPI tools run through
uvx, and tools that import your code (mypy, Nuitka) run inside the project virtualenv.Versions are pinned and re-verified on each use, so a check behaves the same on your laptop and in CI instead of drifting with whatever each machine happens to have installed.
See also
The tools repomatic bridges have standing upstream requests to read [tool.X] from pyproject.toml natively, all still unshipped:
actionlint#623,
biome#9239,
gitleaks#2066,
Nuitka#3909,
zizmor#322.
The same request for shfmt (sh#1268) was declined.
Quick startยถ
Run a tool against your project:
$ repomatic run yamllint -- .
The -- separates repomaticโs own options from the arguments forwarded to the tool. Everything after -- is passed through verbatim.
List all managed tools and their resolved config source:
$ repomatic run --list
โญโโโโโโโโโโโโโโโโโโฌโโโโโโโโโโฌโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฎ
โ Tool โ Version โ Config source โ
โโโโโโโโโโโโโโโโโโโผโโโโโโโโโโผโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโค
โ actionlint โ 1.7.12 โ (bare) โ
โ autopep8 โ 2.3.2 โ (bare) โ
โ biome โ 2.5.0 โ (bare) โ
โ bump-my-version โ 1.2.7 โ (bare) โ
โ gitleaks โ 8.30.1 โ (bare) โ
โ labelmaker โ 0.6.4 โ (bare) โ
โ lychee โ 0.24.2 โ [tool.lychee] in pyproject.toml โ
โ mdformat โ 1.0.0 โ bundled default โ
โ mypy โ 1.19.1 โ [tool.mypy] in pyproject.toml โ
โ nuitka โ 4.1 โ [tool.nuitka] in pyproject.toml โ
โ pyproject-fmt โ 2.25.0 โ (bare) โ
โ ruff โ 0.15.5 โ [tool.ruff] in pyproject.toml โ
โ shfmt โ 3.13.1 โ (bare) โ
โ typos โ 1.47.2 โ [tool.typos] in pyproject.toml โ
โ yamllint โ 1.38.0 โ bundled default โ
โ zizmor โ 1.23.0 โ bundled default โ
โฐโโโโโโโโโโโโโโโโโโดโโโโโโโโโโดโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ
Available toolsยถ
Tool |
Version |
Type |
Config discovery |
|---|---|---|---|
|
Binary |
|
|
|
PyPI |
|
|
|
Binary |
|
|
|
PyPI |
|
|
|
Binary |
|
|
|
Binary |
CLI flags only |
|
|
Binary |
|
|
|
PyPI |
|
|
|
PyPI (venv) |
|
|
|
PyPI (venv) |
|
|
|
PyPI |
|
|
|
PyPI |
|
|
|
Binary |
|
|
|
Binary |
|
|
|
PyPI |
|
|
|
PyPI |
|
Binary: downloaded as platform-specific executables from GitHub Releases.
PyPI: installed via
uvx.PyPI (venv): run inside the project virtualenv via
uv runbecause they need to import project code.
Config resolutionยถ
When repomatic run <tool> is invoked, configuration is resolved through a 4-level precedence chain. The first match wins: no merging across levels.
flowchart TD
run([repomatic run TOOL]) --> l1{native config file?}
l1 -->|yes| u1[Level 1. Use the in-repo config file]
l1 -->|no| l2{tool.X in pyproject.toml?}
l2 -->|yes| u2[Level 2. Use tool.X, translate if needed]
l2 -->|no| l3{bundled default?}
l3 -->|yes| u3[Level 3. repomatic bundled baseline]
l3 -->|no| u4[Level 4. Bare invocation, tool defaults]
Tip
Run repomatic --verbosity INFO run <tool> to see which config level was selected and the exact command line being executed. This is useful for debugging unexpected behavior. For full detail (config file contents, environment, caching), use --verbosity DEBUG.
Level 1: native config fileยถ
If the toolโs own config file exists in the repo (like ruff.toml or .yamllint.yaml), repomatic defers to it entirely. Your repo stays in control.
$ ls ruff.toml
ruff.toml
$ repomatic run ruff -- check .
# Uses ruff.toml directly โ repomatic does nothing special.
Level 2: [tool.X] in pyproject.tomlยถ
If no native config file is found but your pyproject.toml has a [tool.<name>] section, repomatic uses it. For tools that read pyproject.toml natively (ruff, mypy, bump-my-version, etc.), this just works. For tools that donโt, repomatic translates the section into the toolโs native format and passes it via a temporary config file.
Note
When the toolโs native format is also TOML (like gitleaks), the translation keeps the comments from your [tool.X] section and only drops the [tool.X] prefix. Translations to another format (YAML, JSON) carry the values only: a TOML comment has no equivalent to map onto.
# pyproject.toml
[tool.yamllint.rules.line-length]
max = 120
[tool.yamllint.rules.truthy]
check-keys = false
$ repomatic run yamllint -- .
# Translates [tool.yamllint] to YAML, passes via --config-file.
All tools that support [tool.X] sections in pyproject.toml, whether natively or via repomaticโs translation bridge:
Tool |
Customizes |
Section |
Support |
|---|---|---|---|
Workflow linting rules |
repomatic bridge โ YAML |
||
Python code formatting |
Native |
||
JSON/JS formatting and linting |
repomatic bridge โ JSON |
||
Version bump patterns and files |
Native |
||
Code coverage reporting |
Native |
||
Secret detection rules |
repomatic bridge โ TOML |
||
Link checking rules |
Native |
||
Markdown formatting options |
Native (via |
||
Static type checking |
Native |
||
Standalone binary compilation |
repomatic bridge โ CLI flags (native support: Nuitka#3909) |
||
|
Native |
||
Test runner options |
Native |
||
Linting and formatting rules |
Native |
||
Spell-checking exceptions |
Native |
||
Package resolution and build config |
Native |
||
YAML linting rules |
repomatic bridge โ YAML |
||
Workflow security scanning |
repomatic bridge โ YAML |
See Click Extraโs inventory of pyproject.toml-aware tools for a broader list.
Level 3: bundled defaultยถ
If the repo has no config at all, repomatic falls back to its own bundled defaults (stored in repomatic/data/). These provide sensible baseline rules so that tools produce useful results even without any project-specific configuration.
Tools with bundled defaults: mdformat, ruff, yamllint, zizmor.
Level 4: bare invocationยถ
If none of the above applies (no config file, no [tool.X], no bundled default), the tool runs with its own built-in defaults. Tools like autopep8 work this way: all behavior is controlled through CLI flags.
Checking the active config sourceยถ
To see which precedence level is active for each tool in your repo:
$ repomatic run --list
โญโโโโโโโโโโโโโโโโโโฌโโโโโโโโโโฌโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฎ
โ Tool โ Version โ Config source โ
โโโโโโโโโโโโโโโโโโโผโโโโโโโโโโผโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโค
โ actionlint โ 1.7.12 โ (bare) โ
โ autopep8 โ 2.3.2 โ (bare) โ
โ biome โ 2.5.0 โ (bare) โ
โ bump-my-version โ 1.2.7 โ (bare) โ
โ gitleaks โ 8.30.1 โ (bare) โ
โ labelmaker โ 0.6.4 โ (bare) โ
โ lychee โ 0.24.2 โ [tool.lychee] in pyproject.toml โ
โ mdformat โ 1.0.0 โ bundled default โ
โ mypy โ 1.19.1 โ [tool.mypy] in pyproject.toml โ
โ nuitka โ 4.1 โ [tool.nuitka] in pyproject.toml โ
โ pyproject-fmt โ 2.25.0 โ (bare) โ
โ ruff โ 0.15.5 โ [tool.ruff] in pyproject.toml โ
โ shfmt โ 3.13.1 โ (bare) โ
โ typos โ 1.47.2 โ [tool.typos] in pyproject.toml โ
โ yamllint โ 1.38.0 โ bundled default โ
โ zizmor โ 1.23.0 โ bundled default โ
โฐโโโโโโโโโโโโโโโโโโดโโโโโโโโโโดโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ
The โConfig sourceโ column shows whether the tool is using a native config file (level 1), [tool.X] (level 2), a bundled default (level 3), or bare invocation (level 4).
Tutorial: adding yamllint to your projectยถ
This walkthrough covers a common scenario: running yamllint on a project that has no YAML linting configured.
Step 1: run with defaultsยถ
With no config file and no [tool.yamllint] section in pyproject.toml, repomatic uses its bundled default:
$ repomatic run yamllint -- .
The bundled config enforces strict YAML rules. If that produces too many warnings, customize it.
Step 2: customize via pyproject.tomlยถ
Instead of creating a .yamllint.yaml file, add a section to your pyproject.toml:
[tool.yamllint.rules.line-length]
max = 120
[tool.yamllint.rules.truthy]
check-keys = false
Now repomatic run yamllint -- . translates this to YAML, passes it via --config-file, and cleans up the temporary file afterward.
Step 3: graduate to a native config fileยถ
If your yamllint config grows complex, create a .yamllint.yaml directly. Once that file exists, repomatic defers to it (level 1 takes precedence) and the [tool.yamllint] section in pyproject.toml is ignored.
Cleaning up unmodified configsยถ
If you previously ran repomatic init and have a native config file that is identical to the bundled default, repomatic init --delete-unmodified removes it:
$ repomatic init --delete-unmodified
Overriding tool versionsยถ
To test a newer version of a tool before the registry is updated:
$ repomatic run shfmt --version 3.14.0 --skip-checksum -- .
--skip-checksum is required because the registry only stores checksums for the pinned version. For binary tools, --checksum lets you provide the correct SHA-256 for the new version instead of skipping verification entirely:
$ repomatic run shfmt --version 3.14.0 --checksum abc123... -- .
Binary cachingยถ
repomatic run downloads platform-specific binaries (actionlint, biome, gitleaks, labelmaker, lychee, etc.) from GitHub Releases. To avoid re-downloading on every invocation, binaries are cached under a platform-appropriate user cache directory:
Platform |
Default cache path |
|---|---|
Linux |
|
macOS |
|
Windows |
|
Cached binaries are re-verified against their registry SHA-256 checksum on every use. Entries older than 30 days are auto-purged.
Both settings are configurable via [tool.repomatic] (see cache.dir and cache.max-age) or environment variables. The env var takes precedence over the config.
Environment variable |
Config key |
Default |
Description |
|---|---|---|---|
|
(platform-specific) |
Override the cache directory path. |
|
|
|
Auto-purge entries older than this many days. |
Cache management commands:
$ repomatic cache show
$ repomatic cache clean
$ repomatic cache clean --tool ruff --max-age 7
$ repomatic cache path
Use --no-cache on repomatic run to bypass the cache entirely.
Passing extra argumentsยถ
Everything after -- is forwarded to the tool:
$ repomatic run ruff -- check --fix .
$ repomatic run zizmor -- --offline .github/workflows/
$ repomatic run biome -- format --write src/
For tools with subcommands (ruff, biome, gitleaks), the subcommand goes after -- as the first argument.
Tool detailsยถ
actionlintยถ
Installed version: 1.7.12
Installation method: Binary (downloaded from GitHub Releases)
Config files: .github/actionlint.yaml, .github/actionlint.yml
[tool.actionlint] bridge: repomatic translates to YAML and passes via --config-file.
Default flags: -color
Source | Config reference | CLI usage
Try it:
$ repomatic run actionlint
Minimal [tool.actionlint]:
[tool.actionlint.self-hosted-runner]
labels = ["my-linux-runner"]
With no arguments actionlint lints every workflow under .github/workflows. The [tool.actionlint] section is bridged to a temporary YAML config: declaring self-hosted runner labels stops custom runs-on: values being flagged as unknown.
autopep8ยถ
Installed version: 2.3.2
Installation method: PyPI, installed via uvx
Config: [tool.autopep8] in pyproject.toml (native)
Default flags: --recursive --in-place --max-line-length 88 --select E501
Try it:
$ repomatic run autopep8 -- .
autopep8 takes its configuration from CLI flags only. repomatic passes --recursive --in-place --max-line-length 88 --select E501 by default; append more flags after --.
Biomeยถ
Installed version: 2.5.0
Installation method: Binary (downloaded from GitHub Releases)
Config files: biome.json, biome.jsonc, .biome.json, .biome.jsonc
[tool.biome] bridge: repomatic translates to JSON and passes via --config-path.
Source | Config reference | CLI usage
Try it:
$ repomatic run biome -- check .
Minimal [tool.biome]:
[tool.biome.formatter]
indentStyle = "space"
biome check reports formatting and lint issues; add --write after -- to apply fixes. The [tool.biome] section is bridged to a temporary biome.json, so keys keep Biomeโs camelCase spelling.
bump-my-versionยถ
Installed version: 1.2.7
Installation method: PyPI, installed via uvx
Config files: .bumpversion.toml and [tool.bump-my-version] in pyproject.toml (native)
Source | Config reference | CLI usage
Try it:
$ repomatic run bump-my-version -- show-bump
Minimal [tool.bumpversion]:
[tool.bumpversion]
current_version = "1.2.3"
The configuration table is [tool.bumpversion], not [tool.bump-my-version]: the section name predates the projectโs rename. show-bump previews the next versions without writing; repomatic run bump-my-version -- bump minor performs the bump.
Gitleaksยถ
Installed version: 8.30.1
Installation method: Binary (downloaded from GitHub Releases)
Config files: .gitleaks.toml
[tool.gitleaks] bridge: repomatic translates to TOML and passes via --config.
Source | Config reference | CLI usage
Try it:
$ repomatic run gitleaks -- dir .
Minimal [tool.gitleaks]:
[tool.gitleaks.extend]
useDefault = true
[tool.gitleaks.allowlist]
paths = ['''\.env\.sample$''']
gitleaks dir . scans the working tree; gitleaks git scans history instead. The [tool.gitleaks] section is bridged to a temporary .gitleaks.toml: keep extend.useDefault = true, or a custom config silently replaces the built-in rule set.
labelmakerยถ
Installed version: 0.6.4
Installation method: Binary (downloaded from GitHub Releases)
Config: CLI flags only
labelmaker syncs a repositoryโs issue and PR labels from a label-definition file, so unlike the linters it needs a target repository and a GITHUB_TOKEN, not a path in the working tree. There is no [tool.labelmaker] section: the label file is the configuration. See the upstream usage docs for its flags and file schema.
Lycheeยถ
Installed version: 0.24.2
Installation method: Binary (downloaded from GitHub Releases)
Config files: lychee.toml and [tool.lychee] in pyproject.toml (native)
Source | Config reference | CLI usage
Try it:
$ repomatic run lychee -- .
Minimal [tool.lychee]:
[tool.lychee]
max_redirects = 5
lychee checks links found in the given path. Since v0.24 it reads [tool.lychee] from pyproject.toml natively, so repomatic does not translate it.
mdformatยถ
Installed version: 1.0.0
Installation method: PyPI, installed via uvx
Config files: .mdformat.toml and [tool.mdformat] in pyproject.toml (native)
Default flags: --strict-front-matter
Bundled default: mdformat.toml
Plugins:
mdformat_admonmdformat-configmdformat_deflistmdformat_footnotemdformat-front-mattersmdformat-gfmmdformat_gfm_alertsmdformat_mystmdformat-pelicanmdformat_pyprojectmdformat-recover-urlsmdformat-ruffmdformat-shfmtmdformat_simple_breaksmdformat-tocmdformat-webruff
Source | Config reference | CLI usage
Try it:
$ repomatic run mdformat -- .
Minimal [tool.mdformat]:
[tool.mdformat]
wrap = "no"
mdformat rewrites Markdown in place. repomatic bundles a plugin set (GFM, MyST, front-matter, and others) and a baseline mdformat.toml; [tool.mdformat] in your pyproject.toml overrides it.
mypyยถ
Installed version: 1.19.1
Installation method: PyPI, runs in project virtualenv via uv run
Config: [tool.mypy] in pyproject.toml (native)
Default flags: --color-output
Source | Config reference | CLI usage
Try it:
$ repomatic run mypy -- .
Minimal [tool.mypy]:
[tool.mypy]
strict = true
mypy runs inside the project virtualenv (via uv run) so it can import your dependencies. repomatic derives --python-version from requires-python, so the check matches your lowest supported interpreter.
Nuitkaยถ
Installed version: 4.1
Installation method: PyPI, runs in project virtualenv via uv run
Config: [tool.nuitka] in pyproject.toml (translated to CLI flags)
Default flags: --mode=onefile --assume-yes-for-downloads
Source | Config reference | CLI usage
Try it:
$ repomatic run nuitka -- my_app/__main__.py
Minimal [tool.nuitka]:
[tool.nuitka]
onefile = true
output-dir = "build"
repomatic reads every key from [tool.nuitka] and forwards it as a CLI flag: true becomes a bare --flag, a string or number becomes --key=value, and a list repeats the flag once per item. Nuitka does not read [tool.nuitka] natively yet (Nuitka#3909); repomaticโs bridge fills the gap until it does.
pyproject-fmtยถ
Installed version: 2.25.0
Installation method: PyPI, installed via uvx
Config files: pyproject-fmt.toml and [tool.pyproject-fmt] in pyproject.toml (native)
Source | Config reference | CLI usage
Try it:
$ repomatic run pyproject-fmt -- pyproject.toml
Minimal [tool.pyproject-fmt]:
[tool.pyproject-fmt]
indent = 4
pyproject-fmt normalizes and reorders pyproject.toml in place. It reads its own [tool.pyproject-fmt] section natively.
Ruffยถ
Installed version: 0.15.5
Installation method: PyPI, installed via uvx
Config files: .ruff.toml, ruff.toml and [tool.ruff] in pyproject.toml (native)
Bundled default: ruff.toml
Source | Config reference | CLI usage
Try it:
$ repomatic run ruff -- check .
Minimal [tool.ruff]:
[tool.ruff]
line-length = 100
ruff check . lints; ruff format . reformats. Both read [tool.ruff] natively. With no project config, repomatic falls back to its bundled ruff.toml baseline.
shfmtยถ
Installed version: 3.13.1
Installation method: Binary (downloaded from GitHub Releases)
Config files: .editorconfig
Default flags: --write
Source | Config reference | CLI usage
Try it:
$ repomatic run shfmt -- .
shfmt formats shell scripts in place. It has no [tool.shfmt] section: indentation and style come from .editorconfig (indent_size, shell_variant, and the shfmt-specific keys).
typosยถ
Installed version: 1.47.2
Installation method: Binary (downloaded from GitHub Releases)
Config files: typos.toml, _typos.toml, .typos.toml and [tool.typos] in pyproject.toml (native)
Default flags: --write-changes
Source | Config reference | CLI usage
Try it:
$ repomatic run typos -- .
Minimal [tool.typos]:
[tool.typos.files]
extend-exclude = ["*.lock"]
typos scans the tree and, with repomaticโs default --write-changes, fixes what it finds. It reads [tool.typos] natively; use [tool.typos.default.extend-words] to map project-specific terms to their intended spelling.
yamllintยถ
Installed version: 1.38.0
Installation method: PyPI, installed via uvx
Config files: .yamllint, .yamllint.yaml, .yamllint.yml
[tool.yamllint] bridge: repomatic translates to YAML and passes via --config-file.
Default flags: --strict
CI flags: --format github
Bundled default: yamllint.yaml
Source | Config reference | CLI usage
Try it:
$ repomatic run yamllint -- .
Minimal [tool.yamllint]:
[tool.yamllint.rules.line-length]
max = 120
yamllint has no native pyproject.toml support, so repomatic bridges [tool.yamllint] to a temporary YAML config passed via --config-file. With no project config it uses repomaticโs strict bundled yamllint.yaml.
zizmorยถ
Installed version: 1.23.0
Installation method: PyPI, installed via uvx
Config files: .github/zizmor.yml, .github/zizmor.yaml, zizmor.yml, zizmor.yaml
[tool.zizmor] bridge: repomatic translates to YAML and passes via --config.
Default flags: --offline
CI flags: --format github
Bundled default: zizmor.yaml
Source | Config reference | CLI usage
Try it:
$ repomatic run zizmor -- .
zizmor audits GitHub Actions workflows for security issues, offline by default. repomatic bridges [tool.zizmor] to a temporary YAML config (passed via --config); with none, it uses the bundled zizmor.yaml. See the configuration reference for available keys.
Comparisonยถ
Tool |
Stars |
Last release |
Last commit |
Commits |
Dependencies |
Language |
License |
|---|---|---|---|---|---|---|---|
repomatic.tool_runner APIยถ
classDiagram
Enum <|-- ArchiveFormat
Enum <|-- NativeFormat
Unified tool runner with managed config resolution.
Provides repomatic run <tool> โ a single entry point that installs an
external tool at a pinned version, resolves its configuration through a strict
4-level precedence chain, translates [tool.X] sections from
pyproject.toml into the toolโs native format, and invokes the tool with
the resolved config.
Important
Config resolution precedence (first match wins, no merging):
Native config file โ toolโs own config file in the repo.
``[tool.X]`` in ``pyproject.toml`` โ translated to native format.
Bundled default โ from
repomatic/data/.Bare invocation โ no config at all.
- repomatic.tool_runner.GENERATED_HEADER_TEMPLATE = 'Generated by {command} v{version} - https://github.com/kdeldycke/repomatic'ยถ
Template for the first line of generated-file headers.
Used by both CLI commands (e.g.
sync-mailmap) and the tool runner (e.g.run shfmt) to stamp files with provenance. Format fields:command(full command path) andversion(package version).
- repomatic.tool_runner.generated_header(command, comment_prefix='# ')[source]ยถ
Return a generated-by header block with timestamp.
- class repomatic.tool_runner.ArchiveFormat(*values)[source]ยถ
Bases:
EnumArchive format for binary tool downloads.
- RAW = 'raw'ยถ
- TAR_GZ = 'tar.gz'ยถ
- TAR_XZ = 'tar.xz'ยถ
- ZIP = 'zip'ยถ
- tarfile_mode()[source]ยถ
Return the
tarfile.openmode string for this format.- Raises:
ValueError โ If called on a non-tar format.
- Return type:
Literal['r:gz','r:xz']
- class repomatic.tool_runner.NativeFormat(*values)[source]ยถ
Bases:
EnumTarget format for
[tool.X]translation.- YAML = 'yaml'ยถ
- TOML = 'toml'ยถ
- JSON = 'json'ยถ
- EDITORCONFIG = 'editorconfig'ยถ
- FLAGS = 'flags'ยถ
- serialize(data, tool_name='')[source]ยถ
Serialize a config dict to this formatโs string representation.
When data is a live
[tool.X]table parsed frompyproject.toml(atomlrt.Table), the TOML branch keeps the userโs comments by reparenting the section to the document root; see_reroot_section. A plain dict carries no trivia, so it is rendered as-is. The other formats (YAML, JSON, editorconfig) cannot carry TOML comments across the format boundary, so they serialize the values only.- Parameters:
- Raises:
ValueError โ For
FLAGS, which is not a file format.- Return type:
- repomatic.tool_runner.PlatformKeyยถ
A
(platform_or_group, architecture)pair used as binary lookup key.The platform element can be a single
Platform(likeMACOS) or aGroup(likeLINUX, which matches any Linux distribution). The architecture is always a concreteArchitecture.Resolution order in
BinarySpec.resolve_platform():Exact Platform match (
current_platform() == key_platform).Group membership (
current_platform() in key_group), preferring the group with fewest members (most specific).
alias of
tuple[Platform|Group,Architecture]
- class repomatic.tool_runner.BinarySpec(urls, checksums, archive_format, archive_executable=None, strip_components=0)[source]ยถ
Bases:
objectPlatform-specific binary download specification.
Keys are
PlatformKeytuples pairing an extra-platformsPlatformorGroupwith anArchitecture. This lets callers use broad groups (LINUXmatches any distro) or specific platforms (DEBIAN) with full detection heuristics from extra-platforms.Hint
Structural integrity checks (key types, checksum format, URL placeholders, strip_components consistency) are enforced in
test_tool_spec_integrity. If the registry becomes user-configurable in the future, move these checks to__post_init__.- urls: dict[tuple[Platform | Group, Architecture], str]ยถ
Platform key to URL template mapping. URLs use
{version}placeholders.
- checksums: dict[tuple[Platform | Group, Architecture], str]ยถ
Platform key to SHA-256 hex digest mapping.
- archive_format: ArchiveFormat | dict[tuple[Platform | Group, Architecture] | Platform | Group, ArchiveFormat]ยถ
Archive format of the downloaded file.
A single
ArchiveFormatapplies to every platform. A dict maps platform specifiers to formats, allowing mixed archives in one spec:archive_format={ALL_PLATFORMS: ArchiveFormat.TAR_GZ, WINDOWS: ArchiveFormat.ZIP}
Dict keys follow the same resolution as
resolve_platform(): exactPlatformKeytuple first, then barePlatformequality, thenGroupmembership (smallest group wins).
- archive_executable: str | None = Noneยถ
Path of the executable inside the archive.
Nonedefaults to the tool name. ForRAWformat, used as the final filename.
- resolve_platform()[source]ยถ
Match the current environment against registered platform keys.
Uses
current_platform()andcurrent_architecture()from extra-platforms, inheriting its full detection heuristics.- Return type:
tuple[Platform|Group,Architecture]- Returns:
The matching
PlatformKey.- Raises:
RuntimeError โ If no key matches the current environment.
- get_archive_format(key)[source]ยถ
Return the archive format for the given platform key.
When
archive_formatis a singleArchiveFormat, returns it directly. When it is a dict, resolves in order: exactPlatformKeytuple, bare Platform equality, then Group membership (smallest group wins).- Return type:
- class repomatic.tool_runner.ToolSpec(name, display_name=None, version='', package=None, executable=None, module=None, native_config_files=(), config_flag=None, native_format=NativeFormat.YAML, default_config=None, reads_pyproject=False, default_flags=(), ci_flags=(), with_packages=(), needs_venv=False, computed_params=None, config_after_subcommand=False, post_process=None, check_flags=(), binary=None, source_url=None, config_docs_url=None, cli_docs_url=None)[source]ยถ
Bases:
objectSpecification for an external tool managed by repomatic.
Hint
Structural integrity checks (name format, version format, flag conventions, field consistency) are enforced in
test_tool_spec_integrity. If the registry becomes user-configurable in the future, move these checks to__post_init__.Hint
CLI parser quirks for
config_after_subcommandTools that use subcommands (
tool <subcmd> [flags] [files]) may requireconfig_flagto appear after the subcommand name, depending on the CLI parser framework:clap (Rust): global flags accepted before or after the subcommand. No special handling needed. Used by: ruff, labelmaker.
cobra (Go): root-level flags inherited by all subcommands, accepted in both positions. No special handling needed. Used by: gitleaks.
click (Python): global flags accepted before or after the subcommand. No special handling needed. Used by: bump-my-version.
bpaf (Rust):
#[bpaf(external)]fields are scoped inside the subcommand variant, sotool <subcmd> --flagworks buttool --flag <subcmd>does not. Setconfig_after_subcommand=True. Used by: biome.
- name: strยถ
Tool identity: CLI name for
repomatic run <name>, default PyPI package name, and default executable name.
- display_name: str | None = Noneยถ
Human-readable name with proper casing for documentation (like
'Biome','Gitleaks').Nonedefaults toname.
- package: str | None = Noneยถ
PyPI package name.
Nonedefaults toname. Only set when the package name differs from the tool name.
- executable: str | None = Noneยถ
Executable name if different from the tool name.
Nonedefaults to the registry key.
- module: str | None = Noneยถ
Python module name for
-m moduleinvocation, e.g.'nuitka'.When set, the tool is invoked as
python -m <module>instead of the console script. Requiresneeds_venv=True. Use when the toolโs script entry point is not reliably found across platforms (for example, Nuitka installs only a.cmdwrapper on Windows, whichuv run -- nuitkacannot locate).
- native_config_files: tuple[str, ...] = ()ยถ
Config filenames the tool auto-discovers, checked in order.
Paths relative to repo root (e.g.,
'zizmor.yaml','.github/actionlint.yaml'). Empty for tools with no config file.
- config_flag: str | None = Noneยถ
CLI flag to pass a config file path (e.g.,
'--config','--config-file').Noneif the tool only reads from fixed paths.
- native_format: NativeFormat = 'yaml'ยถ
Target format for
[tool.X]translation.NativeFormat.FLAGStranslates the table to CLI flags (viapyproject_table_to_flags) instead of a config file, for tools that expose their config keys as long options but read no config file themselves. It is mutually exclusive withreads_pyproject,config_flag, andnative_config_files.
- default_config: str | None = Noneยถ
Filename in
repomatic/data/for bundled defaults, stored innative_format.Noneif no bundled default exists.
- reads_pyproject: bool = Falseยถ
Whether the tool natively reads
[tool.X]frompyproject.toml.When
Trueand[tool.X]exists inpyproject.toml, repomatic skips Level 2 translation (the tool reads it directly). Resolution still falls through to Level 3 (bundled default) and Level 4 (bare) when no config is found.
- ci_flags: tuple[str, ...] = ()ยถ
Flags added only when
$GITHUB_ACTIONSis set (e.g., output format).
- with_packages: tuple[str, ...] = ()ยถ
Extra packages installed alongside the tool (e.g., mdformat plugins).
Passed as
--with <pkg>to uvx.
- needs_venv: bool = Falseยถ
If
True, useuv run(project venv) instead ofuvx(isolated).Required when the tool imports project code (mypy, pytest).
- computed_params: Callable[[Metadata], list[str]] | None = Noneยถ
Callable that receives a
Metadatainstance and returns extra CLI args derived from project metadata (e.g., mypyโs--python-versionfromrequires-python).Noneif no computed params.
- config_after_subcommand: bool = Falseยถ
Insert
config_flagafter the first token ofextra_args.Needed for tools whose CLI parser (e.g., bpaf) scopes global options inside the subcommand, so
tool subcommand --config-path Xis valid buttool --config-path X subcommandis not. WhenTrue,config_argsare spliced after the first element ofextra_args(the subcommand name).
- post_process: Callable[[Sequence[str]], None] | None = Noneยถ
Callback invoked on
extra_argsafter the tool exits successfully.Intended for temporary workarounds that fix known upstream formatting bugs in-place. Remove the callback once upstream ships the fix.
Note
The callback runs only after a successful write-mode exit (return code 0) and rewrites files on disk, so it cannot apply in check/dry-run mode, which writes nothing. Pair it with
check_flagssorun_toolwarns when a check invocation would silently bypass it. Seecheck_bypasses_post_process().
- check_flags: tuple[str, ...] = ()ยถ
Flags that put the tool in check/dry-run mode, writing no files.
Warning
Check mode bypasses
post_process: that fixup rewrites files on disk, but check mode writes nothing. So when a tool defines both apost_processandcheck_flags, its check-mode exit status is unreliable.run_tooldetects the pairing viacheck_bypasses_post_process()and warns. Verify formatting by running the write path, not the check flag.
- binary: BinarySpec | None = Noneยถ
Platform-specific binary download spec. When set, the tool is downloaded as a binary instead of installed via
uvxoruv run.
- check_bypasses_post_process(extra_args)[source]ยถ
Return
Truewhen a check-mode flag will skippost_process.Check/dry-run flags (
check_flags) make the tool exit without writing files, so thepost_processfixup never runs and the exit status cannot be trusted: it may flag drift the write path would reconcile, or miss drift the write path would introduce.run_toolwarns on this. ReturnsFalsefor tools with nopost_process, where check mode is authoritative.- Return type:
- repomatic.tool_runner.get_data_file_path(filename)[source]ยถ
Yield the filesystem path of a bundled data file.
Unlike
init_project.get_data_content()which returns string content, this yields aPathsuitable for passing to external tools via--config <path>. The path is valid only within the context manager.
- repomatic.tool_runner.load_pyproject_tool_section(tool_name)[source]ยถ
Load
[tool.<tool_name>]frompyproject.tomlin the current directory.Returns the live
tomlrt.Table(adictsubclass) rather than a plain-dict copy, so the section keeps its comment trivia for formats that can preserve it on materialization (seeNativeFormat.serialize()). Callers that only read values or test truthiness are unaffected.
- repomatic.tool_runner.pyproject_table_to_flags(table)[source]ยถ
Translate a
[tool.X]table into long-form CLI flags.For tools whose command-line options mirror their config keys but which cannot read
[tool.X]frompyproject.tomlthemselves and accept no config file. Follows the conventional mapping:key = trueโ--keykey = "value"(or a number) โ--key=valuekey = ["a", "b"]โ--key=a --key=b(one flag per item)key = falseis skipped: there is no universal--no-<key>form.
Keys keep their hyphenated spelling so they map straight onto long options, and flags follow their declaration order in
pyproject.toml.
- repomatic.tool_runner.resolve_config(spec, tool_config=None)[source]ยถ
Resolve config for a tool using the 4-level precedence chain.
- Parameters:
- Return type:
- Returns:
Tuple of (extra CLI args for config, path to clean up). The path is
Nonewhen no cleanup is needed (cache-based configs persist across runs). Non-Nonepaths are CWD files written for tools that have no--configflag.
- repomatic.tool_runner.binary_tool_context(name, no_cache=False)[source]ยถ
Download a binary tool and yield its executable path.
For tools invoked indirectly by repomatic commands (e.g., labelmaker called by
sync-labels) rather than viarun_tool(). Downloads once; the binary stays valid for the contextโs duration. On a cache hit the yielded path points to the cache and the staging directory is empty.
- repomatic.tool_runner.run_tool(name, extra_args=(), version=None, checksum=None, skip_checksum=False, no_cache=False)[source]ยถ
Run an external tool with managed config resolution.
- Parameters:
name (
str) โ Tool name (must be inTOOL_REGISTRY).extra_args (
Sequence[str]) โ Extra arguments passed through to the tool.checksum (
str|None) โ Override the SHA-256 checksum for the current platform.skip_checksum (
bool) โ Skip SHA-256 verification entirely.no_cache (
bool) โ Bypass the binary cache whenTrue.
- Return type:
- Returns:
The toolโs exit code.
- repomatic.tool_runner.resolve_config_source(spec)[source]ยถ
Return a human-readable description of the active config source.
Used by
repomatic run --listto show which precedence level is active for each tool in the current repo.- Return type:
- repomatic.tool_runner.find_unmodified_configs()[source]ยถ
Find native config files identical to their bundled defaults.
Iterates over every tool in
TOOL_REGISTRYthat has adefault_config. For each, checks whether any of itsnative_config_filesexists on disk and is content-identical to the bundled default after trailing-whitespace normalization.The normalization (
rstrip() + "\n"`) matches the convention used by `_init_config_fileswhen writing files duringinit.