터미널에서 가져온 이 스크립트는 _cphetzner
git 저장소를 생성하는 기능을 제공합니다. 이 함수는 _feed_variable
사용자에게 누락된 변수를 묻는 또 다른 함수를 사용합니다.
이 시점에서 내 질문은 다음과 같습니다.
- 이것이 함수에서 복귀하는 올바른 방법입니까
return 1
? - 주석을 제거 하면
set -o errexit
스크립트가 무엇이든return 1
(예: 함수_feed_variable
또는 셸로 반환되는 함수에서) 중단됩니까? - 주석 처리를 제거한 경우
set -o errexit
터미널을 닫지 않고 기능을 종료하는 올바른 방법은 무엇입니까? - 오류를 해결하는 방식으로 스크립트를 디자인하지 않은 것 같습니다. 어떤 제안이 있으십니까?
#!/usr/bin/env bash
# +---------------+
# | Bash settings |
# +---------------+
# abort on nonzero exitstatus
# set -o errexit
# abort on unbound variable
# set -o nounset
# don't hide errors within pipes
# set -o pipefail
# +----------------+
# | Bash Variables |
# +----------------+
# +----------------+
# | Script Content |
# +----------------+
function _feed_variable() {
local prompt_msg=$1
local var_to_set=$2
echo "$prompt_msg"
read answer
case $answer in
[Yy]*)
echo "Enter $2: "
read new_value
eval $var_to_set="'$new_value'"
return 0
;;
[Nn]*)
echo "Clearing up and quitting ..."
return 1
;;
esac
}
function _cphetzner() {
pushd $PWD > /dev/null
POSITIONAL_ARGS=()
REPO_NAME=
# TARGET_FOLDER=$PWD/$REPO_NAME
TARGET_FOLDER=
SOURCE_TEMPLATE=
while [[ $# -gt 0 ]]; do
case $1 in
-r|--repository)
REPO_NAME="$2"
shift
shift
;;
-s|--sourcetemplate)
SOURCE_TEMPLATE="$2"
shift
shift
;;
-t|--targetfolder)
TARGET_FOLDER="$2"
shift
shift
;;
-*|--*)
echo "Unknown option $1"
exit 1
;;
*)
POSITIONAL_ARGS+=("$1") # save positional arg
shift # past argument
;;
esac
done
#=====[ Print Info ]==================================
echo "SOURCE_TEMPLATE is:\t" $SOURCE_TEMPLATE
echo "TARGET_FOLDER is:\t" $TARGET_FOLDER
echo "{POSITIONAL_ARGS is [@]}:\t" ${POSITIONAL_ARGS[@]}
echo "REPO_NAME is:\t" $REPO_NAME
set -- "${POSITIONAL_ARGS[@]}"
if [[ -n $1 ]]; then
echo "Last line of file specified as non-opt/last argument:"
tail -1 "$1"
fi
#=====[ Create folder and initiate repository ]=======
# Check if user input has repo name
if [[ -z "$REPO_NAME" ]]; then
_feed_variable "Do you want to feed me a repository name? [y/n]" "REPO_NAME" ||
return 1
fi
# Check if repository in given path exists
if [[ -z "$TARGET_FOLDER" ]]; then
_feed_variable "Do you want to feed me a target name? [y/n]" "TARGET_FOLDER" ||
return 1
fi
while [[ -d "${TARGET_FOLDER}/${REPO_NAME}" ]]
do
echo "Repository already exist in this folder" &&
_feed_variable "Do you want to feed me another target name? [y/n]" "TARGET_FOLDER" ||
return 1
done
# Create folder
mkdir -p $TARGET_FOLDER/$REPO_NAME
# # Check if template folder exists
# [[ -z $SOURCE_TEMPLATE ]] && echo "No template folder" ||
# cp -a $SOURCE_TEMPLATE/ $TARGET_FOLDER/$REPO_NAME
#====[ Initiate Repo ]================================
cd $TARGET_FOLDER/$REPO_NAME
git init
git remote add origin hetzner:/yolo/$REPO_NAME.git
git add .
git commit -m "Initial commit"
echo -e "[DONE] Repo $REPO_NAME created successfully"
echo -ne "Do you want to push repo to Hetzner now? [y/n]"
read answer
case $answer in
[Yy]*)
echo "Pushing to Hetzner ..."
git push -u -f origin master
;;
[Nn]*)
;;
esac
echo "Have a great day ..."
return 0
popd > /dev/null
}
답변1
쉘은 종료 상태를 다음과 같이 간주합니다.영참/성공, 그리고0이 아닌상태는 거짓/실패입니다. 따라서 해당 기능이 성공하면 return 0
보다 정확한 것으로 간주될 수 있습니다 return 1
.
set -e
/ 의 경우 set -o errexit
, 어떤 명령/함수에서도 실행되지 않습니다. 예를 들어 조건으로 사용되거나 종료 상태가 무시되는 부분(예: 파이프의 가장 오른쪽이 아닌 부분)을 무시합니다. 그러나 오류가 발생하는 중첩 수준은 중요하지 않습니다. 그러나 사용하기 어려운 몇 가지 복잡성이 있습니다.
fail() { return 1; }
# these won't trigger set -e
fail | cat
fail || true
if fail; then echo x; fi
# so this prints
echo one
# these also won't trigger it
try() {
fail
echo two
}
try || true
( fail; echo three ) || true
바라보다https://mywiki.wooledge.org/BashFAQ/105이상한 것들에 대해서 set -e
.