如何使用 docker 高效部署 Node 應(yīng)用
?如何在生產(chǎn)環(huán)境部署一個(gè) Node 應(yīng)用?[1]
?
一個(gè)合理并且高效的部署方案,不僅能夠?qū)崿F(xiàn)快速升級(jí),平滑切換,負(fù)載均衡,應(yīng)用隔離等部署特性,而且配有一套成熟穩(wěn)定的監(jiān)控。
kubernetes 把 Node 應(yīng)用視作一個(gè)服務(wù)端應(yīng)用的黑盒子,完美匹配了以上條件,越來(lái)越多的團(tuán)隊(duì)把 node 部署在 k8s 上。
但在此之前,需要先把 Node 應(yīng)用跑在一個(gè) Docker 容器上,這也是本章的主題。
?關(guān)于前端在 docker 上部署,山月曾寫(xiě)了兩篇文章:
?
如何在 docker 中部署前端[2] 前端部署 Prview 與 Production[3] 前端部署的發(fā)展過(guò)程[4]
一個(gè)簡(jiǎn)單的 Node 應(yīng)用
「index.js」
一個(gè) 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 來(lái)啟動(dòng)應(yīng)用
"scripts":?{
??"start":?"node?index.js"
},
但這僅僅是最簡(jiǎn)單的 Node 應(yīng)用,真實(shí)環(huán)境中還有各種數(shù)據(jù)存儲(chǔ)及定時(shí)任務(wù)調(diào)度等,暫撇開(kāi)不談,這已經(jīng)足夠了。
再稍微復(fù)雜一點(diǎn)點(diǎn)的 Node 應(yīng)用可以查看山月的項(xiàng)目 whoami[5]: 一個(gè)最簡(jiǎn)化的 serverless 與 dockerize 示例。
NODE_ENV=production
在生產(chǎn)環(huán)境中,無(wú)需安裝 devDependecies 中依賴,NODE_ENV 環(huán)境變量設(shè)置為 production 時(shí)將會(huì)跳過(guò) devDep。
#?通過(guò)設(shè)置環(huán)境變量,只安裝生產(chǎn)環(huán)境依賴
$?NODE_ENV=production?npm?ci
#?通過(guò)顯式指定?flag,只安裝生產(chǎn)環(huán)境依賴
$?npm?ci?--production
另一方面,某些第三方模塊會(huì)根據(jù) NODE_ENV 環(huán)境變量做出一些意料不到的配置。因此在生產(chǎn)環(huán)境注意該環(huán)境變量的配置。
一個(gè) Node 應(yīng)用的簡(jiǎn)單部署
一個(gè)典型的、面向服務(wù)端的 Node 應(yīng)用是這么跑起來(lái)的:
npm installnpm run config,從配置服務(wù)(consul/vault)拉取配置 ,如數(shù)據(jù)庫(kù)與緩存的賬號(hào)密碼,此時(shí)構(gòu)建服務(wù)器需要配置服務(wù)權(quán)限npm run migrate,數(shù)據(jù)庫(kù)遷移腳本,執(zhí)行數(shù)據(jù)庫(kù)表列行更改操作,此時(shí)構(gòu)建服務(wù)器需要數(shù)據(jù)庫(kù)訪問(wèn)權(quán)限npm start,啟動(dòng)一個(gè) Node 服務(wù)
把運(yùn)行步驟翻譯為 Dockerfile:
#?選擇一個(gè)體積小的鏡像?(~5MB)
FROM?node:12-alpine
#?環(huán)境變量設(shè)置為生產(chǎn)環(huán)境
ENV?NODE_ENV?production
WORKDIR?/code
#?更好的根據(jù)?Image?Layer?利用緩存
ADD?package.json?package-lock.json?/code
RUN?npm?ci
ADD?.?/code
#?配置服務(wù)及數(shù)據(jù)庫(kù)遷移
RUN?npm?run?config?--if-present?&&?npm?run?migrate?--if-present
EXPOSE?3000
CMD?npm?start
這對(duì)于大部分 Node 應(yīng)用已經(jīng)是足夠了,如果精益求精,可以再走接下來(lái)的多階段構(gòu)建
node-gyp 與 Native Addon
在 Node 中有可能存在著一些 Native Addon,它們通過(guò) node-gyp 進(jìn)行編譯,而它依賴于 python,make 與 g++。
$?apk?--no-cache?add?python?make?g++
在帶有編譯過(guò)程的鏡像構(gòu)建中,源文件與構(gòu)建工具都會(huì)造成空間的浪費(fèi)。借助鏡像的「多階段構(gòu)建」可以高效利用空間。Go App 與 FE App 的構(gòu)建也遵循此規(guī)則。
多階段構(gòu)建 Go 應(yīng)用[6] 多階段構(gòu)建前端應(yīng)用[7]
在構(gòu)建 Node 應(yīng)用鏡像時(shí),第一層鏡像用以構(gòu)造 node_modules。
#?選擇一個(gè)體積小的鏡像?(~5MB)
FROM?node:12-alpine?as?builder
#?環(huán)境變量設(shè)置為生產(chǎn)環(huán)境
ENV?NODE_ENV?production
#?更好的根據(jù)?Image?Layer?利用緩存
ADD?package.json?package-lock.json?./
RUN?npm?ci
#?多階段構(gòu)建之第二階段
#?多階段構(gòu)建之第二階段
#?多階段構(gòu)建之第二階段
FROM?node:12-alpine
WORKDIR?/code
ENV?NODE_ENV?production
ADD?.?.
COPY?--from=builder?node_modules?node_modules
#?配置服務(wù)及數(shù)據(jù)庫(kù)遷移
RUN?npm?run?config?--if-present?&&?npm?run?migrate?--if-present
EXPOSE?3000
CMD?npm?start
相關(guān)文章
N-API and getting started with writing C addons for Node.js[8] Using Docker for Node.js in Development and Production[9]
