Как получить статус systemd в формате json?

Я хочу получить сведения о состоянии службы (загружено, включено, активно , работает, так как, Main PID) в машиночитаемой форме, и я знаю, что инструменты systemd имеют параметр - output = json , но если я это сделаю:

 systemctl status servicename --output=json --plain

я вижу что-то вроде:

● snapd.service - Snappy daemon
   Loaded: loaded (/lib/systemd/system/snapd.service; enabled; vendor preset: enabled)
   Active: active (running) since Mon 2018-04-16 11:20:07 MSK; 4h 45min ago
 Main PID: 738 (snapd)
    Tasks: 10 (limit: 4915)
   CGroup: /system.slice/snapd.service
           └─738 /usr/lib/snapd/snapd

{ "__CURSOR" : "s=461ecb6a4b814913acd57572cd1e1c82;...

Записи журнала находятся в JSON. Но как получить статус сервиса в JSON, если это возможно?

6
задан 16 April 2018 в 16:23
3 ответа

Простой способ получения статуса службы в машиночитаемой форме:

systemctl show servicename --no-page

Эта команда выводит данные в формате ключ = значение :

Type=notify
Restart=always
NotifyAccess=main
...

Если JSON , вы можете использовать следующий простой скрипт Python (совместимый с Python 2 и 3) get_service_info.py :

import os, sys, subprocess, json
key_value = subprocess.check_output(["systemctl", "show", sys.argv[1]], universal_newlines=True).split('\n')
json_dict = {}
for entry in key_value:
    kv = entry.split("=", 1)
    if len(kv) == 2:
        json_dict[kv[0]] = kv[1]
json.dump(json_dict, sys.stdout)

Использование:

get_service_info.py servicename
5
ответ дан 3 December 2019 в 00:12

Разработчики рекомендуют вместо этого использовать dbus API для программного доступа к статусу службы. Если вам действительно нужен JSON, вы можете адаптировать DBUS API для создания нужного JSON.

См. Проблему Github - вывод json не дает вывода json для обсуждения разработчиками того, почему API DBUS рекомендуется вместо вывода JSON.

2
ответ дан 3 December 2019 в 00:12

Для этого также можно использовать jq :

systemctl show --no-page iptables \
    | jq --slurp --raw-input \
        'split("\n")
        | map(select(. != "") 
        | split("=") 
        | {"key": .[0], "value": (.[1:] | join("="))})
        | from_entries'

производит:

{
  "Type": "oneshot",
  "Restart": "no",
  "NotifyAccess": "none",
  "ExecStart": "{ path=/usr/libexec/iptables/iptables.init ; argv[]=/usr/libexec/iptables/iptables.init start ; ignore_errors=no ; start_time=[n/a] ; stop_time=[n/a] ; pid=0 ; code=(null) ; status=0/0 }",
  [... like 60 more entries ...],
  "CollectMode": "inactive"
}

Декодирование команды jq :

--slurp - read the whole thing as a really big string
--raw-input - p.s. it's not json input
split("\n") - break the input into an array of lines
map( ... )  - perform some transformation on each array element
select(. != "") - skip blank lines
split("=") - split the current line into array elements at each "="
(.[1:] | join("=")) - avoid mangling values with equal signs

(будьте осторожны : это приведет к усечению значений, если само значение содержит знак равенства. Это, вероятно, можно обойти, но для этой цели кажется прекрасным)

{"key": .[0], "value": .[1]}  - build an key/value pair object

(на данный момент map возвращает массив объектов ключ / значение)

from_entries - turn an array of "key"/"value" objects into an object
4
ответ дан 3 December 2019 в 00:12

Теги

Похожие вопросы