写在前面

确保容器中服务与应用安全是容器化演进的关键点。容器安全涉及到应用开发与维护的整个生命周期,本文主要从镜像构建的视角来看docker容器的一些安全问题及应对措施。


一、权限管理

1.避免以容器以root身份运行

在Openshift与k8s环境中默认容器需要以非root身份运行,使用root身份运行的情况很少,所以不要忘记在dockerfile中包含USER指令,以将启动容器时默认有效 的UID 更改为非 root 用户。

以非 root 身份运行需要在 Dockerfile 中做的两个步骤:

  • 确保USER指令中指定的用户存在于容器内。
  • 在进程将要读取或写入的位置提供适当的文件系统权限。
FROM alpine
#创建目录,添加myuser用户,目录所有作为myuser
RUN mkdir /server && adduser -D myuser && chown -R myuser /server
USER myuser
WORKDIR /server
COPY myapp ./
CMD ["./myapp"]

2.可执行文件权限应为root用户拥有但不可写

容器中的每个可执行文件都应该由 root 用户拥有,即使它由非 root 用户执行,并且不应该是全局可写的。

通过阻止执行用户修改现有的二进制文件或脚本,可以有效降低攻击,保证容器不变性。不可变容器不会在运行时自动更新其代码,通过这种方式,我们可以防止正在运行的应用程序被意外或恶意修改。

我们在使用COPY时

COPY --chown=myuser:myuser myapp ./
#应改为
COPY myapp ./

二、减少攻击面

避免加载不必要的包、第三方应用或暴露端口以减少攻击面。我们在镜像中包含的组件内容越多,容器暴露的就越多,维护起来就越困难。

1.采用多阶段构建

我们在《Dockerfile 多阶段构建实践》中说到采用多阶段构建,可以此降低构建复杂度,同时有效减小镜像尺寸。

在多阶段构建中,我们创建一个中间容器(阶段),其中包含编译工具及生成最终可执行文件。然后,我们只将生成的工件复制到最终镜像中,而无需额外的开发依赖项、临时构建文件等等。

精心设计的多阶段构建仅包含最终映像中所需的最少二进制文件和依赖项,而不包含构建工具或中间文件。它更为安全,并且还减小了镜像大小。可以有效减少了攻击面,减少了漏洞。

多阶段构建的实现请参考上篇文章《Dockerfile 多阶段构建实践》

2.使用可信赖的镜像

假如我们不是从头开始构建镜像,基镜像建立在不受信任或不受维护的镜像之上会将所有问题和漏洞从该镜像继承到您的容器中。

基础镜像选择的参考:

  • 我们应该选择来自受信任仓库经过验证的官方镜像。
  • 使用自定义镜像时,我们应该检查镜像源和构建的 Dockerfile。更进一步,我们甚至应该以这个Dockerfile来构建自己的基础镜像。因为我们无法保证在dockerhub等公共仓库中发布的映像确实是从指定的 Dockerfile 构建的。也不能保证它是最新的。
  • 有时候在安全性和极简主义方面考虑,官方镜像可能并不非合适的,最优解是我们自己从头构建属于自己的镜像。

2.从头开始构建镜像

假如如果你是从centos镜像开始构建,那么你创建的容器可能将会包含几十个或者上百个漏洞。所以构建一个安全的镜像我们最好需要知道我们的基镜像存在哪些威胁。在生产中通常会从Scratch空镜像或distroless开始。

distroless镜像仅包含应用程序及其运行时依赖项。它们不包括在标准 Linux 发行版中发布应用如包管理器、shell 或任何其他程序。Distroless 镜像非常小。最小的 distroless 图像gcr.io/distroless/static大约为 650 kB。只有alpine(约2.5 MB)大小的 四分之一 ,不到debian(50 MB)大小的 1.5% 。

FROM golang:1.13-buster as build

WORKDIR /go/src/app
ADD . /go/src/app RUN go get -d -v ./... RUN go build -o /go/bin/app # 引用Distroless镜像
FROM gcr.io/distroless/base-debian10
COPY --from=build /go/bin/app /
CMD ["/app"]

gcr.io/distroless/base-debian10只包含一组基本的包,如包括只需要的库,如glibclibsslopenssl 当然对于像 Go 这样不需要libc 的静态编译应用程序我们就可以替换为如下基镜像

