bash 기능에서 올바른 방법으로 복귀하는 방법은 무엇입니까?

bash 기능에서 올바른 방법으로 복귀하는 방법은 무엇입니까?

터미널에서 가져온 이 스크립트는 _cphetznergit 저장소를 생성하는 기능을 제공합니다. 이 함수는 _feed_variable사용자에게 누락된 변수를 묻는 또 다른 함수를 사용합니다.

이 시점에서 내 질문은 다음과 같습니다.

  1. 이것이 함수에서 복귀하는 올바른 방법입니까 return 1?
  2. 주석을 제거 하면 set -o errexit스크립트가 무엇이든 return 1(예: 함수 _feed_variable또는 셸로 반환되는 함수에서) 중단됩니까?
  3. 주석 처리를 제거한 경우 set -o errexit터미널을 닫지 않고 기능을 종료하는 올바른 방법은 무엇입니까?
  4. 오류를 해결하는 방식으로 스크립트를 디자인하지 않은 것 같습니다. 어떤 제안이 있으십니까?

#!/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.

관련 정보