Warning: This project is in an early stage of development. Use at your own risk.
repoactive runs your scripts against a git repository and optionally keeps
the corresponding merge requests up to date. You write the scripts that
produce the code changes; repoactive handles the rest - branches, commits,
and (with --mode publish) the full MR lifecycle.
You configure one or more jobs, each with a script (any shell command or
executable) that modifies the repository's working tree. repoactive runs
each script, captures the resulting diff, and records the change locally.
With --mode publish it also:
- opens a new merge request if one does not already exist for that job, or
- updates the existing merge request branch if the diff has changed.
Branches and MR descriptions are managed automatically - the only code you need to write is the script that produces the change.
[your script] → diff → repoactive → branch
↓ (with --mode push or --mode publish)
git push → merge request
↑ (with --mode publish)
(create or update)
repoactivecreates a new commit on top of the base branch or on top of other repoactive managed branches.- It runs the job's script against the working tree.
- If the script produced a diff, it records the change. With
--mode pushor--mode publish, it pushes the branch; with--mode publish, it also creates or updates the merge request. - If the script produced no diff, the branch is reset to the base. With
--mode pushor--mode publish, the reset branch is pushed without opening or updating an MR.
jj commits the whole working tree. Because
repoactiveuses jj, every new file your script creates inside the working directory is added to the commit unless it is git-ignored. There is no way to select which working-tree changes become part of the commit — they all will. Keep.gitignoreup to date so build artifacts, caches, and other stray files your script produces do not end up in the diff.
repoactive works entirely from the local repository view and never
fetches from the remote on its own. Rebasing onto trunk(), cooldown
throttling, and the unmerged-branch refresh all read the local trunk() /
base branches, so a merge that happened on the remote is invisible until the
local clone advances past it.
Fetch before each run. Run jj git fetch (or git fetch --prune) in
the same cron job or CI pipeline that invokes repoactive, before it. If
you do not, jobs rebase onto a stale base and — most importantly —
cooldown_period never engages,
because the commit that would trigger it has not reached the local base
branch. See
ADR 0005.
- Keeping generated files (API clients, protobuf bindings, lock files) in sync with their sources
- Applying organization-wide refactors or policy changes across many repositories
- Automating any periodic code transformation that should go through a review process
repoactive is configured via .repoactive.toml in the repository root (or
passed via --config). See Config file locations
for how the defaults are discovered and how to split the config across
several files.
Every key in [job-defaults] supplies the default for the matching per-job
key; any job may override it by setting the same key in its [job.<name>]
block.
[job-defaults]
# Prefix prepended to job.name to form the branch name
branch_prefix = "repoactive/"
# Prefix prepended to every MR/PR title (set to "" to disable)
mr_title_prefix = "[repoactive] "
# Prefix prepended to every commit title (set to "" to disable)
commit_title_prefix = "[repoactive] "
# Labels applied to every MR/PR unless overridden per job
labels = ["repoactive"]
# Optional: default target branch for jobs that do not set their own
# (default: repo default branch)
base_branch = "main"
# Optional: default cooldown_period applied to jobs that do not set their own
# (default: none). See "Throttling jobs with cooldown_period" below.
cooldown_period = "7d"
# Optional: default timeout applied to jobs that do not set their own
# (default: "2m"). See "Limiting job runtime with timeout" below.
timeout = "1h"
# The table key is the job's unique name; the branch is always
# <branch_prefix><name>.
[job.regenerate-api-client]
# Script run in the repo working directory; non-zero exit = failure
command = "python scripts/regen_api.py"
# MR/PR title
title = "api: regenerate API client"
# Optional: MR description
description = "Automated regeneration of the API client from the OpenAPI spec."
# Optional: extra labels (merged with job-defaults.labels)
labels = ["automated", "api"]
# Optional: target branch (default: repo default branch)
base_branch = "main"
# Optional: open the MR/PR as a draft (default: false)
draft = false
# Optional: create an MR/PR for this job (default: true). Set to false to
# push the branch without opening an MR/PR.
create_mr = true
# Optional: append the job's command and its output to the commit message
# (default: true). Set to false to keep the commit message clean.
output_in_commit = true
# Optional: skip this job on "run all" invocations (default: false). Sugar for
# tags = ["disabled"]; mutually exclusive with tags. See "Disabling jobs" below.
disabled = false
# Optional: tags for job selection (default: none -> the job carries the
# implicit "enabled" tag and runs in the bare `repoactive run`). Setting any tag
# removes "enabled", so the job runs only via `repoactive run --tag <tag>`. See
# "Selecting jobs with tags" below.
# tags = ["nightly"]
# Optional: override branch_prefix/mr_title_prefix/commit_title_prefix from
# job-defaults for this job only.
mr_title_prefix = "[api] "
# Optional: minimum time between landed changes for this job. If a commit
# from this job landed on the base branch within this window, the job is
# skipped. Format: <number><unit>, unit one of s, m, h, d, w (e.g. "7d").
cooldown_period = "7d"
# Optional: maximum runtime for this job's command. When it expires the
# command's process group is killed and the job fails. Same format as
# cooldown_period.
timeout = "30m"
[job.sync-license-headers]
command = "./scripts/add_license_headers.sh"
title = "sync license headers"
[job.integration-tests-update]
command = "./scripts/update_integration_tests.py"
title = "tests: update integration tests"
# Optional: run this job on top of the merged output of the listed jobs
depends_on = ["regenerate-api-client", "sync-license-headers"]For public GitHub.com or GitLab.com repositories no platform declaration is
needed — repoactive detects the remote URL automatically. To use a
self-hosted instance, add a [[platform]] section:
[[platform]]
# Base URL of the platform instance
url = "https://gitlab.example.com"
# Name of the env var holding the API token
token_env = "GITLAB_TOKEN"
# type must be either "github" or "gitlab"
type = "gitlab"The branch for each job is always branch_prefix + job.name, where
branch_prefix is the job's own value if set, otherwise
job-defaults.branch_prefix. Secrets are kept out of the config file by
referencing environment variable names rather than inline values.
The token named by token_env is stripped from the environment job
commands run in, so a script cannot read the credential repoactive uses
to push and create MRs. A job that needs its own credential must be given a
separate one. repoactive otherwise trusts job commands — they run
arbitrary code against the working tree — so the trust boundary is the
config that defines them; see
ADR 0006.
When depends_on is set, repoactive starts the job's script from a
working tree that has all listed dependency branches merged together, rather
than from the plain base branch. The resulting MR branch will therefore
include both the dependency jobs and the new job on top. Links to the parent
MRs are automatically added to the MR description.
When no --config/-c option is given, repoactive looks for
configuration inside the --repo directory (the current directory by
default):
- the
.repoactive.d/directory, if present, contributes every*.tomlfile it contains, merged in sorted filename order; - the
.repoactive.tomlfile, if present, is merged last so it overrides the directory.
If neither exists, repoactive exits with an error. Splitting configuration
across .repoactive.d/*.toml is handy for dropping in per-job files without
touching a single large config.
--config/-c overrides this discovery. It may point at a file or at a
directory of *.toml files, and may be repeated to merge several sources;
later sources win. Explicit paths are resolved relative to the current
directory, not --repo.
Which jobs a repoactive run touches is decided by tags. Every job has
a set of tags, with a smart default:
- a plain job (no
tags, notdisabled) carries the implicitenabledtag; disabled = trueis sugar fortags = ["disabled"];- setting
tags = [...]uses exactly those tags — and, importantly, drops the implicitenabledtag unless you list it yourself.
repoactive run with no arguments is shorthand for
repoactive run --tag enabled: it runs every job carrying enabled. Pass
--tag/-t (repeatable) to select a different set; a job runs if it
carries any of the requested tags. Naming jobs and passing tags can be
combined — the selection is the union of the two.
# Run all jobs tagged "weekly" (regardless of whether they also have "enabled")
repoactive run --tag weekly
# Union: the "nightly" jobs plus one named job
repoactive run --tag nightly regenerate-api-clientBecause assigning a tag removes the implicit enabled tag, tags are
load-bearing, not free-form labels: tagging a job takes it out of the bare
repoactive run. If you want a job to stay in the default run and belong
to a group, list both: tags = ["enabled", "weekly"]. (For MR/PR labels,
use labels — a separate concept.)
Tag selection is explicit selection, so — like naming a job — it ignores
the enabled/disabled defaults and force-includes dependencies. The bare
repoactive run is implicit selection: a job whose dependency is not
itself selected is dropped
(==> [name] skipped (dependency not in default run)).
The bare repoactive run additionally refreshes any job that currently
has an unmerged branch, regardless of its tags. A branch is "unmerged"
when the job's last commit has not yet landed in trunk(); repoactive finds
these via the Repoactive-Job trailer on unmerged commits and pulls those
jobs (and their dependencies) into the run, so each branch is rebased on the
latest trunk() and the command is re-run against it. (With
--mode publish such a branch has an open MR; with a plain run or
--mode push it is just a branch.)
This means a job's schedule tag governs when a new branch is created,
while the default run keeps an existing branch rebased and current — you
don't have to wait for the next weekly run to resolve a conflict with
trunk(). Once the branch lands, its commit becomes an ancestor of
trunk(), so the job drops back to its normal tag-driven cadence. (A
disabled job's unmerged branch is refreshed too: it was most likely created
by an explicit run, and letting it drift out of date helps no one.)
Set disabled = true on a [job.<name>] to keep it in the config but leave
it out of normal runs; it is exactly sugar for tags = ["disabled"] (so the
two are mutually exclusive). The flag only affects the bare
repoactive run:
- On
repoactive run, disabled jobs are skipped, along with any job thatdepends_onone (its dependency would not be produced). - Naming a job explicitly overrides it:
repoactive run my-jobrunsmy-jobeven when it is disabled. So doesrepoactive run --tag disabled, which runs everything currently turned off.
repoactive is not a daemon and has no built-in scheduler — the cadence of
a job is whatever cadence you invoke it with. To run a job on a fixed
schedule, tag it and have an OS cron job select that tag. The tag keeps the
job out of the bare repoactive run, and the crontab decides the membership
in one place — add or remove weekly jobs by editing the config, not the
crontab:
[job.uv-lock-upgrade]
command = "uv lock --upgrade"
title = "build: upgrade all dependencies"
# Not in the bare `repoactive run`; runs only via `--tag weekly`.
tags = ["weekly"]# Run every job tagged "weekly" each Sunday at 03:00
0 3 * * 0 repoactive run --tag weekly --mode publishBecause the cron is the sole trigger, the command runs exactly when cron
fires — once, whether or not it produces a diff. This is more reliable than
inferring a schedule from repoactive's own history: real cron is stateful
and excludes the other days, whereas repoactive only ever sees what has
landed (see cooldown_period below).
A repoactive run takes an exclusive per-repository lock for its duration,
so two runs against the same repository never interleave (and corrupt each
other's branches and temporary workspaces). If a run is started while
another is still in progress — a slow run overrunning the next cron tick,
say — the second one exits immediately with status code 2 instead of
waiting or racing. That code is distinct from the generic failure code (1),
so a wrapper can treat "already running" as benign:
0 3 * * 0 repoactive run --tag weekly --mode publish || test $? -eq 2The lock is an advisory flock on .jj/repoactive.lock; the OS releases it
automatically if a run is killed, so a crashed run never leaves the
repository locked.
Sometimes the useful set of jobs depends on the repository's contents — one
job per package in a monorepo, one per entry in a manifest — and you don't
want to hand-maintain them. A generator is an ordinary [job.<name>]
with emits_jobs = true. Instead of producing a diff, its command writes
one or more *.toml job fragments into the directory named by the
REPOACTIVE_JOBS_DIR environment variable, and the jobs it emits run in the
same invocation.
[job.per-package]
command = "./scripts/emit_upgrade_jobs.sh"
title = "discover package upgrade jobs"
emits_jobs = true
# Inherited by every emitted job unless the job overrides it (see below).
tags = ["weekly"]
cooldown_period = "7d"The command writes fragments using the normal [job.<name>] syntax, e.g.
# $REPOACTIVE_JOBS_DIR/pkg-a.toml
[job.upgrade-pkg-a]
command = "uv lock --upgrade --package pkg-a"
title = "build: upgrade pkg-a"Key points:
- Selection is unchanged. A generator is selected like any job (by the
bare run, by name, or by
--tag); selecting it runs it and everything it emits. - Inheritance with override. Each emitted job inherits the generator's
tags,cooldown_period,base_branch,timeout,labels,branch_prefix/title prefixes,draftandcreate_mrunless its own fragment sets them. It also defaults todepends_on = ["<generator>"](i.e. built flat ontrunk()); overridedepends_onto a sibling emitted job to stack them into an MR chain. - Stable names are your responsibility. Cooldown, branches and the
Repoactive-Jobtrailer all key on a job'sname, so derive emitted names deterministically from repository state (upgrade-pkg-a, not a random id). - The generator never commits. Any change its command leaves in the working copy is discarded; its only output is the job list.
- Batch cooldown. Each emitted commit carries a second
Repoactive-Jobtrailer with the generator's name, so acooldown_periodon the generator throttles the whole fan-out: it is skipped until enough time has passed since the most recent emitted job landed.
The generator's cooldown is a floor for its jobs. An emitted job is only re-run when the generator re-emits it, and the generator is gated by the same landed commits (via the shared trailer). So overriding an emitted job's
cooldown_periodonly matters when you make it longer than the generator's — then the job stays throttled even after the generator has run again. Making it shorter has no effect: while the generator is on its own cooldown the job is never re-emitted, and by the time the generator runs again the job's shorter window has long since elapsed (nothing landed for that job during the generator's cooldown). To upgrade an individual dependency more often than the batch, lower the generator'scooldown_period, not the emitted job's.
Emitted jobs may not themselves be generators (no recursion), may not reuse
an existing job's name, and may only depends_on jobs that are part of the
same run. See ADR 0004 for the full
design.
Every commit repoactive creates carries a Repoactive-Job: <name> trailer
identifying the job that produced it. When a job sets cooldown_period,
repoactive looks at the base branch for a commit with that job's trailer
and a committer date inside the window before running. If one is found the
job is on cooldown and is skipped for this run (dependents proceed as if it
produced no changes); otherwise the job runs normally. This keeps recurring
jobs - for example a dependency upgrade - from landing more often than the
configured interval.
The signal is the trailer on the base branch, so the cooldown only starts once a change has landed. An open, unmerged MR does not trigger it (the existing MR keeps being updated as usual). Because the check relies on the commit trailer reaching the base branch, MRs for throttled jobs must be merged with a merge commit or rebase - a squash merge discards the commit message and with it the trailer, so the cooldown would never trigger.
The trailer must also be present in the local base branch when the job
runs: repoactive does not fetch, so a clone that has not pulled the merge
will not see the cooldown and will re-run the job. See
Keeping the local clone current.
A job's command can hang or run away. Setting timeout caps how long the
command may run; when the limit is reached repoactive kills the command's
whole process group - the shell and any child processes it spawned - and the
job fails (its workspace is abandoned, no branch or MR is created). The
command runs in its own session/process group to make this possible. The
value uses the same <number><unit> format as cooldown_period (e.g.
"30m", "2h"). timeout may be set per job or in job-defaults; a
per-job value overrides the default. The built-in default is "2m"; set
timeout to a larger value in job-defaults for longer-running commands.
# Print the installed version and exit
repoactive --versionrepoactive run [OPTIONS] [JOBS]...
Run all configured jobs (or a named subset - dependencies are auto-included):
# Apply all jobs locally (no push, no MR creation)
repoactive run
# Apply specific jobs locally
repoactive run regenerate-api-client sync-license-headers
# Run every job carrying a given tag (see "Selecting jobs with tags")
repoactive run --tag weekly
# Push branches to the remote without creating MRs
repoactive run --mode push
# Push branches and create or update merge requests
repoactive run --mode publish
# Enable debug logging
repoactive run --debug| Option | Short | Description |
|---|---|---|
--config PATH |
-c |
Config file or directory of *.toml files; repeat to merge. Default: .repoactive.d/ and .repoactive.toml under --repo |
--repo PATH |
-r |
jj repository path (default: .) |
--mode [local|push|publish] |
-m |
How far to publish: local (default) applies only locally, push also pushes branches, publish also creates/updates MRs |
--tag TAG |
-t |
Run jobs carrying any of these tags (repeatable). With no tags/jobs the default run targets the enabled tag |
--debug |
-d |
Enable debug logging |
A local run (the default --mode local) captures the jj operation id
beforehand and prints a jj op restore <id> command (both before and after
the run, since a run can produce a lot of output). Run it to roll the
repository - commits, bookmarks and colocated git refs - back to the state
it was in before the run. The hint is omitted for
--mode push/--mode publish runs, since restoring local state would not
undo a branch already pushed or an MR already created.
While a job's command runs in an interactive terminal, repoactive shows a
live, scrolling block of its most recent output lines. The block stays on
screen once the command finishes, with the job's status line printed below
it. It defaults to 8 lines; set the REPOACTIVE_PROGRESS_LINES environment
variable to change the count (or to 0 to disable the live block). When
output is not a terminal (piped or in CI) the block is disabled and the
command's output is left untouched.
repoactive recent-commits [OPTIONS] [JOBS]...
List commits produced by repoactive, filtered by a time window and optionally by job name or merge status:
# Show all repoactive commits from the last 2 weeks (default window)
repoactive recent-commits --repo /path/to/repo
# Narrow to a specific window
repoactive recent-commits --within 30d --repo /path/to/repo
# Filter by one or more job names
repoactive recent-commits --within 7d uv-lock-upgrade prek-autoupdate
# Only commits that have landed in trunk
repoactive recent-commits --status merged
# Only commits still on open branches
repoactive recent-commits --status unmerged| Option | Short | Description |
|---|---|---|
--within |
How far back to look (default: 2w; e.g. 7d, 24h) |
|
--repo PATH |
-r |
jj repository path (default: .) |
--status [all|merged|unmerged] |
-s |
Filter by merge status into trunk (default: all) |
To query repoactive commits directly in jj, add these aliases to your
repository config (jj config set --repo) or your global config
(jj config set --user):
[revset-aliases]
'repoactive()' = 'description(regex:"(?m)^Repoactive-Job: ")'
'repoactive_merged()' = 'repoactive() & ::trunk()'
'repoactive_unmerged()' = 'repoactive() & ~(::trunk())'Then use them directly in jj:
jj log -r 'repoactive()'
jj log -r 'repoactive_unmerged()'
jj log -r 'repoactive() & committer_date(after:"2025-01-01")'
jj log -r 'repoactive() & description(regex:"(?m)^Repoactive-Job: uv-lock-upgrade$")'repoactive validate-config [OPTIONS]
Check that a config file is syntactically and semantically valid without running any jobs:
# Validate the discovered defaults (.repoactive.d/ and .repoactive.toml)
repoactive validate-config
# Validate a specific config file or directory
repoactive validate-config --config myconfig.toml
# Validate a merged config (same merging rules as `run`)
repoactive validate-config --config base.toml --config override.tomlOn success the command prints Config OK: N job(s) defined. and exits with
code 0. On failure it prints the validation error to stderr and exits with
code 1.
Validation checks include unknown keys, missing required fields, invalid
depends_on references, and circular job dependencies.
| Option | Short | Description |
|---|---|---|
--config PATH |
-c |
Config file or directory of *.toml files; repeat to merge. Default: .repoactive.d/ and .repoactive.toml under --repo |
--repo PATH |
-r |
jj repository path (default: .) |
- Python 3.11 or later
- jj (Jujutsu) -
repoactiveuses jj to manage branches and commits in the target repository - A GitLab or GitHub API token exposed via the environment variable named in
platform.token_env