专栏首页服务器运维笔记如何使用 docker 高效部署 Node 应用

如何使用 docker 高效部署 Node 应用

❝如何在生产环境部署一个 Node 应用?[1] ❞

一个合理并且高效的部署方案,不仅能够实现快速升级,平滑切换,负载均衡,应用隔离等部署特性,而且配有一套成熟稳定的监控。

kubernetes 把 Node 应用视作一个服务端应用的黑盒子,完美匹配了以上条件,越来越多的团队把 node 部署在 k8s 上。

但在此之前,需要先把 Node 应用跑在一个 Docker 容器上,这也是本章的主题。

❝关于前端在 docker 上部署,山月曾写了两篇文章:

  1. 如何在 docker 中部署前端[2]
  2. 前端部署 Prview 与 Production[3]
  3. 前端部署的发展过程[4]

一个简单的 Node 应用

「index.js」

一个 hello, world 版的 Node Web App

const http = require('http')

const app = async (req, res) => {
  res.end('hello, world')
}

http.createServer(app).listen(3000, () => console.log(3000))

「package.json」

配置 npm start 来启动应用

"scripts": {
  "start": "node index.js"
},

但这仅仅是最简单的 Node 应用,真实环境中还有各种数据存储及定时任务调度等,暂撇开不谈,这已经足够了。

再稍微复杂一点点的 Node 应用可以查看山月的项目 whoami[5]: 一个最简化的 serverlessdockerize 示例。

NODE_ENV=production

在生产环境中,无需安装 devDependecies 中依赖,NODE_ENV 环境变量设置为 production 时将会跳过 devDep

# 通过设置环境变量,只安装生产环境依赖
$ NODE_ENV=production npm ci

# 通过显式指定 flag,只安装生产环境依赖
$ npm ci --production

另一方面,某些第三方模块会根据 NODE_ENV 环境变量做出一些意料不到的配置。因此在生产环境注意该环境变量的配置。

一个 Node 应用的简单部署

一个典型的、面向服务端的 Node 应用是这么跑起来的:

  1. npm install
  2. npm run config,从配置服务(consul/vault)拉取配置 ,如数据库与缓存的账号密码,此时构建服务器需要配置服务权限
  3. npm run migrate,数据库迁移脚本,执行数据库表列行更改操作,此时构建服务器需要数据库访问权限
  4. npm start,启动一个 Node 服务

把运行步骤翻译为 Dockerfile:

# 选择一个体积小的镜像 (~5MB)
FROM node:12-alpine

# 环境变量设置为生产环境
ENV NODE_ENV production

WORKDIR /code

# 更好的根据 Image Layer 利用缓存
ADD package.json package-lock.json /code
RUN npm ci

ADD . /code

# 配置服务及数据库迁移
RUN npm run config --if-present && npm run migrate --if-present

EXPOSE 3000
CMD npm start

这对于大部分 Node 应用已经是足够了,如果精益求精,可以再走接下来的多阶段构建

node-gyp 与 Native Addon

在 Node 中有可能存在着一些 Native Addon,它们通过 node-gyp 进行编译,而它依赖于 pythonmakeg++

$ apk --no-cache add python make g++

在带有编译过程的镜像构建中,源文件与构建工具都会造成空间的浪费。借助镜像的「多阶段构建」可以高效利用空间。Go AppFE App 的构建也遵循此规则。

  • 多阶段构建 Go 应用[6]
  • 多阶段构建前端应用[7]

在构建 Node 应用镜像时,第一层镜像用以构造 node_modules

# 选择一个体积小的镜像 (~5MB)
FROM node:12-alpine as builder

# 环境变量设置为生产环境
ENV NODE_ENV production

# 更好的根据 Image Layer 利用缓存
ADD package.json package-lock.json ./
RUN npm ci

# 多阶段构建之第二阶段
# 多阶段构建之第二阶段
# 多阶段构建之第二阶段
FROM node:12-alpine

WORKDIR /code
ENV NODE_ENV production

