Skip to content

Task Arguments

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.

1. Usage Field (Preferred)

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.

Quick Example

mise.toml
mise-toml
[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_:

shell
# 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"

Help output example:

shellsession
$ 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

Complete Usage Specification Reference

Positional Arguments (arg)

Positional arguments are defined with arg and must be provided in order.

Basic Syntax

kdl
arg "<name>" help="Description"               // Required positional arg
arg "[name]" help="Description"               // Optional positional arg
arg "<file>"                                  // Completed as filename
arg "<dir>"                                   // Completed as directory

With Defaults

kdl
arg "<file>" default="config.toml"            // Default value if not provided
arg "[output]" default="out.txt"              // Optional with default

Variadic Arguments

kdl
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

Environment Variable Backing

kdl
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

Choices (Enum Values)

kdl
arg "<level>" {
  choices "debug" "info" "warn" "error"
}
arg "<shell>" {
  choices "bash" "zsh" "fish"
  help "Shell type"
}

Advanced Features

kdl
arg "<file>" long_help="Extended help text shown with --help"

// Hidden from help output
arg "<file>" hide=#true

// Parse value with external command
arg "<input>" parse="mycli parse-input {}"

Double-Dash Behavior

kdl
// 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"

Flags (flag)

Flags can be defined as booleans or as accepting values.

Boolean Flags

kdl
flag "-f --force"
flag "-v --verbose" help="Enable verbose mode"
flag "--dry-run" help="Preview without executing"

Short-Only or Long-Only

kdl
flag "-f"                                     // Short flag only
flag "--force"                                // Long flag only

Flag With Values

kdl
flag "-o --output <file>" help="Output file"
flag "--port <port>" help="Server port"
flag "--color <when>" {
  choices "auto" "always" "never"
}

Flag With Defaults

kdl
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"
}

Count Flags

kdl
// Can be repeated: -vvv
// $usage_verbose = number of times used (e.g., 3)
flag "-v --verbose" count=#true

Negation

kdl
flag "--color" negate="--no-color" default=#true
// Default: $usage_color = "true"
// With --no-color: $usage_color = "false"

Global Flags

kdl
// Available on all subcommands (if using cmd structure)
flag "-v --verbose" global=#true

Environment Variable and Config Backing

kdl
flag "--color" env="MYCLI_COLOR"              // Can be set via $MYCLI_COLOR
flag "--format <fmt>" config="ui.format"      // Backed by config file value
flag "--port <port>" env="PORT"
flag "--debug" env="DEBUG"

Priority order: CLI flag > Environment variable > Config file > Default value

Conditional Requirements

kdl
// If --output is set, --file must be too
flag "--file <file>" required_if="--output"

// Either --file or --stdin must be set
flag "--file <file>" required_unless="--stdin"

// If --file is set, --stdin is ignored
flag "--file <file>" overrides="--stdin"

Flag Advanced Features

kdl
flag "--verbose" long_help="Extended help text"
flag "--debug" hide=#true                      // Hidden from help
flag "-q --quiet" {
  help "Suppress output"
  alias "--silent"                            // Alternative name
}

Completion (complete)

Custom completion can be defined for any argument or flag by name:

kdl
arg "<plugin>"
complete "plugin" run="mise plugins ls"       // Complete with command output

With Descriptions

kdl
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

Long Help Text

For detailed help text, use multi-line format:

