命令替换在双引号内不起作用

时间:2019-03-25 11:38:34

标签: linux bash shell scripting

这里缺少什么?

这有效(证明文件在那里):

[rundeck@denac166 ~]$ ssh ${SSH_USER}@${BUILD_HOST} "cat ${WLS_E1DOMAIN_LOC}/nodemanager.process.id"
7504

这不是(说没有这样的文件或目录):

[rundeck@denac166 ~]$ ssh ${SSH_USER}@${BUILD_HOST} "kill -9 `cat ${WLS_E1DOMAIN_LOC}/nodemanager.process.id`"
cat: /cygdrive/c/Oracle/Middleware/Oracle_Home/user_projects/domains/E1DevDomain/nodemanager/nodemanager.process.id: No such file or directory

kill: usage: kill [-s sigspec | -n signum | -sigspec] pid | jobspec ... or kill -l [sigspec]

2 个答案:

答案 0 :(得分:1)

您正试图通过ssh将kill命令传递给服务器。

不幸的是,所有替换都在主机端而不是服务器端完成。您从cat得到的错误是在denac166而非BUILD_HOST上生成的错误。如果要将其传递给BUILD_HOST,则在这种情况下必须使用管道。通常,您将使用单引号,但是由于您已经使用了其中的shell变量,因此必须使用管道

rundeck@denac166 ~]$ ssh ${SSH_USER}@${BUILD_HOST} "cat ${WLS_E1DOMAIN_LOC}/nodemanager.process.id | xargs kill -9"

答案 1 :(得分:1)

正如@kvantour所说,问题在于命令扩展正在本地计算机上发生。解决此问题的另一种方法是使用转义符告诉本地Shell通过通常触发命令扩展的字符:

ssh ${SSH_USER}@${BUILD_HOST} "kill -9 \`cat ${WLS_E1DOMAIN_LOC}/nodemanager.process.id\`"
                                       ^                                               ^

顺便说一句,我还建议使用$( )而不是反引号(并且您只需要转义$,而不是括号):

ssh ${SSH_USER}@${BUILD_HOST} "kill -9 \$(cat ${WLS_E1DOMAIN_LOC}/nodemanager.process.id)"
                                       ^

顺便说一句^ 2,您可以在echo命令的命令部分使用ssh来查看将发送到远程计算机的内容。也就是说,它在本地外壳程序完成其解析,替换等之后,但在远程外壳程序完成其解析等之后显示该命令。

$ WLS_E1DOMAIN_LOC=/some/path
$ echo "kill -9 \$(cat ${WLS_E1DOMAIN_LOC}/nodemanager.process.id)"
kill -9 $(cat /some/path/nodemanager.process.id)

在这里,您可以看到$WLS_E1DOMAIN_LOC确实被本地shell替换了,但是命令替换没有被替换。

(这是极少数情况下,echo使用命令可以提供信息,而不是造成误导的情况之一。)