TL;DR:如何从文本文件中导出一组键/值对到shell环境中?
为了记录在案,以下是问题的原始版本,并附有示例。
我在bash中写了一个脚本,它在某个文件夹中解析带有3个变量的文件,这是其中之一:
MINIENTREGA_FECHALIMITE="2011-03-31"
MINIENTREGA_FICHEROS="informe.txt programa.c"
MINIENTREGA_DESTINO="./destino/entrega-prac1"
该文件的存放路径为。/conf/prac1
我的脚本minientrega.sh然后使用以下代码解析文件:
cat ./conf/$1 | while read line; do
export $line
done
但是当我在命令行中执行minientrega.sh prac1时,它不会设置环境变量
我也尝试使用source ./conf/$1,但同样的问题仍然适用
也许还有其他方法可以做到这一点,我只需要使用我传递的文件的环境变量作为脚本的参数。
我构建这个脚本动态地源env vars。
我使用这个脚本是因为我不想记住我在项目中使用的每个变量的名称,我不希望导出命令被存储在历史记录中或完整的.env文件导出到git。
#!/bin/sh
filename=".secret"
secret_var () {
# Parametter 1 : Environnement vars anme
bash -c 'read -p '$1=' -s voila && echo '$1'"=${voila}" > '$filename''
export `cat .secret`
rm $filename
echo ''
}
public_var () {
# Parametter 1 : Environnement vars anme
bash -c 'read -p '$1=' voila && echo '$1'"=${voila}" > '$filename''
export `cat .secret`
rm $filename
}
if [ -e $filename ]
then
echo "A file named '.secret' already exist. Remove it or edit this script."
else
public_var MY_USER_VAR
secret_var MY_PASS_VAR
fi
它非常容易使用:
# To add var MY_VAR_NAME to the env
public_var MY_VAR_NAME
# To add var MY_VAR_NAME secretly into the env
secret_var MY_VAR_NAME
例子:
callmarl@LAPTOP ~ % source set_env.sh
MY_USER_VAR=myusername
MY_PASS_VAR=
callmarl@LAPTOP ~ % env
MY_USER_VAR=myusername
MY_PASS_VAR=mysecretpass
当然,如果希望存储值,可以直接使用export,而不是public_var。
我构建这个脚本动态地源env vars。
我使用这个脚本是因为我不想记住我在项目中使用的每个变量的名称,我不希望导出命令被存储在历史记录中或完整的.env文件导出到git。
#!/bin/sh
filename=".secret"
secret_var () {
# Parametter 1 : Environnement vars anme
bash -c 'read -p '$1=' -s voila && echo '$1'"=${voila}" > '$filename''
export `cat .secret`
rm $filename
echo ''
}
public_var () {
# Parametter 1 : Environnement vars anme
bash -c 'read -p '$1=' voila && echo '$1'"=${voila}" > '$filename''
export `cat .secret`
rm $filename
}
if [ -e $filename ]
then
echo "A file named '.secret' already exist. Remove it or edit this script."
else
public_var MY_USER_VAR
secret_var MY_PASS_VAR
fi
它非常容易使用:
# To add var MY_VAR_NAME to the env
public_var MY_VAR_NAME
# To add var MY_VAR_NAME secretly into the env
secret_var MY_VAR_NAME
例子:
callmarl@LAPTOP ~ % source set_env.sh
MY_USER_VAR=myusername
MY_PASS_VAR=
callmarl@LAPTOP ~ % env
MY_USER_VAR=myusername
MY_PASS_VAR=mysecretpass
当然,如果希望存储值,可以直接使用export,而不是public_var。
source的问题在于它要求文件具有正确的bash语法,而一些特殊字符会破坏它:=、"、'、<、>等。所以在某些情况下你可以
source development.env
它会起作用的。
然而,这个版本可以承受值中的所有特殊字符:
set -a
source <(cat development.env | \
sed -e '/^#/d;/^\s*$/d' -e "s/'/'\\\''/g" -e "s/=\(.*\)/='\1'/g")
set +a
解释:
-a意味着每个bash变量都将成为环境变量
/^#/d删除注释(以#开头的字符串)
/^\s*$/d删除空字符串,包括空格
"s/'/'\\\ " /g"将每个单引号替换为'\ ",这是bash中的一个生成引号的技巧序列:)
"s/=\(.*\)/='\1'/g"将每个a=b转换为a='b'
因此,你可以使用特殊字符:)
要调试这段代码,将source替换为cat,您将看到该命令产生的结果。
如果您想让exec作为脚本的最后一个命令,那么您可以使用execlineb解释器。这是脚本的最后一行:
#!/bin/sh
...
exec envfile -I /etc/default/bla envfile /etc/default/bla-bla my_cmd
envfile……是来自execline套件的命令,它们依赖于“链加载”。
顺便说一句,一旦你进入这个兔子洞,你可能会发现你不再需要壳了(…-)使用execlineb解释器而不是shell来启动服务,以最小的开销是非常有用的,即:
#!/bin/execlineb
...
envfile -I /etc/default/bla
envfile /etc/default/bla-bla
my_cmd