서비스가 스크립트에서 실행 중인지 테스트하는 "올바른" 방법

서비스가 스크립트에서 실행 중인지 테스트하는 "올바른" 방법

내 질문:

주어진 서비스가 실행 중인지 확인하려는 bash 스크립트를 작성 중입니다.

를 사용하여 수동으로 수행하는 방법을 알고 있습니다 $ service [service_name] status.

그러나 (특히 systemd로 전환한 이후) 구문 분석하기 약간 혼란스러운 많은 텍스트가 인쇄됩니다. 간단한 출력이나 확인할 수 있는 반환 값이 있는 스크립트에 대한 명령이 있다고 가정합니다.

하지만 Google에서 검색하면 "아, 그냥"이라는 결과가 많이 나옵니다 ps aux | grep -v grep | grep [service_name]. 그건 모범 사례가 아니죠? 명령의 다른 인스턴스가 실행 중이지만 SysV init 스크립트에 의해 시작된 인스턴스가 아닌 경우에는 어떻게 됩니까?

아니면 그냥 닥치고 약간의 pgrep으로 손을 더럽혀야 할까요?

답변1

systemctl하나 있다is-active이에 대한 하위 명령:

systemctl is-active --quiet service

활성 상태이면 상태 0으로 종료되고 service그렇지 않으면 0이 아닌 상태로 종료되므로 스크립트에 이상적입니다.

systemctl is-active --quiet service && echo Service is running

생략하면 --quiet현재 상태도 표준 출력에 인쇄됩니다.

