如何在 Python 中的 Linux 中监听“USB 设备插入”事件?
- 2024-10-28 08:37:00
- admin 原创
- 64
问题描述:
我想为 Linux 中的 Amarok 编写一个 Python 脚本,以便自动将 stackoverflow 播客复制到我的播放器。当我插入播放器时,它会安装驱动器、复制任何待处理的播客并弹出播放器。我如何监听“插入”事件?我查看了 hald,但找不到一个很好的例子。
解决方案 1:
更新:正如评论中所说,Hal 在最近的发行版中不受支持,现在的标准是 udev,这是一个使用 glib loop 和udev的小例子,出于历史原因,我保留了 Hal 版本。
这基本上是pyudev 文档中的示例,适用于旧版本,并使用 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 绑定并监听DeviceAdded
信号DeviceRemoved
。您必须检查添加的设备的功能,以便仅选择存储设备。
这里有一个小例子,你可以去掉注释然后尝试一下。
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
使用 if 进行调用,您可以阅读 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()
解决方案 2:
以下是 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
在 Python 3.5 上进行了测试pyudev==0.21.0
。
解决方案 3:
我自己还没有尝试编写这样的程序,但是我刚刚看了以下两个链接(感谢谷歌!),我认为它们会有所帮助:
dbus-python 教程(介绍如何使用 Python 访问 D-Bus)
HAL 0.5.10 规范(讨论 HAL 如何将事件发布到 D-Bus)
特别是,阅读有关org.freedesktop.Hal.Manager
界面及其DeviceAdded
事件的内容DeviceRemoved
。:-)
希望这有帮助!
解决方案 4:
我认为 D-Bus 会像 Chris 提到的那样工作,但如果您使用 KDE4,您可能会以类似于 KDE4“新设备通知程序”小程序的方式使用 Solid 框架。
该小程序的 C++ 源代码在此处,其中展示了如何使用 Solid 检测新设备。使用 PyKDE4 将这些库与 Python 绑定,如下所示。
- 2024年20款好用的项目管理软件推荐,项目管理提效的20个工具和技巧
- 2024年开源项目管理软件有哪些?推荐5款好用的项目管理工具
- 项目管理软件有哪些?推荐7款超好用的项目管理工具
- 项目管理软件哪个最好用?盘点推荐5款好用的项目管理工具
- 项目管理软件有哪些最好用?推荐6款好用的项目管理工具
- 项目管理软件有哪些,盘点推荐国内外超好用的7款项目管理工具
- 2024项目管理软件排行榜(10类常用的项目管理工具全推荐)
- 项目管理软件排行榜:2024年项目经理必备5款开源项目管理软件汇总
- 2024年常用的项目管理软件有哪些?推荐这10款国内外好用的项目管理工具
- 项目管理必备:盘点2024年13款好用的项目管理软件