use*_*468 2 linux scripting ssh awk quoting
我正在尝试编写一个脚本,该脚本将从我提供给脚本的列表中收集安装在远程主机上的主机名、IP 和总内存。该脚本将从 Redhat 和 Solaris 机器收集信息。下面是我的脚本的样子:
#!/bin/bash
echo > ip_info.output
echo -e "\n"
for host in `cat ip_adds`
do
echo "Hostname:" $host
sudo ssh -o BatchMode=yes -o ConnectTimeout=5 $host "echo IP Address:; ip route get 1 | awk '{print $NF;exit}'; free -m | grep Mem | awk '{print $1,$2}'"
echo -e "\n"
done
Run Code Online (Sandbox Code Playgroud)
当我运行脚本时,出现以下错误:
awk: {print ,}
awk: ^ syntax error
awk: {print ,}
awk: ^ syntax error
awk: cmd. line:1: {print ,}
awk: cmd. line:1: ^ unexpected newline or end of string
Run Code Online (Sandbox Code Playgroud)
我认为问题出在free -m | grep Mem | awk '{print $1,$2}但不确定如何纠正它。如果我free -m | grep Mem | awk '{print $1,$2}直接运行到 shell 中,我没有问题。就在脚本里面。
这比它需要的要复杂得多。另外,为什么你会运行ssh与sudo?如果您需要以 root 身份登录远程,那么您可以这样做 ( ssh root@$host),但sudo除非您的 ssh 密钥都属于 root ,否则您不太可能需要运行 ssh 。这是一个非常糟糕的主意。
此外,在我的 Arch 系统上,您似乎用来获取 IP 的命令返回了我的用户的 UID:
$ ip route get 1
1.0.0.0 via 192.168.1.1 dev enp0s31f6 src 192.168.1.111 uid 1000
cache
$ ip route get 1 | awk '{print $NF;exit}';
1000
Run Code Online (Sandbox Code Playgroud)
它在我测试的 Ubuntu 上按预期工作:
$ ip route get 1
1.0.0.0 via 123.456.7.8 dev eth0 src 123.456.7.9
$ ip route get 1 | awk '{print $NF;exit}';
123.456.7.9
Run Code Online (Sandbox Code Playgroud)
因此,也许更便携的版本是在以下之后打印字段src:
$ ip route get 1 | sed -nE 's/.* src ([0-9.]+).*/\1/p'
192.168.1.111
Run Code Online (Sandbox Code Playgroud)
你看到的错误确实是因为引用。由于您正在运行ssh $host "command",因此周围的双引号command会导致 shell 扩展在命令中找到的任何变量(例如 awk$2等)。为了避免这种情况并将未展开的符号传递给 awk,您需要将$.
脚本的更简单版本:
#!/bin/bash
sshOpts="BatchMode=yes -o ConnectTimeout=5"
echo > ip_info.output
echo ""
while read host; do
printf "Hostname:%s" $host
ssh -o $sshOpts $host "printf 'Hostname: %s\nIP: %s\nMem: %s\n' $host "$(ip route get 1 | sed -nE 's/.* src ([0-9.]+).*/\1/p')" "$(free -m | awk '/Mem/{print $2}')""
done < ip_adds
Run Code Online (Sandbox Code Playgroud)