无法在我的脚本中找到错误的行为不符合预期

时间:2015-12-08 11:15:41

标签: bash docker

docker run --rm \
/bin/bash -c "\
su - hello && \
echo \"Pre\" && \
python ${M} ${F} ${D} ${B} && \
echo \"Post\" && \
CHECK_VALUE=`$(cat B)` && \
echo ${CHECK_VALUE} && \
"

我的问题是$ {B}是由python代码生成的文件。 但每当我运行这个时,我得到以下日志:

cat: /home/hello/D/W/H/b.txt: No such file or directory
Pre
Post

也是空行 这显然意味着我的python代码未完成,下一行运行。 请帮忙如何在文件写入后将文件读入变量? 为什么这种模棱两可的行为?

1 个答案:

答案 0 :(得分:5)

此部分中有多个错误:

`$(cat B)`

您正在混合两个命令替换。 $(cat B)`cat B`同义,因此您在此处拥有的内容实际上等同于$($(cat B))。换句话说,取B的内容并将其用作命令,然后用替换命令的输出。

此外,因为您在命令周围使用双引号,所以在解析命令时会评估此构造,而不是在执行命令时。你可以通过反斜杠来逃避美元符号来阻止这种情况。

此外,您需要转义${CHECK_VALUE}以防止shell在评估字符串时插入它,就像命令替换一样。我还建议引用该值,除非您希望它包含一个令牌(甚至可能只是为了您自己的理智)。

最后,你有一个悬空&&,后面没有任何东西。这是语法错误。

以下是我的修补程序的脚本:

docker run --rm \
/bin/bash -c "\
    su - hello && \
    echo \"Pre\" && \
    python ${M} ${F} ${D} ${B} && \
    echo \"Post\" && \
    CHECK_VALUE=\"\$(cat B)\" && \
    echo \"\${CHECK_VALUE}\""

我还在cat周围添加了引号,以防它的输出包含shell元字符。

...虽然捕捉变量只是为了echo它是antipattern;可能你只想将cat ${B}作为最后的命令。

如果您希望su - hello导致后续命令以hello运行,那也是错误的(虽然是一个常见的新手错误)。 su生成一个新shell,当该shell 终止(不在其中)时,将执行以下命令。你可能意味着su -c "python $M $F $D $B" hello && cat "$B"简单明了。将其包装在docker命令行中,尝试

docker run --rm \
/bin/bash -c "su -c 'python $M $F $D $B' hello && cat $B"

其中单引号不会阻止变量插值,因为在双引号内部,它们实际上并没有引用任何内容(比较echo "don't"显然撇号会有问题,如果事实上它引用了语义)。