我正在尝试编写一个脚本,它将从我输入脚本的列表中收集远程主机上安装的主机名、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
当我运行脚本时,出现以下错误:
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
我认为问题出在free -m | grep Mem | awk '{print $1,$2}
但不知道如何纠正。如果我free -m | grep Mem | awk '{print $1,$2}
直接进入 shell,我就没有问题。就在脚本里面。
答案1
这比需要的要复杂得多。另外,你为什么要ssh
跑步sudo
?如果您需要以 root 身份登录远程,那么您可以这样做 ( ssh root@$host
),但您不太可能需要运行 ssh,sudo
除非您的 ssh 密钥都属于 root。这是一个非常糟糕的主意。
另外,在我的 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
它在我测试的 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
因此,也许更便携的版本是在之后打印字段src
:
$ ip route get 1 | sed -nE 's/.* src ([0-9.]+).*/\1/p'
192.168.1.111
您看到的错误确实是由于引用造成的。由于您正在运行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
答案2
我已经找到了这个问题的答案。这是与转义有关,所以我在 $1 和 $2 之前添加了 \ ,如下所示:
free -m | grep Mem | awk '{print \$1,\$2}
答案3
这就是最终脚本的样子。我知道它会对你们中的一些人有用。
#!/bin/bash
for host in `cat ip_adds2`
do
echo "Hostname:" $host
sudo ssh -t -o BatchMode=yes -o ConnectTimeout=5 $host 'echo IP: `hostname -i`;read junk total used free shared buffers cached junk < <(free -g | grep ^Mem);echo Memory: $total GiB'
echo -e "\n"
done