mise-toml
[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

Hide arguments from help output (useful for deprecated or internal options):

kdl
arg "<legacy_arg>" hide=#true
flag "--internal-debug" hide=#true

Combining Features Example

mise.toml
mise-toml
[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"
  required_if "--skip-tests"
}

// 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
'''

2. File Task Headers

For file tasks, you can define arguments directly in the file using special #MISE or #USAGE comment syntax:

.mise/tasks/deploy
bash
#!/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

Syntax Options

Use #MISE (uppercase, recommended) or #USAGE for defining arguments in file tasks. # [MISE] or # [USAGE] are also accepted as workarounds for formatters.

Bash Variable Expansion for Usage Variables

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.

Common Patterns

SyntaxBehaviorUse CaseExample
${var?}Error if unsetRequired args or flags with defaults in usage spec${usage_profile?}
${var:?}Error if unset or emptyWhen you need to ensure non-empty values${usage_target:?}
${var:-default}Use default if unsetBoolean flags without default= in usage spec${usage_clean:-false}
${var:=default}Set and use default if unsetWhen you want to set the variable for later use${usage_dir:=.}
${var:+value}Use value if setConditional flag passing${usage_verbose:+--verbose}

Guidelines for Usage Variables

Args and Flags with Defaults

Use ${usage_var?} since usage guarantees they'll be set:

bash
# --profile has default="debug" in usage spec
cargo build --profile "${usage_profile?}"

Boolean Flags without Defaults

Use ${usage_var:-false} to provide a default value:

bash
# --clean flag has no default in usage spec
if [ "${usage_clean:-false}" = "true" ]; then
  cargo clean
fi

Required Arguments

Use ${usage_var:?} to ensure non-empty values:

bash
# <target> is a required positional argument
cargo build --target "${usage_target:?}"

Conditional Flags

Use ${usage_var:+value} to pass flags only when set:

bash
# 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.

Deprecated Method

Tera Template Functions deprecated

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:

  • Two-pass parsing issues: Template functions return empty strings during spec collection, causing unexpected behavior when trying to use them as normal template values
  • Complex escaping rules: Shell escaping rules are confusing and error-prone
  • Inconsistent behavior: Doesn't work the same way between TOML and file tasks
  • Poor user experience: Mixes argument definitions with script logic

Migration required: Please migrate to the usage field method before 2026.11.0.

Click to see deprecated Tera template syntax (not recommended)

Previously, you could define arguments inline in run scripts using Tera template functions:

mise.toml
mise-toml
# ❌ DEPRECATED - Do not use
[tasks.test]
run = 'cargo test {{arg(name="file", default="all")}}'
mise.toml
mise-toml
# ❌ DEPRECATED - Do not use
[tasks.build]
run = [
    'cargo build {{option(name="profile", default="dev")}}',
    './scripts/package.sh {{flag(name="verbose")}}'
]

Problems with this approach:

  1. Empty strings during parsing: During spec collection (first pass), template functions return empty strings, so you can't use them in templates like:

    toml
    # This doesn't work as expected!
    run = 'echo "File: {{arg(name="file")}}" > {{arg(name="file")}}.log'
    # First pass: 'echo "File: " > .log' (invalid!)
  2. Escaping complexity: Different shell types require different escaping:

    toml
    # Escaping behavior varies by shell
    run = 'cmd {{arg(name="file")}}' # May or may not be properly escaped
  3. No help generation: Doesn't generate proper --help output

Migration Guide

Here's how to migrate from Tera templates to the usage field:

Example 1: Simple Arguments

Old (Deprecated):

mise-toml
[tasks.test]
run = '''
cargo test {{arg(
  name="file",
  default="all",
  help="Test file"
)}}
'''

New (Preferred):

mise-toml
[tasks.test]
usage = 'arg "<file>" help="Test file" default="all"'
run = 'cargo test ${usage_file?}'

Example 2: Multiple Arguments with Flags

Old (Deprecated):

mise-toml
[tasks.build]
run = [
  'cargo build {{arg(name="target", default="debug")}}',
  './package.sh {{flag(name="verbose")}}'
]

New (Preferred):

mise-toml
[tasks.build]
usage = '''
arg "<target>" default="debug"
flag "-v --verbose"
'''
run = [
  'cargo build ${usage_target?}',
  './package.sh ${usage_verbose?}'
]

Example 3: Options with Choices

Old (Deprecated):

mise-toml
[tasks.deploy]
run = '''
deploy {{option(
  name="env",
  choices=["dev", "prod"]
)}} {{flag(name="force")}}
'''

New (Preferred):

mise-toml
[tasks.deploy]
usage = '''
flag "--env <env>" {
  choices "dev" "prod"
}
flag "--force"
'''
run = 'deploy --env ${usage_env?} ${usage_force?}'

Example 4: Variadic Arguments

Old (Deprecated):

mise-toml
[tasks.lint]
run = 'eslint {{arg(name="files", var=true)}}'

New (Preferred):

mise-toml
[tasks.lint]
usage = 'arg "<files>" var=#true'
run = 'eslint ${usage_files?}'

See Also

Licensed under the MIT License. Maintained by @jdx and friends.