web-dev-qa-db-ja.com

Linuxでは、Pythonで「usbデバイス挿入」イベントをリッスンするにはどうすればよいですか?

LinuxでAmarokのPythonスクリプトを記述して、stackoverflowポッドキャストを自動的にプレーヤーにコピーします。プレーヤーをプラグインすると、ドライブがマウントされ、保留中のポッドキャストがコピーされます。 「プラグイン」イベントをどのように聞くことができますか?ハルドを調べましたが、良い例が見つかりませんでした。

35
joeforker

Update:コメントで述べたように、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を呼び出します。その場合、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()
55
Jaime Soriano

私はそのようなプログラムを自分で書いたことはありませんが、次の2つのリンク(Googleに感謝します)を見ただけです。

特に、org.freedesktop.Hal.ManagerインターフェースとそのDeviceAddedおよびDeviceRemovedイベント。 :-)

お役に立てれば!

7

これは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 with pyudev==0.21.0

5
ChillarAnand

D-BusはChrisが述べたように機能すると思いますが、KDE4を使用している場合は、KDE4 "New Device Notifier"アプレットと同様の方法でSolidフレームワークを使用できます。

そのアプレットのC++ソースは here で、Solidを使用して新しいデバイスを検出する方法を示しています。これらのライブラリへのPythonバインディング)にはPyKDE4を使用します ここ のように。

4
Matt J