replace属性文件中的环境variables

在Linux中,说我有以下文件(例如conf.properties):

HOST_URL=http://$HOSTNAME STD_CONFIG=http://$HOSTNAME/config USER_CONFIG=http://$HOSTNAME/config/$unconfigured 

我想创build另一个文件,所有的环境variables取代…例如说环境variables$ HOSTNAME是'myhost'和$未configuration未设置,脚本应该产生以下输出:

 HOST_URL=http://myhost STD_CONFIG=http://myhost/config USER_CONFIG=http://myhost/config/ 

我以为这可以通过简单的一行sed / awk魔法来完成,但我不是专家,我的search一直都是静止的,所以感谢任何帮助。

编辑:

我应该提到,该文件可以是任何格式的文本文件,例如xml。 我只是想用当前在环境中设置的东西来replace看起来像envvariables的东西。

 sed 's/$HOSTNAME/myhost/g;s/$unconfigured//g' yourfile.txt > another_file.txt 

更新:

根据你的问题更新,这不是一个好的解决方案。

更新2:

这是基于对相关问题的回答 。 我已经黑了(我不熟悉Perl)删除未定义的变量。

 perl -p -e 's/\$\{([^}]+)\}/defined $ENV{$1} ? $ENV{$1} : $&/eg; s/\$\{([^}]+)\}//eg' yourfile.txt 

应该为任何输入文本文件工作,但是您将需要使用简化字符串匹配的${...}格式定义变量。

关于eval的邪恶转移到一个单独的帖子,以免混淆读者

这是envsubst的用途。

 echo 'Hello $USER' Hello $USER echo 'Hello $USER' | envsubst Hello malvineous 

你可能会更像这样使用它:

 envsubst < input.txt > output.txt 

envsubst似乎是GNU gettext的一部分。

我会这样做:

 # Set the $HOSTNAME and other variables # Now evaluate the properties file as a shell script. . config.properties # Write the values cat >somefile <<EOF HOST_URL=$HOST_URL STD_CONFIG=$STD_CONFIG USER_CONFIG=$USER_CONFIG EOF 

编辑 :或者这非常讨厌的事情(我敢肯定还有更好的办法)

 for name in HOST_URL STD_CONFIG USER_CONFIG echo "$name=$(eval echo `echo '$'$name`)" >>somefile end 

“eval是邪恶的”

这不是一个答案,而是一个响应使用eval来完成这个任务的警告。 你真的真的不想这样做。

图表1:恶意模板文件:

 HOST_URL=http://$HOSTNAME STD_CONFIG=http://$HOSTNAME/config USER_CONFIG=http://$HOSTNAME/config/$unconfigured && cat /etc/redhat-release 

一个毫不怀疑的用户:

 [lsc@aphek]$ cat somefile | while read line; do echo $(eval echo `echo $line`); done HOST_URL=http://xyz STD_CONFIG=http://xyz/config USER_CONFIG=http://xyz/config/ Red Hat Enterprise Linux WS release 4 (Nahant Update 9) 

注意最后一行!

现在,想象一下可能性….

感谢@DarkDust,我想出了这个:

 cat somefile | while read line; do echo $(eval echo `echo $line`); done > somefile.replaced 

我用这个oneliner来替换文件中的${VARIABLE}样式变量:

 TARGET_FILE=/etc/apache2/apache2.conf; for VARNAME in $(grep -P -o -e '\$\{\S+\}' ${TARGET_FILE} | sed -e 's|^\${||g' -e 's|}$||g' | sort -u); do sed -i "s|\${$(echo $VARNAME)}|${!VARNAME}|g" ${TARGET_FILE}; done 

我敢肯定,有人可以做到这一点在三分之一的长度使用awk …感觉挑战! ;)

下面是一个简短的一行,它使用python的大括号格式来安全地实现魔法:

 contents=\"\"\"`cat $file`\"\"\"; python -c "import os;print $contents.format(**os.environ)" 
  • 避免邪恶的评估
  • 允许输出大括号:使用{{而不是{
  • 调用脚本时不需要明确指定变量

例如,给定的属性文件settings.properties

 # my properties file someVar = {MY_ENV_VAR} curlyBraceVar = has {{curly braces}} 

然后,用下面的代码进行替换:

 $ export MY_ENV_VAR="hello" $ file=settings.properties $ contents=\"\"\"`cat $file`\"\"\"; python -c "import os;print $contents.format(**os.environ)" # my properties file someVar = hello curlyBraceVar = has {curly braces} 

脚本在这里: https : //raw.githubusercontent.com/aneilbaboo/machome/master/bin/substenv