有时候在自动化部署的时候,我们可能需要远程在目标服务器执行一些命令。有可能会涉及到切换用户的操作, 在脚本里面可以使用 EOF 来实现
normalcmd.sh:
#!/bin/bash
touch /tmp/whotest
withsu.sh:
#!/bin/bash
sudo su - <<EOF
touch /tmp/whotest
EOF
远程执行一些命令和切换用户执行命令和脚本:
ssh 10.240.34.165 'ls /tmp';
ssh 10.240.34.165 'sudo mv /tmp/upgrade.sh /export/servers/archiver_all/robert_poc/';
# 切换用户执行命令
ssh 10.240.34.165 'sudo su - root -c "touch /tmp/whotest"';
# 切换用户执行脚本:命令行中有 sudo su - root 但是脚本中并没有 sudo
ssh 10.240.34.165 'sudo su - root -c ". /tmp/normalcmd.sh"';
# 切换用户执行脚本:在脚本中用 sudo 和 EOF 实现
ssh 10.240.34.165 '. /tmp/withsu.sh';
shell 扩展:感谢 http://c.biancheng.net/view/739.html 作者的分享
运行 Shell 脚本有两种方法,一种在新进程中运行,一种是在当前 Shell 进程中运行。
在新进程中运行 Shell 脚本
- 将 Shell 脚本作为程序运行
Shell 脚本也是一种解释执行的程序,可以在终端直接调用(需要使用 chmod 命令给 Shell 脚本加上执行权限),如下所示:
[mozhiyan@localhost ~]$ cd demo #切换到 test.sh 所在的目录
[mozhiyan@localhost demo]$ chmod +x ./test.sh #给脚本添加执行权限
[mozhiyan@localhost demo]$ ./test.sh #执行脚本文件
Hello World ! #运行结果
通过这种方式运行脚本,脚本文件第一行的#!/bin/bash一定要写对,好让系统查找到正确的解释器。
- 将 Shell 脚本作为参数传递给 Bash 解释器
你也可以直接运行 Bash 解释器,将脚本文件的名字作为参数传递给 Bash,如下所示:
[mozhiyan@localhost ~]$ cd demo #切换到 test.sh 所在的目录
[mozhiyan@localhost demo]$ /bin/bash test.sh #使用Bash的绝对路径
Hello World ! #运行结果
通过这种方式运行脚本,不需要在脚本文件的第一行指定解释器信息,写了也没用。
更加简洁的写法是运行 bash 命令。bash 是一个外部命令,Shell 会在 /bin 目录中找到对应的应用程序,也即 /bin/bash,
[mozhiyan@localhost ~]$ cd demo
[mozhiyan@localhost demo]$ bash test.sh
Hello World !
这两种写法在本质上是一样的:第一种写法给出了绝对路径,会直接运行 Bash 解释器;第二种写法通过 bash 命令找到 Bash 解释器所在的目录,然后再运行,只不过多了一个查找的过程而已。
检测是否开启了新进程
你怎么知道开启了新进程?你有什么证据吗?既然如此,那我就来给大家验证一下吧。
Linux 中的每一个进程都有一个唯一的 ID,称为 PID,使用 $$ 变量就可以获取当前进程的 PID。$$ 是 Shell 中的特殊变量,读者在此不必深究。
首先编写如下的脚本文件,并命名为 check.sh:
#!/bin/bash
echo $$ #输出当前进程PID
然后使用以上两种方式来运行 check.sh:
[mozhiyan@localhost demo]$ echo $$
2861 #当前进程的PID
[mozhiyan@localhost demo]$ chmod +x ./check.sh
[mozhiyan@localhost demo]$ ./check.sh
4597 #新进程的PID
[mozhiyan@localhost demo]$ echo $$
2861 #当前进程的PID
[mozhiyan@localhost demo]$ /bin/bash check.sh
4584 #新进程的PID
你看,进程的 PID 都不一样,当然就是两个进程了。
在当前进程中运行 Shell 脚本
这里需要引入一个新的命令——source 命令。source 是 Shell 内置命令的一种,它会读取脚本文件中的代码,并依次执行所有语句。你也可以理解为,source 命令会强制执行脚本文件中的全部命令,而忽略脚本文件的权限。
source 命令的用法为:
source filename
也可以简写为:
. filename
两种写法的效果相同。对于第二种写法,注意点号.和文件名中间有一个空格。
例如,使用 source 运行上节的 test.sh:
[mozhiyan@localhost ~]$ cd demo #切换到test.sh所在的目录
[mozhiyan@localhost demo]$ source ./test.sh #使用source
Hello World !
[mozhiyan@localhost demo]$ source test.sh #使用source
Hello World !
[mozhiyan@localhost demo]$ . ./test.sh #使用点号
Hello World !
[mozhiyan@localhost demo]$ . test.sh #使用点号
Hello World !
你看,使用 source 命令不用给脚本增加执行权限,并且写不写./都行,是不是很方便呢?
检测是否在当前 Shell 进程中
我们仍然借助$$变量来输出进程的 PID,如下所示:
[mozhiyan@localhost ~]$ cd demo
[mozhiyan@localhost demo]$ echo $$
5169 #当前进程PID
[mozhiyan@localhost demo]$ source ./check.sh
5169 #Shell脚本所在进程PID
[mozhiyan@localhost demo]$ echo $$
5169 #当前进程PID
[mozhiyan@localhost demo]$ . ./check.sh
5169 #Shell脚本所在进程PID
你看,进程的 PID 都是一样的,当然是同一个进程了。
如果需要在新进程中运行 Shell 脚本,我一般使用bash test.sh这种写法;如果在当前进程中运行 Shell 脚本,我一般使用. ./test.sh这种写法。这是我个人的风格。