ADD . .
COPY --from=builder node_modules node_modules
# 配置服务及数据库迁移
RUN npm run config --if-present && npm run migrate --if-present

EXPOSE 3000
CMD npm start

相关文章

  1. N-API and getting started with writing C addons for Node.js[8]
  2. Using Docker for Node.js in Development and Production[9]

Reference

[1]

如何在生产环境部署一个 Node 应用?: https://github.com/shfshanyue/Daily-Question/issues/420

[2]

如何在 docker 中部署前端: https://shanyue.tech/frontend-engineering/docker.html

[3]

前端部署 Prview 与 Production: https://shanyue.tech/frontend-engineering/feature-deploy.html

[4]

前端部署的发展过程: https://shanyue.tech/frontend-engineering/deploy.html

[5]

whoami: https://github.com/shfshanyue/whoami

[6]

多阶段构建 Go 应用: https://docs.docker.com/develop/develop-images/multistage-build/#use-multi-stage-builds

[7]

多阶段构建前端应用: https://shanyue.tech/frontend-engineering/docker.html#%E5%A4%9A%E9%98%B6%E6%AE%B5%E6%9E%84%E5%BB%BA

[8]

N-API and getting started with writing C addons for Node.js: https://hackernoon.com/n-api-and-getting-started-with-writing-c-addons-for-node-js-cf061b3eae75

[9]

Using Docker for Node.js in Development and Production: https://dev.to/alex_barashkov/using-docker-for-nodejs-in-development-and-production-3cgp

本文分享自微信公众号 - 全栈成长之路(shanyue-road),作者:山月行

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2020-08-25

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 从 LRU Cache 带你看面试的本质

    在人人都知道刷题的今天,面试官也都知道大家会刷题准备面试,代码大家都会写,那面试为什么还在考这些题?那为什么有些人代码写出来了还挂了?

    山月
  • 捉虫记:Unexpected end of JSON input while parsing

    npm 的源码在 https://github.com/npm/cli,找到以下 Issue

    山月
  • 安利几个JS开发小技巧

    除了常规的布尔值true和false之外,JavaScript还将所有其他值视为 ‘truthy’ 或**‘falsy’**。

    山月
  • Ubuntu 16.04下安装nodeJs和npm

    最近折腾了一下使用网上使用人数较多的Ubuntu 16.04搭建了一个基于Linux的开发环境,并准备以后的学习就在Linux平台上了。那么下面讲述一下Linu...

    w候人兮猗
  • .NET Core vs Node.js:你应该选择哪个?

    本文最初发布于 inveritasoft.com 网站,经网站授权由 InfoQ 中文站翻译并分享。

    深度学习与Python
  • Codeforces Round #516 Div.2 D. Labyrinth(双端队列)

    题目连接:http://codeforces.com/contest/1064/problem/D

    Ch_Zaqdt
  • JS 和 Node.js 中的“事件驱动”是什么意思?[每日前端夜话0x106]

    事件驱动架构是建立在软件开发中一种通用模式上的,这种模式被称为发布-订阅或观察者模式。

    疯狂的技术宅
  • 2019前端工程师自检清单与思考

    对于JavaScript,掌握其语法和特性是最基本的,但是这些只是应用能力,最终仍旧考量仍然是计算机体系的理论知识,所以数据结构,算法,软件工程,设计模式等基础...

    super.x
  • 2019 前端工程师自检清单与思考

    对于 JavaScript,掌握其语法和特性是最基本的,但是这些只是应用能力,最终仍旧考量仍然是计算机体系的理论知识,所以数据结构,算法,软件工程,设计模式等基...

    桃翁
  • 如何在2016年成为一个更好的Node.js开发者

    本文主要讨论一些进行Node.js开发的最佳实践和建议,这些建议不仅仅适合开发者,还适合那些管理与维护Node.js基础架构的工作人员。遵循本文提供的这些建议,...

    春哥大魔王

扫码关注云+社区

领取腾讯云代金券