작업을 시작하기 전에 특정 환경 변수가 설정되어 있는지 확인 해야하는 유닉스 셸 스크립트가 몇 가지 있습니다.
if [ -z "$STATE" ]; then
echo "Need to set STATE"
exit 1
fi
if [ -z "$DEST" ]; then
echo "Need to set DEST"
exit 1
fi
많은 타이핑입니다. 환경 변수 세트가 설정되어 있는지 확인하기위한보다 우아한 관용구가 있습니까?
편집 :이 변수에는 의미있는 기본값이 없다는 것을 언급해야합니다. 설정되어 있지 않으면 스크립트가 오류가 발생합니다.
답변
파라미터 확장
분명한 대답은 특수한 형태의 매개 변수 확장 중 하나를 사용하는 것입니다.
: ${STATE?"Need to set STATE"}
: ${DEST:?"Need to set DEST non-empty"}
또는 더 좋습니다 (아래의 ‘큰 따옴표 위치’섹션 참조).
: "${STATE?Need to set STATE}"
: "${DEST:?Need to set DEST non-empty}"
첫 번째 변형 ( ?
) 만 사용 하려면 STATE를 설정해야하지만 STATE = “”(빈 문자열)은 정상입니다. 정확히 원하는 것이 아니라 대체 및 이전 표기법입니다.
두 번째 변형 (을 사용 :?
)은 DEST가 설정되고 비어 있지 않아야합니다.
메시지를 제공하지 않으면 쉘은 기본 메시지를 제공합니다.
${var?}
구조는 버전 7 UNIX 및 Bourne 쉘 (1978 또는 그 부근)에 이식 돌아왔다. 이 ${var:?}
구성은 약간 더 최신입니다. 1981 년경 System III UNIX에 있었지만 그 전에는 PWB UNIX에 있었던 것 같습니다. 따라서 Korn 쉘 및 특히 Bash를 포함한 POSIX 쉘에 있습니다.
일반적으로 쉘의 매뉴얼 페이지에 Parameter Expansion 이라는 섹션에 설명되어 있습니다. 예를 들어, bash
매뉴얼은 다음 과 같이 말합니다.
${parameter:?word}
널 또는 설정되지 않은 경우 오류 표시 매개 변수가 널 (NULL)이거나 설정되지 않은 경우, 단어 확장 (또는 단어가없는 경우 해당 효과에 대한 메시지)이 표준 오류에 기록되고 쉘이 대화식이 아닌 경우 종료됩니다. 그렇지 않으면 매개 변수 값이 대체됩니다.
콜론 사령부
콜론 명령은 단순히 인수를 평가 한 다음 성공한다고 덧붙여 야합니다. 원래 쉘 주석 표기법입니다 ( ‘ #
‘ 이전 줄 끝). 오랫동안 Bourne 쉘 스크립트에는 첫 문자로 콜론이있었습니다. C 셸은 스크립트를 읽고 첫 번째 문자를 사용하여 C 셸 ( ‘ #
‘해시) 또는 Bourne 셸 ( ‘ :
‘콜론) 에 해당하는지 확인합니다 . 그런 다음 커널은 행동을 #!/path/to/program
취하고 ‘ ‘에 대한 지원을 추가 했으며 Bourne 쉘은 ‘ #
‘의견을 얻었 으며 콜론 규칙은 길가에있었습니다. 그러나 콜론으로 시작하는 스크립트를 발견하면 이유를 알 수 있습니다.
큰 따옴표의 위치
이 토론에 대한 의견이 있으십니까? https://github.com/koalaman/shellcheck/issues/380#issuecomment-145872749
토론의 요지는 :
… 그러나
shellcheck
(버전 0.4.1)이 메시지가 나타납니다.In script.sh line 13: : ${FOO:?"The environment variable 'FOO' must be set and non-empty"} ^-- SC2086: Double quote to prevent globbing and word splitting.
이 경우 내가해야 할 일에 대한 조언이 있습니까?
짧은 대답은 ” shellcheck
제안대로”입니다.
: "${STATE?Need to set STATE}"
: "${DEST:?Need to set DEST non-empty}"
이유를 설명하기 위해 다음을 연구하십시오. 점을 유의 :
명령 인수를 (하지만 쉘이 인수를 평가 않음) 에코하지 않습니다. 우리는 인수를보고 싶기 때문에 아래 코드 printf "%s\n"
는 대신에 사용 됩니다 :
.
$ mkdir junk
$ cd junk
$ > abc
$ > def
$ > ghi
$
$ x="*"
$ printf "%s\n" ${x:?You must set x} # Careless; not recommended
abc
def
ghi
$ unset x
$ printf "%s\n" ${x:?You must set x} # Careless; not recommended
bash: x: You must set x
$ printf "%s\n" "${x:?You must set x}" # Careful: should be used
bash: x: You must set x
$ x="*"
$ printf "%s\n" "${x:?You must set x}" # Careful: should be used
*
$ printf "%s\n" ${x:?"You must set x"} # Not quite careful enough
abc
def
ghi
$ x=
$ printf "%s\n" ${x:?"You must set x"} # Not quite careful enough
bash: x: You must set x
$ unset x
$ printf "%s\n" ${x:?"You must set x"} # Not quite careful enough
bash: x: You must set x
$
전체 표현식이 큰 따옴표로 묶이지 않은 경우 값 $x
이 먼저 확장 된 *
다음 파일 이름 목록으로 확장되는 방법에 유의하십시오 . 이것이 shellcheck
권장되는 사항입니다. 표현식이 큰 따옴표로 묶인 양식에 반대하지 않는지 확인하지는 않았지만 괜찮을 것이라는 합리적인 가정입니다.
답변
이 시도:
[ -z "$STATE" ] && echo "Need to set STATE" && exit 1;
답변
귀하의 질문은 사용중인 쉘에 따라 다릅니다.
Bourne shell은 당신이 쫓는 방식에 거의 영향을 미치지 않습니다.
그러나…
거의 모든 곳에서 작동합니다.
그냥 csh에서 멀리 떨어지십시오. 그것은 Bourne 껍질과 비교하여 종과 휘파람에 좋았지 만 지금은 삐걱 거리고 있습니다. 당신이 나를 믿지 않는다면, 그냥 csh에서 STDERR을 분리하십시오! (-:
여기에는 두 가지 가능성이 있습니다. 위의 예, 즉 다음을 사용합니다.
${MyVariable:=SomeDefault}
처음으로 $ MyVariable을 참조해야합니다. 이것은 env를 취합니다. var MyVariable이며 현재 설정되어 있지 않은 경우 나중에 사용할 수 있도록 SomeDefault 값을 변수에 할당합니다.
또한 다음과 같은 가능성이 있습니다.
${MyVariable:-SomeDefault}
이 구문을 사용하는 변수 대신 SomeDefault를 대체합니다. 변수에 SomeDefault 값을 지정하지 않으며이 명령문이 발생한 후에도 MyVariable의 값은 여전히 널입니다.
답변
확실히 가장 간단한 방법은 -u
스위치를 shebang (스크립트 상단의 줄)에 추가하는 것입니다 bash
.
#!/bin/sh -u
바인딩되지 않은 변수가 숨어 있으면 스크립트가 종료됩니다.
답변
${MyVariable:=SomeDefault}
MyVariable
설정되고 null이 아닌 경우 변수 값이 재설정됩니다 (= 아무 일도 일어나지 않음).
그렇지 않으면 MyVariable
로 설정되어 SomeDefault
있습니다.
위의 방법은을 실행하려고 시도 ${MyVariable}
하므로 변수를 설정하려는 경우 :
MyVariable=${MyVariable:=SomeDefault}
답변
제 생각에는 #! / bin / sh에 대한 가장 간단하고 호환 가능한 검사 는 다음과 같습니다.
if [ "$MYVAR" = "" ]
then
echo "Does not exist"
else
echo "Exists"
fi
이 역시 / bin / sh 용이며 이전 Solaris 시스템에서도 호환됩니다.
답변
bash
4.2 -v
는 이름이 빈 문자열을 포함 하여 어떤 값으로 설정되어 있는지 테스트 하는 연산자를 도입했습니다 .
$ unset a
$ b=
$ c=
$ [[ -v a ]] && echo "a is set"
$ [[ -v b ]] && echo "b is set"
b is set
$ [[ -v c ]] && echo "c is set"
c is set