❝ 如何在生产环境部署一个 Node 应用?[1]❞
一个合理并且高效的部署方案,不仅能够实现快速升级,平滑切换,负载均衡,应用隔离等部署特性,而且配有一套成熟稳定的监控。
kubernetes 把 Node 应用视作一个服务端应用的黑盒子,完美匹配了以上条件,越来越多的团队把 node 部署在 k8s 上。
但在此之前,需要先把 Node 应用跑在一个 Docker 容器上,这也是本章的主题。
一个简单的 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]: 一个最简化的 serverless 与 dockerize 示例。
NODE_ENV=production
在生产环境中,无需安装 devDependecies 中依赖,NODE_ENV 环境变量设置为 production 时将会跳过 devDep。
- # 通过设置环境变量,只安装生产环境依赖
- $ NODE_ENV=production npm ci
- # 通过显式指定 flag,只安装生产环境依赖
- $ npm ci --production
另一方面,某些第三方模块会根据 NODE_ENV 环境变量做出一些意料不到的配置。因此在生产环境注意该环境变量的配置。
一个 Node 应用的简单部署
一个典型的、面向服务端的 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 进行编译,而它依赖于 python,make 与 g++。
- $ apk --no-cache add python make g++
在带有编译过程的镜像构建中,源文件与构建工具都会造成空间的浪费。借助镜像的「多阶段构建」可以高效利用空间。Go App 与 FE App 的构建也遵循此规则。
在构建 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
相关文章
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
前言: 在日常使用数据库的过程中,难免会遇到需要修改账号密码的情景,比如密码...
% If Err.Number 0 Then ' 错误处理 . Response.Clear ' 清除缓冲区 . Select Ca...
0、补充 1、Node.js 创建的第一个应用 1、引入 http 模块 var http require ( ht...
网上关于这方面的文章有很多,重复的东西本文不再赘述,仅提供思路,并解释一些...
痴月熊学Python 文章目录 痴月熊学Python 往期文章 前言 一、算术运算 二、比较...
在校招面试中排序算法是经常被问到的。排序算法又比较多很容易遗忘和混淆。建议 ...
1. 什么是库 1.1 库: 本质上来说库是一种可执行的二进制代码 (但不可以独立执行)...
vue项目中openlayers画行政区划(区域范围),供大家参考,具体内容如下 原理 在...
文章目录 一jdbc查询代码 二JDBC存在的问题 三解决方案 一jdbc查询代码 下面这段...
% '****************************** '函数:FormatFileSize(l1) '参数:l1,字节...