容器将应用与环境打包整合,解决了应用外部依赖的痛点,打包后通过窗口可方便地部署到任意环境,用过就知道很香。

创建示例应用

NestJS 为例,先创建一个示例应用。

$ npm i -g @nestjs/cli
$ nest new my-app
$ cd my-app
$ yarn && yarn start
 

然后 app.controller.ts 中添加如下 action:

  @Get('ping')
async ping() {
return 'pong';
}
 

测试一把会得到如下返回,证明我们的 app 一切正常:

$ curl localhost:3000/ping
pong
 

Docker 介绍

先了解 Docker 的两个核心概念:

  • 镜像/image: 本质上是一个文件,里面包含创建容器的指令,可通过 docker images 查看已有的镜像。
  • 容器/container: 通过镜像创建出来运行中的实例即容器,可通过 docker ps 命令查看运行中的容器。

Docker 安装

$ brew install --cask docker
 

如果已经安装过,升级可使用如下命令:

$ brew install --cask docker
 

然后在程序目录或 Spotlight 中找到并启动 Docker,系统状态栏中会有个鲨鱼图标。

启动后命令行工具已经可用,检查安装:

$ docker —version
Docker version 20.10.6, build 370c289
 

使用

通过 docker help 查看帮助。

$ docker help
 

查看具体命令的帮助可在 help 后加上该命令:

$ docker help run
 

打包生成镜像

Docker 中打包后的应用存在于镜像中,其中便包含了应用及依赖的环境。将这个镜像文件进行分发就可以在其他地方加载运行,实现了在新环境中方便部署,无须再关心外部依赖。

创建 Dockerfile

使用 Docker 打包应用需先创建 Dockerfile,其中包含指导 Docker 如何打包的指令。

$ touch Dockerfile
 

一般我们会基于已有镜像来创建自己的镜像,比如这里打包 Node 应用,我们会使用一个已经包含 Node 环境的镜像作为源。通过如下 FROM 语句完成:

FROM node:14
 

创建应用所在的目录:

# Create app directory
WORKDIR /usr/src/app
 

将文件复制到目标路径,然后进行 npm 包依赖的安装:

# Install app dependencies
# A wildcard is used to ensure both package.json AND package-lock.json are copied
# where available (npm@5+)
COPY package*.json ./ RUN npm install
# If you are building your code for production
# RUN npm ci --only=production
 

复制应用中的源码文件:

# Bundle app source
COPY . .
 

依赖和源码都好后,可以编译 Nest 应用,生成 dist 目录了:

npm run build
 

可以把镜像看作一个封闭环境,外界要与其中的应用进行交互,比如这里打包的是 Nest 服务,要能正常访问 Nest 中我们编写的 HTTP 接口,就需要 image 向外暴露端口。

因为默认 Nest 应用起的 3000 端口,这里就将其暴露,

EXPOSE 3000
 

最后一条指令,指导 Docker 启动 Nest 应用:

CMD [ "node", "dist/main" ]
 

所以完整的 Dockerfile 目前长这样了:

FROM node:14

# Create app directory
WORKDIR /usr/src/app # Install app dependencies
# A wildcard is used to ensure both package.json AND package-lock.json are copied
# where available (npm@5+)
COPY package*.json ./ RUN npm install
# If you are building your code for production
# RUN npm ci --only=production
npm run build # Bundle app source
COPY . . EXPOSE 8080
CMD [ "node", "dist/main" ]
 

.dockerignore 文件

可创建 .dockerignore 文件,将日志,本地无用文件排除在复制的文件列表之外。

node_modules
npm-debug.log
 

生成镜像

通过如下命令根据前面创建的 Dockerfile 生成镜像:

$ docker build . -t wayou/my-app
 

其中 -t 指定镜像名称,一般为 <username>/<image_name> 形式,其中 username 与你在 Docker Hub 中的用户名一致。前面提到镜像可进行分发,当然也能分享,同时我们的 Dockerfile 也是基于名为 node:14 的镜像进行创建的,Docker Hub 则是官方一个分享 image 的平台。

生成镜像过程中如果出现如下错误:

Error response from daemon: dial unix docker.raw.sock: connect: connection refused
 

重启一下 Docker 服务即可。

查看镜像

正常的话,可通过如下命令查看到刚刚生成的镜像:

$ docker images
REPOSITORY TAG IMAGE ID CREATED SIZE
wayou/my-app latest 6ba2f1f74d8b 7 hours ago 1.44GB
 

