模板化读取(如 scanf)

模板化读取(如 scanf)

我有一个 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来提取HOSTSHARE、 和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

相关内容