基于 Docker 开发 NodeJS 应用


Posted in NodeJs onJuly 30, 2014

有关这个 Node 应用

此应用包含一个 package.json, server.js 以及一个 .gitignore 文件, 它们简单到可以信手拈来.

.gitignore

node_modules/*

package.json

{
 "name": "docker-dev",
 "version": "0.1.0",
 "description": "Docker Dev",
 "dependencies": {
  "connect-redis": "~1.4.5",
  "express": "~3.3.3",
  "hiredis": "~0.1.15",
  "redis": "~0.8.4"
 }
}

server.js

var express = require('express'),
 app = express(),
 redis = require('redis'),
 RedisStore = require('connect-redis')(express),
 server = require('http').createServer(app);

app.configure(function() {
 app.use(express.cookieParser('keyboard-cat'));
 app.use(express.session({
  store: new RedisStore({
   host: process.env.REDIS_HOST || 'localhost',
   port: process.env.REDIS_PORT || 6379,
   db: process.env.REDIS_DB || 0
  }),
  cookie: {
   expires: false,
   maxAge: 30 * 24 * 60 * 60 * 1000
  }
 }));
});

app.get('/', function(req, res) {
 res.json({
 status: "ok"
 });
});

var port = process.env.HTTP_PORT || 3000;
server.listen(port);
console.log('Listening on port ' + port);

server.js 会拉取所有的依赖并启动一个特定的应用. 这个特定的应用被设定成将会话信息存储到Redis中,并暴露出一个请求端点,其会响应返回一个JSON的状态消息. 这都是非常标准的东西.

需要注意的一件事情就是针对Redis的连接信息可以使用环境变量重写——这将会在稍后从开发环境dev迁移到生产环境prod时起到作用.

Docker file

为了开发的需要,我们将会让Redis和Node在同一个容器中运行。为此,我们将使用一个Dockerfile来配置这个容器。

Dockerfile

FROM dockerfile/ubuntu

MAINTAINER Abhinav Ajgaonkar <abhinav316@gmail.com>

# Install Redis
RUN  \
 apt-get -y -qq install python redis-server

# Install Node
RUN  \
 cd /opt && \
 wget http://nodejs.org/dist/v0.10.28/node-v0.10.28-linux-x64.tar.gz && \
 tar -xzf node-v0.10.28-linux-x64.tar.gz && \
 mv node-v0.10.28-linux-x64 node && \
 cd /usr/local/bin && \
 ln -s /opt/node/bin/* . && \
 rm -f /opt/node-v0.10.28-linux-x64.tar.gz

# Set the working directory
WORKDIR  /src

CMD ["/bin/bash"]

我们一行一行的来理解,

FROM dockerfile/ubuntu
这回告诉docker要使用Docker Inc. 提供的 dockerfile/ubuntu 镜像. 作为构建的基准镜像.

RUN  \
  apt-get -y -qq install python redis-server
基准镜像完全没有包含任何东西——因此我们需要使用apt-get来获取应用运行起来所需的所有东西. 这一句会安装python 和 redis-server. Redis 服务器是必须的,因为我们将会把会话信息存储到它之中,而python的必要性则是通过npm可以构建为Redis node模块所需的C扩展.

RUN \
 cd /opt && \
 wget http://nodejs.org/dist/v0.10.28/node-v0.10.28-linux-x64.tar.gz && \
 tar -xzf node-v0.10.28-linux-x64.tar.gz && \
 mv node-v0.10.28-linux-x64 node && \
 cd /usr/local/bin && \
 ln -s /opt/node/bin/* . && \
 rm -f /opt/node-v0.10.28-linux-x64.tar.gz

这会下载并提取64位的NodeJS二进制文件.

WORKDIR /src

这句会告诉docker一旦容器已经启动,在执行CMD属性指定的东西之前,要做一次 cd /src.

CMD ["/bin/bash"]

作为最后一步,运行 /bin/bash.

构建并运行容器

现在docker文件写好了,让我们来构建一个Docker镜像吧.

docker build -t sqldump/docker-dev:0.1 .

一旦把镜像构建好了,我们就可以使用下面的语句运行一个容器了:

docker run -i -t --rm \
      -p 3000:3000 \
      -v `pwd`:/src \
      sqldump/docker-dev:0.1

让我们来看一看docker运行命令中发生了什么.

-i 会在交互模式下启动容器(对比 -d 是在分离模式下). 这就意味一旦交互会话结束,容器就会退出.

-t 会分配一个pseudo-tty.

--rm 会在退出时移除容器及其文件系统.

-p 3000:3000 会将主机上的端口 3000 转发到容器上的端口3000.

-v `pwd`:/src
这句将会将当前的工作目录挂载到主机上(例如,我们的项目文件)容器中的 /src 里面. 我们将当前目录作为一个卷挂在,而不是使用Dockerfile中的ADD命令,那样我们在文本编辑器中做的任何修改都可以立即在容器中看到了.

sqldump/docker-dev:0.1 是要运行的docker镜像的名称和版本 ? 这跟我们用来构建docker镜像时使用的名称和版本是相同的.

由于Dockerfile指定了CMD ["/bin/bash"], 容器一启动,我们就会登录到一个bash shell环境中. 如果docker运行命令执行成功了,就会像下面这样:

基于 Docker 开发 NodeJS 应用

开始开发

现在容器是运行起来了,在开始写代码之前,我们将需要整理出一些标准的,非docker相关的东西. 首先,要使用下面的语句启动容器里面的redis服务器:

service redis-server start

然后,要安装项目依赖和nodemon. Nodemon  会观察项目文件中的变更,并适时重启服务器.

npm install
npm install -g nodemon

最后,使用如下命令启动服务器:

nodemon server.js

现在,如果你在浏览器中导航到 http://localhost:3000, 你应该会看到像下面这样的东西:

基于 Docker 开发 NodeJS 应用

让我们来像Server.js中加入另外一个端点,以模拟开发流程:

app.get('/hello/:name', function(req, res) {
 res.json({
  hello: req.params.name
 });
});

你会看到nodemon已经侦测到了你所做的修改,并重启了服务器:

基于 Docker 开发 NodeJS 应用

而现在,如果你将浏览器导航到http://localhost:3000/hello/world, 你会看到如下的响应:

基于 Docker 开发 NodeJS 应用

生产环境

当前状态下的容器,还远不能作为产品发布.redis中的数据不会再跨容器重启时仍然保持持久化 , 比方说,如果你重启了容器,所有的会话数据就都灰飞烟灭了. 同样的事情在你销毁容器并开启一个的新的容器时也会发生,明显这不是你想要的。我将会在第二部分的产品化内容中讲到这个问题.

NodeJs 相关文章推荐
Nodejs学习笔记之Stream模块
Jan 13 NodeJs
Nodejs关于gzip/deflate压缩详解
Mar 04 NodeJs
nodejs 中模拟实现 emmiter 自定义事件
Feb 22 NodeJs
NodeJs读取JSON文件格式化时的注意事项
Sep 25 NodeJs
NodeJS整合银联网关支付(DEMO)
Nov 09 NodeJs
nodejs获取微信小程序带参数二维码实现代码
Apr 12 NodeJs
ajax +NodeJS 实现图片上传实例
Jun 06 NodeJs
nodejs 搭建简易服务器的图文教程(推荐)
Jul 18 NodeJs
基于nodejs res.end和res.send的区别
May 14 NodeJs
使用koa-log4管理nodeJs日志笔记的使用方法
Nov 30 NodeJs
Nodejs实现的操作MongoDB数据库功能完整示例
Feb 02 NodeJs
Nodejs核心模块之net和http的使用详解
Apr 02 NodeJs
Google官方支持的NodeJS访问API,提供后台登录授权
Jul 29 #NodeJs
使用nodejs、Python写的一个简易HTTP静态文件服务器
Jul 18 #NodeJs
抛弃Nginx使用nodejs做反向代理服务器
Jul 17 #NodeJs
nodejs的10个性能优化技巧
Jul 15 #NodeJs
提高NodeJS中SSL服务的性能
Jul 15 #NodeJs
在NodeJS中启用ECMAScript 6小结(windos以及Linux)
Jul 15 #NodeJs
nodejs 实现模拟form表单上传文件
Jul 14 #NodeJs
You might like
php foreach正序倒序输出示例代码
2014/07/01 PHP
PHP查询分页的实现代码
2017/06/09 PHP
php设计模式之职责链模式实例分析【星际争霸游戏案例】
2020/03/27 PHP
基于PHP实现邮箱验证激活过程详解
2020/10/28 PHP
Javascript 代码也可以变得优美的实现方法
2009/06/22 Javascript
ext combox 下拉框不出现自动提示,自动选中的解决方法
2010/02/24 Javascript
javascript中用星号表示预录入内容的实现代码
2011/01/08 Javascript
提升你网站水平的jQuery插件集合推荐
2011/04/19 Javascript
jquery获取table中的某行全部td的内容方法
2013/03/08 Javascript
js绑定事件this指向发生改变的问题解决方法
2013/04/23 Javascript
代码触发js事件(click、change)示例应用
2013/12/13 Javascript
js实现div弹出层的方法
2014/11/20 Javascript
Node.js中npm常用命令大全
2016/06/09 Javascript
js图片延迟加载(Lazyload)三种实现方式
2017/03/01 Javascript
将input框中输入内容显示在相应的div中【三种方法可选】
2017/05/08 Javascript
jQuery简单绑定单个事件的方法示例
2017/06/10 jQuery
vue2组件之select2调用的示例代码
2017/10/12 Javascript
webpack实用小功能介绍
2018/01/02 Javascript
浅谈Webpack打包优化技巧
2018/06/12 Javascript
使用vue 国际化i18n 实现多实现语言切换功能
2018/10/11 Javascript
vue实现带复选框的树形菜单
2019/05/27 Javascript
JavaScript实现HSL拾色器
2020/05/21 Javascript
vue实现用户长时间不操作自动退出登录功能的实现代码
2020/07/23 Javascript
[03:46]DOTA2英雄基础教程 维萨吉
2013/12/11 DOTA
Python常用的文件及文件路径、目录操作方法汇总介绍
2015/05/21 Python
windows及linux环境下永久修改pip镜像源的方法
2016/11/28 Python
利用Python如何批量修改数据库执行Sql文件
2018/07/29 Python
Python 获取项目根路径的代码
2019/09/27 Python
Django框架教程之中间件MiddleWare浅析
2019/12/29 Python
pytorch中获取模型input/output shape实例
2019/12/30 Python
css3实现画半圆弧线的示例代码
2017/11/06 HTML / CSS
简述使用ftp进行文件传输时的两种登录方式?它们的区别是什么?常用的ftp文件传输命令是什么?
2016/11/20 面试题
常见的软件开发流程有哪些
2015/11/14 面试题
公司借条范本
2015/05/25 职场文书
运动会100米广播稿
2015/08/19 职场文书
《普罗米修斯》教学反思
2016/02/22 职场文书