web-dev-qa-db-ja.com

Bashの構成ファイルから変数を解析する

ファイルに次のコンテンツがある:

VARIABLE1="Value1"
VARIABLE2="Value2"
VARIABLE3="Value3"

次を出力するスクリプトが必要です。

Content of VARIABLE1 is Value1
Content of VARIABLE2 is Value2
Content of VARIABLE3 is Value3

何か案は?

21
KillDash9

awk -F\= '{gsub(/"/,"",$2);print "Content of " $1 " is " $2}' <filename>

参考までに、別の純粋なbashソリューション

IFS="="
while read -r name value
do
echo "Content of $name is ${value//\"/}"
done < filename
31
abasu

構成ファイルは有効なシェルスクリプトであるため、現在のシェルに取り込むことができます。

. config_file
echo "Content of VARIABLE1 is $VARIABLE1"
echo "Content of VARIABLE2 is $VARIABLE2"
echo "Content of VARIABLE3 is $VARIABLE3"

少し乾燥しているが、扱いにくい

. config_file
for var in VARIABLE1 VARIABLE2 VARIABLE3; do
    echo "Content of $var is ${!var}"
done
58
glenn jackman

これらが必要な場合...

特徴

  • 単一行およびインラインコメント。
  • _=_の周りのスペースのトリミング(つまり_var = value_は失敗しません);
  • 引用符で囲まれた文字列値。
  • DOS行末の理解;
  • 安全を保ち、設定ファイルを入手しないようにしてください。

コード

_shopt -s extglob
configfile="dos_or_unix" # set the actual path name of your (DOS or Unix) config file
tr -d '\r' < $configfile > $configfile.unix
while IFS='= ' read -r lhs rhs
do
    if [[ ! $lhs =~ ^\ *# && -n $lhs ]]; then
        rhs="${rhs%%\#*}"    # Del in line right comments
        rhs="${rhs%%*( )}"   # Del trailing spaces
        rhs="${rhs%\"*}"     # Del opening string quotes 
        rhs="${rhs#\"*}"     # Del closing string quotes 
        declare $lhs="$rhs"
    fi
done < $configfile.unix
_

コメント

_tr -d '\r' ..._は、DOSキャリッジリターンを削除します。
_! $lhs =~ ^\ *#_は単一行のコメントをスキップし、_-n $lhs_は空行をスキップします。
${rhs%%*( )}で末尾のスペースを削除するには、_shopt -s extglob_で拡張グロブを設定する必要があります。 (sedを別に使用)、より複雑な方法でこれを回避できます:

_rhs="${rhs%"${rhs##*[^ ]}"}"  
_

テスト構成ファイル

_## This is a comment 
var1=value1             # Right side comment 
var2 = value2           # Assignment with spaces 

## You can use blank lines 
var3= Unquoted String   # Outer spaces trimmed
var4= "My name is "     # Quote to avoid trimming 
var5= "\"Bob\""         
_

テストコード

_echo "Content of var1 is $var1"
echo "Content of var2 is $var2"
echo "Content of var3 is [$var3]"
echo "Content of var4 + var5 is: [$var4$var5]"
_

結果

_Content of var1 is value1
Content of var2 is value2
Content of var3 is [Unquoted String]
Content of var4 + var5 is: [My name is "Bob"]
_
35
antonio

私はこうする

. $PATH_TO_FILE
7
albert
awk '{print "Content of "$1" is "$3}' FS='[="]'

結果

 VARIABLE1のコンテンツはValue1 
 VARIABLE2のコンテンツはValue2 
 VARIABLE3のコンテンツはValue3 
3
Steven Penny

次のように構成ファイルを指定します:-

[a]
b=C
d=E;rm t1
[b]
g=h

次のワンライナーは値を解析して保持します:-

CFG=path-to-file; for ini in `awk '/^\[/' $CFG`;do unset ARHG;declare -A ARHG;while read A B;do ARHG[$A]=$B;echo "in section $ini, $A is equal to"  ${ARHG["$A"]};done < <(awk -F'=' '/\[/ {x=0} x==1 && $0~/=/ && NF==2 {print $1, $2} $0==INI {x=1}' INI="$ini" $CFG);declare -p ARHG;echo;done;printf "end loop\n\n";declare -p ARHG

今、それを分解しましょう

CFG=path-to-file;
for ini in `awk '/^\[/' $CFG` # finds the SECTIONS (aka "ini")
do 
  unset ARHG # resets ARHG 
  declare -A ARHG # declares an associative array
  while read A B
  do
    ARHG[$A]=$B
    echo "in section $ini, $A is equal to"  ${ARHG["$A"]}
  done < <(awk -F'=' '/\[/ {x=0} x==1 && $0~/=/ && NF==2 {print $1, $2} $0==INI {x=1}' INI="$ini" $CFG)
  # the awk splits the file into sections, 
  # and returns pairs of values separated by "="
  declare -p ARHG # displays the current contents of ARHG
  echo
done
printf "end loop\n\n"
declare -p ARHG

これにより、evalまたはbacktickを使用せずに値を保存できます。 「本当にきれい」にするには、行の先頭と末尾の[:space:]を削除し、「^#」行を無視し、「equals」記号の前後のスペースを削除します。

0
Rich Armstrong
    # 
    #------------------------------------------------------------------------------
    # parse the ini like $0.$Host_name.cnf and set the variables
    # cleans the unneeded during after run-time stuff. Note the MainSection
    # courtesy of : http://mark.aufflick.com/blog/2007/11/08/parsing-ini-files-with-sed
    #------------------------------------------------------------------------------
    doParseConfFile(){
        # set a default cnfiguration file
        cnf_file="$run_unit_bash_dir/$run_unit.cnf"

        # however if there is a Host dependant cnf file override it
        test -f "$run_unit_bash_dir/$run_unit.$Host_name.cnf" \
            && cnf_file="$run_unit_bash_dir/$run_unit.$Host_name.cnf"

        # yet finally override if passed as argument to this function
        # if the the ini file is not passed define the default Host independant ini file
        test -z "$1" || cnf_file=$1;shift 1;


        test -z "$2" || ini_section=$2;shift 1;
        doLog "DEBUG read configuration file : $cnf_file"
        doLog "INFO read [$ini_section] section from config file"

        # debug echo "@doParseConfFile cnf_file:: $cnf_file"
        # coud be later on parametrized ...
        test -z "$ini_section" && ini_section='MAIN_SETTINGS'

        doLog "DEBUG reading: the following configuration file"
        doLog "DEBUG ""$cnf_file"
        ( set -o posix ; set ) | sort >"$tmp_dir/vars.before"

        eval `sed -e 's/[[:space:]]*\=[[:space:]]*/=/g' \
            -e 's/#.*$//' \
            -e 's/[[:space:]]*$//' \
            -e 's/^[[:space:]]*//' \
            -e "s/^\(.*\)=\([^\"']*\)$/\1=\"\2\"/" \
            < $cnf_file \
            | sed -n -e "/^\[$ini_section\]/,/^\s*\[/{/^[^#].*\=.*/p;}"`

        ( set -o posix ; set ) | sort >"$tmp_dir/vars.after"

        doLog "INFO added the following vars from section: [$ini_section]"
        cmd="$(comm -3 $tmp_dir/vars.before $tmp_dir/vars.after | Perl -ne 's#\s+##g;print "\n $_ "' )"
        echo -e "$cmd"
        echo -e "$cmd" >> $log_file
        echo -e "\n\n"
        sleep 1; printf "\033[2J";printf "\033[0;0H" # and clear the screen
    }
    #eof func doParseConfFile
0
Yordan Georgiev