FROM gcr.io/distroless/static-debian10

关于distroless基镜像的更多信息可以参考https://github.com/GoogleContainerTools/distroless

3.及时更新镜像

使用经常更新的基础镜像,在需要时重构你的镜像。随着新的安全漏洞不断被发现,坚持使用最新的安全补丁是一种通用的安全最佳实践。

版本控制策略:

  • 坚持使用稳定或长期支持版本,这些版本会迅速提供安全修复程序。
  • 提前计划。准备好在基本镜像版本达到生命周期结束或停止接收更新之前删除旧版本并迁移。
  • 定期重建自己的镜像,从基础发行版、Node、Golang、Python 等获取最新的包。 大多数包或依赖项管理器,如npmgo mod,将提供指定版本最新的安全更新。

4.端口暴露

容器中每个打开的端口都是通往系统的大门。我们应该仅公开应用程序需要的端口,并且避免公开 SSH (22) 等端口。

我们知道 Dockerfile 提供了EXPOSE 命令有暴露端口,但是该命令仅用于提供信息和用于文档目的。运行容器时,容器不会自动允许所有 EXPOSE 端口的连接(除非在启动容器时使用docker run --publish-all)。

启动容器时,通过-P暴露的端口应与dockerfile中EXPOSE命令指定的端口一致,这样更便于维护。


三、敏感数据管理

1.凭证和密钥

禁止在 Dockerfile 指令(环境变量、参数或其他任何命令中)中放入凭据和密钥。

在复制文件到镜像时,即使文件在 Dockerfile 的后续指令中被删除,它仍然可以在之前的层上访问。因为镜像分层原理,你的文件并没有真正被删除,只是“隐藏”在最终文件系统中。因此在构建镜像时,我们应该遵循以下做法:

关于secrets的使用会在后面文章中详细介绍。

2.ADD、COPY

ADD 和 COPY 指令在 Dockerfile 中提供类似的功能。但是COPY 更为明确。

除非我们确实需要 使用ADD 功能,例如从 URL 或从 tar 文件添加文件。不然最好使用 COPY,COPY 的结果更具可预测性且不易出错。

在某些情况下,最好使用 RUN 指令而不是 ADD 来下载使用curlwget的包,解压缩然后删除原始文件,减少层数。

3.构建上下文与dockerignore

在构建时我们通常使用.作为上下文

#docker build -t images:v1 .

使用 .作为上下文时我们需要谨慎些,因为docker CLI会将上下文中机密或不必要的文件添加到守护进程,甚至到容器中,例如配置文件、凭据、备份、锁定文件、临时文件、源、子文件夹、点文件等等。

在比如:

COPY . /server

此时会将目录下所有内容都添加到镜像中,包括Dockfile本身。

所以正确做法是创建一个包含需要在容器内复制文件的文件夹,将其用作构建上下文,并在可能的情况下明确 COPY 指令(避免使用通配符)。例如:

#docker build -t images:v1 build_files/

为了排除不必要的文件,我们也可以创建一个.dockerignore文件,在其中明确排除的文件和目录。


以上是容器构建时常见安全问题与相关处理措施,容器安全涉及面广,遍布整个devops流程中。有兴趣的同学可以另外一个位面介入深究。

NEXT

  • Docker容器secrets详解
  • Docker容器减小镜像尺寸实践

希望小作文对你有些许帮助,如果内容有误请指正。

您可以随意转载、修改、发布本文章,无需经过本人同意。

