如何解析并将 ini 文件转换为 bash 数组变量?

如何解析并将 ini 文件转换为 bash 数组变量?

我正在尝试将 ini 文件转换为 bash 数组变量。示例 ini 如下:

[foobar]
session=foo
path=/some/path

[barfoo]
session=bar
path=/some/path

因此这些变成:

session[foobar]=foo
path[foobar]=/some/path
session[barfoo]=bar

等等。

现在,我只能想出这个命令

awk -F'=' '{ if ($1 ~ /^\[/) section=$1; else if ($1 !~ /^$/) print $1 section "=" $2 }'

此外,另一个问题是,它没有=考虑附近的空格。我认为sed可能更适合这项工作,但我不知道如何保存和存储部分名称的临时变量sed

那么,您知道如何做到这一点吗?

答案1

Gawk 接受正则表达式作为字段分隔符。以下代码消除了等号周围的空格,但保留了行其余部分的空格。在值周围添加了引号,以便在执行 Bash 赋值时保留这些空格(如果有)。我假设节名称将是数字变量,但如果您使用的是 Bash 4,则可以轻松地将其调整为使用关联数组,并将节名称本身作为索引。

awk -F ' *= *' '{ if ($1 ~ /^\[/) section=$1; else if ($1 !~ /^$/) print $1 section "=" "\"" $2 "\"" }'

请注意,您可能还需要执行 Khaled 显示的空格删除(仅在 $1 和部分),因为 Bash 变量名不能包含空格。

此外,如果值包含等号,此方法将不起作用。

另一种技术是使用 Bashwhile read循环并在读取文件时执行分配,这样declare可以防止大多数恶意内容。

foobar=1
barfoo=2  # or you could increment an index variable each time a section is found
while IFS='= ' read var val
do
    if [[ $var == \[*] ]]
    then
        section=$var
    elif [[ $val ]]
    then
        declare "$var$section=$val"
    fi
done < filename

再次,关联数组可以相当容易地得到支持。

答案2

我将使用简单的 Python 脚本来完成这项工作,因为它内置了 INI解析器

#!/usr/bin/env python

import sys, ConfigParser

config = ConfigParser.ConfigParser()
config.readfp(sys.stdin)

for sec in config.sections():
    print "declare -A %s" % (sec)
    for key, val in config.items(sec):
        print '%s[%s]="%s"' % (sec, key, val)

然后在 bash 中:

#!/bin/bash

# load the in.ini INI file to current BASH - quoted to preserve line breaks
eval "$(cat in.ini  | ./ini2arr.py)"

# test it:
echo ${barfoo[session]}

当然,awk 中有更短的实现,但我认为这个更具可读性且更易于维护。

答案3

如果要消除多余的空格,可以使用内置函数gsub。例如,您可以添加:

gsub(/ /, "", $1);

这将删除所有空格。如果要删除 token 开头或结尾的空格,可以使用

gsub(/^ /, "", $1);
gsub(/ $/, "", $1);

答案4

始终假设有 Python 的 ConfigParser,可以构建一个如下的 shell 辅助函数:

get_network_value()
{
    cat <<EOF | python
import ConfigParser
config = ConfigParser.ConfigParser()
config.read('network.ini')
print (config.get('$IFACE','$param'))
EOF
}

$IFACE$param分别是部分的参数。

然后,该助手允许如下调用:

address=`param=address get_network_value` || exit 1
netmask=`param=netmask get_network_value` || exit 1
gateway=`param=gateway get_network_value` || exit 1

希望这可以帮助!

相关内容