优化基于MAC地址匹配提取子字符串的bash脚本

时间:2016-07-26 07:07:09

标签: linux bash centos

客户端需要根据NIC MAC地址命名50个CentOS 7主机名:

目前我合并了下面的bash脚本,但似乎可以做很多优化,脚本将在Image首次启动时启动,它必须稳定且紧凑:

for i in `ip a |awk '/ether/ {print toupper($2) "_"}' | sed -r 's/:/-/g'`; do grep -i $i* MAC_HOST | awk -F _ '{print $2}'; done

MAC_HOST内容:

22-8A-07-4F-16-A8_BT001
22-8A-07-5C-0F-58_BT002
22-8A-07-5C-0F-5B_BT003
22-8A-07-5D-D1-3A_BT004
22-8A-07-5D-D1-3D_BT005
22-8A-07-5D-D1-4C_BT006
22-8A-07-5D-D1-52_BT007
22-8A-07-5D-D1-58_BT008
22-8A-07-5D-D1-5B_BT009
22-8A-07-5D-D1-5E_BT010
22-8A-07-5D-D1-61_BT011
22-8A-07-5D-D1-67_BT012
22-8A-07-5D-D1-7C_BT013
22-8A-07-5D-D1-82_BT014
22-8A-07-5D-D1-8E_BT015
22-8A-07-5D-D1-91_BT016
22-8A-07-5E-2F-4D_BT017
22-8A-07-5E-2F-5C_BT018
22-8A-07-5E-2F-65_BT019
22-8A-07-5E-2F-7D_BT020
22-8A-07-5F-36-65_BT021
22-8A-07-5F-36-9B_BT022
22-8A-07-5F-36-A1_BT023
22-8A-07-5F-36-A4_BT024
22-8A-07-5F-36-A7_BT025
22-8A-07-5F-36-AA_BT026
22-8A-07-5F-36-AD_BT027
22-8A-07-5F-36-B0_BT028
22-8A-07-5F-36-B3_BT029
22-8A-07-5F-36-B6_BT030
22-8A-07-5F-36-BC_BT031
22-8A-07-5F-D8-0B_BT032
22-8A-07-5F-D8-0E_BT033
22-8A-07-5F-D8-11_BT034
22-8A-07-5F-D8-14_BT035
22-8A-07-5F-D8-17_BT036
22-8A-07-5F-D8-1A_BT037
22-8A-07-5F-D8-1D_BT038
22-8A-07-5F-D8-20_BT039
22-8A-07-5F-D8-26_BT040
22-8A-07-5F-D8-29_BT041
22-8A-07-5F-D8-2F_BT042
22-8A-07-5F-D8-32_BT043
22-8A-07-5F-D8-35_BT044
22-8A-07-5F-D8-38_BT045
22-8A-07-5F-D8-3B_BT046
22-8A-07-5F-D8-3E_BT047
22-8A-07-5F-D8-41_BT048
22-8A-07-5F-D8-44_BT049
22-8A-07-5F-D8-47_BT050

注意:添加到"之前;完成"写找到的主机名。我删除它,所以没有人错误地覆盖他/她的主机名。

  

|猫>>的/ etc /主机名

1 个答案:

答案 0 :(得分:2)

我认为这可以做你想要的,它消除了所有的shell循环和很多进程:

ip a | awk 'FNR==NR{a[$1]=$2;next} /ether/{gsub(/:/, "-", $2); print a[toupper($2)]}' FS=_ MAC_HOST FS=" "  -

如何运作

  • FNR==NR{a[$1]=$2;next}

    这将读取第一个文件MAC_HOST的内容,并将其存储在关联数组a中。

    请注意,无论ip a返回多少接口,文件MAC_HOST只读一次。此外,关联数组具有快速查找功能。

  • /ether/{gsub(/:/, "-", $2); print a[toupper($2)]}

    这将选择包含ether的行并处理第二个字段

  • FS=_ MAC_HOST FS=" " -

    在读取MAC_HOST时将字段分隔符设置为_,然后在读取stdin(来自ip a)时将其设置为空格。

    请注意,在awk中,FS=" "a special case:这意味着任何一个或多个空白字符的序列都被视为字段分隔符。

多行版本

当分布在多行时,相同的代码是:

ip a | awk '

  FNR==NR{
    a[$1]=$2
    next
  }

  /ether/{
    gsub(/:/, "-", $2)
    print a[toupper($2)]
  }
  ' FS=_ MAC_HOST FS=" "  -

变化:忽略不匹配的MAC地址

ip a返回未在MAC_HOST中列出的MAC地址时,我们该怎么办?上面的版本打印一个空行。当缺少MAC_HOST条目时,以下版本不打印任何内容:

ip a | awk 'FNR==NR{a[$1]=$2;next} {b=""} /ether/{gsub(/:/, "-", $2); b=a[toupper($2)]} b {print b}' FS=_ MAC_HOST FS=" "  -

例如,这里的假输入提供了两个MAC地址,一个在MAC_HOST中,一个不在:

$ printf 'ether 22:8a:07:5f:36:a4\n\nether 22:8a:07:5f:36:a5\n\n' | awk 'FNR==NR{a[$1]=$2;next} {b=""} /ether/{gsub(/:/, "-", $2); b=a[toupper($2)]} b {print b}' FS=_ MAC_HOST FS=" "  -
BT024

使用上述内容覆盖/etc/hostname

ip a | awk 'FNR==NR{a[$1]=$2;next} {b=""} /ether/{gsub(/:/, "-", $2); b=a[toupper($2)]} b {print b}' FS=_ MAC_HOST FS=" "  - >/etc/hostname