37

Linux で Amarok 用の Python スクリプトを作成して、stackoverflow ポッドキャストをプレーヤーに自動的にコピーしたいと考えています。プレーヤーを接続すると、ドライブがマウントされ、保留中のポッドキャストがあればコピーされ、プレーヤーが取り出されます。「プラグイン」イベントをリッスンするにはどうすればよいですか? hald を調べましたが、良い例が見つかりませんでした。

4

4 に答える 4

58

更新: コメントで述べたように、Hal は最近のディストリビューションではサポートされていません。現在の標準は udev です。glib ループとudevを利用する小さな例を次に示します。歴史的な理由から Hal バージョンを保持します。

これは基本的にpyudev documentation の例であり、古いバージョンで動作するように調整されており、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 バインディングを使用してDeviceAddedDeviceRemovedシグナルをリッスンできます。ストレージ デバイスのみを選択するには、追加されたデバイスの機能を確認する必要があります。

これは小さな例です。コメントを削除して試してみてください。

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)

機能に基づくフィルターを使用しています。Any を受け入れ、if で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()
于 2009-01-22T22:27:58.820 に答える
6

これが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

于 2016-09-02T09:13:16.737 に答える
4

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

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

于 2009-01-22T15:15:46.133 に答える