Bash-ロギング関数-printf

Aug 30 2020

bash用のロガーを作成しようとしています。問題は、直接印刷は機能しているが、LOGGER_FUNCが配列を正しく処理しないことです。

現在、ログに記録されるはずのデータを印刷できます。

DEBUG_data_ARRAY=(hi ho no bugs here no)
printf "\n%s" "${DEBUG_data_ARRAY[@]}" printf "\n%s %s" "${DEBUG_data_ARRAY[@]}"

printfを次のように置き換える必要がある場所:

LOGGER_FUNC "\n%s" "${DEBUG_data_ARRAY[@]}" LOGGER_FUNC "\n%s %s" "${DEBUG_data_ARRAY[@]}"

ロガー機能:

LOGGER_FUNC () {
    format=$1 message=$2
    if [[ $VERBOSE == 0 ]]; then printf "${format[@]}" "${message[@]}" fi if [[ $VERBOSE == 1 ]]; then
         printf "${format[@]}" "${message[@]}" >> $DEBUG_FILE fi if [[ $VERBOSE == 2 ]]; then
        printf "${format[@]}" "${message[@]}"
        printf "${format[@]}" "${message[@]}" >> $DEBUG_FILE
    fi
}

期待される結果は次のとおりです。

hi 
ho 
no 
bugs
here 
no

hi ho 
no bugs
here no

回答

4 JohnKugelman Aug 30 2020 at 20:35
format=$1 message=$2

これにより、2つのスカラー変数が作成されます。作るためにmessage含む配列を$2$3$4、など、書き込み:

format=$1
message=("${@:2}")

次に、formatはスカラーなので、次のように書くの$formatではなく、単に書くことができます${format[@]}

if [[ $VERBOSE == 0 ]]; then
    printf "$format" "${message[@]}"
fi
if [[ $VERBOSE == 1 ]]; then printf "$format" "${message[@]}" >> "$DEBUG_FILE"
fi
if [[ $VERBOSE == 2 ]]; then printf "$format" "${message[@]}" printf "$format" "${message[@]}" >> "$DEBUG_FILE"
fi
1 LéaGris Aug 30 2020 at 21:08

関数に提供された引数の使用:

#!/usr/bin/env sh

LOGGER_FUNC() {
  # shellcheck disable=SC2059 # Variable format string
  printf "$@" | case $VERBOSE in
    1) cat ;;
    2) cat >>"$DEBUG_FILE" ;; 3) tee -a "$DEBUG_FILE" ;;
  esac
}

または、コンテンツの引数を必要としないストリームロガーを実装しますが、次の場所から取得しstdinます。

#!/usr/bin/env bash

# stream_logger
# Log stdin with options
# &1: Verbose level:
#     1: stdout only
#     2: debug file only
#     3: both stdout and debug file
# &2: Optional debug file path
stream_logger() {
  if [ $# -eq 0 ] || [ "$1" -eq 0 ]; then
    cat >/dev/null
  elif [ $# -eq 1 ] || [ "$1" -eq 1 ]; then
    cat
  elif [ $# -eq 2 ]; then if [ "$1" -eq 2 ]; then
      cat >>"$2" elif [ "$1" -eq 3 ]; then
      tee -a "$2" fi fi } DEBUG_data_ARRAY=(hi ho no bugs here no) echo 'hello' | stream_logger # print nothing # Output to stdout only printf '\n%s' "${DEBUG_data_ARRAY[@]}" | stream_logger 1
printf '\n%s %s' "${DEBUG_data_ARRAY[@]}" | stream_logger 1 # Output to file1.log only printf '\n%s' "${DEBUG_data_ARRAY[@]}" | stream_logger 2 file1.log
printf '\n%s %s' "${DEBUG_data_ARRAY[@]}" | stream_logger 2 file1.log # Output to file2.log and stdout printf '\n%s' "${DEBUG_data_ARRAY[@]}" | stream_logger 3 file2.log
printf '\n%s %s' "${DEBUG_data_ARRAY[@]}" | stream_logger 3 file2.log