我正在尝试将 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
希望这可以帮助!