运行镜像

通过如下命令运行镜像:

$ docker run -p 8000:3000 -d wayou/my-app
 

其中 -p 部分前面为外部环境使用的端口,而 3000 为容器对外 暴露的端口。实际使用时则是使用外部这个 8000。

$ curl localhost:8000/ping
pong
 

通过 docker ps 查看运行中的实例。

镜像启动失败的排查

这里展示下如下 Debug 找出镜像启动失败的原因,即没有生成运行中的容器。

前面启动应用的指令是 CMD [ "node", "dist/main" ],而 dist 目录是通过 npm run build 而来,假如我们的 Dockerfile 中没有 build 这个步骤,很明显就没有 dist 目录所以会导致应用启动失败。

启动失败的话,docker ps 输出为空。

此时可加上 -a 参数,它会列出所有容器,包含停止的实例,以查看其状态。

$ docker ps -a
 

如果看到 STATUSExited,原因就是启动失败了。此时需要 Debug 一下看看启动失败的具体原因。

重新启动,并指定名称,方便后面查看日志:

$ docker run -p 8000:3000 -d --name test wayou/my-app
 

现在查看时可能看一个指定名称为 test 的容器:

$ docker ps -a
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
a9d187c0d665 wayou/my-app "docker-entrypoint.s…" 5 seconds ago Exited (1) 3 seconds ago test
 

然后通过 docker logs 查看其日志:

$ docker logs -t test
2021-05-21T09:58:56.706680291Z internal/modules/cjs/loader.js:888
2021-05-21T09:58:56.706727664Z throw err;
2021-05-21T09:58:56.706735472Z ^
2021-05-21T09:58:56.706739801Z
2021-05-21T09:58:56.706743086Z Error: Cannot find module '/usr/src/app/node dist/main'
2021-05-21T09:58:56.706746265Z at Function.Module._resolveFilename (internal/modules/cjs/loader.js:885:15)
2021-05-21T09:58:56.706751604Z at Function.Module._load (internal/modules/cjs/loader.js:730:27)
2021-05-21T09:58:56.706755609Z at Function.executeUserEntryPoint [as runMain] (internal/modules/run_main.js:72:12)
2021-05-21T09:58:56.706759645Z at internal/main/run_main_module.js:17:47 {
2021-05-21T09:58:56.706762133Z code: 'MODULE_NOT_FOUND',
2021-05-21T09:58:56.706764372Z requireStack: []
2021-05-21T09:58:56.706766508Z }
 

从日志中就清晰地看到原因了。

修正后成功运行的话,通过 docker ps 看到正常运行的容器了。

$ docker ps
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
743cb8b9d604 wayou/my-app "docker-entrypoint.s…" 5 seconds ago Up 3 seconds 0.0.0.0:8000->3000/tcp, :::8000->3000/tcp test
 

镜像及容器的清除

调试过程难免会生成很多无用的测试数据,可通过如下命令进行清除。

单个清除

通过各自对应的 rm 命令来完成。

镜像的删除

$ docker image rm [OPTIONS] IMAGE [IMAGE...]
 

容器的删除

$  docker rm [OPTIONS] CONTAINER [CONTAINER...] 
 

批量清除

也可通过 docker container prune 将全部容器清除掉。

进入容器内

通过如下命令可在容器中开启一个 shell,在 shell 中可查看其中的文件等。

$ docker exec -it <container id> /bin/bash
 

相关资源

The text was updated successfully, but these errors were encountered:

