How to extract service state via Systemd DBus API?

故事扮演 提交于 2020-08-08 06:45:05

问题


I'm trying to extract openvpn.service status using Systemd D-Bus API.

In [1]: import dbus

In [2]: sysbus = dbus.SystemBus()

In [3]: systemd1 = sysbus.get_object('org.freedesktop.systemd1', '/org/freedesktop/systemd1')

In [4]: manager = dbus.Interface(systemd1, 'org.freedesktop.systemd1.Manager')

In [5]: service = dbus.Interface(systemd1, 'org.freedesktop.systemd1.Service')

In [6]: unit = dbus.Interface(systemd1, 'org.freedesktop.systemd1.Unit')

In [7]: unit.ActiveState('openvpn.service')
---------------------------------------------------------------------------
DBusException                             Traceback (most recent call last)
<ipython-input-7-22857e7dcbd7> in <module>()
----> 1 unit.ActiveState('openvpn.service')

/usr/local/lib/python3.4/dist-packages/dbus/proxies.py in __call__(self, *args, **keywords)
     68             # we're being synchronous, so block
     69             self._block()
---> 70             return self._proxy_method(*args, **keywords)
     71 
     72     def call_async(self, *args, **keywords):

/usr/local/lib/python3.4/dist-packages/dbus/proxies.py in __call__(self, *args, **keywords)
    143                                                   signature,
    144                                                   args,
--> 145                                                   **keywords)
    146 
    147     def call_async(self, *args, **keywords):

/usr/local/lib/python3.4/dist-packages/dbus/connection.py in call_blocking(self, bus_name, object_path, dbus_interface, method, signature, args, timeout, byte_arrays, **kwargs)
    649         # make a blocking call
    650         reply_message = self.send_message_with_reply_and_block(
--> 651             message, timeout)
    652         args_list = reply_message.get_args_list(**get_args_opts)
    653         if len(args_list) == 0:

DBusException: org.freedesktop.DBus.Error.UnknownMethod: Unknown method 'ActiveState' or interface 'org.freedesktop.systemd1.Unit'.

In [8]: manager.GetUnit('openvpn.service')
Out[8]: dbus.ObjectPath('/org/freedesktop/systemd1/unit/openvpn_2eservice')

In [9]: u = manager.GetUnit('openvpn.service')

In [10]: u.ActiveState
---------------------------------------------------------------------------
AttributeError                            Traceback (most recent call last)
<ipython-input-10-6998e589f206> in <module>()
----> 1 u.ActiveState

AttributeError: 'dbus.ObjectPath' object has no attribute 'ActiveState'

In [11]: manager.GetUnitFileState('openvpn.service')
Out[11]: dbus.String('enabled')

There's an ActiveState property, which contains a state value that reflects whether the unit is currently active or not. I've succeeded in reading UnitFileState, however I can't figure out how to read ActiveState property.


回答1:


After having a look at some random examples [more permanent link], I've had some luck with

import dbus
sysbus = dbus.SystemBus()
systemd1 = sysbus.get_object('org.freedesktop.systemd1',
    '/org/freedesktop/systemd1')
manager = dbus.Interface(systemd1, 'org.freedesktop.systemd1.Manager')
service = sysbus.get_object('org.freedesktop.systemd1',
    object_path=manager.GetUnit('openvpn.service'))
interface = dbus.Interface(service,
    dbus_interface='org.freedesktop.DBus.Properties')
print(interface.Get('org.freedesktop.systemd1.Unit', 'ActiveState'))

[Edit:] Documenting the versions, just in case:

dbus.version == (1, 2, 14)
systemd 244 (244.1-1-arch)
Python 3.8.1



回答2:


You probably need to call dbus.Interface(sysbus.get_object('org.freedesktop.systemd1', u), 'org.freedesktop.systemd1.Unit') and access the ActiveState property on that object. Your code currently tries to access the ActiveState property on an object path, which is not the object itself.



来源:https://stackoverflow.com/questions/43499880/how-to-extract-service-state-via-systemd-dbus-api

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!