为什么Docker exec命令在Docker容器内部忽略bashrc文件?

9

我有一个带有Ubuntu的Docker镜像(也从该镜像创建了容器)。在我的Docker容器中,我看到以下输出:

root@61ff2a8dbf2d:/# uname -a
Linux 61ff2a8dbf2d 5.3.0-51-generic #44~18.04.2-Ubuntu SMP Thu Apr 23 14:27:18 UTC 2020 x86_64 x86_64 x86_64 GNU/Linux

在我的容器中,文件名为~/.bashrc(即/root/.bashrc),我导出了一些变量:

export VARIABLE1="value1"
export VARIABLE2="value2"

当我尝试使用 "docker exec" 命令在Docker容器上执行命令时,我没有看到来自 .bashrc 文件的变量。
root@HUAWEI-MateBook-X:~/Documents/# docker exec -it sly_compilator /bin/bash -c 'export'
declare -x HOME="/root"
declare -x HOSTNAME="61ff2a8dbf2d"
declare -x OLDPWD
declare -x PATH="/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin"
declare -x PWD="/"
declare -x SHLVL="1"
declare -x TERM="xterm"

为什么我看不到我的变量?如何修复它?

但如果我记录(container)我的容器,这将正常工作。

root@HUAWEI-MateBook-X:~/Documents/# docker exec -it sly_compilator bash
root@61ff2a8dbf2d:/# export
declare -x VARIABLE1="value1"
declare -x VARIABLE2="value2"
declare -x HOME="/root"
declare -x HOSTNAME="61ff2a8dbf2d"
declare -x LS_COLORS="rs=0:di=01;34:ln=01;36:mh=00:pi=40;33:so=01;35:do=01;35:bd=40;33;01:cd=40;33;01:or=40;31;01:mi=00:su=37;41:sg=30;43:ca=30;41:tw=30;42:ow=34;42:st=37;44:ex=01;32:*.tar=01;31:*.tgz=01;31:*.arc=01;31:*.arj=01;31:*.taz=01;31:*.lha=01;31:*.lz4=01;31:*.lzh=01;31:*.lzma=01;31:*.tlz=01;31:*.txz=01;31:*.tzo=01;31:*.t7z=01;31:*.zip=01;31:*.z=01;31:*.Z=01;31:*.dz=01;31:*.gz=01;31:*.lrz=01;31:*.lz=01;31:*.lzo=01;31:*.xz=01;31:*.zst=01;31:*.tzst=01;31:*.bz2=01;31:*.bz=01;31:*.tbz=01;31:*.tbz2=01;31:*.tz=01;31:*.deb=01;31:*.rpm=01;31:*.jar=01;31:*.war=01;31:*.ear=01;31:*.sar=01;31:*.rar=01;31:*.alz=01;31:*.ace=01;31:*.zoo=01;31:*.cpio=01;31:*.7z=01;31:*.rz=01;31:*.cab=01;31:*.wim=01;31:*.swm=01;31:*.dwm=01;31:*.esd=01;31:*.jpg=01;35:*.jpeg=01;35:*.mjpg=01;35:*.mjpeg=01;35:*.gif=01;35:*.bmp=01;35:*.pbm=01;35:*.pgm=01;35:*.ppm=01;35:*.tga=01;35:*.xbm=01;35:*.xpm=01;35:*.tif=01;35:*.tiff=01;35:*.png=01;35:*.svg=01;35:*.svgz=01;35:*.mng=01;35:*.pcx=01;35:*.mov=01;35:*.mpg=01;35:*.mpeg=01;35:*.m2v=01;35:*.mkv=01;35:*.webm=01;35:*.ogm=01;35:*.mp4=01;35:*.m4v=01;35:*.mp4v=01;35:*.vob=01;35:*.qt=01;35:*.nuv=01;35:*.wmv=01;35:*.asf=01;35:*.rm=01;35:*.rmvb=01;35:*.flc=01;35:*.avi=01;35:*.fli=01;35:*.flv=01;35:*.gl=01;35:*.dl=01;35:*.xcf=01;35:*.xwd=01;35:*.yuv=01;35:*.cgm=01;35:*.emf=01;35:*.ogv=01;35:*.ogx=01;35:*.aac=00;36:*.au=00;36:*.flac=00;36:*.m4a=00;36:*.mid=00;36:*.midi=00;36:*.mka=00;36:*.mp3=00;36:*.mpc=00;36:*.ogg=00;36:*.ra=00;36:*.wav=00;36:*.oga=00;36:*.opus=00;36:*.spx=00;36:*.xspf=00;36:"
declare -x OLDPWD
declare -x PATH="/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin"
declare -x PWD="/"
declare -x SHLVL="1"
declare -x TERM="xterm"

请告诉我我必须使用导出变量还是执行什么操作才能成功地从.bashrc中获取变量并在使用docker exec命令时内联执行命令时显示它们?
最好的祝福, 谢尔盖。
1个回答

8

~/.bashrc只有在shell是交互式的时才被调用。这很复杂

当bash作为一个交互式登录shell或者使用--login选项作为非交互式shell来调用时,如果文件/etc/profile存在,它会首先读取并执行该文件中的命令。在读取完该文件之后,它会按照顺序查找文件 ~/.bash_profile、~/.bash_login和~/.profile,并读取并执行第一个存在且可读的文件。当启动shell时使用--noprofile选项可以抑制此行为。

...

当启动一个不是登录shell的交互式shell时,bash从文件~/.bashrc中读取并执行命令,如果该文件存在的话。使用--norc选项可以阻止这种行为。使用--rcfile文件选项将强制bash从文件而不是~/.bashrc中读取并执行命令。

在您的情况下,bash既不作为登录shell启动也不作为交互式shell启动,因此这些文件都不会被加载。当使用docker exec -it时,您确实会得到一个交互式shell,因此会有所不同。
您可以将初始化放在~/.profile中,并传递--login以确保它被使用。
更好的解决方案可能是将变量保存在主机系统上的一个文件(例如variables.env)中,并在用于启动容器的docker run命令中传递--env-file=variables.env。这应该确保它们对容器内启动的每个进程都可用,包括其他shell和非shell程序。

感谢你的帮助,Thomas =) - Sergei Dub

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接