TOML-based Tasks
Tasks can be defined in mise.toml
files in different ways:
[tasks.cleancache]
run = "rm -rf .cache"
hide = true # hide this task from the list
[tasks.clean]
depends = ['cleancache']
run = "cargo clean" # runs as a shell command
[tasks.build]
description = 'Build the CLI'
run = "cargo build"
alias = 'b' # `mise run b`
[tasks.test]
description = 'Run automated tests'
# multiple commands are run in series
run = [
'cargo test',
'./scripts/test-e2e.sh',
]
dir = "{{cwd}}" # run in user's cwd, default is the project's base directory
[tasks.lint]
description = 'Lint with clippy'
env = { RUST_BACKTRACE = '1' } # env vars for the script
# you can specify a multiline script instead of individual commands
run = """
#!/usr/bin/env bash
cargo clippy
"""
[tasks.ci] # only dependencies to be run
description = 'Run CI tasks'
depends = ['build', 'lint', 'test']
[tasks.release]
description = 'Cut a new release'
file = 'scripts/release.sh' # execute an external script
Adding tasks
You can edit the mise.toml
file directly or using mise tasks add
mise task add pre-commit --depends "test" --depends "render" -- echo pre-commit
will add the following to mise.toml
:
[tasks.pre-commit]
depends = ["test", "render"]
run = "echo pre-commit"
Common options
For an exhaustive list, see task configuration.
Run command
Provide the script to run. Can be a single command or an array of commands:
[tasks.test]
run = 'cargo test'
Commands are run in series. If a command fails, the task will stop and the remaining commands will not run.
[tasks.test]
run = [
'cargo test',
'./scripts/test-e2e.sh',
]
You can specify an alternate command to run on Windows by using the run_windows
key:
[tasks.test]
run = 'cargo test'
run_windows = 'cargo test --features windows'
Specifying which directory to use
Tasks are executed with cwd
set to the directory containing mise.toml
. You can use the directory from where the task was run with dir = ""
:
[tasks.test]
run = 'cargo test'
dir = "{{cwd}}"
Also, MISE_ORIGINAL_CWD
is set to the original working directory and will be passed to the task.
Adding a description and alias
You can add a description to a task and alias for a task.
[tasks.build]
description = 'Build the CLI'
run = "cargo build"
alias = 'b' # `mise run b`
- This alias can be used to run the task
- The description will be displayed when running
mise tasks ls
ormise run
` with no arguments.
❯ mise run
Tasks
# Select a tasks to run
# > build Build the CLI
# test Run the tests
Dependencies
You can specify dependencies for a task. Dependencies are run before the task itself. If a dependency fails, the task will not run.
[tasks.build]
run = 'cargo build'
[tasks.test]
depends = ['build']
There are other ways to specify dependencies, see wait_for and depends_post
Environment variables
You can specify environment variables for a task:
[tasks.lint]
description = 'Lint with clippy'
env = { RUST_BACKTRACE = '1' } # env vars for the script
# you can specify a multiline script instead of individual commands
run = """
#!/usr/bin/env bash
cargo clippy
"""
Sources / Outputs
If you want to skip executing a task if certain files haven't changed (up-to-date), you should specify sources
and outputs
:
[tasks.build]
description = 'Build the CLI'
run = "cargo build"
sources = ['Cargo.toml', 'src/**/*.rs'] # skip running if these files haven't changed
outputs = ['target/debug/mycli']
You can use sources
alone if with mise watch
to run the task when the sources change.
Specifying a shell or an interpreter
Tasks are executed with set -e
(set -o erropt
) if the shell is sh
, bash
, or zsh
. This means that the script will exit if any command fails. You can disable this by running set +e
in the script.
[tasks.echo]
run = '''
set +e
cd /nonexistent
echo "This will not fail the task"
'''
You can specify a shell
command to run the script with (default is sh -c
or cmd /c
):
[tasks.lint]
shell = 'bash -c'
run = "cargo clippy"
or use a shebang:
[tasks.lint]
run = """
#!/usr/bin/env bash
cargo clippy
"""
By using a shebang
(or shell
), you can run tasks in different languages (e.g., Python, Node.js, Ruby, etc.):
[tools]
python = 'latest'
[tasks.python_task]
run = '''
#!/usr/bin/env python
for i in range(10):
print(i)
'''
[tools]
uv = 'latest'
[tasks.python_uv_task]
run = """
#!/usr/bin/env -S uv run --script
# /// script
# dependencies = ["requests<3", "rich"]
# ///
import requests
from rich.pretty import pprint
resp = requests.get("https://peps.python.org/api/peps.json")
data = resp.json()
pprint([(k, v["title"]) for k, v in data.items()][:10])
"""
[tools]
node = 'lts'
[tasks.node_task]
shell = 'node -e'
run = [
"console.log('First line')",
"console.log('Second line')",
]
[tools]
bun = 'latest'
[tasks.bun_shell]
description = "https://bun.sh/docs/runtime/shell"
run = """
#!/usr/bin/env bun
import { $ } from "bun";
const response = await fetch("https://example.com");
await $`cat < ${response} | wc -c`; // 1256
"""
[tools]
deno = 'latest'
[tasks.deno_task]
description = "A more complex task using Deno imports"
run = '''
#!/usr/bin/env -S deno run
import ProgressBar from "jsr:@deno-library/progress";
import { delay } from "jsr:@std/async";
if (!confirm('Start download?')) {
Deno.exit(1);
}
const progress = new ProgressBar({ title: "downloading:", total: 100 });
let completed = 0;
async function download() {
while (completed <= 100) {
await progress.render(completed++);
await delay(10);
}
}
await download();
'''
# ❯ mise run deno_task
# [download_task] $ import ProgressBar from "jsr:@deno-library/progress";
# Start download? [y/N] y
# downloading: ...
[tools]
ruby = 'latest'
[tasks.ruby_task]
run = """
#!/usr/bin/env ruby
puts 'Hello, ruby!'
"""
What's a shebang? What's the difference between #!/usr/bin/env
and #!/usr/bin/env -S
A shebang is the character sequence #!
at the beginning of a script file that tells the system which program should be used to interpret/execute the script. The env command comes from GNU Coreutils. mise
does not use env
but will behave similarly.
For example, #!/usr/bin/env python
will run the script with the Python interpreter found in the PATH
.
The -S
flag allows passing multiple arguments to the interpreter. It treats the rest of the line as a single argument string to be split.
This is useful when you need to specify interpreter flags or options. Example: #!/usr/bin/env -S python -u
will run Python with unbuffered output.
Using a file or remote script
You can specify a file to run as a task:
[tasks.release]
description = 'Cut a new release'
file = 'scripts/release.sh' # execute an external script
Remote tasks
Task files can be fetched via http:
[tasks.build]
file = "https://example.com/build.sh"
Currently, they're fetched everytime they're executed, but we may add some cache support later. This could be extended with other protocols like mentioned in this ticket if there were interest.
Arguments
By default, arguments are passed to the last script in the run
array. So if a task was defined as:
[tasks.test]
run = ['cargo test', './scripts/test-e2e.sh']
Then running mise run test foo bar
will pass foo bar
to ./scripts/test-e2e.sh
but not to cargo test
.
You can also define arguments using templates:
[tasks.test]
run = [
'cargo test {{arg(name="cargo_test_args", var=true)}}',
'./scripts/test-e2e.sh {{option(name="e2e_args")}}',
]
Then running mise run test foo bar
will pass foo bar
to cargo test
. mise run test --e2e-args baz
will pass baz
to ./scripts/test-e2e.sh
. If any arguments are defined with templates then mise will not pass the arguments to the last script in the run
array.
TIP
Using templates to define arguments will make them work with completion and help messages.
Positional Arguments
These are defined in scripts with {{arg()}}
. They are used for positional arguments where the order matters.
Example:
[tasks.test]
run = 'cargo test {{arg(name="file")}}'
# execute: mise run test my-test-file
# runs: cargo test my-test-file
i
: The index of the argument. This can be used to specify the order of arguments. Defaults to the order they're defined in the scripts.name
: The name of the argument. This is used for help/error messages.var
: Iftrue
, multiple arguments can be passed.default
: The default value if the argument is not provided.
Options
These are defined in scripts with {{option()}}
. They are used for named arguments where the order doesn't matter.
Example:
[tasks.test]
run = 'cargo test {{option(name="file")}}'
# execute: mise run test --file my-test-file
# runs: cargo test my-test-file
name
: The name of the argument. This is used for help/error messages.var
: Iftrue
, multiple values can be passed.default
: The default value if the option is not provided.
Flags
Flags are like options except they don't take values. They are defined in scripts with {{flag()}}
.
Examples:
[tasks.echo]
run = 'echo {{flag(name=("myflag")}}'
# execute: mise run echo --myflag
# runs: echo true
[tasks.maybeClean]
run = """
if [ '{{flag(name='clean')}}' = 'true' ]; then
echo 'cleaning'
fi
"""
# execute: mise run maybeClean --clean
# runs: echo cleaning
name
: The name of the flag. This is used for help/error messages.
The value will be true
if the flag is passed, and false
otherwise.
Usage spec
More advanced usage specs can be added to the task's usage
field:
[tasks.add-user]
description = "Add a user"
usage = '''
arg "<user>" default="unknown"
complete "user" run="mise run list-users-completion"
'''
run = 'echo {{arg(name="user")}}'
[tasks.list-users-completion]
hide = true
quiet = true # this is mandatory to make completion work (makes the mise command just print "alice bob charlie")
description = "List users"
run = 'echo "alice bob charlie"'