Подтвердить что ты не робот

Как я могу прослушивать события "usb device insert" в Linux, на Python?

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

4b9b3361

Ответ 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 с помощью System Bus.

        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()

Ответ 2

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

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

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

Ответ 3

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

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

Ответ 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

Протестировано на Python 3.5 с помощью pyudev==0.21.0.