Как я могу прослушивать события «вставлено USB-устройство» в Linux, в Python? - PullRequest
35 голосов
/ 22 января 2009

Я хотел бы написать скрипт Python для Amarok в Linux, чтобы автоматически копировать подкаст stackoverflow в мой проигрыватель. Когда я подключаю плеер, он монтирует диск, копирует все ожидающие подкасты и извлекает плеер. Как я могу прослушать событие «подключен»? Я просмотрел Hald, но не смог найти хороший пример.

Ответы [ 4 ]

53 голосов
/ 23 января 2009

Обновление : Как сказано в комментариях, Hal не поддерживается в последних дистрибутивах, теперь стандартом является udev. Вот небольшой пример, который использует цикл glib и 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()
7 голосов
/ 22 января 2009

Я сам не пытался написать такую ​​программу, однако я только что посмотрел следующие две ссылки (спасибо Google!), Которые, я думаю, будут полезны:

В частности, прочитайте об интерфейсе org.freedesktop.Hal.Manager и его событиях DeviceAdded и DeviceRemoved. : -)

Надеюсь, это поможет!

4 голосов
/ 02 сентября 2016

Вот решение в 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.

4 голосов
/ 22 января 2009

Я думаю, что D-Bus будет работать, как упоминал Крис, но если вы используете KDE4, вы можете использовать платформу Solid таким же образом, как в апплете "New Device Notifier" KDE4.

Источник C ++ для этого апплета - здесь , который показывает, как использовать Solid для обнаружения новых устройств. Используйте PyKDE4 для привязок Python к этим библиотекам, как показано здесь .

...