我如何在Python中用Linux监听'usb device inserted'事件?

时间:2009-01-22 14:10:00

标签: python linux usb

我想在Linux中为Amarok编写一个Python脚本,以自动将stackoverflow播客复制到我的播放器。当我插入播放器时,它会挂载驱动器,复制任何待处理的播客,然后弹出播放器。我该如何收听“插入”事件?我已经查看过了,但找不到一个好的例子。

4 个答案:

答案 0 :(得分:53)

更新:正如评论中所述,Hal在最近的发行版中不受支持,标准现在是udev,这是一个利用glib循环和 udev ,我保留Hal版本的历史原因。

这基本上是example in the pyudev documentation,适用于旧版本,并且对于glib循环,请注意应根据您的特定需求定制过滤器:

import glib

from pyudev import Context, Monitor

try:
    from pyudev.glib import MonitorObserver

    def device_event(observer, device):
        print 'event {0} on device {1}'.format(device.action, device)
except:
    from pyudev.glib import GUDevMonitorObserver as MonitorObserver

    def device_event(observer, action, device):
        print 'event {0} on device {1}'.format(action, device)

context = Context()
monitor = Monitor.from_netlink(context)

monitor.filter_by(subsystem='usb')
observer = MonitorObserver(monitor)

observer.connect('device-event', device_event)
monitor.start()

glib.MainLoop().run()

使用Hal和d-bus的旧版本:

您可以使用D-Bus绑定并收听DeviceAddedDeviceRemoved信号。 您必须检查已添加设备的功能,才能选择存储设备。

这是一个小例子,您可以删除评论并尝试。

import dbus
import gobject

class DeviceAddedListener:
    def __init__(self):

您需要使用系统总线连接到Hal Manager。

        self.bus = dbus.SystemBus()
        self.hal_manager_obj = self.bus.get_object(
                                              "org.freedesktop.Hal", 
                                              "/org/freedesktop/Hal/Manager")
        self.hal_manager = dbus.Interface(self.hal_manager_obj,
                                          "org.freedesktop.Hal.Manager")

您需要将听众连接到您感兴趣的信号,在本例中为DeviceAdded

        self.hal_manager.connect_to_signal("DeviceAdded", self._filter)

我正在使用基于功能的过滤器。它将接受任何volume并将调用do_something,如果您可以阅读Hal文档以找到更适合您需求的查询,或者有关Hal设备属性的更多信息。

    def _filter(self, udi):
        device_obj = self.bus.get_object ("org.freedesktop.Hal", udi)
        device = dbus.Interface(device_obj, "org.freedesktop.Hal.Device")

        if device.QueryCapability("volume"):
            return self.do_something(device)

显示有关卷的一些信息的示例函数:

     def do_something(self, volume):
        device_file = volume.GetProperty("block.device")
        label = volume.GetProperty("volume.label")
        fstype = volume.GetProperty("volume.fstype")
        mounted = volume.GetProperty("volume.is_mounted")
        mount_point = volume.GetProperty("volume.mount_point")
        try:
            size = volume.GetProperty("volume.size")
        except:
            size = 0

        print "New storage device detectec:"
        print "  device_file: %s" % device_file
        print "  label: %s" % label
        print "  fstype: %s" % fstype
        if mounted:
            print "  mount_point: %s" % mount_point
        else:
            print "  not mounted"
        print "  size: %s (%.2fGB)" % (size, float(size) / 1024**3)

if __name__ == '__main__':
    from dbus.mainloop.glib import DBusGMainLoop
    DBusGMainLoop(set_as_default=True)
    loop = gobject.MainLoop()
    DeviceAddedListener()
    loop.run()

答案 1 :(得分:7)

我自己没有尝试过编写这样的程序,但是我只看了以下两个链接(感谢Google!),我认为这会有所帮助:

请特别阅读org.freedesktop.Hal.Manager界面及其DeviceAddedDeviceRemoved事件。 : - )

希望这有帮助!

答案 2 :(得分:4)

我认为D-Bus可以像Chris提到的那样工作,但是如果您使用的是KDE4,则可以使用类似于KDE4“New Device Notifier”小程序的方式使用Solid框架。

该applet的C ++源代码为here,其中显示了如何使用Solid检测新设备。使用PyKDE4进行Python绑定到这些库,如here所示。

答案 3 :(得分:4)

以下是5行的解决方案。

import pyudev

context = pyudev.Context()
monitor = pyudev.Monitor.from_netlink(context)
monitor.filter_by(subsystem='usb')

for device in iter(monitor.poll, None):
    if device.action == 'add':
        print('{} connected'.format(device))
        # do something very interesting here.

将其保存到文件usb_monitor.py,然后运行python monitor.py。插入任何USB,它将打印设备详细信息

→ python usb_monitor.py 
Device('/sys/devices/pci0000:00/0000:00:14.0/usb1/1-6/1-6:1.0') connected
Device('/sys/devices/pci0000:00/0000:00:14.0/usb1/1-1/1-1:1.0') connected

使用pyudev==0.21.0在Python 3.5上测试。