使用 Docker 部署 Node 应用的更多相关文章

  1. 使用 Docker 部署 Node 应用 - 镜像文件尺寸的优化

    前面 使用 Docker 部署 Node 应用 一文中完成了镜像的创建和运行,不过生成的镜像还有些粗糙,需要进一步优化. 镜像的优化 通过 docker images 看到简单的一个 node 服务端 ...

  2. docker部署node.js

    1.dockerfile FROM node:14.16.0 RUN mkdir -p /var/log/lily/ RUN mkdir -p /opt/node # 工作目录 WORKDIR /op ...

  3. docker部署nodejs项目应用

    之前笔者弄了一套nestjs项目放在自己服务器上,并用pm2管理进程. 现在要把pm2停止,尝试一下用docker容器,那么首先要安装docker 一.安装docker 由于笔者服务器的系统是cent ...

  4. Docker部署Elasticsearch集群

    http://blog.sina.com.cn/s/blog_8ea8e9d50102wwik.html Docker部署Elasticsearch集群 参考文档: https://hub.docke ...

  5. docker 部署cassandra

    摘要 本文主要介绍在redhat7 平台,利用docker 部署cassandra 集群,除了介绍基本的部署步骤,另外主要 讨论类似于cassandra 这种分布式集群系统部署 docker如何进行网 ...

  6. 【原创】运维基础之Docker(2)通过docker部署zookeeper nginx tomcat redis kibana/elasticsearch/logstash mysql kafka mesos/marathon

    通过docker可以从头开始构建集群,也可以将现有集群(配置以及数据)平滑的迁移到docker部署: 1 docker部署zookeeper # usermod -G docker zookeeper ...

  7. 一步步学会用docker部署应用(nodejs版)

    一步步学会用docker部署应用 docker是一种虚拟化技术,可以在内核层隔离资源.因此对于上层应用而言,采用docker技术可以达到类似于虚拟机的沙盒环境.这大大简化了应用部署,让运维人员无需陷入 ...

  8. 教你使用docker部署淘宝rap2服务

    什么是rap2 先来说说起因,在上一个星期的分享会上,谈到前后端联调上,有同事提到了rap2,特意去了解了一下,觉得使用这个东西来进行前后端的接口联调来真是太方便了,对比我们之前公司内部开发的API ...

  9. 【Docker江湖】之docker部署与理解

    转载请注明出处:http://blog.csdn.net/gamer_gyt 博主微博:http://weibo.com/234654758 Github:https://github.com/thi ...

随机推荐

  1. java之I/O流

    I/O流的使用情况多种多样,首先它的数据源就可能是文件.控制台.服务器等,它的单位可能是按字节.按字符.按行等.为了涵盖所有的可能,java类库中创建了大量的类,如此多的类让我们在使用时感觉有点难以选 ...

  2. 【SpringMVC】添加操作时返回400

    本博客老魏原创,如需转载请留言 问题描述: springmvc向数据库添加新的记录时,发生400错误,控制台没有抛出异常. 问题原因: 视图中的提交数据的某一个字段不不匹配导致. 解决方法: 不要怀疑 ...

  3. C#事件总线

    目录 简介 实现事件总线 定义事件基类 定义事件参数基类 定义EventBus 使用事件总线 事件及事件参数 定义发布者 定义订阅者 实际使用 总结 参考资料 简介 事件总线是对发布-订阅模式的一种实 ...

  4. LINQ之方法语法

    上节讲到使用linq的查询关键字进行查询,这节讲一下linq查询的另一种方式--linq方法. 使用linq方法语法,必须要会用lambda表达式,配合lambda表达式才能体会到linq的优雅便捷. ...

  5. [并发编程 - 多线程:信号量、死锁与递归锁、时间Event、定时器Timer、线程队列、GIL锁]

    [并发编程 - 多线程:信号量.死锁与递归锁.时间Event.定时器Timer.线程队列.GIL锁] 信号量 信号量Semaphore:管理一个内置的计数器 每当调用acquire()时内置计数器-1 ...

  6. [物联网] 电气 & 工控

    原理 一次回路和二次回路 一次回路:强电部分(380伏---22万伏),连接发电机.电动机.变压器.电网线路.电网开关.电网避雷器等等 二次回路:弱电部分,指的是控制线路.保护线路.测量线路.计量线路 ...

  7. 阿里云上安装 OpenStack 是什么体验

    阿里云上跑火车(安装 OpenStack Train 版本),猜猜最终花了多少钱? 前言 前面给大家提供了用虚拟机安装 OpenStack 的镜像,虽然已经很简便了,但还是略显笨重.一来镜像文件比较大 ...

  8. 常用的HTML标记

    一.格式标记 1.<br> <br>是一个单标记,用来强制换行. 2.<p> <p>是双标记.用来换分段落. 3.<center> < ...

  9. [bug] Junit initializationError

    原因 导包错误 解决 先删除 import org.junit.Test; 再导入正确的包 参考 https://blog.csdn.net/javae100/article/details/7978 ...

  10. 什么是CPU缓存

    一.什么是CPU缓存 1. CPU缓存的来历 众所周知,CPU是计算机的大脑,它负责执行程序的指令,而内存负责存数据, 包括程序自身的数据.在很多年前,CPU的频率与内存总线的频率在同一层面上.内存的 ...