Just:告别 Makefile 的现代命令行任务运行器
本文推荐的一个轻量级命令行工具--Just,它提供了一种简单高效的方式来管理项目任务,类似于传统的 Make 工具,但具有更简洁的语法和更现代化的功能。
我目前在一些小项目中开始使用它来管理一些日常的任务,非常方便,它是基于Rust开发的工具,
所以它在Windows系统中使用也非常方便。
Just简介
Just是一个简单而强大的命令运行器,它允许你将项目中常用的命令保存在一个名为justfile的文件中。
与传统的Make工具类似, 但是Just提供了更简洁的语法和更灵活的功能。
主要包括:
- 跨平台支持(
Linux、macOS、Windows) - 错误信息清晰具体
- 支持命令行参数和环境变量
- 自动补全功能
- 无需声明
.PHONY目标 - 可从项目任意子目录调用
如果你平时也用Makefile来管理项目,那么上手Just会非常快。
Just支持多种操作系统,包括 Linux、MacOS 和 Windows,并且无需额外依赖,它的目标是让命令管理变得简单、直观且易于维护。
它常用命令组织在名为justfile的文件中,通过简洁的语法定义任务(称为"配方"),然后通过 just 命令运行这些任务。
主要组成部分
justfile是Just的核心,它包含了所有可运行的命令(也称为配方)。
一个典型的justfile包括以下几个部分:
- 配方定义:每个配方定义了一个可执行的命令序列。
- 变量定义:用于存储和复用的值。
- 设置:用于控制
Just行为的配置选项。 - 注释:用于解释配方或变量的作用。
以下是一个简单的justfile示例:
# 这是一个注释
build:
echo "Building project..."
cargo build
test:
echo "Running tests..."
cargo test
在这个例子中,build和test是两个配方,分别用于构建和测试项目。
基本语法
justfile的语法非常直观,而且和Makefile非常类似。
最简单的例子如下:
# 这是一个注释
hello:
@echo "Hello, Just!"
命令行中运行结果:
$ just
Hello, Just!
变量的使用
变量是我们编写配方时最常用的功能,Just支持多种变量的类型。
变量可以通过:= 进行赋值,并在配方中通过{{变量名}}的方式进行引用。
- 普通变量
version := "1.0.0"
test:
@echo {{version}}
运行结果:
$ just test
1.0.0
- 路径拼接
config_dir := "config"
config_file := config_dir / "settings.toml"
test:
@echo {{config_dir}}
@echo {{config_file}}
运行结果:
$ just test
config
config/settings.toml
- 变量插值
version := "1.0.0"
deploy:
scp build/app-{{version}}.tar.gz server:/apps
执行时,scp命令中的{{version}}会替换成version变量的实际值。
- 命令行覆盖变量
写在justfile中的变量并不是固定的,可以通过命令行来传入不同的值,比如第一个示例中:
version := "1.0.0"
test:
@echo {{version}}
$ just test
1.0.0
$ just version=2.0.0 test
2.0.0
- 环境变量
path := env_var('PATH')
# 访问环境变量
show_path:
echo "PATH Var: {{path}}"
通过内置的env_var函数可以访问环境变量。
条件判断
条件判断可以让我们编写配方时更加灵活,比如针对不同的平台编译不同的二进制。
build_type := if os() == "windows" { "exe" } else { "bin" }
build:
@echo "可执行文件的后缀: {{build_type}}"
我是windows系统,执行结果如下:
$ just build
可执行文件的后缀: exe
如果条件分支比较多,可以使用else if,比如:
arch := if arch() == "x86_64" {
"amd64"
} else if arch() == "aarch64" {
"arm64"
} else {
error("Unsupported architecture")
}
内置函数
Just中内置了很多函数,方便我们在编写任务时,获取一些必要的信息。
比如上面示例中使用的os()和arch(),都是内置函数。
| 函数 | 描述 | 备注 |
|---|---|---|
| arch() | 指令集结构 | "aarch64", "arm", "asmjs", "hexagon", "mips", "msp430", "powerpc", "powerpc64", "s390x", "sparc", "wasm32", "x86", "x86_64", 和 "xcore" |
| os() | 操作系统 | "android", "bitrig", "dragonfly", "emscripten", "freebsd", "haiku", "ios", "linux", "macos", "netbsd", "openbsd", "solaris", 和 "windows" |
| os_family() | 操作系统系列 | "unix" 和 "windows" |
| env_var(key) | 获取名称为 key 的环境变量 | 如果不存在则终止 |
| env_var_or_default(key, default) | 获取名称为 key 的环境变量 | 如果不存在则返回 default |
| invocation_directory() | 获取 just 被调用时当前目录所对应的绝对路径 | - |
| justfile() | 取得当前 justfile 的路径 | - |
| justfile_directory() | 取得当前 justfile 文件父目录的路径 | - |
| just_executable() | just 可执行文件的绝对路径 | - |
| quote(s) | 用单引号包裹字符串并转义内部单引号 | |
| replace(s, from, to) | 替换字符串中的所有匹配项 | |
| replace_regex(s, regex, repl) | 使用正则表达式替换字符串 | |
| trim(s) | 移除首尾空白字符 | |
| trim_end(s) | 移除尾部空白字符 | |
| trim_end_match(s, substr) | 移除匹配的后缀 | |
| trim_end_matches(s, substr) | 重复移除匹配的后缀 | |
| trim_start(s) | 移除首部空白字符 | |
| trim_start_match(s, substr) | 移除匹配的前缀 | |
| trim_start_matches(s, substr) | 重复移除匹配的前缀 | |
| capitalize(s) | 首字母大写其余小写 | "hello"->"Hello" |
| kebabcase(s) | 转换为 kebab-case | "fooBar"->"foo-bar" |
| lowercamelcase(s) | 转换为 lowerCamelCase | "Foo_bar"->"fooBar" |
| lowercase(s) | 转换为全小写 | "HeLLo"->"hello" |
| shoutykebabcase(s) | 转换为 SHOUTY-KEBAB-CASE | "fooBar"->"FOO-BAR" |
| shoutysnakecase(s) | 转换为 SHOUTY_SNAKE_CASE | "fooBar"->"FOO_BAR" |
| snakecase(s) | 转换为 snake_case | "fooBar"->"foo_bar" |
| titlecase(s) | 转换为 Title Case | "hello world"->"Hello World" |
| uppercamelcase(s) | 转换为 UpperCamelCase | "foo_bar"->"FooBar" |
| uppercase(s) | 转换为全大写 | "hello"->"HELLO" |
| absolute_path(path) | 将相对路径转为绝对路径 | |
| extension(path) | 获取文件扩展名 | |
| file_name(path) | 获取文件名(含扩展名) | |
| file_stem(path) | 获取文件名(不含扩展名) | |
| parent_directory(path) | 获取父目录路径 | |
| without_extension(path) | 获取不含扩展名的路径 | |
| clean(path) | 简化路径(移除冗余部分) | |
| join(a, b…) | 拼接路径(自动处理分隔符) | |
| path_exists(path) | 检查路径是否存在 | |
| error(message) | 终止执行并显示错误消息 | |
| sha256(string) | 计算字符串的SHA-256哈希值(十六进制) | |
| sha256_file(path) | 计算文件的SHA-256哈希值(十六进制) | |
| uuid() | 生成随机UUID |
执行系统命令
justfile中,可以通过反引号来执行系统命令,比如:
current_branch := `git branch --show-current`
deploy:
echo "Deploying branch {{current_branch}}"
与Makefile的比较
最后,整理了justfile与传统的Makefile相比,在特性方面的不同和改进,供参考:
| 特性 | Just | Make |
|---|---|---|
| 语法 | 简洁直观 | 较为复杂 |
| 跨平台 | 原生支持 | 需要兼容层 |
| 错误处理 | 清晰详细 | 有时晦涩 |
| 变量作用域 | 统一简单 | 多种类型 |
| 自动补全 | 完善支持 | 有限 |
| .PHONY 声明 | 不需要 | 需要 |
| 环境变量 | 内置支持 | 需要额外处理 |
| 现代功能 | 条件、函数等 | 有限 |
Just的优势主要体现在:
- 避免 Make 的"up to date"问题
- 更友好的错误信息
- 内置函数和环境变量支持
- 更简洁的变量系统
- 无需声明伪目标
如果你的项目也有Makefile的需求,不妨试试用justfile来试试!
Just:告别 Makefile 的现代命令行任务运行器的更多相关文章
- 无法从命令行或调试器启动服务,必须首先安装Windows服务(使用installutil.exe),然后用ServerExplorer、Windows服务器管理工具或NET START命令启动它
无法从命令行或调试器启动服务,必须首先安装Windows服务(使用installutil.exe),然后用ServerExplorer.Windows服务器管理工具或NET START命令启动它 1. ...
- 在命令行中运行eclipse中创建的java项目
在命令行中运行eclipse中创建的java项目 博客分类: java相关 javaeclipse命令行 由于项目要求,需要对eclipse中的项目进行打包,似的可以在客户机上不装eclipse的情 ...
- 命令行编译运行Java
首先要安装JDK,然后设置环境变量Path,添加C:\Program Files (x86)\Java\jdk1.8.0_66\bin 然后建立一个名为j.java的文件,里面加入如下代码: publ ...
- 命令行下运行php的方法和技巧
linux中直接用"php"命令来执行php文件 一般在linux命令行下运行php文件的代码: XML/HTML代码 linux下执行:#php安装路径 -f php文件路径 例 ...
- SoapUI命令行方式运行
http://stackoverflow.com/questions/9220132/soapui-groovy-script-calls-to-command-line SoapUI支持用命令行方式 ...
- 在命令行终端运行 Elisp 脚本
通常,我们在 Emacs 中运行 Elisp 代码片段,但是也可能需要在命令行终端运行 Elisp 脚本程序.在命令行终端运行 Elisp 脚本需要使用 --script 选项,例如: emacs - ...
- 终端命令行编译运行C#程序HelloWorld
终端命令行编译运行C#程序HelloWorld 今天忽然有人问我,刚学习java程序的时候,一般老师都会建议我们从配置环境变量开始从命令行编译运行程序,那么学习c#的时候基本是直接安装visual s ...
- jmeter命令行模式运行,实时获取压测结果
jmeter命令行模式运行,实时获取压测结果 jmeter很小,很快,使用方便,可以在界面运行,可以命令行运行.简单介绍下命令行运行的方式: sh jmeter.sh -n -t my-script. ...
- 从命令行模式运行Windows管理工具。
从命令行模式运行Windows管理工具. 分类: Play Windows 2004-08-06 16:39 6076人阅读 评论(3) 收藏 举报 1.可以直接在开始-〉运行里面输入的管理工具: 文 ...
- 您可以从 Windows 命令行上运行 gcc、g++、ar、ranlib、dlltool 和其他一些 GNU 工具
Windows 上的安装为了在 Windows 上安装 GCC,您需要安装 MinGW.为了安装 MinGW,请访问 MinGW 的主页 www.mingw.org,进入 MinGW 下载页面,下载最 ...
随机推荐
- Pydantic配置继承抽象基类模式
title: Pydantic配置继承抽象基类模式 date: 2025/3/21 updated: 2025/3/21 author: cmdragon excerpt: Pydantic模型配置系 ...
- 归并排序(递归)(NB)
博客地址:https://www.cnblogs.com/zylyehuo/ 递归思路 # _*_coding:utf-8_*_ import random def merge(li, low, mi ...
- selenium爬虫学习1
简介 Selenium是广泛使用的模拟浏览器运行的库,它是一个用于Web应用程序测试的工具. Selenium测试直接运行在浏览器中,就像真正的用户在操作一样,并且支持大多数现代 Web 浏览器. 函 ...
- MTR工具使用说明与结果分析
免责声明: 本文档可能包含第三方产品信息,该信息仅供参考.阿里云对第三方产品的性能.可靠性以及操作可能带来的潜在影响,不做任何暗示或其他形式的承诺. 概述 当客户端访问目标服务器或负载均衡,使用pin ...
- Python提取pdf文字信息
目录 Python提取pdf文字信息 需求 代码 总结 Python提取pdf文字信息 需求 今天教务处导出来我们全年级的成绩,一看吓一跳,我们的名字怎么不在文件名里,只能一个个找吗.事情开始变得离谱 ...
- 【SpringCloud】Consul服务注册与发现
Consul服务注册与发现 Consul简介 是什么 https://www.consul.io/intro/index.html Consul是一开源的分布式服务发现和配置管理系统,由HashiCo ...
- Linux权限之基础权限
介绍 Linux是多用户的操作系统,允许多个用户同时登录和工作,Linux权限是操作系统用来限制不同用户对资源的访问机制.这里暂且将Linux的权限分为三类: 基本权限:给文件和目录的所属者.所属组. ...
- Graph4Stream:基于图的流计算加速
作者:汪煜 之前在「姊妹篇」<Stream4Graph:动态图上的增量计算>中,向大家介绍了在图计算技术中引入增量计算能力「图+流」,GeaFlow流图计算相比Spark GraphX取得 ...
- java中运行指令浅析
后续业务可能需要在程序中运行指令, 所以这里简单探究了一下, 分别从win和linux两个平台进行研究, 又以为java是跨平台语言, 可能二者之间的区别应该只是返回内容与输入指令的不同. (还不是在 ...
- Lua 的os.date()
Lua os.date() os.date## 原型:os.date ([format [, time]]) 解释:返回一个按format格式化日期.时间的字串或表. usage## 参数格式: 由原 ...