프로세스를 종료할 때 불법 PID가 표시되는 이유는 무엇입니까?

프로세스를 종료할 때 불법 PID가 표시되는 이유는 무엇입니까?

이제 이 스크립트를 사용하여 프로세스를 다시 시작합니다.

PID=`ps -ef|grep -w ${APP_NAME}|grep -v grep|cut -c 9-15`
if [[ ${PID} -gt 1 ]]; then
  kill -9 ${PID}
else
  echo "Process not found"
fi

하지만 GitHub Actions의 원격 서버에서 이 스크립트를 실행하면 다음 오류가 표시됩니다.

======CMD======
cd /opt/apps/dolphin-acientbay/libs
. /opt/apps/dolphin-acientbay/libs/upgrade.sh

======END======
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:13> JAVA_HOME=/***/.sdkman/candidates/java/11.0.11.hs-adpt 
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:14> APP_HOME=/opt/apps/dolphin-acientbay/libs 
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:15> APP_NAME=dolphin-acientbay-service-1.0.0-SNAPSHOT.jar 
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:17> PID=+/opt/apps/dolphin-acientbay/libs/upgrade.sh:17> ps -ef
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:17> PID=+/opt/apps/dolphin-acientbay/libs/upgrade.sh:17> grep -w dolphin-acientbay-service-1.0.0-SNAPSHOT.jar
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:17> PID=+/opt/apps/dolphin-acientbay/libs/upgrade.sh:17> grep -v grep
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:17> PID=+/opt/apps/dolphin-acientbay/libs/upgrade.sh:17> cut -c 9-15
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:17> PID=' 19882 ' 
2021/05/30 11:46:21 Process exited with status 1
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:18> [[ ' 19882 ' -gt 1 ]]
err: +/opt/apps/dolphin-acientbay/libs/upgrade.sh:19> kill -9 ' 19882 '
err: /opt/apps/dolphin-acientbay/libs/upgrade.sh:kill:19: illegal pid:  19882 

전체 스크립트는 다음과 같습니다 upgrade.sh.

#!/usr/bin/env bash

set -u

set -e

set -x

JAVA_HOME="/root/.sdkman/candidates/java/11.0.11.hs-adpt"
APP_HOME="/opt/apps/dolphin-acientbay/libs"
APP_NAME="dolphin-acientbay-service-1.0.0-SNAPSHOT.jar"

PID=`ps -ef|grep -w ${APP_NAME}|grep -v grep|cut -c 9-15`
if [[ ${PID} -gt 1 ]]; then
  kill -9 ${PID}
else
  echo "Process not found"
fi

sleep 5

count=`ps -ef | grep ${APP_NAME} | grep -v "grep" | wc -l`
if [[ ${count} -lt 1 ]]; then
  nohup ${JAVA_HOME}/bin/java -Xmx128M -Xms128M -jar \
  -Xdebug -Xrunjdwp:transport=dt_socket,suspend=n,server=y,address=0.0.0.0:5021 \
  -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=/opt/apps/dolphin-acientbay/ \
  ${APP_HOME}/${APP_NAME} >> ./acientbay.log &
  sleep 5
else
  echo "process aready exists!"
  exit 1
fi

pid에 대한 공백을 제거하기 위해 다음과 같이 kill process 명령을 조정해 보았습니다(xargs는 pid에 대한 공백 자르기).

 echo "${PID}" | xargs | kill -9

그것은 나에게 말한다 :

err: /opt/apps/dolphin-acientbay/libs/upgrade.sh:kill:19: not enough arguments

답변1

이것이 문제의 원인입니다:

