简介

在Linux系统中,我们可以通过ifconfig,route等shell命令来查看系统接口配置,网关和路由等信息。通过shell的正则表达式功能,通过系列复杂操作,我们可以从字符串中提取出相关的信息。现在,通过Python的netifaces模块,可以很容易的获取这些信息。本文主要介绍netifaces的使用。

 

安装

当前的版本是0.10.4。

参考下面的步骤进行安装:

  1. tar xvzf netifaces-0.10.4.tar.gz   
  2. cd netifaces-0.10.4   
  3. 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'