Я хотел бы написать Python script для Amarok в Linux, чтобы автоматически скопировать подкаст stackoverflow на мой плеер. Когда я подключаю плеер, он монтирует диск, копирует любые ожидающие подкасты и выталкивает плеер. Как я могу прослушать событие "подключен"? Я просмотрел hald, но не смог найти хороший пример.
Как я могу прослушивать события "usb device insert" в Linux, на Python?
Ответ 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!), которые, я думаю, помогут:
- учебник dbus-python (в котором говорится о том, как использовать Python для доступа к D-Bus)
- HAL 0.5.10 Спецификация (в котором говорится о том, как HAL публикует события для D-Bus)
В частности, читайте о интерфейсе 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
.