【发布时间】:2010-09-15 09:00:20
【问题描述】:
是否可以在不执行的情况下检查 bash 脚本语法?
使用 Perl,我可以运行 perl -c 'script name'。 bash 脚本是否有任何等效命令?
【问题讨论】:
标签: linux bash unix syntax gnu
是否可以在不执行的情况下检查 bash 脚本语法?
使用 Perl,我可以运行 perl -c 'script name'。 bash 脚本是否有任何等效命令?
【问题讨论】:
标签: linux bash unix syntax gnu
bash -n scriptname
也许有一个明显的警告:这会验证语法,但不会检查您的 bash 脚本是否尝试执行不在您的路径中的命令,例如 ech hello 而不是 echo hello。
【讨论】:
set 所做的所有单字符选项。跨度>
if ["$var" == "string" ] 而不是 if [ "$var" == "string" ] 引起的错误
type [ 说“[ 是内置的 shell”。它最终委托给test 程序,但它也需要一个右括号。所以它就像if test"$var",这不是作者的意思,但在语法上是有效的(假设$var 的值为“a”,那么我们将看到“bash: testa: command not found”)。关键是在语法上,没有空格。
$var 恰好扩展为空字符串 时,才会调用内置[。如果$var 扩展为非空 字符串,[ 与该字符串连接 并被解释为命令 名称(不是function name) 由 Bash 编写,是的,这是 syntactically 有效的,但是,正如您所说,显然不是意图。如果您使用[[ 而不是[,即使[[ 是一个shell 关键字(而不是内置),您也会得到相同的结果,因为意外的字符串连接仍然会覆盖识别关键字。
["$var" 是 syntactically 一个有效的 command-name 表达式;同样,标记== 和"$string" 是有效的命令arguments。 (通常,内置 [ 使用 command 语法解析,而 [[ - 作为 shell keyword - 解析方式不同。)内置 shell [ 确实 不 委托到“test程序”(外部实用程序):bash、dash、ksh、zsh都有builtin 两个版本的 [ 和 test,它们确实不调用它们的外部实用程序对应项。
sh -n script-name
运行这个。如果脚本中有任何语法错误,那么它会返回相同的错误消息。
如果没有错误,那么它就会出来而不给出任何消息。您可以立即使用echo $?进行检查,它将返回0确认成功,没有任何错误。
它对我很有效。我在 Linux 操作系统、Bash Shell 上运行。
【讨论】:
sh -n 可能不会检查脚本是否是有效的 Bash 脚本。它可能会给出假阴性。 sh 是一些 Bourne shell 变体,通常不是 Bash。例如在 Ubuntu Linux 中 realpath -e $(command -v sh) 给出 /bin/dash
我还在我编写的每个 bash 脚本上启用了“u”选项,以便进行一些额外的检查:
set -u
这将报告未初始化变量的使用情况,如以下脚本'check_init.sh'
#!/bin/sh
set -u
message=hello
echo $mesage
运行脚本:
$ check_init.sh
将报告以下内容:
./check_init.sh[4]:消息:未设置参数。
非常有助于捕捉错别字
【讨论】:
set -u 虽然这并不能真正回答问题,因为您必须运行脚本才能获取错误消息。甚至bash -n check_init.sh 都没有显示警告
时间改变一切。 Here is a web site 为 shell 脚本提供在线语法检查。
我发现检测常见错误非常强大。
ShellCheck 是一个用于 sh/bash 脚本的静态分析和 linting 工具。它主要侧重于处理典型的初学者和中级语法错误和陷阱,其中 shell 只是给出一个神秘的错误消息或奇怪的行为,但它也报告了一些更高级的问题,其中极端情况可能导致延迟失败。
Haskell 源代码在 GitHub 上可用!
【讨论】:
brew install shellcheck 安装 shellcheck.net CLI,shellcheck。
apt-get install shellcheck
trusty-backports 安装。
null 命令 [冒号] 在调试查看变量值时也很有用
set -x
for i in {1..10}; do
let i=i+1
: i=$i
done
set -
【讨论】:
set -x 在执行之前显示每一行
BashSupport plugin 用于检查语法的 IntelliJ IDEA。
【讨论】:
.sh 或与 Bash 脚本相关的其他扩展名结尾(如果您使用诸如 ERB 之类的模板工具(然后它们以.erb 结尾)。如果你想修复它,请投票给youtrack.jetbrains.com/issue/IDEA-79574!
我实际上在没有使用find 工具运行它们的情况下检查当前目录中的所有 bash 脚本是否存在语法错误:
示例:
find . -name '*.sh' -print0 | xargs -0 -P"$(nproc)" -I{} bash -n "{}"
如果您想将它用于单个文件,只需使用文件名编辑通配符即可。
【讨论】:
如果你需要一个变量中所有文件的有效性(git pre-commit hook,build lint script),你可以捕获“sh -n”或“bash -n”命令的stderr输出(见其他答案)在一个变量中,并有一个基于它的“if/else”
bashErrLines=$(find bin/ -type f -name '*.sh' -exec sh -n {} \; 2>&1 > /dev/null)
if [ "$bashErrLines" != "" ]; then
# at least one sh file in the bin dir has a syntax error
echo $bashErrLines;
exit;
fi
根据您的需要将“sh”更改为“bash”
【讨论】:
【讨论】:
如果您启用语法检查,Bash shell 脚本将运行语法检查
set -o noexec
如果你想关闭语法检查
set +o noexec
【讨论】: