Dotfiles experimental
mise can manage dotfiles from the [dotfiles] section of mise.toml. Entries can either own a whole file or directory, or manage one small piece of a file something else owns.
[settings]
dotfiles.root = "~/.dotfiles"
dotfiles.default_mode = "symlink"
[dotfiles]
"~/.zshrc" = {} # ~/.dotfiles/.zshrc
"~/.gitconfig" = "dotfiles/gitconfig" # explicit source
"~/.config/alacritty.toml" = { mode = "copy" } # ~/.dotfiles/.config/alacritty.toml
"~/.config/starship.toml" = { source = "dotfiles/starship.toml", mode = "copy" }
"~/.ssh/config" = { source = "dotfiles/ssh_config.tmpl", mode = "template" }
"~/.config/nvim" = "dotfiles/nvim" # symlink the directory itself
"~/.local/bin" = { source = "dotfiles/bin", mode = "symlink-each" } # symlink each file within
"~/hosts/dev" = { line = "127.0.0.1 dev.local" } # edit one line in ~/hostsDotfiles are only applied when explicitly requested with mise dotfiles apply or mise bootstrap. They are never applied implicitly by mise install or mise bootstrap packages.
Whole-file entries
Whole-file entries are keyed by the target path — absolute or starting with ~/ — and may point at a source file or directory. If source is omitted, mise mirrors the home-relative target path under dotfiles.root: ~/.zshrc uses ~/.dotfiles/.zshrc, and ~/.config/foo.toml uses ~/.dotfiles/.config/foo.toml. Targets outside $HOME must specify source.
String entries are shorthand for an explicit source with dotfiles.default_mode. Commands that write [dotfiles] always write table form with mode, even when it is the default:
[dotfiles]
"~/.zshrc" = { mode = "symlink" }
"~/.ssh/config" = { source = "ssh/config", mode = "copy" }Relative explicit sources resolve against the directory of the config file that declares the entry, so a global ~/.config/mise/config.toml can manage dotfiles kept next to it, and a project config can ship machine setup from the repo.
Source paths may contain glob wildcards like *, **, ?, or [ab]. When a wildcard source matches multiple paths, the target path must contain matching wildcards so each source expands to a unique target:
[dotfiles]
"~/.config/*.toml" = "dotfiles/config/*.toml"
"~/.local/share/app/**/*.json" = { source = "dotfiles/app/**/*.json", mode = "copy" }
"~/.config/app?.toml" = "dotfiles/config/app?.toml"
"~/.config/theme-[ab].toml" = "dotfiles/config/theme-[ab].toml"Modes
| Mode | Behavior |
|---|---|
symlink | Symlink the target to the source. Works for files and directories — a directory source gets one link for the whole directory. This is the default. |
symlink-each | Source must be a directory: recreate its directory structure under the target and symlink each file individually, so the target directory (say, ~/.config) can also hold files mise doesn't manage. |
copy | Copy the source file (or directory, recursively). Use when the target must be a real file — e.g. tools that rewrite their config in place. Directory copies are additive: matching files are overwritten, files mise doesn't manage are left in place. |
template | Render the source through the mise template engine and write the result. Permissions are taken from the source file (and repaired if they drift). |
Templates get the same context as other mise templates (env, vars, exec(), etc.), which is the main reason to use them: one source file, per-machine output.
Detecting whether a template's output has drifted requires rendering it, so mise dotfiles status and a real apply evaluate templates — including any exec() calls — from your trusted config, just like [env] templates. --dry-run is the exception: it promises to execute nothing, so it skips template rendering and lists those entries as (if changed).
Edit entries
Edit entries manage one piece of a file: the mise activate block in your shell rc, an entry in /etc/hosts, or a small snippet in a config file. They are keyed by target path plus an id naming each edit within the file:
[dotfiles]
"~/.zshrc/activate" = { block = 'eval "$(mise activate zsh)"' }
"~/.zshrc/aliases" = { block = '''
alias ll='ls -l'
alias la='ls -la'
''' }
"/etc/hosts/dev" = { line = "127.0.0.1 dev.local" }
"~/.gitconfig/identity" = { source = "snippets/git-identity.tmpl", template = "tera" }For edit entries, source is paired with template = "tera" to make the entry unambiguously an edit. A table with only source is a whole-file entry using dotfiles.default_mode.
A block is delimited by marker comments in the target file, named by the entry's id:
# >>> mise:activate >>> managed by mise - do not edit between markers
eval "$(mise activate zsh)"
# <<< mise:activate <<<The markers are the ownership record, stored in the file itself, so the design stays stateless: applying replaces only what's between them or appends the block if absent, and everything else in the file is untouched.
Ids may contain letters, digits, _, -, and .. The marker comment prefix is inferred from the file extension (# for shell/config files, -- for Lua, // for C-like languages, ; for INI, " for vim) and can be overridden with comment = "...". Files that can't hold line comments at all (strict JSON, XML) aren't a fit for blocks — use a whole-file entry instead.
A line ensures an exact line exists somewhere in the file, appending it at the end if absent. It never modifies or removes other lines, which is what makes it safely idempotent. The value must be a single line; use a block for multi-line content.
Semantics
- Declarative and additive — entries merge across the config hierarchy (global → project). Whole-file entries merge by target path; edit entries merge by
(path, id). - Manual application only — nothing is written implicitly. Only
mise dotfiles applyormise bootstrapapplies dotfiles. - Idempotent — entries already in their desired state are skipped; re-running is always safe.
- Unknown modes and operations are ignored with a warning so configs using features from newer mise versions still parse.
Conflicts
mise refuses to replace existing files it doesn't manage: a real file or directory where a symlink should go, or a directory where a file should go, is an error listing the conflicting paths. Pass mise dotfiles apply --force to replace them.
Content updates are not conflicts: a copy or template entry overwrites the target file's content without --force — that is the declared intent of those modes. Symlinks are re-pointed freely, since a symlink is never data.
Edit entries never need --force: a block owns only what's between its markers, and a line only ever appends. Two cases are refused with an error instead of guessed at: corrupted markers and targets that are symlinks. An edit through a symlink would modify whatever the link points at, often a [dotfiles] source, so point the edit at the real file instead.
Removing an entry from config leaves its file, block, or line in place because mise keeps no state database. Delete unmanaged leftovers by hand.
Commands
mise dotfiles status # shows applied/missing/differs/source missing
mise dotfiles status --missing # exit 1 if anything is out of sync
mise dotfiles apply # apply files and edits
mise dotfiles apply --dry-run # print what would be done
mise dotfiles apply --dry-run --verbose # include diff-like details
mise dotfiles apply --yes # skip the confirmation prompt
mise dotfiles apply --force # also replace conflicting files
mise dotfiles add ~/.zshrc # capture a live file into dotfiles.root
mise dotfiles edit ~/.zshrc # edit the managed source or owning config
mise dotfiles edit --apply ~/.zshrcmise dotfiles status reports each entry as applied, missing, differs with a reason, or source missing.
Capturing changes
If you edit a copied dotfile in place and want to store those changes back in your dotfiles, run mise dotfiles add again:
$EDITOR ~/.config/starship.toml
mise dotfiles add ~/.config/starship.tomlFor an unmanaged target, add creates a [dotfiles] entry and seeds the source under dotfiles.root. For an already-managed target, it updates the existing source from the live target.
Self-managing mise config
You can manage the mise config and the dotfiles root as dotfiles too:
[settings]
dotfiles.root = "~/.dotfiles"
[dotfiles]
"~/.dotfiles" = "~/src/dotfiles"
"~/.config/mise/config.toml" = "~/.dotfiles/mise/config.toml"This is a bootstrap pattern: clone the real repo (for example ~/src/dotfiles) before the first mise dotfiles apply or mise bootstrap. Replacing ~/.config/mise/config.toml affects future mise invocations, so make sure the source contains a valid config before applying it.
Root-owned files
Dotfiles write as the current user — there is no sudo here. Managing /etc/hosts works when running as root (containers, CI); otherwise mise fails with an ordinary permission error.
Windows
File symlinks require elevation on Windows, so symlink and symlink-each fall back to copying for files there; directory symlinks use junctions.