我正在使用一个名为dotdrop的点文件管理软件。使用名为 的配置文件.env
,内容如下:git_folder="~/code/Git"
.
在第一次使用它之前,也可以使用脚本来设置这个 dotfile 工具,如下所示,
#!/bin/bash
env1="~/Dropbox/.env"
env2="/mnt/d/Dropbox/.env"
if [ -f $env1 ]; then
echo "Found dotdrop env file, installing dotfiles..."
source $env1
eval $(grep -v "^#" $env1) dotdrop --cfg=${git_folder}/dotfiles/config.yaml install
elif [ -f $env2 ]; then
echo "Found dotdrop env file, installing dotfiles..."
source $env2
eval $(grep -v "^#" $env2) dotdrop --cfg=${git_folder}/dotfiles/config.yaml install
else echo "Pls sync environment files first!"
fi
如果我将我的配置文件存储在~/Dropbox/.env
,当我运行脚本时,我就得到了"Pls sync environment files first!"
(如果预期条件则运行)。如果配置文件存储在/mnt/d/Dropbox/.env
,脚本将通过 elif 条件,这是预期的。
找到原因,直到我在 dubug 模式下运行脚本,并得到区别:
➜ scripts git:(master) ✗ bash -x dotdrop_setup.sh
+ env1='~/Dropbox/.env'
+ env2=/mnt/d/Dropbox/.env
+ '[' -f '~/Dropbox/.env' ']'
+ '[' -f /mnt/d/Dropbox/.env ']'
+ echo 'Pls sync environment files first!'
Pls sync environment files first!
~
所以,我认为和之间的区别/home/user
是原因。
在我更改~
为/home/roach
(roach 是用户名)后,它可以工作。
➜ scripts git:(master) ✗ bash -x dotdrop_setup.sh
+ env1=/home/roach/Dropbox/.env
+ env2=/mnt/d/Dropbox/.env
+ '[' -f /home/roach/Dropbox/.env ']'
+ echo 'Found dotdrop env file, installing dotfiles...'
Found dotdrop env file, installing dotfiles...
+ source /home/roach/Dropbox/.env
++ git_folder='~/code/Git'
++ grep -v '^#' /home/roach/Dropbox/.env
+ eval git_folder='~/code/Git'
++ dotdrop '--cfg=~/code/Git/dotfiles/config.yaml' install
_ _ _
__| | ___ | |_ __| |_ __ ___ _ __
/ _` |/ _ \| __/ _` | '__/ _ \| '_ |
\__,_|\___/ \__\__,_|_| \___/| .__/ v0.22.0
|_|
0 dotfile(s) installed.
调试显示''
环绕+ env1='~/Dropbox/.env'
已删除,我认为这是原因。
但是,为什么?
附加问题,
alias dotdrop="eval $(grep -v "^#" $env1) /usr/bin/dotdrop --cfg=${git_folder}/dotfiles/config.yaml install"
是一个配置添加到bashrc
,zshrc
等。如果我直接将它添加到我的脚本中它不起作用!
最后,发现我必须添加source $env
,
那么为什么它在 bashrc 文件中工作?
~
是您的主目录的快捷方式,但仅当它出现在引号之外的字符串开头时。赋值运算符右侧的开头是字符串的开头,因此env1=~/Dropbox/.env
可以工作(env1=~"/Dropbox/.env"
或任何数量的变体)。它设置env1
为/home/roach/Dropbox/.env
。但是env1="~/Dropbox/.env"
设置env1
为确切的 string~/Dropbox/.env
,它作为文件名~
在当前目录中具有单字符名称的目录中查找。波浪号是缩写,而不是通配符。
$env1
外部引号扩展 的值中的通配符env1
,但它不扩展波浪号,因为波浪号不是通配符。您也可以使用
env1="$HOME/Dropbox/.env"
. 这相当于env1=~/Dropbox/.env
. 字符$
(美元)在双引号内具有特殊含义(与引号外的含义相同):它启动变量替换(或命令或算术替换)。~
另一方面,字符(波浪号)在引号内时只是一个普通字符,甚至是双引号。至于别名,它在 bash 脚本中不起作用的原因是 bash 默认情况下不会在脚本中扩展别名。除非您在该脚本中包含或获取别名的定义,否则它不会在任何脚本中工作,因为别名是每个 shell 实例的属性。它们不是流程环境的一部分。
别名在您的第二个脚本中完全起作用的原因是 dotdrop 本身在读取其配置值时会扩展波浪号。