需求:设备有6个千电口,2个万兆光口。 需要按类型和接口顺序排列后为它们分配网卡名称(请注意不是mac顺序,而是busid号)。
例如:
00-10-f3-ff-ff-xx-a1-35 --eth0
00-10-f3-ff-ff-xx-a1-36 --eth1
00-10-f3-ff-ff-xx-a1-37 --eth2
00-10-f3-ff-ff-xx-a1-38 --eth3
00-10-f3-ff-ff-xx-a1-39 --eth4
00-10-f3-ff-ff-xx-a1-3a --eth5
00-10-f3-ff-ff-oo-bb-97 --fb10
00-10-f3-ff-ff-oo-bb-97 --fb11
linux版本:centos6.2 32位
起初我是想通过python3的 psutil 模块 可以获得mac地址,然后将mac地址转换成十进制进行排序后对 /etc/udev/rules.d/70-persistent-net.rules 文件修改来实现。也考虑过mac地址不能代表io顺序的问题,最后发现linux命令lspci可以获取对应关系从而放弃这个思路。
#-*-coding:utf8-*-
macs=[]
interface={}
from psutil import net_if_addrs
for k, v in net_if_addrs().items():
for item in v:
address = item[1]
if ":" in address and len(address) == 17:
idkey = address.replace(':', '')
interface.update({address:(int(address.replace(':', ''), 16))})
elif "-" in address and len(address) == 17:
idkey = address.replace('-', '')
interface.update({address:(int(address.replace('-', ''), 16))})
for k in sorted(interface,key=interface.__getitem__):
print(k,interface[k])
lspci -D -n -v |egrep "0200|Device Serial Number"
这里是两个过滤条件,0200代表网卡,包含Device Serial Number行会显示mac地址 。
我的想法是通过busid顺序修改 “/etc/udev/rules.d/70-persistent-net.rules”文件的命名顺序
[root@new-b3 ~]# lspci -D -n -v |egrep "0200|Number" 0000:02:00.0 0200: 8086:150c Capabilities: [140] Device Serial Number 00-10-f3-ff-ff-xx-a1-35 0000:03:00.0 0200: 8086:150c Capabilities: [140] Device Serial Number 00-10-f3-ff-ff-xx-a1-36 0000:04:00.0 0200: 8086:150c Capabilities: [140] Device Serial Number 00-10-f3-ff-ff-xx-a1-37 0000:05:00.0 0200: 8086:150c Capabilities: [140] Device Serial Number 00-10-f3-ff-ff-xx-a1-38 0000:06:00.0 0200: 8086:150c Capabilities: [140] Device Serial Number 00-10-f3-ff-ff-xx-a1-39 0000:07:00.0 0200: 8086:150c Capabilities: [140] Device Serial Number 00-10-f3-ff-ff-xx-a1-3a 0000:09:00.0 0200: 8086:10fb (rev 01) Capabilities: [140] Device Serial Number 00-10-f3-ff-ff-oo-bb-97 0000:09:00.1 0200: 8086:10fb (rev 01) Capabilities: [140] Device Serial Number 00-10-f3-ff-ff-oo-bb-97
python通过 lspci命令 提取busid 和macimport subprocess
id = []
mac = []
x = 'lspci -D -n -v | egrep "0200|Serial"'
p = subprocess.Popen(x, shell=True, stdout=subprocess.PIPE)
out, err = p.communicate()
for i in out.splitlines():
if "\t" in i:
v1 = i.replace('''\tCapabilities: [140] Device Serial Number ''', '')
v2 = v1.replace("-ff-ff","") # 发现lspci提出来的mac地址多了两位 -ff-ff (00-10-f3-ff-ff-oo-bb-97)
mac.append(v2.replace("-", ":"))
elif "0200:" in i:
v1 = i[5:12]
v2 = int(v1.split(":")[0]) + int(v1.split(".")[-1])
id.append(v2)
print(zip(id, mac))
执行后得到如下结果:
[(2, '00:10:f3:xx:a1:c9'), (3, '00:10:f3:xx:a1:ca'), (4, '00:10:f3:xx:a1:cb'), (5, '00:10:f3:xx:a1:cc'), (6, '00:10:f3:xx:a1:cd'), (7, '00:10:f3:xx:a1:ce'), (9, '00:10:f3:oo:bb:c3'), (10, '00:10:f3:oo:bb:c3')]
import subprocess
id = []
mac = []
x = 'lspci -D -n -v | egrep "0200|Serial"'
p = subprocess.Popen(x, shell=True, stdout=subprocess.PIPE)
out, err = p.communicate()
for i in out.splitlines():
if "\t" in i:
v1 = i.replace('''\tCapabilities: [140] Device Serial Number ''', '')
v2 = v1.replace("-ff-ff","")
mac.append(v2.replace("-", ":"))
elif "0200:" in i:
v1 = i[5:12]
v2 = int(v1.split(":")[0]) + int(v1.split(".")[-1])
id.append(v2)
mac_dic = dict(zip(id, mac))
mac_key = sorted(mac_dic)
for k in mac_key:
print(mac_dic[k] )
00:10:f3:xx:a1:35
00:10:f3:xx:a1:36
00:10:f3:xx:a1:37
00:10:f3:xx:a1:38
00:10:f3:xx:a1:39
00:10:f3:xx:a1:3a
00:10:f3:oo:b1:97
00:10:f3:oo:b1:97
这块万兆光口为什么两个口mac地址是相同的?
通过观察这个重复的mac地址最后一位加一就可以了(目前发现的规律是这样)
今天还发现了有个有趣的事情,python在linux下使用 os模块的 system向系统执行命令无法使用 export 添加变量
例如在python shell下 import os模块,然后os.system('export AAA=\"hello!\“')。
执行后在系统的bash下 echo $AAA,会发现没有生效的。
原本想使用 /lib/udev/write_net_rules系统脚本添加的,发现变量不能生效就放弃了。
这些是 export 系统变量的参数,比如 添加个注释、指定网卡名称、指定对应mac地址,执行 /lib/udev/write_net_rules 后就会在 /etc/udev/rules.d/70-persistent-net.rules 文件中添加一条规则。
# MATCHADDR MAC address used for the match
# MATCHID bus_id used for the match
# MATCHDEVID dev_id used for the match
# MATCHDRV driver name used for the match
# MATCHIFTYPE interface type match
# COMMENT comment to add to the generated rule
# INTERFACE_NAME requested name supplied by external tool
# INTERFACE_NEW new interface name returned by rule writer
#!/usr/bin/python
#-*-coding:utf8-*-
import subprocess
import os
macs = []
eth_names = []
def lspci_mac():
x = 'lspci -D -n -v | egrep "0200|Serial"'
p = subprocess.Popen(x, shell=True, stdout=subprocess.PIPE)
out, err = p.communicate()
for line in out.splitlines():
if "\t" in line:
value = line.replace("-",":")[51-9:].replace(":ff:ff","")
if value in macs:
up = str(hex(int(value[-2:],16)+1)[-2:])
fib = value.replace(value[-2:], str(up))
macs.append(fib)
else:
macs.append(value)
elif "0200:" in line:
pass
def eth_name():
for id in range(len(macs)):
name = ("eth"+str(id))
if id <6:
eth_names.append(name)
elif name =="eth6":
eth_names.append("eth10")
elif name == "eth7":
eth_names.append("eth11")
def out_file():
path = "/etc/udev/rules.d/70-persistent-net.rules"
os.system("rm -rf %s"%(path))
for conf_line in range(len(macs)):
mac_add = macs[conf_line]
ethnum = eth_names[conf_line]
cfg = '''SUBSYSTEM=="net", ACTION=="add", DRIVERS=="?*", ATTR{address}=="%s", ATTR{type}=="1", KERNEL=="eth*", NAME="%s"\n''' %(mac_add, ethnum)
f = open(path, 'aw')
f.write(cfg)
f.close()
if __name__ == "__main__":
lspci_mac()
eth_name()
out_file()