我有一个 shell 脚本变量,当前定义如下:
tsource=/backup/%HOST%/%SHARE%/%PERIOD%
我想将此模板与另一个变量匹配,该变量可能具有如下值:
psource=/backup/somehost/someshare/monthly.1
我的目标是生成以下分配,以便我可以在脚本中稍后使用它们进行替换:
vars[HOST]=somehost
vars[SHARE]=someshare
vars[PERIOD]=monthly.1
我应该指出,用户可以覆盖这两个值,我们最终可能会得到不同的(但仍然匹配)形状,这意味着简单的“分割/
”(甚至“分割标点”)是不够的:
tsource=/backup/%PERIOD/where/%HOST%-%SHARE%
psource=/backup/monthly.1/where/somehost-someshare
目的是能够$psource
根据提供的模板进行解析$tsource
。由于它是实用软件,我并不关心用户是否试图破坏它 - 如果提供的参数不足或无效,它会在稍后的时间点退出。
看看可能的解决方案,我认为类似的东西sscanf
在这里可能有用,这是我可以间接使用的工具。我可以很容易地操作$tsource
来提取HOST
、SHARE
、 和PERIOD
,并派生出一个sscanf
模板:
grep -Po "(?<=%)[[:upper:]]+(?=%)" <<<"$tsource" # "HOST" "SHARE" "PERIOD"
tscanf=$(sed -re 's/%[[:upper:]]+%/%s/g' <<<"$tsource") # "/backup/%s/%s/%s"
这将允许我在 Perl 中应用模板,如下例所示:
perl -MString::Scanf -e '
$psource = "/backup/somehost/someshare/monthly.1";
$tscanf = "/backup/%s/%s/%s";
($host, $source, $period) = sscanf($tscanf, $psource);
print "host=$host, share=$share, period=$period\n"
'
# "host=somehost, share=someshare, period=monthly.1"
(如果我要深入研究 Perl,我可能也会sscanf
在 Perl 部分中进行模板重写和生成,但我们现在先把它搁置一下。)
然而,编写需要perl
.
是否有替代(更好)的解决方案允许我将字符串中的值映射到模板中相当任意的标签,而无需快速深入了解 Perl?
答案1
使用正则表达式是一种选择吗?不过,我仍然需要花费 to 来sed
干净地替换%...%
为(.*)
:
$ psource=/backup/monthly.1/where/somehost-someshare
$ tsource=/backup/%PERIOD%/where/%HOST%-%SHARE%
$ # replace labels with '(.*)', need sed to avoid greediness of * in ${../../..}
$ tsource_re=$(sed 's/%[^%]*%/(.*)/g' <<<"${tsource}")
$ # match against tsource to get template names
$ [[ $tsource =~ $tsource_re ]]; tmatch=(${BASH_REMATCH[@]:1})
$ # match against psource to get template values
$ [[ $psource =~ $tsource_re ]]; pmatch=(${BASH_REMATCH[@]:1})
$ for i in ${!tmatch[@]}; do printf "%s:\t%s\n" "${tmatch[$i]}" "${pmatch[$i]}"; done
%PERIOD%: monthly.1
%HOST%: somehost
%SHARE%: someshare
答案2
这应该可以解决问题:
tsource='/backup/%HOST%/%SHARE%/%PERIOD%'
psource='/backup/somehost/someshare/monthly.1'
IFS=$'\n' read -d '' -ra var_names <<< $(grep -Po "(?<=%)[[:upper:]]+(?=%)" <<< "$tsource")
IFS='/' read -ra var_values <<< $(echo ${psource#/*/})
declare -A vars=( ["${var_names[0]}"]="${var_values[0]}" ["${var_names[1]}"]="${var_values[1]}" ["${var_names[2]}"]="${var_values[2]}" )
echo ${vars[HOST]} ${vars[SHARE]} ${vars[PERIOD]}
>somehost someshare monthly.1