docker中怎样设置开机启动--随容器的启动而启动服务?

谁说我不能喝 提交于 2020-08-18 05:33:26

  docker可以说给我们的部署带来极大的方便和可逢凶化吉性!(懂的同学自然懂)

  在初步了解之后,我们就能简单使用docker了。

  刚开始玩docker时,可以基于系统级别的镜像做定制,比如基于  centos 镜像使用docker;

docker pull centos:7        # 把镜像拉下来
docker run -it centos:7        # 创建一个容器即可运行

  你可以在拉下来的容器里安装任何需要的应用(必要的);然后,一切都看起来很美好!

  但是,这时,我们最担心一个问题,那就是docker挂了怎么办?机器重启后怎么办?

  是的,这个担心是有道理的,如果你没有去查细细看过官方文档,那么多半你是吃过这亏,才会清楚其中的坑!

所以,我们来提几个问题? 简单而不简约!

1. 如何备份当前容器的修改?备份是可靠性的一个保证!

  1. 使用commit 命令保存到本地

docker commit -m "msg" abcommitid myimage:1.0

  其好处是操作方便,本机永久保存,方便下次快速操作。坏处,也不算,就是你如果想放到远程,你必须注册一个docker hub 账号,然后保存过去。而且这样的镜像,一般只适合自己使用,不适合团队传播!(不过也不是绝对的)

  2. 使用docker save 保存压缩包到本地

docker save -o myimage_save.tar abcontainerid

  这样备份好之后,就可以将改该容器到处分发到其他机器或者做备份了。这样做都好处是,传播方便,不会导致隐私泄露。坏处是都是本地包,没有存储在云端,必须到处携带该包。

  导入时使用 docker load 即可!

docker load -i myimage_save.tar

  然后就可以看到镜像了,使用docker run 运行。操作步骤稍微繁琐了点。注意: 只有 docker run 才能自定义各种参数哦。(docker run 聚合了 create 和 start 的功能)

  在部署应用时可能存在需要将新应用部署好后,再删除原容器的操作,所以容器重命名就很有必要的, 使用 docker rename

docker rename 原容器名  新容器名

  3. 使用 docker export 保存到本地, 操作如同 docker save, 但是功能受限,个人不是很喜欢使用

docker export -o mycontainer_save.tar tmp_container

2. 如何设置应用开机启动?只要启动容器就够了!

  应用场景可以说是刚需: 我把第一个镜像安装好后,希望下次创建容器之后就能自动运行,而不是还要操作N多繁杂步骤,从而降低移植性带来的方便性!

  1. 能想得最简单的是,使用系统的开机启动功能

    比如在 /etc/rc.d/rc.local 中添加相应的启动,但是这里有个前提,那就是你必须先把容器启起来; (启不启得来另说)

  2. 使用docker的开机自动运行功能,使用简单;

    这里说的简单是指使用,而在操作的时候则看个人了。在docker创建时,使用启动脚本,自动运行。主要的命令有: CMD, RUN, ADD . 创建一个start.sh,可以定复杂的启动逻辑;但是这在后期中,就很难更改其逻辑了哦!

CMD ["sh", "-c", "service httpd start;bash"]        # 开机启动 apache 服务

3. 如何自定义端口映射?容器需要与外部通信!

  Docker 自带了端口映射功能,使用docker run -p 进行操作!

docker run -d -p 81:80 --name container_name myimage:1.0

  多个端口映射使用多个-p即可;

4. 如何自定义hostname?订制你的机器名而不是随机数!

  自定义host也很有用,比如我想看我当前气息环境,hostname就很有用,还有hostname的固定可以不致让自己迷失; 只需要使用-h参数就可以了。

docker run -d -h myapi1 myimage:1.0

5. 如何插入host的解析?容器内订制自己的 hosts !

  这种应用场景是,比如你其他应用的服务,为了防止ip经常变更导致的麻烦性,这种服务一般是以内网域名形式出现,所以需要加入域名解析。

  方法一是,你给每个容器定义一个通用的域名解析器dns; 

  方法二是,为各自的hosts里加入解析。而这在docker中,则操作是不会被保存的,每个新容器老是新 hosts. 可以通过 --add-host 添加自定义hosts解析:

docker run --add-host a.com:1.2.3.4 myimage:1.0

  这种方式仅用于学习,其中更有用的是加入一个dns; 其操作步骤就是安装个bind 软件,然后配制 named.conf 即可。使用时,vim /etc/resolv.conf

nameserver 100.1.1.1

