Docker是PaaS供应商dotCloud开源的一个基于LXC 的高级容器引擎,源代码托管在 GitHub 上, 基于Go语言开发并遵从Apache 2.0协议开源。Docker提供了一种在安全、可重复的环境中自动部署软件的方式,它的出现拉开了基于云计算平台发布产品方式的变革序幕。为了更好的促进Docker在国内的发展以及传播,我们决定开设《深入浅出Docker》专栏,邀请Docker相关的布道师、开发人员、技术专家来讲述Docker的各方面内容,让读者对Docker有更深入的了解,并且能够积极投入到新技术的讨论和实践中。另外,欢迎加入InfoQ Docker技术交流群交流Docker的最佳实践,QQ群号:365601355。1. Docker命令行Docker官方为了让用户快速了解Docker,提供了一个交互式教程,旨在帮助用户掌握Docker命令行的使用方法。但是由于Docker技术的快速发展,此交互式教程已经无法满足Docker用户的实际使用需求,所以让我们一起开始一次真正的命令行学习之旅。首先,Docker的命令清单可以通过运行 docker ,或者 docker help 命令得到: $ sudo docker 在Docker容器技术不断演化的过程中,Docker的子命令已经达到34个之多,其中核心子命令(例如:run)还会有复杂的参数配置。笔者通过结合功能和应用场景方面的考虑,把命令行划分为4个部分,方便我们快速概览Docker命令行的组成结构:
1.1 参数约定单个字符的参数可以放在一起组合配置,例如 docker run -t -i --name test busybox sh 可以用这样的方式等同: docker run -ti --name test busybox sh 1.2 BooleanBoolean参数形式如: -d=false。注意,当你声明这个Boolean参数时,比如 docker run -d=true,它将直接把启动的Container挂起放在后台运行。 1.3 字符串和数字参数如 --name=“” 定义一个字符串,它仅能被定义一次。同类型的如-c=0 定义一个数字,它也只能被定义一次。 1.4 后台进程Docker后台进程是一个常驻后台的系统进程,值得注意的是Docker使用同一个文件来支持客户端和后台进程,其中角色切换通过-d来实现。这个后台进程是用来管理容器的,使用Docker --help可以得到更详细的功能参数配置, 如下图:
注意,其中带有[] 的启动参数可以指定多次,例如 $ docker run -a stdin -a stdout -a stderr -i -t ubuntu /bin/bash 2. Docker命令行探秘2.1 环境信息相关info 使用方法: docker info 例子: [fedora@docker-devel-cli docker]$ sudo docker -D infoContainers: 0Images: 32Storage Driver: devicemapper Pool Name: docker-252:1-130159-pool Data file: /var/lib/docker/devicemapper/devicemapper/data Metadata file: /var/lib/docker/devicemapper/devicemapper/metadata Data Space Used: 1616.9 Mb Data Space Total: 102400.0 Mb Metadata Space Used: 2.4 Mb Metadata Space Total: 2048.0 MbExecution Driver: native-0.2Kernel Version: 3.11.10-301.fc20.x86_64Debug mode (server): falseDebug mode (client): trueFds: 11Goroutines: 14EventsListeners: 0Init SHA1: 2c5adb59737b8a01fa3fb968519a43fe140bc9c9Init Path: /usr/libexec/docker/dockerinitSockets: [fd://] 使用说明: 这个命令在开发者报告Bug时会非常有用,结合docker vesion一起,可以随时使用这个命令把本地的配置信息提供出来,方便Docker的开发者快速定位问题。 version 使用方法: docker version 使用说明: 显示Docker的版本号,API版本号,Git commit, Docker客户端和后台进程的Go版本号。 2.2 系统运维相关attach 使用方法: docker attach [OPTIONS] CONTAINER 例子: $ ID=$(sudo docker run -d ubuntu /usr/bin/top -b)$ sudo docker attach $IDtop - 17:21:49 up 5:53, 0 users, load average: 0.63, 1.15, 0.78Tasks: 1 total, 1 running, 0 sleeping, 0 stopped, 0 zombie%Cpu(s): 1.0 us, 0.7 sy, 0.0 ni, 97.7 id, 0.7 wa, 0.0 hi, 0.0 si, 0.0 stKiB Mem: 2051644 total, 723700 used, 1327944 free, 33032 buffersKiB Swap: 0 total, 0 used, 0 free. 565836 cached Mem PID USER PR NI VIRT RES SHR S %CPU %MEM TIME+ COMMAND 1 root 20 0 19748 1280 1008 R 0.0 0.1 0:00.04 top$ sudo docker stop $ID 使用说明: 使用这个命令可以挂载正在后台运行的容器,在开发应用的过程中运用这个命令可以随时观察容器內进程的运行状况。开发者在开发应用的场景中,这个命令是一个非常有用的命令。 build 使用方法:docker build [OPTIONS] PATH | URL | - 例子: $ docker build .Uploading context 18.829 MBUploading contextStep 0 : FROM busybox ---> 769b9341d937Step 1 : CMD echo Hello world ---> Using cache ---> 99cc1ad10469Successfully built 99cc1ad10469 使用说明: 这个命令是从源码构建新Image的命令。因为Image是分层的,最关键的Base Image是如何构建的是用户比较关心的,Docker官方文档给出了构建方法,请参考这里。 commit 使用方法:docker commit [OPTIONS] CONTAINER [REPOSITORY[:TAG]] 例子: $ sudo docker psID IMAGE COMMAND CREATED STATUS PORTSc3f279d17e0a ubuntu:12.04 /bin/bash 7 days ago Up 25 hours197387f1b436 ubuntu:12.04 /bin/bash 7 days ago Up 25 hours$ docker commit c3f279d17e0a SvenDowideit/testimage:version3f5283438590d$ docker images | headREPOSITORY TAG ID CREATED VIRTUAL SIZESvenDowideit/testimage version3 f5283438590d 16 seconds ago 335.7 MB 使用说明: 这个命令的用处在于把有修改的container提交成新的Image,然后导出此Imange分发给其他场景中调试使用。Docker官方的建议是,当你在调试完Image的问题后,应该写一个新的Dockerfile文件来维护此Image。commit命令仅是一个临时创建Imange的辅助命令。 cp 使用方法: cp CONTAINER:PATH HOSTPATH 使用说明: 使用cp可以把容器內的文件复制到Host主机上。这个命令在开发者开发应用的场景下,会需要把运行程序产生的结果复制出来的需求,在这个情况下就可以使用这个cp命令。 diff 使用方法:docker diff CONTAINER 例子: $ sudo docker diff 7bb0e258aefeC /devA /dev/kmsgC /etcA /etc/mtabA /goA /go/srcA /go/src/github.comA /go/src/github.com/dotcloud.... 使用说明: diff会列出3种容器内文件状态变化(A - Add, D - Delete, C - Change )的列表清单。构建Image的过程中需要的调试指令。 export 使用方法:docker export CONTAINER 例子: $ sudo docker export red_panda > latest.tar 使用说明: 把容器系统文件打包并导出来,方便分发给其他场景使用。 images 使用方法:docker images [OPTIONS] [NAME] 例子: $ sudo docker images | headREPOSITORY TAG IMAGE ID CREATED VIRTUAL SIZE<none> <none> 77af4d6b9913 19 hours ago 1.089 GBcommittest latest b6fa739cedf5 19 hours ago 1.089 GB<none> <none> 78a85c484f71 19 hours ago 1.089 GB$ docker latest 30557a29d5ab 20 hours ago 1.089 GB<none> <none> 0124422dd9f9 20 hours ago 1.089 GB<none> <none> 18ad6fad3402 22 hours ago 1.082 GB<none> <none> f9f1e26352f0 23 hours ago 1.089 GBtryout latest 2629d1fa0b81 23 hours ago 131.5 MB<none> <none> 5ed6274db6ce 24 hours ago 1.089 GB 使用说明: Docker Image是多层结构的,默认只显示最顶层的Image。不显示的中间层默认是为了增加可复用性、减少磁盘使用空间,加快build构建的速度的功能,一般用户不需要关心这个细节。 import / save / load 使用方法: docker import URL|- [REPOSITORY[:TAG]]docker save IMAGEdocker load 使用说明: 这一组命令是系统运维里非常关键的命令。加载(两种方法: import, load),导出(一种方法: save)容器系统文件。 inspect 使用方法: docker inspect CONTAINER|IMAGE [CONTAINER|IMAGE...] 例子: $ sudo docker inspect --format='{{.NetworkSettings.IPAddress}}' $INSTANCE_ID 使用说明: 查看容器运行时详细信息的命令。了解一个Image或者Container的完整构建信息就可以通过这个命令实现。 kill 使用方法: docker kill [OPTIONS] CONTAINER [CONTAINER...] 使用说明: 杀掉容器的进程。 port 使用方法: docker port CONTAINER PRIVATE_PORT 使用说明: 打印出Host主机端口与容器暴露出的端口的NAT映射关系 pause / unpause 使用方法: docker pause CONTAINER 使用说明: 使用cgroup的freezer顺序暂停、恢复容器里的所有进程。详细freezer的特性,请参考官方文档。 ps 使用方法: docker ps [OPTIONS] 例子: $ docker psCONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES4c01db0b339c ubuntu:12.04 bash 17 seconds ago Up 16 seconds webappd7886598dbe2 crosbymichael/redis:latest /redis-server --dir 33 minutes ago Up 33 minutes 6379/tcp redis,webapp/db 使用说明: docker ps打印出正在运行的容器,docker ps -a打印出所有运行过的容器。 rm 使用方法: docker rm [OPTIONS] CONTAINER [CONTAINER...] 例子: $ sudo docker rm /redis/redis 使用说明: 删除指定的容器。 rmi 使用方法: docker rmi IMAGE [IMAGE...] 例子: $ sudo docker imagesREPOSITORY TAG IMAGE ID CREATED SIZEtest1 latest fd484f19954f 23 seconds ago 7 B (virtual 4.964 MB)test latest fd484f19954f 23 seconds ago 7 B (virtual 4.964 MB)test2 latest fd484f19954f 23 seconds ago 7 B (virtual 4.964 MB)$ sudo docker rmi fd484f19954fError: Conflict, cannot delete image fd484f19954f because it is tagged in multiple repositories2013/12/11 05:47:16 Error: failed to remove one or more images$ sudo docker rmi test1Untagged: fd484f19954f4920da7ff372b5067f5b7ddb2fd3830cecd17b96ea9e286ba5b8$ sudo docker rmi test2Untagged: fd484f19954f4920da7ff372b5067f5b7ddb2fd3830cecd17b96ea9e286ba5b8$ sudo docker imagesREPOSITORY TAG IMAGE ID CREATED SIZEtest latest fd484f19954f 23 seconds ago 7 B (virtual 4.964 MB)$ sudo docker rmi testUntagged: fd484f19954f4920da7ff372b5067f5b7ddb2fd3830cecd17b96ea9e286ba5b8Deleted: fd484f19954f4920da7ff372b5067f5b7ddb2fd3830cecd17b96ea9e286ba5b8 使用说明: 指定删除Image文件。 run 使用方法: docker run [OPTIONS] IMAGE [COMMAND] [ARG...] 例子: $ sudo docker run --cidfile /tmp/docker_test.cid ubuntu echo "test" 使用说明: 这个命令是核心命令,可以配置的参数多达28个参数。详细的解释可以通过docker run --help列出。官方文档中提到的 Issue 2702:"lxc-start: Permission denied - failed to mount" could indicate a permissions problem with AppArmor. 在最新版本的Dcoker中已经解决。 start / stop / restart 使用方法: docker start CONTAINER [CONTAINER...] 使用说明: 这组命令可以开启(两个:start, restart),停止(一个:stop)一个容器。 tag 使用方法: docker tag [OPTIONS] IMAGE[:TAG] [REGISTRYHOST/][USERNAME/]NAME[:TAG] 使用说明: 组合使用用户名,Image名字,标签名来组织管理Image。 top 使用方法: docker top CONTAINER [ps OPTIONS] 使用说明: 显示容器內运行的进程。 wait 使用方法: docker wait CONTAINER [CONTAINER...] 使用说明: 阻塞对指定容器的其他调用方法,直到容器停止后退出阻塞。 2.3 日志信息相关events 使用方法: docker events [OPTIONS] 使用说明: 打印容器实时的系统事件。 history 使用方法: docker history [OPTIONS] IMAGE 例子: $ docker history dockerIMAGE CREATED CREATED BY SIZE3e23a5875458790b7a806f95f7ec0d0b2a5c1659bfc899c89f939f6d5b8f7094 8 days ago /bin/sh -c #(nop) ENV LC_ALL=C.UTF-8 0 B8578938dd17054dce7993d21de79e96a037400e8d28e15e7290fea4f65128a36 8 days ago /bin/sh -c dpkg-reconfigure locales && locale-gen C.UTF-8 && /usr/sbin/update-locale LANG=C.UTF-8 1.245 MBbe51b77efb42f67a5e96437b3e102f81e0a1399038f77bf28cea0ed23a65cf60 8 days ago /bin/sh -c apt-get update && apt-get install -y git libxml2-dev python build-essential make gcc python-dev locales python-pip 338.3 MB4b137612be55ca69776c7f30c2d2dd0aa2e7d72059820abf3e25b629f887a084 6 weeks ago /bin/sh -c #(nop) ADD jessie.tar.xz in / 121 MB750d58736b4b6cc0f9a9abe8f258cef269e3e9dceced1146503522be9f985ada 6 weeks ago /bin/sh -c #(nop) MAINTAINER Tianon Gravi <admwiggin@gmail.com> - mkimage-debootstrap.sh -t jessie.tar.xz jessie http://http.debian.net/debian 0 B511136ea3c5a64f264b78b5433614aec563103b4d4702f3ba7d4d2698e22c158 9 months ago 0 B
使用说明: 打印指定Image中每一层Image命令行的历史记录。 logs 使用方法: docker logs CONTAINER 使用说明: 批量打印出容器中进程的运行日志。 2.4 Dcoker Hub服务相关login 使用方法: docker login [OPTIONS] [SERVER] 使用说明: 登录Hub服务。 pull / push 使用方法: docker push NAME[:TAG] 使用说明: 通过此命令分享Image到Hub服务或者自服务的Registry服务。 search 使用方法: docker search TERM 使用说明: 通过关键字搜索分享的Image。 3. 总结通过以上Docker命令行的详细解释,可以强化对Docker命令的全面理解。考虑到Docker命令行的发展变化非常快,读者可以参考官方的命令行解释文档更新相应的命令行解释。另外,通过以上Docker命令行的分析,可以知道Docker命令行架构设计的特点在于客户端和服务端的运行文件是同一个文件,内部实现代码应该是重用的设计。笔者希望开发者在开发类似的命令行应用时参考这样的设计,减少前后台容错的复杂度。 参考文献[1] https://docs.docker.com/reference/commandline/cli/ [2] https://en.wikipedia.org/wiki/Cross-Origin_Resource_Sharing [3] https://en.wikipedia.org/wiki/CIDR_notation#CIDR_notation |