Releases: jdx/hk
v1.45.0: Buildifier built-ins and smarter auto-batching
A small feature release: Bazel users get first-class buildifier built-ins, and hk's auto-batching is now smart enough to leave steps alone when their commands don't actually reference the file list.
Added
-
buildifier_formatandbuildifier_lintbuilt-ins (@plx) #896. Two new built-ins for Bazel projects, modeled onbuf_format/buf_lint. They coverBUILD,BUILD.bazel,WORKSPACE,WORKSPACE.bazel,MODULE.bazel,*.bzl,*.star, and*.skyfiles, and ship with the usual project-indicator metadata so they're auto-suggested for Bazel repos.import "package://github.com/jdx/hk/releases/download/v1.45.0/[email protected]#/Builtins.pkl" hooks { ["pre-commit"] { steps = new { ["buildifier-format"] = Builtins.buildifier_format ["buildifier-lint"] = Builtins.buildifier_lint } } }
Fixed
-
Auto-batching no longer splits jobs whose command doesn't reference
{{files}}(@jdx) #901. Previously, hk decided whether to split a step into multipleARG_MAX-safe batches purely from the size of the file-list expansion. On Windows — whereARG_MAXfalls back to 128KB — a step like:local vscodeCommitHint = new Step { exclusive = true check = "echo If you see this message in a pop-up, the pre-commit steps failed." }
…against a ~20K-file repo would be fanned out into ~29 jobs, printing the message 29 times even though the command never used
{{files}}. Auto-batching now happens at execution time with the full tera context available, renders the real run command for each candidate batch, and only splits when the rendered command exceeds the safe limit. Byte estimation is kept as a fallback if rendering fails. The split path also now correctly preservescheck_firstandworkspace_indicatoracross batches (the old code dropped them with a TODO).
Documentation
New Contributors
Full Changelog: v1.44.3...v1.45.0
💚 Sponsor hk
hk is developed by @jdx at en.dev — a small independent studio behind developer tools like mise, aube, hk, and more. Work on hk is funded by sponsorships.
If hk has sped up your pre-commit loop or made linting feel less painful, please consider sponsoring at en.dev. Sponsorships are what keep hk moving and the project independent.
v1.44.3: Honest fail_on_fix and readable CI logs
A small patch release fixing two notable rough edges: fail_on_fix=true no longer silently re-stages the fixer's output over your git add, and hk's text-mode progress output is finally readable in CI logs.
Fixed
-
fail_on_fix=trueno longer overwrites your staged changes with the fix (@jdx) #892. Previously, when a hook hadfail_on_fix = true, the step's auto-staging would silently merge the fixer's output into the index over your explicitgit addchoices. After the failed commit, the fix was no longer visible as an unstaged change for review, and a re-commit would silently succeed with the fix baked in — defeating the entire point offail_on_fix.should_stageis now forced off forRunType::Fixruns whenfail_on_fixis set, so the fixer's output stays in the worktree as an unstaged change for you to inspect, and the commit keeps failing until you accept it. Fixes #888. -
Text-mode progress output is usable in CI again (@jdx) #890. hk's output in GitHub Actions and other piped-stderr environments was a mess: raw
[9A[80D[0Jcursor-control escapes leaked into the log, every status change was duplicated, failure stderr was suppressed, and a step matching hundreds of files dumped ~4KB of paths into every progress line. This release fixes the lot:- Bumps
clxto 2.0.1, which makesrefresh_once()a no-op in text mode (no more leaked UI escape codes) and dedupes consecutive identical job lines per job. - Failure summaries are now emitted in text mode by default. Successful steps stay quiet (their output already streamed during execution), but failed steps get a full diagnostic block at the end so you can see the failure in one place.
HK_SUMMARY_TEXT=1still forces every step's summary to print. - Per-step progress messages are bounded. A new display-only tera context truncates
files/workspace_filestofirst_file …when more than one file matches, and the rendered message itself is capped at 2048 printable chars (ANSI-aware). The execution command is rendered against the full file list as before — only the human-readable progress line is truncated. - Stops truncating text-mode messages at 60 chars. The previous
truncate_textfilter clamped toterm_width - 20, which is 60 in non-TTY environments — exactly enough to hide the diagnostic detail you actually need to debug a CI failure.
A typical
dbgstep matching 98.rsfiles now reads:dbg – 98 files – **/*.rs – ! rg -e 'dbg!' bin/generate_docs.rs …instead of unrolling all 98 paths on every prop update.
- Bumps
Full Changelog: v1.44.2...v1.44.3
💚 Sponsor hk
hk is developed by @jdx at en.dev — a small independent studio behind developer tools like mise, aube, hk, and more. Work on hk is funded by sponsorships.
If hk has sped up your pre-commit loop or made linting feel less painful, please consider sponsoring at en.dev. Sponsorships are what keep hk moving and the project independent.
v1.44.2: pklr cache freshness and quieter Builtins
A small patch release focused on two HK_PKL_BACKEND=pklr rough edges: edits to hk.pkl are now picked up immediately, and loading Builtins.pkl no longer spams deprecation warnings on every run.
Fixed
-
hk.pkledits are now picked up underHK_PKL_BACKEND=pklr(@jdx) #879. The two pkl backends return different things fromanalyze_imports— thepklCLI happens to include the source file inresolvedImports, butpklronly returns transitiveimportURIs. As a result, withpklrthe mainhk.pklwas missing from the config cache'sfresh_files, so edits didn't invalidate the cache andhkkept reusing the staleConfiguntil you ranhk cache clear. The main config path is now always added tofresh_files. Fixes #877. -
No more
pklrdeprecation warnings on everyBuiltins.pklload (@jdx) #880. Previously, every invocation underHK_PKL_BACKEND=pklrprinted:[pklr] WARNING: property 'check_byte_order_marker' is deprecated [pklr] WARNING: property 'fix_byte_order_marker' is deprecated…even when your
hk.pkldidn't reference those aliases. This release bumpspklrto 0.4.2 (which evaluates@Deprecatedlazily, on field access) and reworksBuiltins.pklso its own internal bindings no longer touch the deprecated aliases at load time. The migration nudge still fires if you explicitly referenceBuiltins.check_byte_order_markerorBuiltins.fix_byte_order_marker. Fixes #878. -
Mobile docs banner layout (@jdx) #865, #867. At
<=640px, the banner now stacks the message and "Read more" link vertically, with the close button pinned to the top-right corner instead of floating in the middle of the taller stacked layout.
Documentation
- The VitePress site nav now surfaces the current release version (parsed from
Cargo.toml) and a GitHub star counter, matching the mise and aube docs (@jdx) #872.
Full Changelog: v1.44.1...v1.44.2
💚 Sponsor hk
hk is developed by @jdx at en.dev — a small independent studio behind developer tools like mise, aube, hk, and more. Work on hk is funded by sponsorships.
If hk has sped up your pre-commit loop or made linting feel less painful, please consider sponsoring at en.dev. Sponsorships are what keep hk moving and the project independent.
v1.44.1: post-commit / pre-rebase and faster YADM-style worktrees
A small patch release fixing two rough edges introduced with the v1.44.0 global-hooks work: post-commit and pre-rebase now have proper hk run subcommands, and HK_STASH_UNTRACKED=false finally skips the untracked-file scan (not just the stash), which makes hk usable on YADM-style dotfile repos where GIT_WORK_TREE is $HOME.
Fixed
-
hk run post-commitandhk run pre-rebaseare now first-class subcommands (@jdx) #858. Both events are written byhk install, but previously fell through to the genericotherhandler — so they didn't show up inhk run --helpand their arguments got mixed into the positional file collector, occasionally producing confusingUsage: hk run --from-ref <FROM_REF> [FILES]...errors duringgit rebase.pre-rebasenow has a typed<upstream> [branch]signature matching git's spec, andpost-commitis a proper no-args handler. -
HK_STASH_UNTRACKED=falsenow also skips the untracked scan ingit status(@jdx) #861. Before this, the flag only suppressed stashing — hk still rangit status --untracked-files=allon every invocation, which could take tens of seconds and emit hundreds of megabytes of output whenGIT_WORK_TREEpoints at a large directory like$HOME(as in YADM). Both the libgit2 and shell-git code paths now honor the setting, so large-worktree users can opt out of the scan entirely. Fixes #860.export HK_STASH_UNTRACKED=false hk check --all # no longer scans the entire worktree for untracked files
Documentation
- Getting-started docs now lead with
hk install --globalas the recommended setup path, since the--from-hookshort-circuit added in v1.44.0 makes it safe to enable once per machine (@jdx) #855. - Added a dismissible cross-site announcement banner to hk.jdx.dev, with an optional
expiresfield so banners auto-hide on their own (@jdx) #857, #862.
Full Changelog: v1.44.0...v1.44.1
💚 Sponsor hk
hk is developed by @jdx at en.dev — a small independent studio behind developer tools like mise, aube, hk, and more. Work on hk is funded by sponsorships.
If hk has sped up your pre-commit loop or made linting feel less painful, please consider sponsoring at en.dev. Sponsorships are what keep hk moving and the project independent.
v1.44.0: Install Globally, Plan Before You Run
This release is all about understanding and controlling where hk runs. hk check --plan lets you dry-run a hook and see exactly which steps would execute and why, hk install --global registers hk against every repo on your machine using Git 2.54's new config-based hooks, and bare-repo dotfile managers like YADM are now supported via GIT_DIR/GIT_WORK_TREE.
Highlights
hk check --plan/--why/--json— dry-run any hook to see which steps would run, which would skip, and why, with JSON output for toolinghk install --global— install hooks once in~/.gitconfigand have hk apply to every repo (Git 2.54+)- Bare-repo dotfile support — hk now respects
GIT_DIRandGIT_WORK_TREE, so YADM and similar setups work out of the box - New
cocogitto_commit_msgbuiltin for Conventional Commits validation
Added
-
hk check --plan/-P,--why [STEP]/-W,--json/-J: You can now dry-run a hook to see what hk would do without executing any commands.--planprints the parallel groups, matched file counts, and included/skipped steps with reasons;--whydrills into the skip reasons for every step (or a specific one);--jsonemits the plan as structured JSON for tooling. (@jdx) #848$ hk check --plan Plan: check Run type: check [parallel group] group_0 ○ actionlint (no files matched filters) ✓ cargo-fmt (6 files matched) ○ cargo-clippy (required profile(s) not enabled: slow) ✓ cargo-check (6 files matched)
The planner reuses hk's real job-building and skip-evaluation logic, so the plan accurately reflects what would happen — including filter matches, profile gating,
conditionevaluation,dependsOn, and--step/--skip-stepselections. It never executes step commands. -
Git 2.54 config-based hook installation with
--global: On Git 2.54+,hk installnow writes config-based hooks (hook.hk-<event>.command/.event) instead of shell shims in.git/hooks/. The hooks directory is left untouched, and hk composes cleanly with other hook managers. Use--legacyto force the old shim behavior; older Git falls back automatically. (@jdx) #853More importantly,
hk install --globalwrites those entries to your~/.gitconfigso hk runs in every repository on your machine:$ hk install --globalIn repos without an
hk.pkl(or without a matching event), the invocation is a silent no-op via a new hiddenhk run --from-hookflag — install once, forget, and repos that don't use hk are unaffected.hk uninstallnow cleans up both script shims and config entries regardless of current Git version, andhk uninstall --globalremoves the global entries. -
GIT_DIR/GIT_WORK_TREEsupport for bare-repo dotfile managers: hk now honors these environment variables during repository discovery, so it works with YADM and similar bare-repo dotfile setups where there is no.gitdirectory in the work tree. When libgit2 opens a bare repo, hk falls back to shellgitfor status/diff operations (libgit2 refuses those on bare repos). As a bonus,hk builtinsno longer loads project settings, so it runs outside a repo instead of panicking. Fixes #831. (@jdx) #847 -
cocogitto_commit_msgbuiltin: A new builtin linter that validates commit messages against the Conventional Commits spec using cocogitto'scog verify. Uses the{{commit_msg_file}}template variable, making it a drop-in for thecommit-msghook. (@hituzi-no-sippo) #838
Fixed
- Text progress in CI: Some CI systems allocate a pseudo-TTY, which made
console::user_attended_stderr()report an interactive stderr while the log collector stripped cursor-control escapes and recorded spinner frames as noisy log rows. hk now detects CI environments viais_ciand forces clx progress into text mode, while leaving local interactive behavior unchanged. (@jdx) #845
Changed
Full Changelog: v1.43.0...v1.44.0
💚 Sponsor hk
hk is developed by @jdx at en.dev — a small independent studio behind developer tools like mise, aube, hk, and more. Work on hk is funded by sponsorships.
If hk has sped up your pre-commit loop or made linting feel less painful, please consider sponsoring at en.dev. Sponsorships are what keep hk moving and the project independent.
v1.43.0: Stdin forwarding, harper builtin, and musl binaries
This release adds {{ hook_stdin }} for forwarding git hook stdin to step commands (completing git-lfs support started in v1.42.0), introduces a built-in harper grammar checker, and ships Linux musl binaries for Alpine and other musl-based distributions.
Highlights
{{ hook_stdin }}template variable completes git-lfs pre-push support -- LFS objects are now properly uploaded duringgit push- harper-cli builtin adds grammar checking as a first-class linter
- Linux musl release binaries for Alpine and other musl-based distros
- hk is back on crates.io -- installable via
cargo install hkagain
Added
-
{{ hook_stdin }}template variable: Step commands can now receive the raw stdin that git passes to hook scripts via thestdinfield. This is essential forgit lfs pre-push, which needs the ref data piped through stdin to know which LFS objects to upload. Without this,git lfs pre-pushwould silently succeed but upload nothing, causing the remote to reject the push. The variable is available inpre-pushandpost-rewritehooks. (@JohanLorenzo) #825hooks { ["pre-push"] { steps { ["git-lfs"] { check = "git lfs pre-push {{ hook_args }}" stdin = "{{ hook_stdin }}" } } } }
For
pre-push,{{ hook_stdin }}contains the ref lines that git pipes in (e.g.,refs/heads/main <local-sha> refs/heads/main <remote-sha>). Forpost-rewrite, it contains the old/new SHA mapping lines. When stdin is a terminal (no piped data), it expands to an empty string. -
Built-in
harperandharper_commit_messagelinter steps: harper-cli is now available as a builtin linter for grammar checking prose and documentation. Theharperstep runs against text files, whileharper_commit_messagechecks commit messages. (@hituzi-no-sippo) #714 -
Linux musl release binaries: Pre-built binaries for
x86_64-unknown-linux-muslandaarch64-unknown-linux-muslare now included in releases, making hk easy to install on Alpine Linux and other musl-based distributions. (@jdx) #829 -
hk is published to crates.io again: The crate had been stuck at v1.10.1 since August 2025 after the publish step was accidentally dropped during a build system migration. Starting with this release,
cargo install hkwill get the latest version. (@jdx) #830
New Contributors
- @hituzi-no-sippo made their first contribution in #714
Full Changelog: v1.42.0...v1.43.0
v1.42.0: Hook args template and Windows quoting fix
This release adds a new {{ hook_args }} template variable for forwarding git hook arguments to downstream commands, and fixes a Windows-specific bug where {{files}} expansion silently broke file-based checks.
Added
-
{{ hook_args }}template variable: Step commands can now access the arguments that git passes to hook scripts via{{ hook_args }}. This is essential for tools like git-lfs, whose hooks (post-checkout,post-merge,pre-push) require the original positional arguments from git to function correctly. Without this, commands likegit lfs post-checkoutwould fail with "This should be run through Git's post-checkout hook." (@JohanLorenzo) #807hooks { ["post-checkout"] { steps { ["git-lfs"] { check = "git lfs post-checkout {{ hook_args }}" } } } ["post-merge"] { steps { ["git-lfs"] { check = "git lfs post-merge {{ hook_args }}" } } } ["pre-push"] { steps { ["git-lfs"] { check = "git lfs pre-push {{ hook_args }}" } } } }
The variable is populated for all hook types:
pre-pushgets<remote-name> <remote-url>,commit-msggets the message file path,post-checkoutgets<prev-head> <new-head> <is-branch>, and so on. For hooks that receive no arguments (likepre-commit), it expands to an empty string. -
First-class
post-checkout,post-merge, andpost-rewritehooks: These three hook types now have dedicated subcommands (hk run post-checkout,hk run post-merge,hk run post-rewrite) with proper argument parsing, rather than being handled as generic hooks. (@JohanLorenzo) #807
Fixed
{{files}}expansion on Windows no longer silently breaks checks: On Windows, Rust'sCommand::argapplies MSVCRT-style argv escaping that collides withcmd.exe's own quoting rules. This caused the already-quoted{{files}}payload to reach tools with literal"characters embedded in arguments. Tools like ruff, biome, and others would silently exit 0 while processing zero files, making hk report success on broken invocations. The fix switches the Windowscmd.exe /ccode path to useraw_arg, passing the rendered command string verbatim so cmd.exe can parse its own quoting without Rust interference. This also affects{{workspace_files}}. (@jdx) #824
New Contributors
- @JohanLorenzo made their first contribution in #807
Full Changelog: v1.41.1...v1.42.0
v1.41.1: Cleaner hook failure output
A patch release focused on fixing hook failure output. Previously, failing steps could produce duplicated or missing diagnostic output depending on the combination of output_summary, check_first, and fail_fast settings. These fixes ensure that failure output is shown exactly once, includes both stdout and stderr so no diagnostics are lost, and preserves the configured summary label.
Fixed
-
No more duplicate output on failure: The end-of-run error handler was reprinting the first failing step's output after the per-step summary had already displayed it. This caused confusing duplication and misattribution -- for example, one tool's errors could appear to be part of another tool's section. The redundant
handle_script_failedoutput has been removed. (@nkakouros) #784 -
Combined output shown for failed steps: When a step fails, hk now shows combined stdout+stderr output regardless of the
output_summarysetting (unless set to"hide"). Previously, ifoutput_summarywas set to"stderr"but the tool wrote diagnostics to stdout (as eslint, flake8, prettier, and many others do), those diagnostics were invisible in the failure summary. (@nkakouros) #772 -
Configured output summary label preserved on failure: The combined-output-on-failure fix from #772 was changing the summary header from the configured label (e.g.,
lint stderr:) tolint combined:. The label now matches the configuredoutput_summaryvalue while still using combined content underneath. (@jdx) #808 -
check_firstdiagnostics preserved when cancelled byfail_fast: For steps usingcheck_difforcheck_list_fileswithcheck_first, diagnostic output from the check phase was lost if another step failed first and triggeredfail_fastcancellation before the fix phase could run. The check output is now saved so it appears in the summary. (@nkakouros) #784
New Contributors
Full Changelog: v1.41.0...v1.41.1
v1.41.0: Per-worktree hooks, XDG config path fix, and Go multi-package fixes
This release adds support for per-worktree git hook configurations, fixes the global config path to consistently use XDG_CONFIG_HOME across all platforms, and resolves a common issue where Go package-level analysis tools would fail in multi-package projects.
Added
-
Per-worktree hook support: When
extensions.worktreeConfigis enabled and a per-worktreecore.hooksPathis set,hk installandhk uninstallnow respect that worktree-local directory instead of always using the shared hooks directory. This allows different worktrees to have independent hook configurations without conflicting with each other. (@nkakouros) #789# Enable per-worktree config in the main repo git config extensions.worktreeConfig true # In each worktree, point hooks to a worktree-local directory cd /path/to/worktree git config --worktree core.hooksPath "$(git rev-parse --git-dir)/hooks" hk install
Fixed
-
Config path now uses
XDG_CONFIG_HOMEon all platforms: Previously, hk useddirs::config_dir()which resolves to~/Library/Application Support/on macOS, meaning the documented path~/.config/hk/config.pklwould not be found. hk now uses$XDG_CONFIG_HOME(defaulting to~/.config) on all platforms, consistent with mise and matching what the documentation describes. (@fukuchancat) #801 -
Go package-level builtins work in multi-package projects: Seven Go analysis builtins (
go_vet,golangci_lint,go_sec,go_vuln_check,staticcheck,revive,err_check) now useworkspace_indicator = "go.mod"and run./...from the module root instead of passing individual file paths. This fixes the "named files must all be in one directory" error that occurred when staged.gofiles spanned multiple packages -- a Go toolchain constraint that affects any tool built ongo/packages. File-level formatters (go_fmt,go_fumpt,go_imports,go_lines) are unchanged. (@jdx) #803
Breaking Changes
- macOS config path changed: On macOS, hk now looks for global config at
~/.config/hk/config.pklinstead of~/Library/Application Support/hk/config.pkl. If you had a config file at the old macOS-specific path, move it to~/.config/hk/config.pkl(or setHK_CONFIG_DIRto point to your preferred location). This was never documented or intended behavior.
New Contributors
- @fukuchancat made their first contribution in #801
Full Changelog: v1.40.0...v1.41.0
v1.40.0: pklr backend, required env fields, and five new builtins
This release introduces an experimental pure-Rust Pkl evaluator (pklr) as an opt-in backend, adds a required field for steps that need specific environment variables, and expands the builtins library with five new linter/scanner configurations. It also fixes an important staging bug where hk fix could inadvertently stage pre-existing untracked files.
Highlights
- Experimental pklr backend: Set
HK_PKL_BACKEND=pklrto evaluate.pklconfig files using a built-in Rust evaluator instead of shelling out to thepklCLI. This eliminates the pkl binary dependency entirely. Proxy, CA certificate, and HTTP rewrite settings are forwarded automatically. Note: pklr is experimental and may not support every pkl feature yet. requiredfield for steps: Steps can now declare environment variables that must be present for the step to run. If any are missing, the step is gracefully skipped with a clear message. This is designed for builtins likeaddlicensewhere running without user-provided configuration would produce incorrect results.- Five new builtins:
google_java_format,dclint,gitleaks,betterleaks, andmdschemajoin the built-in linter registry. - Staging correctness fix:
hk fixno longer stages untracked files that existed before the hook ran -- only files newly created by fixers are staged.
Added
-
pklrPkl backend: A pure-Rust Pkl evaluator is now available as an opt-in alternative to the externalpklCLI. Enable it withHK_PKL_BACKEND=pklr. Supports proxy settings, custom CA certificates viaHK_PKL_CA_CERTIFICATES, and HTTP rewrites viaHK_PKL_HTTP_REWRITE. (@jdx) #768, #769export HK_PKL_BACKEND=pklr hk run check # no pkl CLI needed
-
requiredfield on steps: Declare environment variables that must be set for a step to run. If any are missing, the step is skipped with a message likeskipped: missing required environment variable(s): LICENSE_FILE. Variables can be satisfied by the process environment, the globalenvblock, or the step's ownenvblock. (@timothysparg) #785["addlicense"] { required = List("LICENSE_FILE") check = "addlicense --check -f $LICENSE_FILE {{files}}" fix = "addlicense -f $LICENSE_FILE {{files}}" }
To see skip messages for missing required env vars, add
"missing-required-env"todisplay_skip_reasons. -
google_java_formatbuiltin: Format Java files using google-java-format. Matches**/*.java. (@timothysparg) #777 -
dclintbuiltin: Lint and fix Docker Compose files using dclint. Auto-detected viacompose.yml,docker-compose.yml, and variants. (@timothysparg) #779 -
gitleaksbuiltin: Scan for secrets in Git repositories using gitleaks. Auto-detected via.gitleaks.toml. (@hituzi-no-sippo) #749 -
betterleaksbuiltin: Scan for secrets using betterleaks. Auto-detected via.gitleaks.tomlor.betterleaks.toml. (@hituzi-no-sippo) #750 -
mdschemabuiltin: Validate Markdown documents against schemas using mdschema. Auto-detected via.mdschema.yml. (@hituzi-no-sippo) #748
Changed
-
excludedefaults toList(): Theexcludefield on steps now defaults to an empty list instead ofnull. This means you no longer need the null-coalescing operator when extending exclusions in Pkl configs. (@timothysparg) #781// Before (required null check) exclude = (Builtins.actionlint.exclude ?? List()) + List("**/ignored-dir/**") // After (just concatenate) exclude = Builtins.actionlint.exclude + List("**/ignored-dir/**")
Fixed
-
Pre-existing untracked files no longer staged by
hk fix: When using the defaultstage=<JOB_FILES>behavior, untracked files that existed before the hook started are no longer added to the git index. Only files newly created by fixers during the run are staged. Explicitstageglobs still opt into staging all matching untracked files. (@jdx) #788 -
Pkl package URIs use correct versioned format: Documentation and error messages now use the correct versioned Pkl package URI format (
/releases/download/vX.Y.Z/[email protected]) instead of invalid/latest/paths. Error messages also now dynamically reflect the running hk version. (@jdx) #770
New Contributors
- @timothysparg made their first contribution in #777
Full Changelog: v1.39.0...v1.40.0