# 小巧玲珑的ubuntu镜像 **This translation was last updated October 24, 2014.** Baseimage-docker是一个特殊的 [Docker](https://www.docker.com) 镜像,可以很优雅的将它用于docker镜像。相对于 Ubuntu, 有这些新增: * 修改之后,可以很友好的使用docker * 修复了一些docker的bug [some Docker bugs](#workaroud_modifying_etc_hosts). * 很有用的管理员工具 你可以用它作为你自己的基础docker镜像。 Baseimage-docker 可以从[the Docker registry](https://registry.hub.docker.com/u/phusion/baseimage/)获取到! ### 原生的ubuntu基础镜像有什么问题呢? 原生ubuntu不是设计为运行docker的。它是假设运行到真实硬件或者虚拟硬件上的,但不是docker容器。你运行docker容器,不需要一个完整的系统,你需要一个小巧的系统。如果你不熟悉unix系统,那么配制出来的精简系统会有很奇怪的问题。这样就可以导致很多奇奇怪怪的问题。 Baseimage-docker 将错误都修正了。在"内容"部分,可以看到哪些内容修改过了. ### 为什么使用 baseimage-docker? 你可以用原生`ubuntu`镜像作为你的Dockerfile,那为什么又麻烦的使用baseimage-docker呢? * 配置一个友好的基础镜像不是一个简单的任务.正如之前提到的,会很有的坑在里面.当你处理那些坑之后,你又重复造了一个baseimage-docker.使用baseimage-docker会拯救你的努力. * 它减少了一个写正确Dockerfile文件的时间.你不用担心基础系统,你可以专注你的业务和你的项目. * 它减少了运行`docker build`的时间,让你更快的迭代Dockerfile. * 它减少了多次下载的时间.Docker只需要下载一次基础镜像:在第一次部署的时候.在之后的部署中,只需要对你最近的基础镜像进行修改. ----------------------------------------- **相关资源** [Website](http://phusion.github.io/baseimage-docker/) | [Github](https://github.com/phusion/baseimage-docker) | [Docker registry](https://index.docker.io/u/phusion/baseimage/) | [Discussion forum](https://groups.google.com/d/forum/passenger-docker) | [Twitter](https://twitter.com/phusion_nl) | [Blog](http://blog.phusion.nl/) **Table of contents** **目录** * [镜像里面包含什么?](#whats_inside) * [概述](#whats_inside_overview) * [等等,我认为docker在一个容器中只能允许一个进程?](#docker_single_process) * [细细看看baseimage-docker](#inspecting) * [用baseimage-docker作为基础镜像](#using) * [开始](#getting_started) * [增加后台进行](#adding_additional_daemons) * [在启动容器过程中允许脚本](#running_startup_scripts) * [环境变量](#environment_variables) * [集中定义你自己的环境变量](#envvar_central_definition) * [保存环境变量](#envvar_dumps) * [修改环境变量](#modifying_envvars) * [安全性](#envvar_security) * [解决docker没有办法修改/etc/hosts的问题](#workaroud_modifying_etc_hosts) * [禁用ssh](#disabling_ssh) * [容器管理](#container_administration) * [在一个新容器中运行单条命令](#oneshot) * [在正在运行的的容器中运行命令](#run_inside_existing_container) * [通过nsenter登录容器](#login_nsenter) * [用法](#nsenter_usage) * [`docker-bash`工具](#docker_bash) * [使用ssh登录容器](#login_ssh) * [在容器中使用不安全的key](#using_the_insecure_key_for_one_container_only) * [支持长久不变的不安全的key](#enabling_the_insecure_key_permanently) * [使用你自己的key](#using_your_own_key) * [`docker-ssh`工具](#docker_ssh) * [构建你自己的镜像](#building) * [总结](#conclusion) ----------------------------------------- ## 镜像里面包含什么? ### 概述 *想看一个完整的基础镜像?这里有一个包含Ruby,Python Node.js and Meteor的.可以看看[passenger-docker](https://github.com/phusion/passenger-docker).* | 模块 | 包含什么 / 备注 | | ---------------- | ------------------- | | Ubuntu 14.04 LTS | 基础系统. | | 一个**正确**的初始化进程 | 根据Unix进程模型,[初始化进程](https://en.wikipedia.org/wiki/Init) -- PID 1 -- 继承了所有[孤立的子进行],并且必须[监控他们](https://en.wikipedia.org/wiki/Wait_(system_call)).大多数Docker容器没有一个正确的初始化进程,结果是他们的容器就出现了大量的[僵尸进程](https://en.wikipedia.org/wiki/Zombie_process).