6. 如何自定义自己都的变量以满足容器内动态修改的需求?特殊场景定制化!

  其实如上的配置基本能满足大部分情况下的生产需求了。但是难免还是个性的,比如我想定义nginx访问某个的另外端口,这时使用dns 就不好搞了。在不改变外部环境的情况下,我们只能自定义修改了。最直接的方式是在 nginx 中直接改掉即可。但是这样做还有个,如果我想让同一个容器灵活地指向任意端口怎么办?那就只能自定义变量操作了,在创建容器的时候指定该端口即可。

  这样的自定义变量可以用于设置 nginx 的自定义端口, java 堆大小,日志目录设置等等.

  具体做法步骤如下:

# 1. docker run -it myimage:1.0, 修改 nginx 配置文件,使其可以方便被替换,如下
# vim /usr/local/nginx/conf/conf.d/www.conf
server {
    listen       80;
    server_name  localhost;
    access_log  /var/log/nginx/host.access.log  main;
    location / {
        root   /www/webapp/html/app1;
        try_files $uri $uri/ /index.html;
        index  index.html index.htm;
    }
    location /api {
        # 设置点位符使外部可替换
        set $API_HOST 192.168.1.1;
        set $API_PORT 8083;
        proxy_pass   http://$API_HOST:$API_PORT;
        index  index.html index.htm;
    }
}
# 2. 自己尝试启动无误后,将新变更提交到原镜像, docker commit
    docker commit -m 'conf update' abcommitid myimage:1.0
# 3. vim Dockerfile, 设置启动脚本
    FROM myimage:1.0
    MAINTAINER xxx <abc@abc.com>
    ENV NGINX_CONFD_PATH /usr/local/nginx/conf/conf.d
    ENV PATH /usr/local/nginx/sbin:$PATH
    ADD ./start.sh /usr/local/bin/start.sh
    RUN chmod +x /usr/local/bin/start.sh
    EXPOSE 80
    # 设置启动脚本,脚本来源本机,可随时修改
    CMD [ "/usr/local/bin/start.sh" ]
# 4. 缩写启动脚本,使在启动时执行动态配置变更 awk
# vim start.sh
    #!/bin/bash
    # override port variable if set
    if [ ! -z "$API_PORT" ]; then
        echo "------- api port replacing to $API_PORT -----------";
        awk -v PORT="set \$API_PORT $API_PORT;" '{ sub(/set.*\$API_PORT.*/, PORT); print; }' ${NGINX_CONFD_PATH}/default.conf \
            > ${NGINX_CONFD_PATH}/default.conf.new && mv ${NGINX_CONFD_PATH}/default.conf.new ${NGINX_CONFD_PATH}/default.conf
    fi
    if [ ! -z "$API_HOST" ]; then
        echo "------- api host replacing to $API_HOST -----------";
        awk -v HOST="set \$API_HOST $API_HOST;" '{ sub(/set.*\$API_HOST.*/, HOST); print; }' ${NGINX_CONFD_PATH}/default.conf \
            > ${NGINX_CONFD_PATH}/default.conf.new && mv ${NGINX_CONFD_PATH}/default.conf.new ${NGINX_CONFD_PATH}/default.conf
    fi

    # start nginx
    /bin/sh -c 'nginx -g "daemon off;" '
# 5. 都操作好后,重新构建一个新的镜像,使用就可以了
    docker build -t myimage:1.1 .
# 6. 使用时,用 --env 来指定自定义变量
    docker run --env API_HOST=192.168.1.112 --env API_PORT=8090 -it myimage:1.1
# 此时,进入查看时,nginx已经运行在不同的api端口下了

  注意: 本处使用的是比较原始的 docker 版本,如果使用到 docker-compose 等高级工具,可能就不需要这么麻烦了!

7. 如何将宿主机目录与容器内目录进行交换?可视化你的容器内容!

  这样的场景是比较多的: 

    比如为了统一管理安装包,不让所有安装包散乱在各个容器的各个目录;
    比如为了让容器的数据存储使用一块新买的磁盘;
    比如我想复制同一份代码到新容器使用,从而方便后续独立修改;
    比如你无法进入你的容器,却想拿到其中的数据等等;


  所以,我们需要使用到目录映射功能,这是 docker 自带的功能,方便实用: -v 参数设置即可:

docker run -d -v /opt/docker/webapps:/www/webapp myimage:1.0

  

  经过上面这些问答,相信你已经能简单应付docker常用运维场景了,尽情享受docker带来的可移植性方便以及其隔离性吧。

 

标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!