docs/tasks/task-arguments.md
Task arguments allow you to pass parameters to tasks, making them more flexible and reusable. There are three ways to define task arguments in mise, but only two are recommended for current use.
The usage field is the recommended approach for defining task arguments. It provides a clean, declarative syntax that works with both TOML tasks and file tasks.
See Complete Usage Specification Reference for more details.
[tasks.deploy]
description = "Deploy application"
usage = '''
arg "<environment>" help="Target environment" {
choices "dev" "staging" "prod"
}
flag "-v --verbose" help="Enable verbose output"
flag "--region <region>" help="AWS region" default="us-east-1" env="AWS_REGION"
'''
run = '''
echo "Deploying to ${usage_environment?} in ${usage_region?}"
[[ "${usage_verbose?}" == "true" ]] && set -x
./deploy.sh "${usage_environment?}" "${usage_region?}"
'''
Arguments defined in the usage field are automatically available as environment variables prefixed with usage_:
# Execute with arguments
$ mise run deploy staging --verbose --region us-west-2
# Inside the task, these are available as:
# $usage_environment = "staging"
# $usage_verbose = "true"
# $usage_region = "us-west-2"
In addition to environment variables, usage values are available inside Tera
templates in task run scripts via a usage map:
[tasks.deploy]
description = "Deploy application"
usage = '''
arg "<environment>" help="Target environment"
flag "-v --verbose" help="Enable verbose output"
flag "--region <region>" help="AWS region" default="us-east-1"
'''
run = '''
echo "Deploying to {{ usage.environment }} in {{ usage.region }}"
{% if usage.verbose %}
echo "Verbose mode enabled"
{% endif %}
'''
The usage map uses snake_case argument/flag names as keys (just like the
usage_ environment variables). Names with - are converted to _, so a flag
like --dry-run becomes available as <span v-pre>{{ usage.dry_run }}</span>
and $usage_dry_run. Variadic arguments/flags are exposed as arrays and can be
used with Tera's for loops and filters like length. The usage map is
separate from the deprecated Tera template functions (arg(), option(),
flag()) described later on this page—you should not mix the two approaches in
the same task.
<span v-pre>{{usage.*}}</span> templates can also be used in depends, depends_post, and
wait_for to forward arguments to dependency tasks. See
Passing parent task arguments to dependencies
for details.
Help output example:
$ mise run deploy --help
Deploy application
Usage: deploy <environment> [OPTIONS]
Arguments:
<environment> Target environment [possible values: dev, staging, prod]
Options:
-v, --verbose Enable verbose output
--region <region> AWS region [env: AWS_REGION] [default: us-east-1]
-h, --help Print help
For file tasks, you can define arguments directly in the file using special #MISE or #USAGE comment syntax:
#!/usr/bin/env bash
#MISE description "Deploy application"
#USAGE arg "<environment>" help="Deployment environment" {
#USAGE choices "dev" "staging" "prod"
#USAGE }
#USAGE flag "--dry-run" help="Preview changes without deploying"
#USAGE flag "--region <region>" help="AWS region" default="us-east-1" env="AWS_REGION"
ENVIRONMENT="${usage_environment?}"
REGION="${usage_region?}"
DRY_RUN="${usage_dry_run:-false}"
if [[ "$DRY_RUN" == "true" ]]; then
echo "DRY RUN: Would deploy to $ENVIRONMENT in $REGION"
else
echo "Deploying to $ENVIRONMENT in $REGION..."
./scripts/deploy.sh "$ENVIRONMENT" "$REGION"
fi
::: tip Syntax Options
Use #MISE (uppercase, recommended) or #USAGE for defining arguments in file tasks. # [MISE] or # [USAGE] are also accepted as workarounds for formatters.
:::
arg)Positional arguments are defined with arg and must be provided in order.
arg "<name>" help="Description" // Required positional arg
arg "[name]" help="Description" // Optional positional arg
arg "<file>" // Completed as filename
arg "<dir>" // Completed as directory
arg "<file>" default="config.toml" // Default value if not provided
arg "[output]" default="out.txt" // Optional with default
arg "[files]" var=#true // 0 or more files
arg "<files>" var=#true // 1 or more files (required)
arg "<files>" var=#true var_min=2 // At least 2 files required
arg "<files>" var=#true var_max=5 // Maximum 5 files allowed
arg "<files>" var=#true var_min=1 var_max=3 // Between 1 and 3 files
::: tip Handling Variadic Args with Spaces in Bash Variadic arguments are passed as a shell-escaped string. To properly handle arguments containing spaces as a bash array, wrap the variable in parentheses:
# Convert to bash array:
eval "files=($usage_files)"
# Use as array:
for f in "${files[@]}"; do
echo "Processing: $f"
done
# Or pass to commands:
touch "${files[@]}"
:::
arg "<token>" env="API_TOKEN" // Can be set via $API_TOKEN
arg "<host>" env="API_HOST" default="localhost"
Priority order: CLI argument > Environment variable > Default value
arg "<level>" {
choices "debug" "info" "warn" "error"
}
arg "<shell>" {
choices "bash" "zsh" "fish"
help "Shell type"
}
arg "<file>" long_help="Extended help text shown with --help"
// Hidden from help output
arg "<file>" hide=#true
// Must use: mycli -- file.txt
arg "<file>" double_dash="required"
// Both work: mycli file.txt or mycli -- file.txt
arg "<file>" double_dash="optional"
// After first arg, behaves as if -- was used
arg "<files>" double_dash="automatic"
flag)Flags can be defined as booleans or as accepting values.
flag "-f --force"
flag "-v --verbose" help="Enable verbose mode"
flag "--dry-run" help="Preview without executing"
flag "-f" // Short flag only
flag "--force" // Long flag only
flag "-o --output <file>" help="Output file"
flag "--port <port>" help="Server port"
flag "--color <when>" {
choices "auto" "always" "never"
}
flag "--force" default=#true
flag "--format <format>" help="Output format" default="json"
flag "--port <port>" help="Server port" default="8080"
flag "--color <when>" {
choices "auto" "always" "never"
default "auto"
}
// Can be repeated: -vvv
// $usage_verbose = number of times used (e.g., 3)
flag "-v --verbose" count=#true
flag "--color" negate="--no-color" default=#true
// Default: $usage_color = "true"
// With --no-color: $usage_color = "false"
// Available on all subcommands (if using cmd structure)
flag "-v --verbose" global=#true
flag "--verbose" long_help="Extended help text"
flag "--debug" hide=#true // Hidden from help
complete)Custom completion can be defined for any argument or flag by name:
arg "<plugin>"
complete "plugin" run="mise plugins ls" // Complete with command output
complete "plugin" run="mycli plugins list" descriptions=#true
Output format (split on : for value and description):
nodejs:JavaScript runtime
python:Python language
ruby:Ruby language
For detailed help text, use multi-line format:
[tasks.complex]
usage = '''
arg "<input>" {
help "Input file to process"
long_help """
The input file should be in JSON or YAML format.
Supported schemas:
- schema-v1: Legacy format
- schema-v2: Current format (recommended)
- schema-v3: Experimental format
Example:
mise run complex data.json
"""
}
flag "--format <fmt>" {
help "Output format"
long_help """
Supported output formats:
- json: JSON output (default)
- yaml: YAML output
- toml: TOML output
"""
choices "json" "yaml" "toml"
default "json"
}
'''
run = 'process-data "${usage_input?}" --format "${usage_format?}"'
Hide arguments from help output (useful for deprecated or internal options):
arg "<legacy_arg>" hide=#true
flag "--internal-debug" hide=#true
[tasks.deploy]
description = "Deploy application to cloud"
usage = '''
// Positional arguments
arg "<environment>" {
help "Deployment environment"
choices "dev" "staging" "prod"
}
arg "[services]" {
help "Services to deploy (default: all)"
var #true
var_min 0
}
// Flags
flag "-v --verbose" {
help "Enable verbose logging"
count #true
default 0
}
flag "--dry-run" help="Show what would be deployed without doing it"
flag "--region <region>" {
help "Cloud region"
env "AWS_REGION"
default "us-east-1"
choices "us-east-1" "us-west-2" "eu-west-1"
}
flag "--skip-tests" help="Skip running tests before deploy"
flag "--force" help="Force deployment even with warnings"
// Custom completions
complete "services" run="mycli list-services"
'''
run = '''
#!/usr/bin/env bash
set -euo pipefail
# Handle verbosity
if [[ "${usage_verbose?}" -ge 2 ]]; then
set -x
elif [[ "${usage_verbose?}" -ge 1 ]]; then
export VERBOSE=1
fi
# Validate environment
ENVIRONMENT="${usage_environment?}"
REGION="${usage_region?}"
DRY_RUN="${usage_dry_run:-false}"
SKIP_TESTS="${usage_skip_tests:-false}"
FORCE="${usage_force:-false}"
echo "Deploying to $ENVIRONMENT in $REGION"
# Run tests unless skipped
if [[ "$SKIP_TESTS" != "true" ]]; then
echo "Running tests..."
npm test
fi
# Deploy services
if [[ -n "${usage_services?}" ]]; then
echo "Deploying services: ${usage_services?}"
for service in ${usage_services?}; do
deploy_service "$service" "$ENVIRONMENT" "$REGION" "$DRY_RUN"
done
else
echo "Deploying all services"
deploy_all "$ENVIRONMENT" "$REGION" "$DRY_RUN"
fi
'''
When accessing usage-defined variables in bash scripts, use parameter expansion syntax to help shellcheck understand these variables and provide default values for boolean flags.
| Syntax | Behavior | Use Case | Example |
|---|---|---|---|
${var?} | Error if unset | Required args or flags with defaults in usage spec | ${usage_profile?} |
${var:?} | Error if unset or empty | When you need to ensure non-empty values | ${usage_target:?} |
${var:-default} | Use default if unset | Boolean flags without default= in usage spec | ${usage_clean:-false} |
${var:=default} | Set and use default if unset | When you want to set the variable for later use | ${usage_dir:=.} |
${var:+value} | Use value if set | Conditional flag passing | ${usage_verbose:+--verbose} |
Use ${usage_var?} since usage guarantees they'll be set:
# --profile has default="debug" in usage spec
cargo build --profile "${usage_profile?}"
Use ${usage_var:-false} to provide a default value:
# --clean flag has no default in usage spec
if [ "${usage_clean:-false}" = "true" ]; then
cargo clean
fi
Use ${usage_var:?} to ensure non-empty values:
# <target> is a required positional argument
cargo build --target "${usage_target:?}"
Use ${usage_var:+value} to pass flags only when set:
# Only add --verbose if the flag was provided
mycli deploy ${usage_verbose:+--verbose}
These expansions help shellcheck understand your script and prevent warnings about potentially unset variables while maintaining proper error handling.
::: danger Deprecated - Removal in 2026.11.0 The Tera template method for defining task arguments is deprecated and will be removed in mise 2026.11.0.
Why it's being removed:
Migration required: Please migrate to the usage field method before 2026.11.0.
Opt-out setting: If you want to disable the two-pass parsing behavior immediately (before removal), you can set:
# ~/.config/mise/config.toml
[settings]
task.disable_spec_from_run_scripts = true
Or via environment variable: MISE_TASK_DISABLE_SPEC_FROM_RUN_SCRIPTS=1
When enabled, mise will only use the usage field for spec generation, ignoring any arg(), option(), or flag() functions in run scripts. See Settings for more details.
:::
Previously, you could define arguments inline in run scripts using Tera template functions:
# ❌ DEPRECATED - Do not use
[tasks.test]
run = 'cargo test {{arg(name="file", default="all")}}'
# ❌ DEPRECATED - Do not use
[tasks.build]
run = [
'cargo build {{option(name="profile", default="dev")}}',
'./scripts/package.sh {{flag(name="verbose")}}'
]
Problems with this approach:
Empty strings during parsing: During spec collection (first pass), template functions return empty strings, so you can't use them in templates like:
# This doesn't work as expected!
run = 'echo "File: {{arg(name="file")}}" > {{arg(name="file")}}.log'
# First pass: 'echo "File: " > .log' (invalid!)
Escaping complexity: Different shell types require different escaping:
# Escaping behavior varies by shell
run = 'cmd {{arg(name="file")}}' # May or may not be properly escaped
No help generation: Doesn't generate proper --help output
Here's how to migrate from Tera templates to the usage field:
Old (Deprecated):
[tasks.test]
run = '''
cargo test {{arg(
name="file",
default="all",
help="Test file"
)}}
'''
New (Preferred):
[tasks.test]
usage = 'arg "<file>" help="Test file" default="all"'
run = 'cargo test ${usage_file?}'
Old (Deprecated):
[tasks.build]
run = [
'cargo build {{arg(name="target", default="debug")}}',
'./package.sh {{flag(name="verbose")}}'
]
New (Preferred):
[tasks.build]
usage = '''
arg "<target>" default="debug"
flag "-v --verbose"
'''
run = [
'cargo build ${usage_target?}',
'./package.sh ${usage_verbose?}'
]
Old (Deprecated):
[tasks.deploy]
run = '''
deploy {{option(
name="env",
choices=["dev", "prod"]
)}} {{flag(name="force")}}
'''
New (Preferred):
[tasks.deploy]
usage = '''
flag "--env <env>" {
choices "dev" "prod"
}
flag "--force"
'''
run = 'deploy --env ${usage_env?} ${usage_force?}'
Old (Deprecated):
[tasks.lint]
run = 'eslint {{arg(name="files", var=true)}}'
New (Preferred):
[tasks.lint]
usage = 'arg "<files>" var=#true'
run = 'eslint ${usage_files?}'
::: tip Handling Arguments with Spaces If your variadic arguments may contain spaces, convert the variable to a bash array:
[tasks.process]
usage = 'arg "<files>" var=#true'
run = '''
eval "files=($usage_files)"
for f in "${files[@]}"; do
process "$f"
done
'''
:::