使用Docker数据卷容器发布前端应用

自从使用Docker作为部署工具后,给开发人员提供了很大的便利。
近日在项目中前端应用的部署也使用了Docker,现将方法记录如下。

最初的想法:使用安装有Nginx的容器存储静态页面

  将前端页面直接build到安装有Nginx的容器中,这种应该是比较直观的想法。项目使用webpack构建,需要npm环境,因此使用node的镜像较为方便。

# 镜像选择node:8-slim
FROM node:8-slim
# 更新源并安装nginx
RUN apt-get update && apt-get install -y nginx
# 使用cnpm加快npm包的安装速度
RUN npm install -g cnpm --registry=https://registry.npm.taobao.org
WORKDIR /app
# 单独提取加载package.json并安装npm包,与代码构建分离,避免因为代码变更导致镜像构建需要重新安装npm包
ADD package.json /app/package.json
RUN cnpm install
COPY . /app/
# 构建,生成最终部署文件,/var/www/html是构建结果存放的目录
RUN npm run build  && cp -r dist/* /var/www/html && rm -rf /app
# 复制不同环境的nginx配置文件到镜像的nginx配置目录
COPY ./nginx.ci.conf /etc/nginx/
COPY ./nginx.lt.conf /etc/nginx/
COPY ./nginx.ot.conf /etc/nginx/
# 不在命令中使用 -g daemon off,可以在配置文件中配置
ENTRYPOINT ["nginx"]

其中一个环境的配置文件大概就是下面这个样子:

user www-data;
worker_processes 4;
pid /run/nginx.pid;
# Importent
daemon off;

events {
    worker_connections 2048;
    use epoll;
}

http {
    ##
    # Basic Settings
    ##
    sendfile on;
    tcp_nopush on;
    tcp_nodelay on;
    keepalive_timeout 65;
    types_hash_max_size 2048;
    # server_tokens off;
    # server_names_hash_bucket_size 64;
    # server_name_in_redirect off;

    include /etc/nginx/mime.types;
    default_type application/octet-stream;

    ##
    # SSL Settings
    ##
    ssl_protocols TLSv1 TLSv1.1 TLSv1.2; # Dropping SSLv3, ref: POODLE
    ssl_prefer_server_ciphers on;

    ##
    # Logging Settings
    ##
    log_format main      '$remote_addr - $remote_user [$time_local] '
                         '"$request" $status $bytes_sent '
                         '"$http_referer" "$http_user_agent" '
                         '"$gzip_ratio"';

    access_log /dev/stdout main;
    error_log /dev/stderr;

    ##
    # Gzip Settings
    ##
    gzip on;
    gzip_disable "msie6";
    gzip_vary on;
    gzip_proxied any;
    gzip_comp_level 6;
    gzip_buffers 16 8k;
    gzip_http_version 1.1;
    gzip_types text/plain text/css application/json application/javascript text/xml application/xml application/xml+rss text/javascript;

    ##
    # Virtual Host Configs
    ##
    server {
        listen       9001;
        server_name  127.0.0.1;
        location / {
            root   /var/www/html;
            index  index.html index.htm;
        }
    }

    include /etc/nginx/conf.d/*.conf;
    include /etc/nginx/sites-enabled/*;
}

以下是容器启动的命令,使用 -c 选项指定生效的nginx配置文件:

docker run -d --name web -p 9001:9001 xxxx/xxxx/xxxx:latest -c /etc/nginx/nginx.ci.conf

换一种思路:使用数据卷的方式发布前端页面

  直观的想法往往不是较好的解法,将前端页面放置在nginx的容器中一同发布,我觉得不符合关注点分离。
  发布的时候,不仅需要部署前端页面,还需要对nginx进行配置。如果这二者发布在一个容器中,当前容器的nginx只负责本容器的页面服务,而web server理应还有其他用途,比如反向代理后端服务、做负载均衡,势必在更上一层还需要启动一个web server,部署容器自带的nginx其实是一种资源浪费。
  所以,使用数据卷容器发布前端页面,用外部的nginx来提供服务,应该是一个好的选择。如果有更好的方法,不吝赐教。

Dockerfile改成了构建数据卷容器:

FROM node:8-slim
RUN npm install -g cnpm --registry=https://registry.npm.taobao.org
WORKDIR /app
ADD package.json /app/package.json
RUN cnpm install
COPY . /app/
RUN npm run build && mkdir /var/www/html -p && cp -r dist/* /var/www/html && rm -rf /app
VOLUME [ "/var/www/html" ]

包含有前端页面的数据卷容器启动:

docker run --name web -d xxxx/xxxx/xxxx:latest

启动nginx容器,加载数据卷:

docker run --name nginx -d --volumes-from web -v /etc/nginx/nginx.conf:/etc/nginx/nginx.conf -p 9001:9001 nginx:stable-alpine

附一小段nginx.conf的配置:

    ##
    # Virtual Host Configs
    ##
    server {
        listen       9001;
        server_name  127.0.0.1;
        location / {
            root   /var/www/html;
            index  index.html index.htm;
        }
    }

页面访问也没有问题。

你可能感兴趣的:(使用Docker数据卷容器发布前端应用)