PID=`ps -ef|grep -w ${APP_NAME}|grep -v grep|cut -c 9-15`
  1. 변수를 사용할 때는 큰따옴표를 사용하세요. 예를 들어 "$APP_NAME" 또는 "${APP_NAME}"입니다.

    그런데 중괄호는 문자열의 다른 텍스트와 변수를 구별해야 하는 경우에만 필요합니다. 예를 들어, 변수가 실제로 있지만 $APP문자열에서 사용해야 하는 경우 _NAME사용할 수 있습니다. "${APP}_NAME"이렇게 하면 _NAME쉘이 해당 변수를 변수 이름의 일부로 해석하지 못하게 됩니다.

  2. 대신 백틱을 사용하고 있습니다 $(). 이는 수년간 더 이상 사용되지 않으며 그럴 만한 이유가 있습니다. 그것은 문제의 원인이 아니라, 버려야 할 나쁜 습관일 뿐입니다.

  3. cut1개 이상의 문자로 구분될 수 있는 필드를 추출하는 데는 좋은 도구가 아닙니다. 필드 사이에 구분 기호가 하나만 있는 경우에만 유효합니다. 많은 텍스트 파일은 필드 구분 기호로 1개 이상의 공백(및/또는 탭 및/또는 기타 공백 문자)을 사용하므로 또는 대신 쉽게 사용할 수 없습니다 cut.awkperl

    이를 사용하면 cut -c 9-15$PID에 최소한 하나의 추가 공백 문자가 캡처됩니다. PID를 추출하려면 대신 를 ps -ef사용하십시오 .awk '{print $2}'cut -c 9-15

  4. pgrep "$APP_NAME"이름으로 프로세스의 PID를 얻는 데 사용됩니다 . 또는 pgrep -f "$APP_NAME"검색 중인 문자열이 인수인 경우(예: 스크립트 이름이 인터프리터에 인수로 전달되는 경우)

    화물 숭배는 ps | grep ... | grep -v grep수십 년 동안 유행에 뒤떨어져 왔으며 어떤 면에서도 좋은 방식은 아닙니다. 항상 더 나은 접근 방식이 있었지만( ps -ef | awk '/[p]rocess_name/ {print $2}'패턴의 첫 번째 문자 주위에 를 배치하면 awk, grep 또는 기타 항목이 ps 출력에서 ​​일치하는 것을 방지할 수 있음) 이제 그 접근 방식도 .[]pgrep

    ps자체에는 -C프로세스 이름 일치 옵션, h헤더 억제 옵션, -o원하는 출력 지정 옵션이 있습니다 ps. 예를 들어ps h -o pid -C "$APP_NAME"

요약하면 다음을 사용합니다.

    PID=$(pgrep "$APP_NAME")
or
    PID=$(pgrep -f "$APP_NAME")
or
    PID=$(ps h -o pid -C "$APP_NAME")

그런데 만약 있다면어느pgrep여러 PID를 반환할 가능성이 있으므로 ps출력을 스칼라 변수가 아닌 배열로 캡처해야 합니다. 예를 들어, 다음 명령은 표시되는 모든 bash 프로세스의 PID를 배열로 캡처합니다 $BASHPIDS.

$ BASHPIDS=( $(ps h -o pid -C bash) )

$ typeset -p BASHPIDS
declare -a BASHPIDS=([0]="68910" [1]="71059" [2]="71634" [3]="71641" [4]="71643"
[5]="71680" [6]="71683" [7]="71684" [8]="71687" [9]="71693" [10]="71712" [11]="72394"
[12]="72568" [13]="72589" [14]="970222" [15]="974740" [16]="1078757" [17]="1278073"
[18]="1365082" [19]="1405642" [20]="1458889" [21]="2278763" [22]="2466442" [23]="2876831"
[24]="2955565" [25]="3260896" [26]="3261235" [27]="3269020" [28]="3281961" [29]="3702104" 
[30]="4038149")

어떤 이유로 그들을 모두 죽이고 싶다면 다음과 같이 할 수 있습니다.

kill "${BASHPIDS[@]}"

답변2

이 명령으로 문제가 해결되었습니다.

 echo "${PID}" | xargs kill -9

관련 정보