简介
在Linux系统中,我们可以通过ifconfig,route等shell命令来查看系统接口配置,网关和路由等信息。通过shell的正则表达式功能,通过系列复杂操作,我们可以从字符串中提取出相关的信息。现在,通过Python的netifaces模块,可以很容易的获取这些信息。本文主要介绍netifaces的使用。
安装
当前的版本是0.10.4。
参考下面的步骤进行安装:
- tar xvzf netifaces-0.10.4.tar.gz
- cd netifaces-0.10.4
- sudo python setup.py install
或者
pip install netifaces
提示:通过源码安装此模块,需要安装python-dev开发包,可以通过“apt-get install python-dev”进行安装。
使用
netifaces模块使用起来非常简单,我们来逐一介绍其支持的功能特性。
地址类型编码
Netifaces定义了接口地址类型字典,你可以很方便的查询各类地址类型对应的地址类型编码,在后面的介绍中,我们会发现查询结果仅显示接口地址类型编码,可以通过address_families查询地址编码的字符串表示形式。
bob@192.168.1.100:~$ python
Python 2.6.2 (release26-maint, Apr 19 2009, 01:56:41)
[GCC 4.3.3] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> import netifaces
>>> netifaces.address_families
{0: 'AF_UNSPEC', 1: 'AF_FILE', 2: 'AF_INET', 3: 'AF_AX25', 4: 'AF_IPX',
5: 'AF_APPLETALK', 6: 'AF_NETROM', 7: 'AF_BRIDGE', 8: 'AF_ATMPVC', 9: 'AF_X25',
10: 'AF_INET6', 11: 'AF_ROSE', 12: 'AF_DECnet', 13: 'AF_NETBEUI', 14: 'AF_SECURITY',
15: 'AF_KEY', 16: 'AF_NETLINK', 17: 'AF_PACKET', 18: 'AF_ASH', 19: 'AF_ECONET',
20: 'AF_ATMSVC', 22: 'AF_SNA', 23: 'AF_IRDA', 24: 'AF_PPPOX', 25: 'AF_WANPIPE',
31: 'AF_BLUETOOTH', 34: 'AF_ISDN'}
接口查询
>>> netifaces.interfaces()
['lo', 'eth0', 'eth1', 'eth2_rename', 'vlan11']
通过interfaces()函数接口,可以很容易的获取系统当前的接口列表。Duang,Duang,Duang,是不是很任性呢。
接口配置查询
我们获取了系统的接口信息后,就可以继续查询指定接口的地址配置信息。
>>> netifaces.ifaddresses('eth0')
{17: [{'broadcast': 'ff:ff:ff:ff:ff:ff', 'addr': '00:0c:29:3e:6b:c8'}],
2: [{'broadcast': '10.220.33.255', 'netmask': '255.255.255.0', 'addr': '10.220.33.101'}],
10: [{'netmask': 'ffff:ffff:ffff:ffff::', 'addr': 'fe80::20c:29ff:fe3e:6bc8%eth0'}]}
使用ifaddresses()接口,可以这么容易的获取接口的地址配置信息,返回的信息以字典的形式返回。每一项都指向了一个特定的地址簇配置,如
2: [{'broadcast': '10.220.33.255', 'netmask': '255.255.255.0', 'addr': '10.220.33.101'}]
地址类型为2,查询address_families,我们可以知道它是AF_INET的类型,是关于IPv4的配置。在你实际使用中,不要直接使用2这种魔数,实际上该模块已经定义了一些常量,正确的用法,我们可以参考下面的例子:
>>> netifaces.ifaddresses('eth0')[netifaces.AF_INET]
[{'broadcast': '10.220.33.255', 'netmask': '255.255.255.0', 'addr': '10.220.33.101'}]
网关和路由信息查询
为了方便使用,使用‘default’可以引用缺省网关配置,如果查询特定地址的网关信息,也非常容易,由于查询结果是字典形式表示,输入对应的接口地址类型,就可以查询指定的结果:
>>> netifaces.gateways()
{'default': {2: ('10.220.33.1', 'eth0')}, 2: [('10.220.33.1', 'eth0', True)]}
>>>
>>> netifaces.gateways()[netifaces.AF_INET]
[('10.220.33.1', 'eth0', True)]
源码实例:获取本机IP
import netifaces
def get_most_likely_ip():
for interface_name in netifaces.interfaces():
if interface_name.startswith('lo'):
continue
# TODO: continue if network interface is down
#如果网络接口关闭,继续
addresses = netifaces.ifaddresses(interface_name)
if netifaces.AF_INET in addresses:
for item in addresses[netifaces.AF_INET]:
if 'addr' in item:
logger.debug('Found likely IP {0} on interface {1}'.format(item['addr'], interface_name))
return item['addr']
# well, actually the interface could have more IP's... But for now we assume that the IP
# we want is the first in the list on the IF.
#实际上,接口可以有更多的IP…但是现在我们假设我们想要的IP是if列表中的第一个。
default_ip = '127.0.0.1'
logger.warning('Count not detect likely IP, returning {0}'.format(default_ip))
return '127.0.0.1'