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代码未完成,下一行运行。 请帮忙如何在文件写入后将文件读入变量? 为什么这种模棱两可的行为?
答案 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"
显然撇号会有问题,如果事实上它引用了语义)。