此外,`docker stop`发送终止信号给初始化进程,然后停止所有的服务进程.不幸的是,由于他们主机关闭了容器,大多初始系统没有正确运行.

Baseimage-docker包含了一个初始进程`/sbin/my_init`,来正确的执行这些任务. | | 修复了APT 与docker不兼容的问题 | See https://github.com/dotcloud/docker/issues/1024. | | 修复某些Docker bugs | [Learn more.](#workaroud_modifying_etc_hosts) | | syslog-ng | 必须要有一个监控很多服务的系统日志进程,包括内核本身,以便可以正确的输出日志到/var/log/syslog.如果没有后台日志进程,那么很多重要的信息就会丢失了.

只监听本地服务. | | logrotate | logrotate 程序是一个日志文件管理工具,可以定期转存和压缩日志. | | SSH server | 允许你可以很容易的登录到你的容器中进行[管理](#login_ssh).

_baseimage-docker提供了一个SSH的方法.还有其他方法,比如通过[nsenter](#login_nsenter).SSH提供了一个方法,因为nsenter还有很多的问题._

通过密码和challenge-response的方式,默认是禁用的.只有通过key认证的方式是允许的.

如果你想禁用SSH访问的方式,禁用也是很容易的.可以阅读下面的说明. | | cron | 定时任务进程保证定时任务的运行. | | [runit](http://smarden.org/runit/) | 替代Ubuntu的Upstart. 用于监控服务和管理.比SysV init更容易使用,同时支持当有服务挂掉之后,重启这些服务.比Upstart更易使用,也更加的轻量级. | | `setuser` |使用另一个用户运行命令的工具,比`su`更容易使用,减少使用`sudo`的安全性,不像使用`chpst`,需要正确的设置`$HOME`.`/sbin/setuser`这样使用就可以了. | Baseimage-docker is very lightweight: it only consumes 6 MB of memory. Baseimage-docker是非常轻量级的:仅仅小号6MB内存. ### 等等,我认为Docker在一个容器中就运行一个进程吗? 绝对不是这样的. 在一个docker容器中,运行多个进程也是很好的. 事实上,没有什么技术原因限制你只运行一个进程,运行很多的进程,只会把容器中系统的基本功能搞的更乱,比如syslog. Baseimage-docker *鼓励* 通过runit来运行多进程. ## 检测一下baseimage-docker 要检测镜像,执行下面的命令: docker run --rm -t -i phusion/baseimage: /sbin/my_init -- bash -l `` 是[baseimage-docker的版本号](https://github.com/phusion/baseimage-docker/blob/master/Changelog.md). 你不用手动去下载任何文件.上面的命令会自动从docker仓库下载baseimage-docker镜像. ## 使用baseimage-docker作为基础镜像 ### 入门指南 The image is called `phusion/baseimage`, and is available on the Docker registry. 镜像名字叫`phusion/baseimage`,在Docker仓库上也是可用的. 下面的这个是一个Dockerfile的模板. # 使用phusion/baseimage作为基础镜像,去构建你自己的镜像,需要下载一个明确的版本,千万不要使用`latest`. # 查看https://github.com/phusion/baseimage-docker/blob/master/Changelog.md,可用看到版本的列表. FROM phusion/baseimage: # 设置正确的环境变量. ENV HOME /root # 生成SSH keys,baseimage-docker不包含任何的key,所以需要你自己生成.你也可以注释掉这句命令,系统在启动过程中,会生成一个. RUN /etc/my_init.d/00_regen_ssh_host_keys.sh # 初始化baseimage-docker系统 CMD ["/sbin/my_init"] # 这里可以放置你自己需要构建的命令 # 当完成后,清除APT. RUN apt-get clean && rm -rf /var/lib/apt/lists/* /tmp/* /var/tmp/* ### 增加后台进程 你可以通过runit工具向你的镜像中添加后台进程(例如:你自己的某些应用).你需要编写一个运行你需要的后台进程的脚步就可以了,runit工具会保证它的正常运行,如果进程死掉,runit也会重启它的. 脚本的名称必须是`run`,必须是可以运行的,它需要放到`/etc/service/`. 这里有一个例子,向你展示如果运行memcached服务的. ### memcached.sh(确定文件的权限是chmod +x): #!/bin/sh # `/sbin/setuser memcache` 指定一个`memcache`用户来运行命令.如果你忽略了这部分,就会使用root用户执行. exec /sbin/setuser memcache /usr/bin/memcached >>/var/log/memcached.log 2>&1 ### 在Dockerfile中: RUN mkdir /etc/service/memcached ADD memcached.sh /etc/service/memcached/run 注意脚本必须运行在后台的,**不能让他们进程进行daemonize/fork**.通常,后台进程会提供一个标志位或者配置文件. ### 在容器启动的时候,运行脚本. baseimage-docker的初始化脚本 `/sbin/my_init`,在启动的时候进程运行,按照下面的顺序: * 如果`/etc/my_init.d`存在,则按照字母顺序执行脚本. * 如果`/etc/rc.local`存在,则执行里面的脚本. 所有的脚本都是正确退出的,例如:退出的code是0.如果有任何脚本以非0的code退出,启动就会失败. 下面的例子向你展示了怎么添加一个启动脚本.这个脚本很简单的记录的一个系统启动时间,将启动时间记录到/tmp/boottime.txt. ### 在 logtime.sh (文件权限chmod +x): #!/bin/sh date > /tmp/boottime.txt ### 在 Dockerfile中: RUN mkdir -p /etc/my_init.d ADD logtime.sh /etc/my_init.d/logtime.sh ### 环境变量 如果你使用`/sbin/my_init`作为主容器命令,那么通过`docker run --env`或者在Dockerfile文件中设置的`ENV`环境变量,都会被`my_init`读取. * 在Unix系统中,环境变量都会被子进程给继承.这就意味着,子进程不可能修改环境变量或者修改其他进程的环境变量. * 由于上面提到的一点,这里没有一个可以为所有应用和服务集中定义环境的地方.Debian提供了一个`/etc/environment` 文件,解决一些问题. * 某些服务更改环境变量是为了给子进程使用.Nginx有这样的一个例子:它移除了所有的环境变量,除非你通过`env`进行了配置,明确了某些是保留的.如果你部署了任何应用在Nginx镜像(例如:使用[passenger-docker](https://github.com/phusion/passenger-docker)镜像或者使用Phusion Passenger作为你的镜像.),那么你通过Docker,你不会看到任何环境变量. `my_init`提供了一个办法来解决这些问题. #### 集中定义你的环境变量 在启动的时候,在执行[startup scripts](#running_startup_scripts),`my_init`会从`/etc/container_environment`导入环境变量.这个文件夹下面,包含的文件,文件被命名为环境变量的名字.文件内容就是环境变量的值.这个文件夹是因此是一个集中定义你的环境变量的好地方,它会继承到所有启动项目和Runit管理的服务中. 给个例子,在你的dockerfile如何定义一个环境变量: RUN echo Apachai Hopachai > /etc/container_environment/MY_NAME 你可以按照下面这样验证: $ docker run -t -i /sbin/my_init -- bash -l ... *** Running bash -l... # echo $MY_NAME Apachai Hopachai **换行处理** 如果你观察仔细一点,你会注意到'echo'命令,实际上在它是在新行打印出来的.为什么$MY_NAME没有包含在一行呢? 因为`my_init`在尾部有个换行字符.如果你打算让你的值包含一个新行,你需要增*另外*一个新字符,像这样: RUN echo -e "Apachai Hopachai\n" > /etc/container_environment/MY_NAME #### 环境变量存储 上面提到集中定义环境变量,它不会从子服务进程改变父服务进程或者重置环境变量.而且,`my_init`也会很容易的让你查询到原始的环境变量是什么. 在启动的时候,`/etc/container_environment`, `my_init`中的变量会存储起来,并且导入到环境变量中,例如一下的格式: * `/etc/container_environment` * `/etc/container_environment.sh`- 一个bash存储的环境变量格式.你可以从这个命令中得到base格式的文件. * `/etc/container_environment.json` - 一个json格式存储的环境变量格式. 多种格式可以让你不管采用什么语言/apps都可以很容易使用环境变量. 这里有个例子,展示怎么使用: $ docker run -t -i \ --env FOO=bar --env HELLO='my beautiful world' \ phusion/baseimage: /sbin/my_init -- \ bash -l ... *** Running bash -l... # ls /etc/container_environment FOO HELLO HOME HOSTNAME PATH TERM container # cat /etc/container_environment/HELLO; echo my beautiful world # cat /etc/container_environment.json; echo {"TERM": "xterm", "container": "lxc", "HOSTNAME": "f45449f06950", "HOME": "/root", "PATH": "/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin", "FOO": "bar", "HELLO": "my beautiful world"} # source /etc/container_environment.sh # echo $HELLO my beautiful world #### 修改环境变量 通过修改`/etc/container_environment`这个文件,很有可能修改了`my_init`中的环境变量.之后,每次`my_init`启动[启动脚本](#running_startup_scripts),就会重置掉我们自己`/etc/container_environment`中的环境变量,也就会导致`container_environment.sh`和`container_environment.json`重新存储. 但是记住这些: * 修改`container_environment.sh` 和 `container_environment.json`是没有效果的. * Runit 的服务是不能像这样修改环境变量的.`my_init`运行的时候,只对`/etc/container_environment`中的修改是生效的. #### 安全 因为环境变量可能包含敏感信息, `/etc/container_environment`和它的bash文件和JSON文件,默认都是root,都是可以被`docker_env`群组可以访问的(所以任何用户只要添加到群组中,都可以自动的获取这些信息). 如果你确定你的环境变量中没有什么敏感信息,那么你可以放松管理权限,将文件夹和文件分配下面的权限: RUN chmod 755 /etc/container_environment RUN chmod 644 /etc/container_environment.sh /etc/container_environment.json ### 解决Docker没有办法解决的/etc/hosts的问题 当前是没有办法在docker容器中修改`/etc/hosts`,这个是因为[Docker bug 2267](https://github.com/dotcloud/docker/issues/2267).Baseimage-docker包含了解决这个问题的办法,你必须明白是怎么修改的. 修改的办法包含在系统库中的` libnss_files.so.2`文件,这个文件使用`/etc/workaround-docker-2267/hosts`来代替系统使用`/etc/hosts`.如果需要修改`/etc/hosts`,你只要修改`/etc/workaround-docker-2267/hosts`就可以了. 增加这个修改到你的Dockerfile.下面的命令修改了文件`libnss_files.so.2`. RUN /usr/bin/workaround-docker-2267 (其实你不用在Dockerfile文件中运行这个命令,你可以在容器中运行一个shell就可以了.) 验证一下它是否生效了,[在你的容器中打开一个shell](#inspecting),修改`/etc/workaround-docker-2267/hosts`,检查一下是否生效了: bash# echo 127.0.0.1 my-test-domain.com >> /etc/workaround-docker-2267/hosts bash# ping my-test-domain.com ...should ping 127.0.0.1... **注意apt-get升级:** 如果Ubuntu升级,就有可能将`libnss_files.so.2`覆盖掉,那么修改就会失效.你必须重新运行`/usr/bin/workaround-docker-2267`.为了安全一点,你应该在运行`apt-get upgrade`之后,运行一下这个命令. ### 禁用SSH Baseimage-docker默认是支持SSH的,所以可以[使用SSH](#login_ssh)来[管理你的容器](#container_administration).万一你不想支持SSH,你可以只要禁用它: RUN rm -rf /etc/service/sshd /etc/my_init.d/00_regen_ssh_host_keys.sh ## 容器管理 一个优秀的docker想法,就是docker是一个无状态的,容易启动的容器,就想一个黑盒子.然而,你可能遇到某种情况,需要登录到容器,或者运行命令在容器中.或者为了开发,需要查看或者debug的目的.这章就给你讲解怎么管理容器. ### 在一个新容器中运行一个一闪而过的命令 _**备注:** 这章讲解怎么在一个-新-容器中运行命令.要在一个存在的容器中运行命令,请查看[在一个存在的容器中,运行一个命令](#run_inside_existing_container)._ 正常情况下,当你创建了一个新容器,为了在容器中运行一个单独的命令,而且在运行之后会立即退出的,你会这样调用docker命令: docker run YOUR_IMAGE COMMAND ARGUMENTS... 然而下面的方法初始化系统的进行是不会启动.它是这样的,当调用`COMMAND`的时候,重要的后台进程,例如定时任务和系统日志都是不运行的.同样,子进程也是不会出现的,因为`COMMAND`的pid是1. Baseimage-docker提供了一个灵活的方式运行只要一闪而过的命令,同时也解决了上述所说的问题.以一下的方式运行一条命令: docker run YOUR_IMAGE /sbin/my_init -- COMMAND ARGUMENTS ... 他们会按照下面的流程执行: * 运行所有的启动文件,例如 /etc/my_init.d/* and /etc/rc.local. * 运行所有的runit服务 * 运行指定的命令 * 运行指定的命令结束之后,结束所有runit服务. 例如: $ docker run phusion/baseimage: /sbin/my_init -- ls *** Running /etc/my_init.d/00_regen_ssh_host_keys.sh... No SSH host key available. Generating one... Creating SSH2 RSA key; this may take some time ... Creating SSH2 DSA key; this may take some time ... Creating SSH2 ECDSA key; this may take some time ... *** Running /etc/rc.local... *** Booting runit daemon... *** Runit started as PID 80 *** Running ls... bin boot dev etc home image lib lib64 media mnt opt proc root run sbin selinux srv sys tmp usr var *** ls exited with exit code 0. *** Shutting down runit daemon (PID 80)... *** Killing all processes... 你会发现默认的启动的流程太负责.或者你不希望执行启动文件.你可以自定义所有通过给`my_init`增加参数.调用`docker run YOUR_IMAGE /sbin/my_init --help`可以看到帮助信息. 例如上面运行`ls`命令,同时要求不运行启动脚本,减少信息打印,运行runit所有命令. $ docker run phusion/baseimage: /sbin/my_init --skip-startup-files --quiet -- ls bin boot dev etc home image lib lib64 media mnt opt proc root run sbin selinux srv sys tmp usr var ### 在一个已经运行的容器中,运行一条命令 这里有两种办法去在一个已经运行的容器中运行命令. * 通过`nseneter`工具.这个工具用于Linux内核调用在内嵌容器中运行命令.可以查看[通过`nsenter`,登录容器或者在容器内执行命令](#login_nsenter). * 通过SSH.这种办法需要在容器中运行ssh服务,而且需要你创建自己的sshkey.可以查看[通过`ssh`,登录容器或者在容器内执行命令](#login_ssh). 两种方法都是他们各自的优点和确定,你可以学习他们各自的章节来了他们. ### 通过`nsenter`,登录容器或者在容器内执行命令 你可以使用在docker主机上面的`nsenter`工具,来登录任何基于baseimage-docker的docker容器.你可以使用它在你的容器中运行命令. 这里有个和[通过`ssh`,登录容器或者在容器内执行命令](#login_ssh)的优缺点的比较: * 优点 * 不需要在容器中运行ssh服务. * 不需要ssh key. * 运行在任何容器上,甚至不是基于baseimage-docker的容器. * 缺点 * 通过`nsenter`运行的进程会和正常运行稍微有不同.例如,他们不同结束掉在容器中正常运行的进程.这适用于所有的子进程. * 如果`nsenter`进程被其他命令(如`kill`命令)给终止,然后由nsenter所执行的命令,是*不会*被结束的.你将不得不手动清理.(备注:终端控制命令像Ctrl-C *会* 清理所有的子进程,因为终端信号被发送到所有流程的终端会话) * 需要学习新工具. * 需要在docker主机上面提供root权限. * 需要在docker主机上面是可用的.在写这篇文字的时候(2014年7月),大多数linux发行版没有加载它.然而,baseimage-docker提供了预编译的二进制文件,允许你通过[docker-bash](#docker_bash)工具,来很容易的使用它. * 不可能没有登录到docker主机,就登录到docker容器中.(也就是说,你必须登录到docker主机,通过docker主机登录到容器.) #### 用例 第一,确定`nsenter`已经安装了.在写这篇文字的时候(2014年7月),大多数linux发行版没有加载它.然而,baseimage-docker提供了预编译的二进制文件,允许你通过[docker-bash](#docker_bash)工具,让任何人都可以使用. 接着,启动一个容器. docker run YOUR_IMAGE 找出你刚才运行容器的`ID`. docker ps 一旦拥有容器的id,找到运行容器的主要进程额`PID`. docker inspect -f "{{ .State.Pid }}" 现在你有的容器的主进程的PID,就可以使用`nsenter`来登录容器,或者在容器里面执行命令: # 登录容器 nsenter --target
--mount --uts --ipc --net --pid bash -l # 在容器中执行命令 nsenter --target
--mount --uts --ipc --net --pid -- echo hello world #### `docker-bash`工具 查找一个容器的主要进程的PID和输入这么长的nsenter命令很快会变得乏味无论.幸运的是,我们提供了一个`docker-bash` 工具,它可以自动完成只要的工具.这个工具是运行在*docker主机*上面,不是在docker容器中. 该工具还附带了一个预编译的二进制`nsenter`,这样你不需要自己安装`nsenter`了.`docker-bash`是很简单的使用的. 首先,在docker主机上安装这个工具: curl --fail -L -O https://github.com/phusion/baseimage-docker/archive/master.tar.gz && \ tar xzf master.tar.gz && \ sudo ./baseimage-docker-master/install-tools.sh 运行这个工具登录到容器中: docker-bash YOUR-CONTAINER-ID 你可以通过`docker ps`来查找你的容器ID. 默认,`docker-bash`会打开一个bash 回话.你可以告诉运行什么命令,之后就会自动退出: docker-bash YOUR-CONTAINER-ID echo hello world ### 通过`ssh`,登录容器或者在容器内执行命令 你可以使用ssh来登录任何基于baseimage-docker的容器.你可以使用它在容器中执行命令. 这里有个和[通过`nsenter`,登录容器或者在容器内执行命令](#login_nsenter)的优缺点的比较: * 优点 * 不像`nsenter`一样,运行在docker主机上面.几乎每个人都会安装一个ssh客户端. * 不想使用`nsenter`,运行的进程和正在的进程会不一样. * 不需要docker主机提供root权限. * 运行你让用户登录到容器,而不需要登录到docker主机.然而,默认这是不启用的,因为baseimage-docker默认不是开放ssh服务的. * 缺点 * 需要设置ssh key.然而,baseimage-docker会提供一中办法,会让key的生成会很容器.阅读更多信息. 第一件事情,就是你需要确定你在容器中已经安装设置了ssh key. 默认,没有任何安装key的,所有你无法登录.为了方便的原因,我们提供了一个[已经生成的key](https://github.com/phusion/baseimage-docker/blob/master/image/insecure_key) [(PuTTY format)](https://github.com/phusion/baseimage-docker/blob/master/image/insecure_key.ppk),为了让你使用方便.然后,请注意这个key仅仅是为方便.他没有任何安全行,因为它的key是在网络上提供的.**在生产环境,你必须使用你自己的key.** #### 在容器中使用key 你可以临时的使用key仅仅作为容器使用.这就以为这key是安装在容器上的.如果你使用`docker stop`和`docker start`控制容器,那么key是在容器中,但是如果你使用`docker run`开启一个新容器,那么这个容器是不包含key的. 启动新容器包含key`--enable-insecure-key`: docker run YOUR_IMAGE /sbin/my_init --enable-insecure-key 找出你的刚才运行的容器的ID: docker ps 一旦你拥有容器的ID,就能找到容器使用的IP地址: docker inspect -f "{{ .NetworkSettings.IPAddress }}" 现在你有得了IP地址,你就看通过SSH来登录容器,或者在容器中执行命令了: # 下载key curl -o insecure_key -fSL https://github.com/phusion/baseimage-docker/raw/master/image/insecure_key chmod 600 insecure_key # 登录容器 ssh -i insecure_key root@ # 在容器中执行命令 ssh -i insecure_key root@ echo hello world #### 支持一个长久的key 在一个长久存在的镜像中支持一个key是很可能的.一般是不推荐这么做,但是对于临时开始或者做demo演示,对安全要求不高,还是很合适的. 编辑你的dockerfile,来安装永久的key: RUN /usr/sbin/enable_insecure_key 在容器中怎么使用,同[在容器中使用key](#using_the_insecure_key_for_one_container_only)的章节说的一样. #### 使用你自己的key 编辑你的dockerfile,来安装ssh public key: ## 安装你自己的public key. ADD your_key.pub /tmp/your_key.pub RUN cat /tmp/your_key.pub >> /root/.ssh/authorized_keys && rm -f /tmp/your_key.pub 重新创建你的镜像.一旦你创建成功,启动基于这个镜像的容器. docker run your-image-name 找出你的刚才运行的容器的ID: docker ps 一旦你拥有容器的ID,就能找到容器使用的IP地址: docker inspect -f "{{ .NetworkSettings.IPAddress }}" 现在你有得了IP地址,你就看通过SSH来登录容器,或者在容器中执行命令了: # 登录容器 ssh -i /path-to/your_key root@ # 在容器中执行命令 ssh -i /path-to/your_key root@ echo hello world #### `docker-ssh`工具 找到容器的IP,运行ssh命令,很快会变得乏味无聊.幸运的是,我们提供了一个`docker-ssh`,可以自动完成这些事情.这个工具是运行在*Docker 主机*上的,不是安装在docker容器中的. 首先,在docker主机上面安装这个工具. curl --fail -L -O https://github.com/phusion/baseimage-docker/archive/master.tar.gz && \ tar xzf master.tar.gz && \ sudo ./baseimage-docker-master/install-tools.sh 使用这个工具通过ssh登录容器: docker-ssh YOUR-CONTAINER-ID 你可以使用`docker ps`找到`YOUR-CONTAINER-ID`. 默认,`docker-bash`会打开一个bash 回话.你可以告诉运行什么命令,之后就会自动退出: docker-ssh YOUR-CONTAINER-ID echo hello world ## 创建你自己的镜像 如果某些原因,你需要创建你自己的镜像,来替代从docker仓库下载镜像,可以按照的说明. 克隆仓库: git clone https://github.com/phusion/baseimage-docker.git cd baseimage-docker 创建一个包含docker在的虚拟机.你可以使用我们提供的Vagrantfile. vagrant up vagrant ssh cd /vagrant 编译镜像: make build 如果你想把创建的镜像名字,叫其他名字,通过`NAME`变量可以设置: make build NAME=joe/baseimage ## 总结 * Using baseimage-docker? [Tweet about us](https://twitter.com/share) or [follow us on Twitter](https://twitter.com/phusion_nl). * Having problems? Want to participate in development? Please post a message at [the discussion forum](https://groups.google.com/d/forum/passenger-docker). * Looking for a more complete base image, one that is ideal for Ruby, Python, Node.js and Meteor web apps? Take a look at [passenger-docker](https://github.com/phusion/passenger-docker). [](http://www.phusion.nl/) Please enjoy baseimage-docker, a product by [Phusion](http://www.phusion.nl/). :-)