Docker容器 关于镜像构建的安全问题的更多相关文章

  1. 7.云原生之Docker容器Dockerfile镜像构建浅析与实践

    转载自:https://www.bilibili.com/read/cv15220707/?from=readlist Dockerfile 镜像构建浅析与实践 描述:Dockerfile是一个文本格 ...

  2. [转]图解Docker容器和镜像

    本文转自:https://www.cnblogs.com/wangqiaomei/p/5818636.html 图解Docker容器和镜像 这篇文章希望能够帮助读者深入理解Docker的命令,还有容器 ...

  3. 图解Docker容器和镜像

    图解Docker容器和镜像 这篇文章希望能够帮助读者深入理解Docker的命令,还有容器(container)和镜像(image)之间的区别,并深入探讨容器和运行中的容器之间的区别. 当我对Docke ...

  4. Docker容器和镜像的区别

    docker容器和镜像区别  转自 https://www.cnblogs.com/bethal/p/5942369.html 这篇文章希望能够帮助读者深入理解Docker的命令,还有容器(conta ...

  5. 理解docker容器和镜像(layer,ufs)和docker命令解释

    博客好文1:http://blog.csdn.net/x931100537/article/details/49633107(理解docker容器和镜像,理解简单,从原理入手,什么是layer,什么是 ...

  6. 【原创】深入理解Docker容器和镜像 -- 分析了docker的命令含义

    10张图带你深入理解Docker容器和镜像 镜像(Image)就是一堆只读层(read-only layer)的统一视角 要点:容器 = 镜像 + 读写层.并且容器的定义并没有提及是否要运行容器. 一 ...

  7. docker容器和镜像区别

    这篇文章希望能够帮助读者深入理解Docker的命令,还有容器(container)和镜像(image)之间的区别,并深入探讨容器和运行中的容器之间的区别. 当我对Docker技术还是一知半解的时候,我 ...

  8. docker之NGINX镜像构建

    Nginx是一个高性能的Web和反向代理服务器,它具有很多非常优越的特性:1.作为Web服务器.2.作为负载均衡服务器.3.作为邮件代理服务器.4.安装及配置简单.接下来我们介绍在docker构建ng ...

  9. 10张图带你深入理解Docker容器和镜像

    http://dockone.io/article/783 [编者的话]本文用图文并茂的方式介绍了容器.镜像的区别和Docker每个命令后面的技术细节,能够很好的帮助读者深入理解Docker. Doc ...

随机推荐

  1. netcore一键nssm发布为windows服务

    AntDeploy 是我开发一款开源一键部署工具包 发布功能支持: docker容器一键部署 docker镜像一键发布 支持iis一键部署 windows服务一键部署 linux服务一键部署 支持增量 ...

  2. (Opencv06)绘制轮廓函数

    (Opencv06)绘制轮廓函数 cv2.drawContours(image, contours, contourIdx, color, thickness) img: 指在哪副图像上绘制轮廓 会改 ...

  3. spring Data Rest 远程命令执行漏洞(CVE-2017-8046)

    参考 文章https://blog.csdn.net/weixin_41438728/article/details/110425174 影响版本 Spring Framework 5.0 to 5. ...

  4. 漏洞分析:CVE 2021-3156

    漏洞分析:CVE 2021-3156 漏洞简述 漏洞名称:sudo堆溢出本地提权 漏洞编号:CVE-2021-3156 漏洞类型:堆溢出 漏洞影响:本地提权 利用难度:较高 基础权限:需要普通用户权限 ...

  5. 小白学习vue第五天(理解使用组件开发,组件第一弹)

    组件怎么从创建到使用? 第一步创建组件构造器对象 感觉个人理解就是创建一个模板,和创建MongoDB数据模板相似 const cpnC = Vue.extend({ template: ` <d ...

  6. 关于表单重复提交之验证码 和谷歌Kaptcha图片验证码的使用

    表单重复提交之-----验证码 表单重复提交有三种常见的情况: 一:提交完表单.服务器使用请求转来进行页面跳转.这个时候,用户按下功能键 F5,就会发起最后一次的请求. 造成表单重复提交问题.解决方法 ...

  7. Blazor 模板化组件开发指南

    翻译自 Waqas Anwar 2021年4月15日的文章 <A Developer's Guide To Blazor Templated Components> [1] 在我之前的一篇 ...

  8. SpringCloud-技术专区-从源码层面让你认识Feign工作流程和运作机制

    Feign工作流程源码解析 什么是feign:一款基于注解和动态代理的声明式restful http客户端. 原理 Feign发送请求实现原理 微服务启动类上标记@EnableFeignClients ...

  9. S3C2440—12.按键中断

    文章目录 一. 总体 二. CPSR设置 三. 中断源设置 四. 中断控制器设置 五. C中断处理函数 六. 汇编IRQ异常处理程序 七. 源码 一. 总体 要驱动按键中断控制LED亮灭,程序要进行如 ...

  10. SpringCloud升级之路2020.0.x版-18.Eureka的客户端核心设计和配置

    本系列代码地址:https://github.com/HashZhang/spring-cloud-scaffold/tree/master/spring-cloud-iiford Eureka 客户 ...