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,但同样的问题仍然适用
也许还有其他方法可以做到这一点,我只需要使用我传递的文件的环境变量作为脚本的参数。
对于那些使用ruby的人,我制作了一个名为dotenv_export的小型实用工具gem。
使用
Dotenv_export是一个小的实用程序命令,它读取。env文件,并使用ruby dotenv实现将其转换为导出语句。
# first install `dotenv_export`
gem install dotenv_export
然后,在你的.bash_profile中,或者任何你想要加载环境变量的shell环境中,执行以下命令:
eval "$(dotenv-export /path/to/.env)"
posix兼容的解决方案(不依赖于bash)
正如其他人所注意到的,在这里使用for/while循环的问题是,变量在shell及其子shell之间不共享。然而,我们能做的是使用args/stdin/stdout在shell之间传递文本。
在subshell中设置环境变量在我们获取脚本源代码时是没有帮助的
变量不会向上传播,但我们知道可以将文本发送回去。这个文本也可以是代码,我们可以用eval在当前shell中求值。
如果我们生成用于设置所有环境变量的代码,然后对结果进行计算呢?
create_exports_script() {
echo "$1" | while read line; do
echo "export $line"
done
}
file_contents=$(cat "./conf/myconf.env")
eval $(create_exports_script "$file_contents")
bash中的这种函数式元编程非常灵活。您还可以用这种方式生成bash/sh以外的其他语言。
值中的空格
这里有很多很好的答案,但我发现它们都缺乏对值中空白的支持:
DATABASE_CLIENT_HOST=host db-name db-user 0.0.0.0/0 md5
我发现了2个解决方案,这些值支持空行和注释。
一个基于sed和@javier-buzzi的答案:
source <(sed -e /^$/d -e /^#/d -e 's/.*/declare -x "&"/g' .env)
和一个读行在循环基于@john1024答案
while read -r line; do declare -x "$line"; done < <(egrep -v "(^#|^\s|^$)" .env)
这里的关键是使用声明-x并将line放在双引号中。我不知道为什么,但当你将循环代码重新格式化为多行时,它不会工作-我不是bash程序员,我只是狼吞虎咽地把这些放在一起,它对我来说仍然是魔法:)
我用这个:
source <(cat .env \
| sed -E '/^\s*#.*/d' \
| tr '\n' '\000' \
| sed -z -E 's/^([^=]+)=(.*)/\1\x0\2/g' \
| xargs -0 -n2 bash -c 'printf "export %s=%q;\n" "${@}"' /dev/null)
首先删除评论:
sed -E '/^\s*#.*/d'
然后转换为空分隔符而不是换行符:
tr '\n' '\000'
然后用null替换equal:
sed -z -E 's/^([^=]+)=(.*)/\1\x0\2/g'
然后打印对到有效的带引号的bash导出(使用bash printf for %q):
xargs -0 -n2 bash -c 'printf "export %s=%q;\n" "${@}"' /dev/null
最后是来源。
它应该适用于所有特殊字符的所有情况。
当我试图在shell中重用Docker -env-files时,我遇到了这个线程。它们的格式与bash不兼容,但很简单:name=value,没有引号,没有替换。它们还会忽略空行和#注释。
我不能完全让它与posix兼容,但这里有一个应该在bash-like shell中工作(在OSX 10.12.5的zsh和Ubuntu 14.04的bash中测试):
while read -r l; do export "$(sed 's/=.*$//' <<<$l)"="$(sed -E 's/^[^=]+=//' <<<$l)"; done < <(grep -E -v '^\s*(#|$)' your-env-file)
它不会处理上面文档链接的例子中的三种情况:
Bash: export: ' 123qwe=bar':不是有效的标识符
Bash: export: `org.spring。Config =something':不是有效的标识符
并且它不会处理传递语法(一个纯FOO)