일부 유닛은 서비스를 제공하기 위해 실행 중인 것이 아무것도 없어도 활성화될 수 있습니다. "RemainAfterExit"라고 표시된 유닛은 성공적으로 종료되면 활성 상태로 간주됩니다. 아이디어는 데몬이 필요하지 않은 서비스를 제공한다는 것입니다(예를 들어시스템의 특정 측면을 구성합니다. 그러나 데몬과 관련된 장치는 데몬이 계속 실행되는 동안에만 활성화됩니다.

일회용 장치아니요"RemainAfterExit"는 활성 유닛 상태에 들어가지 않으므로 is-active이러한 유닛을 처리하는 데 성공하지 못합니다. is-active텍스트 출력을 구문 분석할 수 있습니다.

systemctl is-active service

현재 실행 중인 일회용 장치에 대해서는 "활성화", 현재 실행 중이 아니지만 마지막으로 성공적으로 실행된(있는 경우) 일회용 장치에 대해서는 "비활성", 현재 실행되지 않는 일회용 장치에 대해서는 "실패"를 출력합니다. 실행 중이며 마지막으로 실행했을 때 실패했습니다. is-active이러한 단위는 항상 0이 아닌 상태를 반환하므로 다음을 실행하십시오.

systemctl is-active service ||:

필요한 경우 무시하십시오.

답변2

systemctl실제로 스크립팅을 위한 패턴이 있습니다. show대신 에 / 및 옵션을 status추가하여 원하는 출력만 얻으십시오.-p--properties--value

다음은 Ubuntu 17.04 시스템의 예입니다.

$ systemctl show -p SubState --value NetworkManager
running

실행 중(또는 기타)은 입니다 SubState. 서비스가 활성 상태인지 알고 싶다면 이 속성을 사용하세요.ActiveState

$ systemctl show -p ActiveState --value x11-common
inactive
$ systemctl show -p SubState --value x11-common
dead

댓글 man:

show [PATTERN...|JOB...]
           Show properties of one or more units, jobs, or the manager
           itself. If no argument is specified, properties of the
           manager will be shown. If a unit name is specified, properties
           of the unit are shown, and if a job ID is specified,
           properties of the job are shown. By default, empty properties
           are suppressed. Use --all to show those too. To select specific
           properties to show, use --property=. This command is intended
           to be used whenever computer-parsable output is required. Use
           status if you are looking for formatted human-readable output.

-p, --property=
           When showing unit/job/manager properties with the show command,
           limit display to properties specified in the argument. The
           argument should be a comma-separated list of property names,
           such as "MainPID". Unless specified, all known properties are
           shown. If specified more than once, all properties with the
           specified names are shown. Shell completion is implemented for
           property names.

--value
           When printing properties with show, only print the value, and
           skip the property name and "=".

서비스의 사용 가능한 속성을 보려면 다음을 실행합니다(예: polkit).

systemctl show -a polkit

LoadState안타깝게도 , ActiveState및 에 대한 가능한 값은 SubState맨페이지에 문서화되어 있지 않고 대신 D-Bus 인터페이스 설명에 문서화되어 있습니다 org.freedesktop.systemd1.https://www.freedesktop.org/software/systemd/man/org.freedesktop.systemd1.html

LoadState장치의 구성 파일이 로드되었는지 여부를 반영하는 상태 값을 포함합니다. 현재 정의된 상태는 " loaded", " error" 및 " masked"입니다. " loaded"는 구성이 성공적으로 로드되었음을 나타냅니다. " error"는 구성 로드에 실패했음을 나타냅니다. 이 LoadError필드(아래 참조)에는 이 실패의 원인에 대한 정보가 포함되어 있습니다. " masked"는 해당 단위가 현재 마스크되어 있음을 나타냅니다(즉, 심볼릭 링크되어 있거나 /dev/null비어 있음). 유효한 로드된 구성이 없는 장치가 활성화될 수 있으므로(아래 참조) LoadState과 완전히 직교합니다 ( ActiveState장치가 이미 활성화된 동안 구성이 다시 로드되었을 수 있으므로).

ActiveState장치가 현재 활성 상태인지 여부를 반영하는 상태 값을 포함합니다. 현재 정의된 상태는 " active", " reloading", " inactive", " failed", " activating" 및 " deactivating"입니다. " active"는 장치가 활성화되었음을 의미합니다(분명히...). " reloading"는 장치가 활성 상태이고 현재 구성을 다시 로드하고 있음을 나타냅니다. " inactive"는 비활성 상태이고 이전 실행이 성공했거나 이전 실행이 아직 발생하지 않았음을 의미합니다. " failed"는 비활성 상태이고 이전 실행이 성공하지 못했음을 의미합니다(이에 대한 자세한 내용은 결과 속성의 서비스와 같은 장치 유형별 인터페이스에서 찾을 수 있습니다. 아래 참조). " activating"는 해당 장치가 이전에 비활성 상태였지만 현재 활성화되고 있음을 나타냅니다. 반면에 " deactivating"는 해당 장치가 현재 비활성화되고 있음을 나타냅니다.

SubStateActiveState셀 유형과 관련된 보다 세분화된 상태를 재정의하지만 이해하는 동일한 상태 머신의 상태를 인코딩합니다. ActiveState6개의 상위 수준 상태 만 다루며, SubState6개의 상위 수준 상태에 매핑된 더 많은 하위 수준 단위 유형 특정 상태를 다룰 수 있습니다. 여러 하위 수준 상태가 동일한 상위 수준 상태에 매핑될 수 있지만 그 반대는 불가능합니다. 모든 상위 수준 상태가 모든 단위 유형에 대해 하위 수준 상태를 갖는 것은 아닙니다. 현재 하위 수준 상태는 여기에 문서화되어 있지 않으며 위에서 설명한 일반적인 상위 수준 상태보다 나중에 확장될 가능성이 높습니다.

거기에는 많은 부동산이 있으므로, 당신이 찾고 있는 것이 무엇인지 알고 있다면...

$ systemctl show - polkit | grep Active
ActiveState=active
ActiveEnterTimestamp=Thu 2020-07-02 07:24:40 IST
ActiveEnterTimestampMonotonic=6682102
ActiveExitTimestamp=
ActiveExitTimestampMonotonic=0

답변3

Zanna의 답변을 보완하기 위해 --value옵션이 도입되었습니다.systemctl show시스템 버전 230. 따라서 일부 배포판(예: debian jessie)에서는 사용하지 못할 수도 있습니다.

이 경우 sed를 사용하여 이 옵션을 에뮬레이트할 수 있습니다.

$ systemctl show -p ActiveState sshd | sed 's/ActiveState=//g'
active
$ systemctl show -p SubState sshd | sed 's/SubState=//g'  
running

답변4

명령줄 실행이나 스크립트 작성 시 이것이 유용하다고 생각합니다.

@StephenKitt에서 복사함

서비스가 다운되었는지 확인하고 서비스를 다시 시작합니다.

systemctl is-active --quiet <service name> || <service name> restart

여기서 ||systemctl의 반환 값이 0이 아닌지 확인합니다. 즉, 작성자가 설명한 대로 활성화되지 않았음을 의미합니다.

관련 정보