在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
根据你的问题更新,这不是一个好的解决方案。
这是基于对相关问题的回答 。 我已经黑了(我不熟悉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