赞
踩
前面我们为大家介绍了Docker支持容器root用户的 Capability 能力限制、镜像签名、Apparmor的MAC访问控制、使用Seccomp限制系统调用等安全性支持,这篇文章我们会为大家介绍Docker其他安全性特性支持。上篇文章回顾:浅谈Docker的安全性支持(上篇)
Linux 命名空间为运行中的进程提供了隔离,限制他们对系统资源的访问,而进程没有意识到这些限制。为防止容器内的特权升级攻击的最佳方法是将容器的应用程序配置为非特权用户运行,对于其进程必须作为容器中的 root 用户运行的容器,可以将此用户重新映射到 Docker 主机上权限较低的用户。映射的用户被分配了一系列 UID,这些 UID 在命名空间内作为从 0 到 65536 的普通 UID 运行,但在主机上没有特权。重新映射由两个文件处理:/etc/subuid 和 /etc/subgid,其中前者关注用户 ID 范围,后者关注用户组 ID 范围。
例如,如下 /etc/subuid 中的条目:
testuser:231072:65536
这意味着 testuser 将从 231072 开始,在后面的 65536 个整数中按顺序为用户分配一个 ID。例如,命名空间中的 UID 231072 映射到容器中的 UID 0(root),UID 231073 映射为 UID 1,依此类推。如果某个进程尝试提升特权到命名空间外部,则该进程将作为主机上无特权的高数字 UID 运行,该 UID 甚至不映射到真实用户,这意味着该进程完全没有主机系统的权限。
在Docker1.10以后,可以通过在Dockerd启动参数中指定userns-remap 来启用这个功能。
下面我们做一下演示:
1、查看Docker Daemon是否以root用户身份运行
lynzabo@ubuntu:~$ ps -ef | grep dockerdroot 1557 1 0 12:54 ? 00:05:08 /usr/bin/dockerd -H fd://lynzabo 36398 23696 0 21:41 pts/1 00:00:00 grep --color=auto dockerdlynzabo@ubuntu:~$
2、运行容器,指定 id 命令```
lynzabo@ubuntu:~$ docker run --rm alpine idUnable to find image ‘alpine:latest’ locallylatest: Pulling from library/alpine4fe2ade4980c: Pull complete Digest: sha256:621c2f39f8133acb8e64023a94dbdf0d5ca81896102b9e57c0dc184cadaf5528Status: Downloaded newer image for alpine:latestuid=0(root) gid=0(root) groups=0(root),1(bin),2(daemon)…lynzabo@ubuntu:~$
上面输出的最后一行显示容器以root身份运行:uid = 0(root)和gid = 0(root)。3、执行docker run 指定参数--user ,指定容器以当前用户身份来运行
lynzabo@ubuntu:~$ iduid=1000(lynzabo) gid=1000(lynzabo) groups=1000(lynzabo)…lynzabo@ubuntu:~$ docker run --rm --user 1000:1000 alpine iduid=1000 gid=1000lynzabo@ubuntu:~$
可以看到容器使用的我们设置的用户和组来运行。
有时候,我们更希望容器里面是以root用户来运行,但是并不需要具有宿主机上root权限,可以使用User Namespace做到这些。使用User Namespace,容器中的root用户会被重新映射到宿主机上一个非特权用户,这意味着该进程完全没有主机系统的权限。
下面我们带大家一起演示一下:
1、停止Docker Daemon
lynzabo@ubuntu:~$ sudo systemctl stop dockerlynzabo@ubuntu:~$
2、指定在User Namespace模式下运行Docker Daemon
lynzabo@ubuntu:~$ sudo dockerd --userns-remap=default &lynzabo@ubuntu:~$
当你将 Docker 配置为使用 userns-remap 功能时,可以指定为现有用户或组,也可以指定为 default。如果指定为default,则会为此创建并使用用户和组 dockremap 。也可以在 daemon.json 配置文件中指定。
通过 id 命令验证 Docker 已经创建了这个用户。
lynzabo@ubuntu:~$ id dockremapuid=123(dockremap) gid=132(dockremap) groups=132(dockremap)lynzabo@ubuntu:~$
验证条目已经添加到了 /etc/subuid 和 /etc/subgid 文件中。
lynzabo@ubuntu:~$ grep dockremap /etc/subuiddockremap:165536:65536lynzabo@ubuntu:~$ grep dockremap /etc/subgiddockremap:165536:65536lynzabo@ubuntu:~$
如果这些条目不存在,需要以 root 用户身份编辑文件,并且分配起始的 UID 和 GID(在最高的已经分配的值的基础上加上偏移,65536)。注意不要使范围重叠。
3、使用 docker info 命令验证Docker是否正确启用了用户命名空间支持
lynzabo@ubuntu:~$ docker info…Docker Root Dir: /home/docker/165536.165536…lynzabo@ubuntu:~$ lynzabo@ubuntu:~$ ls -ld /home/docker/165536.165536drwx------ 14 165536 165536 4096 Sep 17 21:44 /home/docker/165536.165536lynzabo@ubuntu:~$ sudo ls -l /home/docker/165536.165536/total 48drwx------ 2 165536 165536 4096 Sep 17 21:44 volumesdrwx–x–x 3 root root 4096 Sep 17 21:44 containerddrwx------ 2 165536 165536 4096 Sep 17 21:44 containersdrwx------ 3 root root 4096 Sep 17 21:44 imagedrwxr-x— 3 root root 4096 Sep 17 21:44 networkdrwx------ 4 165536 165536 4096 Sep 17 21:44 overlay2…lynzabo@ubuntu:~$
可以看到Docker 工作目录在原有/var/lib/docker/ 目录下多了一层以“用户UID.GID”命名的目录。查看该目录下各个子目录权限,有些子目录仍有 root 拥有,有些子目录已经继承了上级目录权限。4、查看本地镜像
lynzabo@ubuntu:~$ docker imagesREPOSITORY TAG IMAGE ID CREATED SIZElynzabo@ubuntu:~$
可以看到本地没有任何镜像,很奇怪,我们在上面使用的alpine镜像消失了。
5、下面我们以交互模式运行一个容器,将宿主机的/bin目录挂载到容器中
lynzabo@ubuntu:~$ docker run -it --rm -v /bin:/host/bin busybox /bin/shUnable to find image ‘busybox:latest’ locallylatest: Pulling from library/busybox8c5a7da1afbc: Pull complete Digest: sha256:cb63aa0641a885f54de20f61d152187419e8f6b159ed11a251a09d115fdff9bdStatus: Downloaded newer image for busybox:latest/ # iduid=0(root) gid=0(root) groups=10(wheel)/ #
上面的输出显示容器内部是以root用户的安全上下文下运行。6、下面我们尝试执行命令
/ # rm /host/bin/shrm: can’t remove ‘sh’: Permission denied
操作失败并显示权限被拒绝,这是因为要删除的文件存在于Docker宿主机的本地文件系统中,并且容器在其所在的命名空间之外没有root访问权限。如果未启用User Namespace,执行相同的操作,操作将成功。 **SELinux支持** ------------- 我们知道系统的用户主要分为系统管理员与一般用户,而这两种身份能否使用系统文件资源与 rwx 的权限设置有关,这种存取文件系统的方式被称为“自主式存取控制(DAC)”。不过你要注意的是,各种权限设置对 root 是无效的,这个时候就可以使用委任式存取控制(MAC)了,使用MAC可以针对特定的程序与特定的文件资源来进行权限的控管,也就是说,即使是root用户,那么在使用不同的程序时,你所能取得的权限并不一定是root,而要根据当时程序的设置而定。SELinux 就是通过 MAC 的方式来控管程序,他控制的主体是程序, 而目标则是该程序能否读写的“文件资源”。下面是使用SeLinux基本流程: <img src="https://p1-jj.byteimg.com/tos-cn-i-t2oaga2asx/gold-user-assets/2019/5/14/16ab582d813ca5fb~tplv-t2oaga2asx-zoom-in-crop-mark:4536:0:0:0.image" style="margin: auto" /> 由上图我们可以发现:(1) 主体程序必须要通过 SELinux 政策内的规则放行后,就可以与目标资源进行安全性本文的比对。 (2) 比对安全性本文,比对成功就可以访问目标,比对失败,记录拒绝信息。 SELinux的工作模式一共有三种 Enforcing、Permissive和Disabled : * Enforcing 模式:将受限主体进入规则比对、安全本文比对,如果失败,抵挡主体程序的读写行为,并且记录这一行为。 如果成功,这才进入到 rwx 权限的判断。* Permissive模式:不会抵挡主体程序读写行为,只是将该动作记录下来。* Disabled 的模式:禁用SELinux,直接去判断 rwx。Docker守护进程的SELinux功能默认是禁用的,需要使用--selinux-enabled来启用,容器的标签限制可使用-security-opt加载SELinux或者AppArmor的策略进行配置。 下面演示使用SELinux: 1、我们在宿主机上开启 SELinux,尝试启动一个 Nginx 容器并将 nginx.conf 挂载到容器内。
使用 docker ps 查看容器状态,容器已经退出,退出日志为“Permission denied”```
[root@localhost conf]# docker ps -aCONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMESbbef34e4caa4 nginx "nginx -g 'daemon of…" 15 seconds ago Exited (1) 13 seconds ago test-selinux-nginx[root@localhost conf]# docker logs -f bbef34e4caa42018/09/17 15:16:02 [emerg] 1#1: open() "/etc/nginx/nginx.conf" failed (13: Permission denied)nginx: [emerg] open() "/etc/nginx/nginx.conf" failed (13: Permission denied)[root@localhost conf]#
可以看到错误信息好像是权限被拒绝,那么我们检查一下nginx.conf 的权限是否符合我们的要求。使用 ls -Z 查看 nginx.conf的 DAC 与 MAC 权限信息。
[root@localhost conf]# ls -Z-rw-r--r--. root root unconfined_u:object_r:admin_home_t:s0 nginx.conf[root@localhost conf]#
文件的权限为 644。我们在上面查看 Docker 进程,Docker 进程的权限为 root,对于644的权限文件是可读可写的。 看来,问题应该是出在 MAC 权限上。分析 ls -Z 的结果,nginx.conf 对应的安全性文本的类型为 admin_home_t:s0,在启用SELinux后,我们的主体是无法操作这种类型的object的,所以无论 Docker 容器的权限是否是 root,Docker 容器进程都没有权限读取宿主上的 nginx.conf。
Docker 官方提供了一种解决方案专门用来解决与 SELinux 相关的权限问题,在将 SELinux 上的文件挂载到容器中时,在挂载的路径最后加上:z。如:
docker run -v /var/db:/var/db:z rhel7 /bin/sh
Docker 会自动将被挂载的宿主目录的安全性文本配置为目标可读。
[root@localhost conf]# docker run --name test-selinux-z-nginx -v /root/nginx/conf/nginx.conf:/etc/nginx/nginx.conf:z -d nginxdb49bbe352ff1ab800274a17fd18f9c7d86c281e60ac3ffa36ba14e12949285d[root@localhost conf]# docker psCONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMESdb49bbe352ff nginx "nginx -g 'daemon of…" 5 seconds ago Up 2 seconds 80/tcp test-selinux-z-nginx[root@localhost conf]#
Linux内核会限制所有进程可以打开的文件总数,同时为了防止某个进程消耗过多文件资源,也会对单个进程设置限制,这个时候ulimit就派上了用场,使用ulimit命令可以限制进程最多打开文件句柄数、最多打开进程数、线程栈大小等等。Docker对ulimit也提供了支持,Docker 1.6之前,Docker容器的ulimit设置,继承自Docker daemon,Docker 1.6之后,既可以设置全局默认的ulimit,也可以对单个容器指定ulimit。
如下,指定容器最多可打开文件句柄数为2048,最多打开100个进程。
[lynzabo@VM_0_6_centos ~]$ docker run -it --ulimit nofile=2048 --ulimit nproc=100 busybox sh / # ulimit -a-f: file size (blocks) unlimited-t: cpu time (seconds) unlimited-d: data seg size (kb) unlimited-s: stack size (kb) 8192-c: core file size (blocks) unlimited-m: resident set size (kb) unlimited-l: locked memory (kb) 64-p: processes 100-n: file descriptors 2048-v: address space (kb) unlimited-w: locks unlimited-e: scheduling priority 0-r: real-time priority 0/ #
容器进程数限制坑介绍说起进程数限制,大家可能都知道ulimit的nproc这个配置,nproc是存在坑的,与其他ulimit选项不同的是,nproc是一个以用户为管理单位的设置选项,即他调节的是属于一个用户UID的最大进程数之和。如下面输出:
# 我们使用daemon用户启动4个容器,并设置允许的最大进程数为3$ docker run -d -u daemon --ulimit nproc=3 busybox top$ docker run -d -u daemon --ulimit nproc=3 busybox top$ docker run -d -u daemon --ulimit nproc=3 busybox top# 这个容器会失败并报错,资源不足$ docker run -d -u daemon --ulimit nproc=3 busybox top
我们指定使用daemon用户来在容器中启动top进程,结果启动到第4个容器的时候就报错了。而实际上,我们本来是想限制每个容器里用户最多只能创建3个进程。另外,默认情况下,Docker在容器中启动进程是以root用户身份启动的,而ulimit的nproc参数是无法对root用户进行限制。
Docker从1.10以后,支持为容器指定–pids-limit 限制容器内进程数,和容器里用户无关。如下面例子:
[lynzabo@VM_0_6_centos ~]$ docker run -d --name test-pids-limit --pids-limit=5 busybox top5693c8c31284b0f3cb4eb10d4f67e13ad98d1972a27dab094f0ad96154a5ce6a[lynzabo@VM_0_6_centos ~]$ docker exec -ti test-pids-limit sh / # ps -efPID USER TIME COMMAND1 root 0:00 top5 root 0:00 sh9 root 0:00 ps -ef/ # nohup top &/ # nohup: appending output to nohup.out/ # nohup top &/ # nohup: appending output to nohup.out/ # nohup top &/ # nohup: appending output to nohup.out/ # nohup top &sh: can't fork: Resource temporarily unavailable/ #
在容器生态的周围,还有很多工具可以为容器安全性提供支持。1、可以使用 docker-bench-security检查你的Docker运行环境,如Docker daemon配置,宿主机配置
2、使用Sysdig Falco(地址:https://sysdig.com/opensource/falco/)可以监视容器的行为,检测容器中是否有异常活动。
3、使用GRSEC 和 PAX来加固系统内核,还可以使用GRSecurity为系统提供更丰富的安全限制。等等。
这篇文章的分享就到这里,希望本次分享对大家有所帮助,欢迎留言与我们交流。
这是一份网络安全从零基础到进阶的学习路线大纲全览,小伙伴们记得点个收藏!
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dKvWFouG-1676464883938)()]编辑
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-e6daLnGU-1676464883938)()]
网络安全导论
渗透测试基础
网络基础
操作系统基础
Web安全基础
数据库基础
编程基础
CTF基础
该阶段学完即可年薪15w+
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-vmRxdPNJ-1676464883939)()]
弱口令与口令爆破
XSS漏洞
CSRF漏洞
SSRF漏洞
XXE漏洞
SQL注入
任意文件操作漏洞
业务逻辑漏洞
该阶段学完年薪25w+
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xsM5MaI8-1676464883939)()]
反序列化漏洞
RCE
综合靶场实操项目
内网渗透
流量分析
日志分析
恶意代码分析
应急响应
实战训练
该阶段学完即可年薪30w+
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-lyj3ZV2u-1676464883940)()]
蓝队基础
蓝队进阶
该部分主攻蓝队的防御,即更容易被大家理解的网络安全工程师。
攻防兼备,年薪收入可以达到40w+
需要上述路线图对应的网络安全配套视频、源码以及更多网络安全相关书籍&面试题等内容
同学们可以扫描下方二维码获取哦!
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。