--- title: Tasks - Run Commands in Zed description: Run and rerun shell commands from Zed with task definitions. Supports variables, templates, and language-specific tasks. --- # Tasks Zed supports ways to spawn (and rerun) commands using its integrated [terminal](./terminal.md) to output the results. These commands can read a limited subset of Zed state (such as a path to the file currently being edited or selected text). ```json [tasks] [ { "label": "Example task", "command": "for i in {1..5}; do echo \"Hello $i/5\"; sleep 1; done", //"args": [], // Env overrides for the command, will be appended to the terminal's environment from the settings. "env": { "foo": "bar" }, // Current working directory to spawn the command into, defaults to current project root. //"cwd": "/path/to/working/directory", // Whether to use a new terminal tab or reuse the existing one to spawn the process, defaults to `false`. "use_new_terminal": false, // Whether to allow multiple instances of the same task to be run, or rather wait for the existing ones to finish, defaults to `false`. "allow_concurrent_runs": false, // What to do with the terminal pane and tab, after the command was started: // * `always` — always show the task's pane, and focus the corresponding tab in it (default) // * `no_focus` — always show the task's pane, add the task's tab in it, but don't focus it // * `never` — do not alter focus, but still add/reuse the task's tab in its pane "reveal": "always", // What to do with the terminal pane and tab, after the command has finished: // * `never` — Do nothing when the command finishes (default) // * `always` — always hide the terminal tab, hide the pane also if it was the last tab in it // * `on_success` — hide the terminal tab on task success only, otherwise behaves similar to `always` "hide": "never", // Which shell to use when running a task inside the terminal. // May take 3 values: // 1. (default) Use the system's default terminal configuration in /etc/passwd // "shell": "system" // 2. A program: // "shell": { // "program": "sh" // } // 3. A program with arguments: // "shell": { // "with_arguments": { // "program": "/bin/bash", // "args": ["--login"] // } // } "shell": "system", // Whether to show the task line in the output of the spawned task, defaults to `true`. "show_summary": true, // Whether to show the command line in the output of the spawned task, defaults to `true`. "show_command": true, // Which edited buffers to save before running the task: // * `all` — save all edited buffers // * `current` — save currently active buffer only // * `none` — don't save any buffers "save": "none" // Represents the tags for inline runnable indicators, or spawning multiple tasks at once. // "tags": [] } ] ``` There are two actions that drive the workflow of using tasks: {#action task::Spawn} and {#action task::Rerun}. {#action task::Spawn} opens a modal with all available tasks in the current file. {#action task::Rerun} reruns the most recently spawned task. You can also rerun tasks from the task modal. By default, rerunning tasks reuses the same terminal (due to the `"use_new_terminal": false` default) but waits for the previous task to finish before starting (due to the `"allow_concurrent_runs": false` default). Keep `"use_new_terminal": false` and set `"allow_concurrent_runs": true` to allow cancelling previous tasks on rerun. ## Task templates Tasks can be defined: - in the global `tasks.json` file; such tasks are available in all Zed projects you work on. This file is usually located in `~/.config/zed/tasks.json`. You can edit them by using the {#action zed::OpenTasks} action. - in the worktree-specific (local) `.zed/tasks.json` file; such tasks are available only when working on a project with that worktree included. You can edit worktree-specific tasks by using the {#action zed::OpenProjectTasks} action. - on the fly with [oneshot tasks](#oneshot-tasks). These tasks are project-specific and do not persist across sessions. - by language extension. ## Variables Zed tasks act just like your shell; that also means that you can reference environmental variables via sh-esque `$VAR_NAME` syntax. A couple of additional environmental variables are set for your convenience. These variables allow you to pull information from the current editor and use it in your tasks. The following variables are available: - `ZED_COLUMN`: current line column - `ZED_ROW`: current line row - `ZED_FILE`: absolute path of the currently opened file (e.g. `/Users/my-user/path/to/project/src/main.rs`) - `ZED_FILENAME`: filename of the currently opened file (e.g. `main.rs`) - `ZED_DIRNAME`: absolute path of the currently opened file with file name stripped (e.g. `/Users/my-user/path/to/project/src`) - `ZED_RELATIVE_FILE`: path of the currently opened file, relative to `ZED_WORKTREE_ROOT` (e.g. `src/main.rs`) - `ZED_RELATIVE_DIR`: path of the currently opened file's directory, relative to `ZED_WORKTREE_ROOT` (e.g. `src`) - `ZED_STEM`: stem (filename without extension) of the currently opened file (e.g. `main`) - `ZED_SYMBOL`: currently selected symbol; should match the last symbol shown in a symbol breadcrumb (e.g. `mod tests > fn test_task_contexts`) - `ZED_SELECTED_TEXT`: currently selected text - `ZED_LANGUAGE`: language of the currently opened buffer (e.g. `Rust`, `Python`, `Shell Script`) - `ZED_WORKTREE_ROOT`: absolute path to the root of the current worktree. (e.g. `/Users/my-user/path/to/project`) - `ZED_MAIN_GIT_WORKTREE`: absolute path to the main git worktree's working directory. For normal checkouts this equals `ZED_WORKTREE_ROOT`; for linked git worktrees this is the original repository's working directory. - `ZED_CUSTOM_RUST_PACKAGE`: (Rust-specific) name of the parent package of $ZED_FILE source file. To use a variable in a task, prefix it with a dollar sign (`$`): ```json [tasks] { "label": "echo current file's path", "command": "echo $ZED_FILE" } ``` You can also use verbose syntax that allows specifying a default if a given variable is not available: `${ZED_FILE:default_value}` These environmental variables can also be used in tasks' `cwd`, `args`, and `label` fields. ### Variable Quoting When working with paths containing spaces or other special characters, please ensure variables are properly escaped. For example, instead of this (which will fail if the path has a space): ```json [tasks] { "label": "stat current file", "command": "stat $ZED_FILE" } ``` Provide the following: ```json [tasks] { "label": "stat current file", "command": "stat", "args": ["$ZED_FILE"] } ``` Or explicitly include escaped quotes like so: ```json [tasks] { "label": "stat current file", "command": "stat \"$ZED_FILE\"" } ``` ### Task filtering based on variables Task definitions with variables which are not present at the moment the task list is determined are filtered out. For example, the following task will appear in the spawn modal only if there is a text selection: ```json [tasks] { "label": "selected text", "command": "echo \"$ZED_SELECTED_TEXT\"" } ``` Set default values to such variables to have such tasks always displayed: ```json [tasks] { "label": "selected text with default", "command": "echo \"${ZED_SELECTED_TEXT:no text selected}\"" } ``` ## Oneshot tasks The same task modal opened via {#action task::Spawn} supports arbitrary bash-like command execution: type a command inside the modal text field, and use `opt-enter` to spawn it. The task modal persists these ad-hoc commands for the duration of the session, {#action task::Rerun} will also rerun such tasks if they were the last ones spawned. You can also adjust the currently selected task in a modal (`tab` is the default key binding). Doing so will put its command into a prompt that can then be edited & spawned as a oneshot task. ### Ephemeral tasks You can use the `cmd` modifier when spawning a task via a modal; tasks spawned this way will not have their usage count increased (thus, they will not be respawned with {#action task::Rerun} and they won't have a high rank in the task modal). The intended use of ephemeral tasks is to stay in the flow with continuous {#action task::Rerun} usage. ### More task rerun control By default, tasks capture their variables into a context once, and this "resolved task" is being rerun always. This can be controlled with the `"reevaluate_context"` argument to the task: setting it to `true` will force the task to be reevaluated before each run. ```json [keymap] { "context": "Workspace", "bindings": { "alt-t": ["task::Rerun", { "reevaluate_context": true }] } } ``` ## Custom keybindings for tasks You can define your own keybindings for your tasks via an additional argument to `task::Spawn`. If you wanted to bind the aforementioned `echo current file's path` task to `alt-g`, you would add the following snippet in your [`keymap.json`](./key-bindings.md) file: ```json [keymap] { "context": "Workspace", "bindings": { "alt-g": ["task::Spawn", { "task_name": "echo current file's path" }] } } ``` Note that these tasks can also have a 'target' specified to control where the spawned task should show up. This could be useful for launching a terminal application that you want to use in the center area: ```json [tasks] // In tasks.json { "label": "start lazygit", "command": "lazygit -p $ZED_WORKTREE_ROOT" } ``` ```json [keymap] // In keymap.json { "context": "Workspace", "bindings": { "alt-g": [ "task::Spawn", { "task_name": "start lazygit", "reveal_target": "center" } ] } } ``` ## Hooks In addition to being spawned manually, tasks can be configured to run automatically in response to certain Zed events by adding a hook to the `hooks` field on a task template. A task with a matching hook will be resolved and spawned when that event fires. The following hooks are currently supported: - `create_worktree` — runs after Zed creates a new linked Git worktree, either directly through the CLI or from the [worktree picker](./git.md#git-worktrees). The task is spawned with `ZED_WORKTREE_ROOT` pointing at the newly created worktree and `ZED_MAIN_GIT_WORKTREE` pointing at the original repository's working directory, which makes these hooks well-suited to copying untracked files (such as `.env` files) or running per-worktree setup commands. Hook tasks are resolved from the same global and worktree-local `tasks.json` files as manually spawned tasks, and multiple tasks may register for the same hook; they all run when the hook fires. A hook task still benefits from the usual task configuration fields — `cwd`, `env`, `reveal`, `hide`, and so on — so you can control how much of the terminal UI is shown while it runs. ```json [tasks] [ { "label": "copy .env into new worktree", "command": "cp", "args": ["$ZED_MAIN_GIT_WORKTREE/.env", "$ZED_WORKTREE_ROOT/.env"], "hooks": ["create_worktree"], "reveal": "no_focus", "hide": "on_success" } ] ``` Tasks that define `hooks` are still available from the task modal like any other task, so the same template can be reused for manual runs. ## Custom Git Commands The Git Graph supports running custom Git command tasks from the commit context menu. To add a command, define a task in your global `tasks.json` file with the `git-command` tag (worktree-local tasks are not supported yet). When shown from a commit's context menu, the task is resolved against the selected commit and repository, and runs from the selected repository root by default. Git Graph command tasks support the Git-specific task variables below. These variables are provided only when resolving Git Graph command tasks. Other task variables, such as `ZED_FILE`, `ZED_SELECTED_TEXT`, `ZED_WORKTREE_ROOT`, and `ZED_MAIN_GIT_WORKTREE`, are not provided to Git Graph command tasks unless they use default values. - `ZED_GIT_SHA`: full SHA of the selected commit. - `ZED_GIT_SHA_SHORT`: short SHA of the selected commit. - `ZED_GIT_REPOSITORY_NAME`: name of the selected Git repository. - `ZED_GIT_REPOSITORY_PATH`: absolute path to the selected Git repository's working directory. For example: ```json [tasks] [ { "label": "Branches containing commit: $ZED_GIT_SHA_SHORT", "command": "git", "args": ["branch", "-a", "--contains", "$ZED_GIT_SHA"], "tags": ["git-command"] } ] ``` ## VS Code Task Format When importing VS Code tasks from `.vscode/tasks.json`, you can omit the `label` field. Zed automatically generates labels based on the task type: - **npm tasks**: `npm: