Назад | Перейти на главную страницу

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

Я хочу получить подробную информацию о состоянии службы (загружен, включен, активен, запущен, с момента, основной 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, если это возможно?

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

systemctl show servicename --no-page

Эта команда выводит данные в key=value формат:

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

Ты можешь использовать 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

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

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