34 KiB
slug | sidebar_position |
---|---|
/usage/ | 3 |
使用指南
快速入门
在项目的根目录中创建一个名为 Taskfile.yml
的文件。 cmds
属性应包含 task 的命令。 下面的示例允许编译 Go 应用程序并使用 esbuild 将多个 CSS 文件合并并缩小为一个文件。
version: '3'
tasks:
build:
cmds:
- go build -v -i main.go
assets:
cmds:
- esbuild --bundle --minify css/index.css > public/bundle.css
运行 task 就这样简单:
task assets build
Task 使用 mvdan.cc/sh,一个原生的 Go sh 解释器。 因此,您可以编写 sh/bash 命令,它甚至可以在 Windows 上运行,而 sh
或 bash
通常不可用。 请记住,任何被调用的可执行文件都必须在操作系统或 PATH 中可用。
如果不传 task 的名字,默认会调用 "default"。
支持的文件名称
Task 会按以下顺序查找配置文件:
- Taskfile.yml
- taskfile.yml
- Taskfile.yaml
- taskfile.yaml
- Taskfile.dist.yml
- taskfile.dist.yml
- Taskfile.dist.yaml
- taskfile.dist.yaml
使用 .dist
变体的目的是允许项目有一个提交版本 (.dist
),同时仍然允许个人用户通过添加额外的 Taskfile.yml
(将在 .gitignore
上)来覆盖 Taskfile。
从子目录运行 Taskfile
如果在当前工作目录中找不到 Taskfile,它将沿着文件树向上查找,直到找到一个(类似于 git
的工作方式)。 当从这样的子目录运行 Task 时,它的行为就像从包含 Taskfile 的目录运行它一样。
您可以将此功能与特殊的 {{.USER_WORKING_DIR}}
变量一起使用来创建一些非常有用的可重用 task。 例如,如果你有一个包含每个微服务目录的 monorepo,你可以 cd
进入一个微服务目录并运行一个 task 命令来启动它,而不必创建多个 task 或具有相同内容的 Taskfile。 例如:
version: '3'
tasks:
up:
dir: '{{.USER_WORKING_DIR}}'
preconditions:
- test -f docker-compose.yml
cmds:
- docker-compose up -d
在此示例中,我们可以运行 cd <service>
和 task up
,只要 <service>
目录包含 docker-compose.yml
,就会启动 Docker Compose。
运行全局 Taskfile
如果您使用 --global
(别名 -g
)标志调用 Task,它将查找您的 home 目录而不是您的工作目录。 In short, Task will look for a Taskfile that matches $HOME/{T,t}askfile.{yml,yaml}
.
这对于您可以在系统的任何地方运行的自动化很有用!
:::info
当使用 -g
运行全局 Taskfile 时,task 将默认在 $HOME
上运行,而不是在您的工作目录上!
如前一节所述,{{.USER_WORKING_DIR}}
特殊变量在这里可以非常方便地在您从中调用 task -g
的目录中运行内容。
version: '3'
tasks:
from-home:
cmds:
- pwd
from-working-directory:
dir: '{{.USER_WORKING_DIR}}'
cmds:
- pwd
:::
环境变量
Task
你可以使用 env
给每个 task 设置自定义环境变量:
version: '3'
tasks:
greet:
cmds:
- echo $GREETING
env:
GREETING: Hey, there!
此外,您可以设置可用于所有 task 的全局环境变量:
version: '3'
env:
GREETING: Hey, there!
tasks:
greet:
cmds:
- echo $GREETING
:::info
env
支持扩展和检索 shell 命令的输出,就像变量一样,如您在 变量 部分中看到的那样。
:::
.env 文件
您还可以使用 dotenv:
设置要求 tasks 包含 .env
之类的文件
KEYNAME=VALUE
ENDPOINT=testing.com
version: '3'
env:
ENV: testing
dotenv: ['.env', '{{.ENV}}/.env.', '{{.HOME}}/.env']
tasks:
greet:
cmds:
- echo "Using $KEYNAME and endpoint $ENDPOINT"
也可以在 task 级别指定 .env 文件:
version: '3'
env:
ENV: testing
tasks:
greet:
dotenv: ['.env', '{{.ENV}}/.env.', '{{.HOME}}/.env']
cmds:
- echo "Using $KEYNAME and endpoint $ENDPOINT"
在 task 级别明确指定的环境变量将覆盖点文件中定义的变量:
version: '3'
env:
ENV: testing
tasks:
greet:
dotenv: ['.env', '{{.ENV}}/.env.', '{{.HOME}}/.env']
env:
KEYNAME: DIFFERENT_VALUE
cmds:
- echo "Using $KEYNAME and endpoint $ENDPOINT"
:::info
请注意,您目前无法在包含的 Taskfile 中使用 dotenv
键。
:::
包含其他 Taskfile
如果要在不同项目(Taskfile)之间共享任务,可以使用导入机制使用 includes
关键字包含其他任务文件:
version: '3'
includes:
docs: ./documentation # will look for ./documentation/Taskfile.yml
docker: ./DockerTasks.yml
给定的 Taskfile 中描述的任务将在指定的命名空间中提供。 因此,您可以调用 task docs:serve
从 documentation/Taskfile.yml
运行 serve
task,或者调用 task docker:build
从 DockerTasks.yml
文件运行 build
task。
相对路径是相对于包含包含 Taskfile 的目录解析的。
操作系统特定 Taskfile
在 version: '2'
中,task 会自动尝试引入 Taskfile_{{OS}}.yml
文件 (例如Taskfile_windows.yml
, Taskfile_linux.yml
或 Taskfile_darwin.yml
)。 但是因为过于隐晦,在版本 3 中被移除了, 在版本 3 可以通过明确的引用来实现类似功能:
version: '3'
includes:
build: ./Taskfile_{{OS}}.yml
包含 Taskfile 的目录
默认情况下,包含的 Taskfile 的 task 在当前目录中运行,即使 Taskfile 在另一个目录中,但您可以使用以下替代语法强制其 task 在另一个目录中运行:
version: '3'
includes:
docs:
taskfile: ./docs/Taskfile.yml
dir: ./docs
:::info
包含的 Taskfile 必须使用与主 Taskfile 使用的相同规则版本。
:::
可选 includes
如果包含文件丢失,标记为可选的包含将允许 task 继续正常执行。
version: '3'
includes:
tests:
taskfile: ./tests/Taskfile.yml
optional: true
tasks:
greet:
cmds:
- echo "This command can still be successfully executed if
./tests/Taskfile.yml does not exist"
内部 includes
标记为 internal 的包含会将包含文件的所有 task 也设置为内部 task(请参阅下面的 内部-tasks 部分)。 这在包含不打算由用户直接使用的实用程序任务时很有用。
version: '3'
includes:
tests:
taskfile: ./taskfiles/Utils.yml
internal: true
包含 Taskfile 的变量
您还可以在包含 Taskfile 时指定变量。 这对于拥有可以调整甚至多次包含的可重用 Taskfile 可能很有用:
version: '3'
includes:
backend:
taskfile: ./taskfiles/Docker.yml
vars:
DOCKER_IMAGE: backend_image
frontend:
taskfile: ./taskfiles/Docker.yml
vars:
DOCKER_IMAGE: frontend_image
命名空间别名
包含 Taskfile 时,您可以为命名空间提供一个 aliases
列表。 这与 task 别名 的工作方式相同,可以一起使用来创建更短且更易于键入的命令。
version: '3'
includes:
generate:
taskfile: ./taskfiles/Generate.yml
aliases: [gen]
:::info
在包含的 Taskfile 中声明的变量优先于包含 Taskfile 中的变量! 如果您希望包含的 Taskfile 中的变量可被覆盖,请使用 默认方法:MY_VAR: '{{.MY_VAR | default "my-default-value"}}'
。
:::
内部 tasks
内部 task 是用户不能直接调用的 task。 运行 task --list|--list-all
时,它们不会出现在输出中。 其他 task 可以照常调用内部 task。 这对于创建在命令行上没有用处的可重用、类似函数的 task 很有用。
version: '3'
tasks:
build-image-1:
cmds:
- task: build-image
vars:
DOCKER_IMAGE: image-1
build-image:
internal: true
cmds:
- docker build -t {{.DOCKER_IMAGE}} .
Task 目录
默认情况下,tasks 将在 Taskfile 所在的目录中执行。 但是您可以轻松地让 task 在另一个目录中运行,指定 dir
:
version: '3'
tasks:
serve:
dir: public/www
cmds:
# run http server
- caddy
如果该目录不存在,task
会创建它。
Task 依赖
依赖项并行运行,因此一项 task 的依赖项不应相互依赖。 如果您想强制任务顺序运行,请查看下面的 调用另一个 task 部分。
您可能有依赖于其它的 task。 将它们指向 deps
将使它们在运行父 task 之前自动运行:
version: '3'
tasks:
build:
deps: [assets]
cmds:
- go build -v -i main.go
assets:
cmds:
- esbuild --bundle --minify css/index.css > public/bundle.css
在上面的示例中,如果您运行 task build
,assets
将始终在 build
之前运行。
一个 task 只能有依赖关系,没有命令来将 task 组合在一起:
version: '3'
tasks:
assets:
deps: [js, css]
js:
cmds:
- esbuild --bundle --minify js/index.js > public/bundle.js
css:
cmds:
- esbuild --bundle --minify css/index.css > public/bundle.css
如果有多个依赖项,它们总是并行运行以获得更好的性能。
:::tip
您还可以使用 --parallel
标志(别名 -p
)使命令行给出的 task 并行运行。 例如: task --parallel js css
。
:::
如果你想将信息传递给依赖项,你可以像 调用另一个 task 一样以相同的方式进行:
version: '3'
tasks:
default:
deps:
- task: echo_sth
vars: { TEXT: 'before 1' }
- task: echo_sth
vars: { TEXT: 'before 2' }
silent: true
cmds:
- echo "after"
echo_sth:
cmds:
- echo {{.TEXT}}
平台特定的 tasks 和 cmds
如果您想将 task 的运行限制在明确的平台上,可以使用 platforms:
键来实现。 Task 可以限制在特定的操作系统、架构或两者的组合中。 如果不匹配,任务或命令将被跳过,并且不会抛出任何错误。
允许作为 OS 或 Arch 的值是有效的 GOOS
和 GOARCH
值,正如 此处 的 Go 语言所定义的那样。
下面的 build-windows
task 将仅在 Windows 所有架构上运行:
version: '3'
tasks:
build-windows:
platforms: [windows]
cmds:
- echo 'Running command on Windows'
这可以限制为特定的架构,如下所示:
version: '3'
tasks:
build-windows-amd64:
platforms: [windows/amd64]
cmds:
- echo 'Running command on Windows (amd64)'
也可以将 task 限制在特定的架构中:
version: '3'
tasks:
build-amd64:
platforms: [amd64]
cmds:
- echo 'Running command on amd64'
可以指定多个平台,如下所示:
version: '3'
tasks:
build:
platforms: [windows/amd64, darwin]
cmds:
- echo 'Running command on Windows (amd64) and macOS'
个别命令也可以限制在特定平台上:
version: '3'
tasks:
build:
cmds:
- cmd: echo 'Running command on Windows (amd64) and macOS'
platforms: [windows/amd64, darwin]
- cmd: echo 'Running on all platforms'
调用另一个 task
当一个 task 有很多依赖时,它们是并发执行的。 这通常会导致更快的构建管道。 但是,在某些情况下,您可能需要串行调用其他 task。 在这种情况下,请使用以下语法:
version: '3'
tasks:
main-task:
cmds:
- task: task-to-be-called
- task: another-task
- echo "Both done"
task-to-be-called:
cmds:
- echo "Task to be called"
another-task:
cmds:
- echo "Another task"
使用 vars
和 silent
属性,您可以选择在逐个调用的基础上传递变量和切换 静默模式:
version: '3'
tasks:
greet:
vars:
RECIPIENT: '{{default "World" .RECIPIENT}}'
cmds:
- echo "Hello, {{.RECIPIENT}}!"
greet-pessimistically:
cmds:
- task: greet
vars: { RECIPIENT: 'Cruel World' }
silent: true
deps
也支持上述语法。
:::tip
注意:如果您想从 包含的 Taskfile 中调用在根 Taskfile 中声明的 task,请像这样添加 :
前缀:task: :task-name
。
:::
减少不必要的工作
通过指纹识别本地生成的文件及其来源
如果一个 task 生成了一些东西,你可以通知 task 源和生成的文件,这样 task 就会在不需要的时候阻止运行它们。
version: '3'
tasks:
build:
deps: [js, css]
cmds:
- go build -v -i main.go
js:
cmds:
- esbuild --bundle --minify js/index.js > public/bundle.js
sources:
- src/js/**/*.js
generates:
- public/bundle.js
css:
cmds:
- esbuild --bundle --minify css/index.css > public/bundle.css
sources:
- src/css/**/*.css
generates:
- public/bundle.css
sources
和 generates
可以配置具体文件或者使用匹配模式。 设置后, Task 会根据源文件的 checksum 来确定是否需要执行当前任务。 如果不需要执行, 则会输出像 Task "js" is up to date
这样的信息。
如果您希望通过文件的修改 timestamp 而不是其 checksum(内容)来进行此检查,只需将 method
属性设置为 timestamp
即可。
version: '3'
tasks:
build:
cmds:
- go build .
sources:
- ./*.go
generates:
- app{{exeExt}}
method: timestamp
在需要更大灵活性的情况下,可以使用 status
关键字。 您甚至可以将两者结合起来。 有关示例,请参阅 状态 文档。
:::info
默认情况,task 在本地项目的 .task
目录保存 checksums 值。 一般都会在 .gitignore
(或类似配置)中忽略掉这个目录,这样它就不会被提交。 (如果您有一个已提交的代码生成任务,那么提交该任务的校验和也是有意义的)。
如果你想要将这些文件存储在另一个目录中,你可以在你的机器中设置一个 TASK_TEMP_DIR
环境变量。 可以使用相对路径,比如 tmp/task
,相对项目根目录,也可以用绝对路径、用户目录路径,比如 /tmp/.task
或 ~/.task
(每个项目单独创建子目录)。
export TASK_TEMP_DIR='~/.task'
:::
:::info
每个 task 只为其 sources
存储一个 checksum。 如果您想通过任何输入变量来区分 task,您可以将这些变量添加为 task 标签的一部分,它将被视为不同的 task。
如果您想为每个不同的输入集运行一次 task,直到 sources 实际发生变化,这将很有用。 例如,如果 sources 依赖于变量的值,或者您希望在某些参数发生变化时重新运行 task,即使 sources 没有发生变化也是如此。
:::
:::tip
将 method 设置为 none
会跳过任何验证并始终运行任务。
:::
:::info
要使 checksum
(默认)或 timestamp
方法起作用,只需要通知 source 文件即可。 当使用 timestamp
方法时,最后一次运行 task 被认为是一次生成。
:::
使用程序检查来表示任务是最新的
或者,您可以通知一系列测试作为 status
。 如果没有错误返回(退出状态 0),task 被认为是最新的:
version: '3'
tasks:
generate-files:
cmds:
- mkdir directory
- touch directory/file1.txt
- touch directory/file2.txt
# test existence of files
status:
- test -d directory
- test -f directory/file1.txt
- test -f directory/file2.txt
通常,您会将 sources
与 generates
结合使用 - 但对于生成远程工件(Docker 映像、部署、CD 版本)的 task,checksum source 和 timestamps 需要访问工件或 .checksum
指纹文件。
两个特殊变量 {{.CHECKSUM}}
和 {{.TIMESTAMP}}
可用于 status
命令中的插值,具体取决于分配给 sources 的指纹方法。 只有 source
块才能生成指纹。
请注意,{{.TIMESTAMP}}
变量是一个“实时”Go time.Time
结构,可以使用 time.Time
响应的任何方法进行格式化。
有关详细信息,请参阅 Go Time 文档。
如果你想强制任务运行,即使是最新的,你也可以使用 --force
或 -f
。
此外,如果任何 task 不是最新的,task --status [tasks]...
将以非零退出代码退出。
如果 source/generated 的工件发生变化,或者程序检查失败,status
可以与 指纹 结合以运行任务:
version: '3'
tasks:
build:prod:
desc: Build for production usage.
cmds:
- composer install
# Run this task if source files changes.
sources:
- composer.json
- composer.lock
generates:
- ./vendor/composer/installed.json
- ./vendor/autoload.php
# But also run the task if the last build was not a production build.
status:
- grep -q '"dev": false' ./vendor/composer/installed.json
使用程序检查取消任务及其依赖项的执行
除了 status
检查之外,preconditions
检查是 status
检查的逻辑逆过程。 也就是说,如果您需要一组特定的条件为 true,您可以使用 preconditions
。 preconditions
类似于 status
行,除了它们支持 sh
扩展,并且它们应该全部返回 0。
version: '3'
tasks:
generate-files:
cmds:
- mkdir directory
- touch directory/file1.txt
- touch directory/file2.txt
# test existence of files
preconditions:
- test -f .env
- sh: '[ 1 = 0 ]'
msg: "One doesn't equal Zero, Halting"
先决条件可以设置特定的失败消息,这些消息可以使用 msg
字段告诉用户要采取什么步骤。
如果一个 task 依赖于一个具有前提条件的子 task,并且不满足该前提条件 - 调用 task 将失败。 请注意,除非给出 --force
,否则以失败的前提条件执行的 task 将不会运行。
与 status
判断 task 是最新状态时会跳过并继续执行不同, precondition
失败会导致 task 失败,以及所有依赖它的 task 都会失败。
version: '3'
tasks:
task-will-fail:
preconditions:
- sh: 'exit 1'
task-will-also-fail:
deps:
- task-will-fail
task-will-still-fail:
cmds:
- task: task-will-fail
- echo "I will not run"
在任务运行时限制
如果 task 由多个 cmd
或多个 deps
执行,您可以使用 run
控制何时执行。 run
也可以设置在 Taskfile 的根目录以更改所有任务的行为,除非被明确覆盖。
run
支持的值:
always
(默认)总是尝试调用 task,无论先前执行的次数如何once
只调用一次这个任务,不管引用的数量when_changed
只为传递给 task 的每个唯一变量集调用一次 task
version: '3'
tasks:
default:
cmds:
- task: generate-file
vars: { CONTENT: '1' }
- task: generate-file
vars: { CONTENT: '2' }
- task: generate-file
vars: { CONTENT: '2' }
generate-file:
run: when_changed
deps:
- install-deps
cmds:
- echo {{.CONTENT}}
install-deps:
run: once
cmds:
- sleep 5 # long operation like installing packages
变量
在进行变量插值时,Task 将查找以下内容。 它们按权重顺序列在下面(即最重要的第一位):
- task 内部定义的变量
- 被其它 task 调用时传入的变量(查看 调用另一个 task)
- 包含其他 Taskfile 中的变量 (当包含其他 task 时)
- 包含 Taskfile 的变量(包含 task 时)
- 全局变量 (在 Taskfile 的
vars:
中声明) - 环境变量
使用环境变量传输参数的示例:
$ TASK_VARIABLE=a-value task do-something
:::tip
包含任务名称的特殊变量 .TASK
始终可用。
:::
由于某些 shell 不支持上述语法来设置环境变量 (Windows),task 在不在命令开头时也接受类似的样式。
$ task write-file FILE=file.txt "CONTENT=Hello, World!" print "MESSAGE=All done!"
本地声明的变量示例:
version: '3'
tasks:
print-var:
cmds:
- echo "{{.VAR}}"
vars:
VAR: Hello!
Taskfile.yml
中的全局变量示例:
version: '3'
vars:
GREETING: Hello from Taskfile!
tasks:
greet:
cmds:
- echo "{{.GREETING}}"
动态变量
以下语法 (sh:
prop in a variable) 被认为是动态变量。 该值将被视为命令并产生输出结果用于赋值。 如果有一个或多个尾随换行符,最后一个换行符将被修剪。
version: '3'
tasks:
build:
cmds:
- go build -ldflags="-X main.Version={{.GIT_COMMIT}}" main.go
vars:
GIT_COMMIT:
sh: git log -n 1 --format=%h
这适用于所有类型的变量。
将 CLI 参数转发到 cmds
如果 --
在 CLI 中给出,则所有以下参数都将添加到特殊的 .CLI_ARGS
变量中。 这对于将参数转发给另一个命令很有用。
下面的示例将运行 yarn install
。
$ task yarn -- install
version: '3'
tasks:
yarn:
cmds:
- yarn {{.CLI_ARGS}}
使用 defer
做 task 清理
使用 defer
关键字,可以安排在 task 完成后运行清理。 与仅将其作为最后一个命令的不同之处在于,即使 task 失败,该命令也会运行。
在下面的示例中,即使第三个命令失败,rm -rf tmpdir/
也会运行:
version: '3'
tasks:
default:
cmds:
- mkdir -p tmpdir/
- defer: rm -rf tmpdir/
- echo 'Do work on tmpdir/'
使用其它 task 作为清理任务的命令时,可以这样:
version: '3'
tasks:
default:
cmds:
- mkdir -p tmpdir/
- defer: { task: cleanup }
- echo 'Do work on tmpdir/'
cleanup: rm -rf tmpdir/
:::info
由于 Go 自身的 defer
工作方式 的性质,如果您安排多个 defer命令,则 defer 命令将以相反的顺序执行。
:::
Go 的模板引擎
Task 在执行命令之前将命令解析为 Go 的模板引擎。 可以通过点语法 (.VARNAME
) 访问变量。
Go 的 slim-sprig 库 的所有功能都可用。 以下示例按照给定格式获取当前日期:
version: '3'
tasks:
print-date:
cmds:
- echo {{now | date "2006-01-02"}}
Task 还增加了以下功能:
OS
:返回操作系统。 可能的值为“windows”、“linux”、“darwin”(macOS) 和“freebsd”。ARCH
:返回 Task 的编译架构为:“386”、“amd64”、“arm”或“s390x”。splitLines
:拆分 Unix (\n) 和 Windows (\r\n) 样式的换行符。catLines
:用空格替换 Unix (\n) 和 Windows (\r\n) 样式的换行符。toSlash
:在 Unix 上不执行任何操作,但在 Windows 上将字符串从\
路径格式转换为/
。fromSlash
:与toSlash
相反。 在 Unix 上不执行任何操作,但在 Windows 上将字符串从/
路径格式转换为\
。exeExt
:返回当前操作系统的正确可执行文件扩展名(Windows 为“.exe”
,其他操作系统为“”
)。shellQuote
:引用一个字符串以使其在 shell 脚本中安全使用。 Task 为此使用了 这个 Go 函数。 假定使用 Bash 语法。splitArgs
:将字符串作为命令的参数进行拆分。 Task 使用 这个 Go 函数
示例:
version: '3'
tasks:
print-os:
cmds:
- echo '{{OS}} {{ARCH}}'
- echo '{{if eq OS "windows"}}windows-command{{else}}unix-command{{end}}'
# This will be path/to/file on Unix but path\to\file on Windows
- echo '{{fromSlash "path/to/file"}}'
enumerated-file:
vars:
CONTENT: |
foo
bar
cmds:
- |
cat << EOF > output.txt
{{range $i, $line := .CONTENT | splitLines -}}
{{printf "%3d" $i}}: {{$line}}
{{end}}EOF
帮助
运行 task --list
(或 task -l
)列出所有带有描述的任务。 以下 Taskfile:
version: '3'
tasks:
build:
desc: Build the go binary.
cmds:
- go build -v -i main.go
test:
desc: Run all the go tests.
cmds:
- go test -race ./...
js:
cmds:
- esbuild --bundle --minify js/index.js > public/bundle.js
css:
cmds:
- esbuild --bundle --minify css/index.css > public/bundle.css
将打印以下输出:
* build: Build the go binary.
* test: Run all the go tests.
如果您想查看所有任务,还有一个 --list-all
(别名 -a
)标志。
显示任务摘要
运行 task --summary task-name
将显示任务的摘要。 以下 Taskfile:
version: '3'
tasks:
release:
deps: [build]
summary: |
Release your project to github
It will build your project before starting the release.
Please make sure that you have set GITHUB_TOKEN before starting.
cmds:
- your-release-tool
build:
cmds:
- your-build-tool
运行 task --summary release
将打印以下输出:
task: release
Release your project to github
It will build your project before starting the release.
Please make sure that you have set GITHUB_TOKEN before starting.
dependencies:
- build
commands:
- your-release-tool
如果缺少摘要,将打印描述。 如果任务没有摘要或描述,则会打印一条警告。
请注意:显示摘要不会执行命令。
Task 别名
Aliases 是 task 的替代名称。 它们可以使运行具有长名称或难以键入名称的 task 变得更加容易和快速。 您可以在命令行上使用它们,在您的 Taskfile 中 调用子任务 时以及在 包含来自另一个 Taskfile 的别名 task 时。 它们也可以与 命名空间别名 一起使用。
version: '3'
tasks:
generate:
aliases: [gen]
cmds:
- task: gen-mocks
generate-mocks:
aliases: [gen-mocks]
cmds:
- echo "generating..."
覆盖 Task 名称
有时你可能想覆盖打印在摘要上的 task 名称,最新消息到 STDOUT 等。 在这种情况下,你可以只设置 label:
,也可以用变量进行插值:
version: '3'
tasks:
default:
- task: print
vars:
MESSAGE: hello
- task: print
vars:
MESSAGE: world
print:
label: 'print-{{.MESSAGE}}'
cmds:
- echo "{{.MESSAGE}}"
Warning Prompts
Warning Prompts to prompt a user for confirmation before a task is executed.
Below is an example using prompt
with a dangerous command, that is called between two safe commands:
version: '3'
tasks:
example:
cmds:
- task: not-dangerous
- task: dangerous
- task: another-not-dangerous
not-dangerous:
cmds:
- echo 'not dangerous command'
another-not-dangerous:
cmds:
- echo 'another not dangerous command'
dangerous:
prompt: This is a dangerous command... Do you want to continue?
cmds:
- echo 'dangerous command'
❯ task dangerous
task: "This is a dangerous command... Do you want to continue?" [y/N]
Warning prompts are called before executing a task. If a prompt is denied Task will exit with exit code 205. If approved, Task will continue as normal.
❯ task example
not dangerous command
task: "This is a dangerous command. Do you want to continue?" [y/N]
y
dangerous command
another not dangerous command
To skip warning prompts automatically, you can use the --yes
(alias -y
) option when calling the task. By including this option, all warnings, will be automatically confirmed, and no prompts will be shown.
:::caution
Tasks with prompts always fail by default on non-terminal environments, like a CI, where an stdin
won't be available for the user to answer. In cases like, use --yes
(-y
) to force all tasks with a prompt to run.
:::
静默模式
静默模式在 Task 运行命令之前禁用命令回显。 对于以下 Taskfile:
version: '3'
tasks:
echo:
cmds:
- echo "Print something"
通常这将打印:
echo "Print something"
Print something
开启静默模式后,将打印以下内容:
Print something
开启静默模式有四种方式:
- 在 cmds 级别:
version: '3'
tasks:
echo:
cmds:
- cmd: echo "Print something"
silent: true
- 在 task 级别:
version: '3'
tasks:
echo:
cmds:
- echo "Print something"
silent: true
- 在 Taskfile 全局级别:
version: '3'
silent: true
tasks:
echo:
cmds:
- echo "Print something"
- 或者全局使用
--silent
或-s
标志
如果您想改为禁止 STDOUT,只需将命令重定向到 /dev/null
:
version: '3'
tasks:
echo:
cmds:
- echo "This will print nothing" > /dev/null
试运行模式
试运行模式 (--dry
) 编译并逐步完成每个 task,打印将运行但不执行它们的命令。 这对于调试您的 Taskfile 很有用。
忽略错误
您可以选择在命令执行期间忽略错误。 给定以下 Taskfile:
version: '3'
tasks:
echo:
cmds:
- exit 1
- echo "Hello World"
Task 将在运行 exit 1
后中止执行,因为状态代码 1
代表 EXIT_FAILURE
。 但是,可以使用 ignore_error
继续执行:
version: '3'
tasks:
echo:
cmds:
- cmd: exit 1
ignore_error: true
- echo "Hello World"
也可以为 task 设置 ignore_error
,这意味着所有命令的错误都将被忽略。 不过,请记住,此选项不会传播到由 deps 或 cmds 调用的其他 task!
输出语法
默认情况下,Task 只是将正在运行的命令的 STDOUT 和 STDERR 实时重定向到 shell。 这有利于通过命令打印日志记录的实时反馈,但如果同时运行多个命令并打印大量内容,输出可能会变得混乱。
为了使其更具可定制性,目前您可以选择三种不同的输出选项:
interleaved
(默认)group
prefixed
要选择另一个,只需在 Taskfile 根目录中设置即可:
version: '3'
output: 'group'
tasks:
# ...
group
输出将在命令完成后打印一次命令的全部输出,因此您不会对需要很长时间运行的命令有实时反馈。
使用 group
输出时,您可以选择提供模板化消息以在组的开始和结束处打印。 这对于指示 CI 系统对给定任务的所有输出进行分组非常有用,例如使用 GitHub Actions 的 ::group::
命令 或 Azure Pipelines。
version: '3'
output:
group:
begin: '::group::{{.TASK}}'
end: '::endgroup::'
tasks:
default:
cmds:
- echo 'Hello, World!'
silent: true
$ task default
::group::default
Hello, World!
::endgroup::
使用 group
输出时,如果没有失败(零退出代码),您可以在标准输出和标准错误上执行命令的输出。
version: '3'
silent: true
output:
group:
error_only: true
tasks:
passes: echo 'output-of-passes'
errors: echo 'output-of-errors' && exit 1
$ task passes
$ task errors
output-of-errors
task: Failed to run task "errors": exit status 1
prefix
输出将为命令打印的每一行添加前缀 [task-name]
作为前缀,但您可以使用 prefix:
属性自定义命令的前缀:
version: '3'
output: prefixed
tasks:
default:
deps:
- task: print
vars: { TEXT: foo }
- task: print
vars: { TEXT: bar }
- task: print
vars: { TEXT: baz }
print:
cmds:
- echo "{{.TEXT}}"
prefix: 'print-{{.TEXT}}'
silent: true
$ task default
[print-foo] foo
[print-bar] bar
[print-baz] baz
:::tip
output
选项也可以由 --output
或 -o
标志指定。
:::
交互式 CLI 应用
Task 执行包含交互式的命令时有时会出现奇怪的结果, 尤其当 输出模式 设置的不是 interleaved
(默认), 或者当交互式应用与其它 task 并发执行时。
interactive: true
告诉 Task 这是一个交互式应用程序,Task 将尝试针对它进行优化:
version: '3'
tasks:
default:
cmds:
- vim my-file.txt
interactive: true
如果您在通过 Task 运行交互式应用程序时仍然遇到问题,请打开一个关于它的 Issue。
短 Task 语法
从 Task v3 开始,如果 task 具有默认设置(例如:没有自定义 env:
、vars:
、desc:
、silent:
等),您现在可以使用更短的语法编写task:
version: '3'
tasks:
build: go build -v -o ./app{{exeExt}} .
run:
- task: build
- ./app{{exeExt}} -h localhost -p 8080
set
和 shopt
可以为 set
和 shopt
内置函数指定选项。 这可以在全局、task 或命令级别添加。
version: '3'
set: [pipefail]
shopt: [globstar]
tasks:
# `globstar` required for double star globs to work
default: echo **/*.go
:::info
请记住,并非所有选项在 Task 使用的 shell 解释器库 中都可用。
:::
观察 task
使用 --watch
或 -w
参数可以观察文件变化,然后重新执行 task。 这需要配置 sources
属性,task 才知道观察哪些文件。
默认监控的时间间隔是 5 秒,但可以通过 Taskfile 中根属性 interval: '500ms'
设置,也可以通过命令行 参数 --interval=500ms
设置。