使用pyudev仅查找磁盘驱动器

时间:2013-04-11 06:10:24

标签: python udev pyudev

我希望使用udev枚举计算机上的所有硬盘,特别是pyudev来枚举所有内容:

import pyudev

context = pyudev.Context()

for device in context.list_devices(subsystem='block', DEVTYPE='disk'):
    print "{}, ({})".format(device.device_node, device.device_type)

这打印出以下内容:

/dev/sdb (disk)
/dev/sdc (disk)
/dev/sda (disk)
/dev/sr0 (disk)
/dev/loop0 (disk)
/dev/loop1 (disk)
/dev/loop2 (disk)
/dev/loop3 (disk)
/dev/loop4 (disk)
/dev/loop5 (disk)
/dev/loop6 (disk)
/dev/loop7 (disk)
/dev/ram0 (disk)
/dev/ram1 (disk)
/dev/ram10 (disk)
/dev/ram11 (disk)
/dev/ram12 (disk)
/dev/ram13 (disk)
/dev/ram14 (disk)
/dev/ram15 (disk)
/dev/ram2 (disk)
/dev/ram3 (disk)
/dev/ram4 (disk)
/dev/ram5 (disk)
/dev/ram6 (disk)
/dev/ram7 (disk)
/dev/ram8 (disk)
/dev/ram9 (disk)

由于我主要关注的是实际的驱动器,而不是光盘驱动器,环回设备或ram设备,如何将结果过滤到仅获得真实的物理介质?

2 个答案:

答案 0 :(得分:4)

由于udev下的设备管理非常灵活,因此可能取决于/lib/udev/rules.d/etc/udev/rules.d中安装的udev规则,这些规则决定了您可以过滤哪些密钥。

您可以使用udevadm获取设备列表及其相关的udev密钥。

sudo udevadm info --export-db

同样,您的脚本可以修改为输出此信息,如下所示:

#!/usr/bin/env python
import pyudev

context = pyudev.Context()

for device in context.list_devices(subsystem='block', DEVTYPE='disk'):
    for key, value in device.iteritems():
        print '{key}={value}'.format(key=key, value=value)
    print

这将允许您确定可用于过滤的键。在我的Debian系统上,一些条目看起来像这样:

# Loop device
UDEV_LOG=3
DEVPATH=/devices/virtual/block/loop0
MAJOR=7
MINOR=0
DEVNAME=/dev/loop0
DEVTYPE=disk
SUBSYSTEM=block
UDISKS_PRESENTATION_NOPOLICY=1
DEVLINKS=/dev/block/7:0

#cdrom
UDEV_LOG=3
DEVPATH=/devices/pci0000:00/0000:00:01.1/host1/target1:0:0/1:0:0:0/block/sr0
MAJOR=11
MINOR=0
DEVNAME=/dev/sr0
DEVTYPE=disk
SUBSYSTEM=block
ID_CDROM=1
ID_CDROM_DVD=1
ID_CDROM_MRW=1
ID_CDROM_MRW_W=1
ID_SCSI=1
ID_VENDOR=VBOX
ID_VENDOR_ENC=VBOX\x20\x20\x20\x20
ID_MODEL=CD-ROM
ID_MODEL_ENC=CD-ROM\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20
ID_REVISION=1.0
ID_TYPE=cd
ID_BUS=scsi
ID_PATH=pci-0000:00:01.1-scsi-1:0:0:0
GENERATED=1
UDISKS_PRESENTATION_NOPOLICY=0
DEVLINKS=/dev/block/11:0 /dev/scd0 /dev/disk/by-path/pci-0000:00:01.1-scsi-1:0:0:0 /dev/cdrom /dev/dvd
TAGS=:udev-acl:

# hard disk
UDEV_LOG=3
DEVPATH=/devices/pci0000:00/0000:00:0d.0/host2/target2:0:0/2:0:0:0/block/sda
MAJOR=8
MINOR=0
DEVNAME=/dev/sda
DEVTYPE=disk
SUBSYSTEM=block
ID_ATA=1
ID_TYPE=disk
ID_BUS=ata

可以过滤这些键/值对,只返回那些真实的磁盘。例如,UDISKS_PRESENTATION_NOPOLICY条目可用于过滤掉循环设备。您可以仔细阅读udev规则,以确定它如何尝试区分真实/虚拟磁盘。

这样的东西会滤掉cdroms和循环设备:

devices = []
for device in context.list_devices(subsystem='block', DEVTYPE='disk'):
    # Filter out cd drives, loop devices.
    if device.get('ID_TYPE', '') == 'cd':
        continue
    if device.get('UDISKS_PRESENTATION_NOPOLICY', '0') == '1':
        continue
    devices.append(device)

print "HARD DISKS:"

答案 1 :(得分:2)

由于udev仅在Linux内核上运行(至少截至目前),您可以按MAJOR number 8过滤掉,代表所有基于SCSI / SATA磁盘驱动程序的设备。

for device in context.list_devices(MAJOR='8'):
    if (device.device_type == 'disk'):
        print "{}, ({})".format(device.device_node, device.device_type)

在我的系统上,您的代码输出以下内容:

/dev/sda, (disk)
/dev/sdf, (disk)
/dev/sdb, (disk)
/dev/sdc, (disk)
/dev/sdd, (disk)
/dev/sde, (disk)
/dev/sr0, (disk)
/dev/loop0, (disk)
/dev/loop1, (disk)
/dev/loop2, (disk)
/dev/loop3, (disk)
/dev/loop4, (disk)
/dev/loop5, (disk)
/dev/loop6, (disk)
/dev/loop7, (disk)

按主要编号8过滤后,我看到以下输出:

/dev/sda, (disk)
/dev/sdf, (disk)
/dev/sdb, (disk)
/dev/sdc, (disk)
/dev/sdd, (disk)
/dev/sde, (disk)

请注意,您还可以在列表中获得USB硬盘和USB记忆棒,因为它们也倾向于使用相同的SCSI磁盘驱动程序。

我不确定IDE硬盘是否被映射为sdXhdX与最新的2.6或3.x内核。没有IDE硬盘驱动器可以验证,并且我有一个。 :d

更新:相同的设备编号页面列出了/dev/hdX是IDE硬盘驱动器使用的版本(也可能是IDE cdroms?)。如果你想过滤这些,我相信你可以这样做:

for device in context.list_devices(DEVTYPE='disk'):
    major = device['MAJOR']
    if major == '8' or major == '3':
        print "{}, ({})".format(device.device_node, device.device_type)