问题描述
index.=10200,intf.10200.4=eth1,vlan.10200.4=500,IP-Addr.10200.4=C0A80A78,uptime.10200.4=120days,duplex.10200.4=full,status.10200.4=up
index.=10201,intf.10201.25=eth1,vlan.10201.25=500,IP-Addr.10201.25=C0A80A79,uptime.10201.25=80days,duplex.10201.25=full,status.10201.25=up
index.=10202,intf.10202.6=eth1,vlan.10202.6=500,IP-Addr.10202.6=C0A80A80,uptime.10202.6=240days,duplex.10202.6=full,status.10202.6=up
我正在尝试通过就地编辑文件将所有十六进制 IP 地址转换为点分十进制。 我已经在网上看到了所有 bash 十六进制到 dec ip 转换器,但那些用于输入值然后转换。
我将如何在文件中就地编辑值?
更详细的数据集:
index.=10200,CDP-Local-Intf.10200=FastEthernet0/0/1,CDP-IP-Addr.10200.44=C0A80A78,CDP-Uptime.10200.44=Timeticks:(342342322)52days,CDP-Opposite-Intf.10200.44=FastEthernet0/1/0,CDP-VLAN.10200.44=100,CDP-Duplex.10200.44=3,CDP-Other.10200.44=00000032
index.=10201,CDP-Local-Intf.10201=FastEthernet2/0/1,CDP-IP-Addr.10201.25=C0A80A79,CDP-Uptime.10201.25=Timeticks:(342342322)52days,CDP-Opposite-Intf.10201.25=FastEthernet0/2/0,CDP-VLAN.10201.25=101,CDP-Duplex.10201.25=3,CDP-Other.10201.25=00000032
index.=10202,CDP-Local-Intf.10202=FastEthernet2/1/1,CDP-IP-Addr.10202.11=C0A80A80,CDP-Uptime.10202.11=Timeticks:(342342322)52days,CDP-Opposite-Intf.10202.11=FastEthernet0/3/0,CDP-VLAN.10202.11=101,CDP-Duplex.10202.11=3,CDP-Other.10202.11=00000032
解决方法
如果 perl
是您的选择,您是否会尝试以下操作:
perl -i -pe 's/(CDP-IP-Addr[\d.]+=)([[:xdigit:]]{8})/$1 . join(".",unpack("C4",pack("H8",$2)))/e' datafile
-
-i
的perl
选项启用就地编辑。 -
-pe
选项告诉 perl 将输入文件逐行处理为sed
或awk
可以。 - 正则表达式
(CDP-IP-Addr[\d.]+=)([[:xdigit:]]{8})
提取子字符串 “CDP-IP-Addr.xx.xx=”和以下十六进制字符串。他们是$1
和$2
按顺序捕获。 - 函数
unpack("C4",$2))
转换 8 位十六进制 将$2
字符串转换为四个十进制值的数组。 - 切换到
e
运算符的s/regex/replacement/
告诉 perl 将替换评估为 perl 表达式。
不确定这是否会有所帮助,但如果您不介意使用 Python 进行快速转换,请查看此链接,我找到了一个非常简单的 Python 脚本来完成十六进制到十进制的转换。
http://pythonldap.blogspot.com/2014/09/python-script-to-convert-hexadecimal.html
如果您需要转换十六进制文件,请尝试添加一个 for 循环来遍历十六进制值并将该文件用作参数。
,这是 gawk
测试工具 - 它使用 gawk
特定功能和扩展。
$ cat tst.awk
match($0,/.*=([[:xdigit:]]+),(CDP-Uptime|uptime).*/,a) {
str = strtonum(sprintf("0x%s",a[1]))
ip=sprintf ("%d.%d.%d.%d",rshift(and(str,0xff000000),24),0x00ff0000),16),0x0000ff00),08),0x000000ff),00))
$0=substr($0,1,a[1,"start"]-1) ip substr($0,"start"]+a[1,"length"])
}
1
$ gawk -f tst.awk myFile myMoreDetailedFile
index.=10200,intf.10200.4=eth1,vlan.10200.4=500,IP-Addr.10200.4=192.168.10.120,uptime.10200.4=120days,duplex.10200.4=full,status.10200.4=up
index.=10201,intf.10201.25=eth1,vlan.10201.25=500,IP-Addr.10201.25=192.168.10.121,uptime.10201.25=80days,duplex.10201.25=full,status.10201.25=up
index.=10202,intf.10202.6=eth1,vlan.10202.6=500,IP-Addr.10202.6=192.168.10.128,uptime.10202.6=240days,duplex.10202.6=full,status.10202.6=up
index.=10200,CDP-Local-Intf.10200=FastEthernet0/0/1,CDP-IP-Addr.10200.44=192.168.10.120,CDP-Uptime.10200.44=Timeticks:(342342322)52days,CDP-Opposite-Intf.10200.44=FastEthernet0/1/0,CDP-VLAN.10200.44=100,CDP-Duplex.10200.44=3,CDP-Other.10200.44=00000032
index.=10201,CDP-Local-Intf.10201=FastEthernet2/0/1,CDP-IP-Addr.10201.25=192.168.10.121,CDP-Uptime.10201.25=Timeticks:(342342322)52days,CDP-Opposite-Intf.10201.25=FastEthernet0/2/0,CDP-VLAN.10201.25=101,CDP-Duplex.10201.25=3,CDP-Other.10201.25=00000032
index.=10202,CDP-Local-Intf.10202=FastEthernet2/1/1,CDP-IP-Addr.10202.11=192.168.10.128,CDP-Uptime.10202.11=Timeticks:(342342322)52days,CDP-Opposite-Intf.10202.11=FastEthernet0/3/0,CDP-VLAN.10202.11=101,CDP-Duplex.10202.11=3,CDP-Other.10202.11=00000032
这绝对可以进一步简化。 试一试!