基于 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 url验证(url-valid)的使用方法
Nov 18 NodeJs
NodeJS Web应用监听sock文件实例
Feb 18 NodeJs
详解nodejs与javascript中的aes加密
May 22 NodeJs
NodeJS遍历文件生产文件列表功能示例
Jan 22 NodeJs
nodejs读写json文件的简单方法(必看)
Mar 09 NodeJs
Nodejs搭建wss服务器教程
May 24 NodeJs
docker中编译nodejs并使用nginx启动
Jun 23 NodeJs
NodeJS使用七牛云存储上传文件的方法
Jul 24 NodeJs
nodejs用gulp管理前端文件方法
Jun 24 NodeJs
基于Koa(nodejs框架)对json文件进行增删改查的示例代码
Feb 02 NodeJs
nodejs dgram模块广播+组播的实现示例
Nov 04 NodeJs
一文秒懂nodejs中的异步编程
Jan 28 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
Zend Studio 无法启动的问题解决方法
2008/12/04 PHP
php 仿Comsenz安装效果代码打包提供下载
2010/05/09 PHP
PHP逐行输出(ob_flush与flush的组合)
2012/02/04 PHP
深入理解Yii2.0乐观锁与悲观锁的原理与使用
2017/07/26 PHP
关于Javascript 的 prototype问题。
2007/01/03 Javascript
HTML颜色选择器实现代码
2010/11/23 Javascript
ScrollDown的基本操作示例
2013/06/09 Javascript
javascript每日必学之条件分支
2016/02/17 Javascript
浅谈javascript:两种注释,声明变量,定义函数
2016/10/05 Javascript
AngularJS入门教程之Helloworld示例
2016/12/25 Javascript
js时间戳和c#时间戳互转方法(推荐)
2017/02/15 Javascript
js中document.referrer实现移动端返回上一页
2017/02/22 Javascript
Js实现京东无延迟菜单效果实例(demo)
2017/06/02 Javascript
jQuery复合事件用法示例
2017/06/10 jQuery
浅析vue插槽和作用域插槽的理解
2019/04/22 Javascript
Vue通过for循环随机生成不同的颜色或随机数的实例
2019/11/09 Javascript
浅谈Python中的可变对象和不可变对象
2017/07/07 Python
基于使用paramiko执行远程linux主机命令(详解)
2017/10/16 Python
python中实现精确的浮点数运算详解
2017/11/02 Python
Django跨域请求问题的解决方法示例
2018/06/16 Python
在pycharm中设置显示行数的方法
2019/01/16 Python
Python 简单计算要求形状面积的实例
2020/01/18 Python
python GUI库图形界面开发之PyQt5浏览器控件QWebEngineView详细使用方法
2020/02/26 Python
python统计字符串中字母出现次数代码实例
2020/03/02 Python
mac安装python3后使用pip和pip3的区别说明
2020/09/01 Python
python 爬取百度文库并下载(免费文章限定)
2020/12/04 Python
CSS3 滤镜 webkit-filter详细介绍及使用方法
2012/12/27 HTML / CSS
AmazeUI 按钮交互的实现示例
2020/08/24 HTML / CSS
加拿大建筑和装修专家:Reno-Depot
2017/12/21 全球购物
如何为DataGridView添加一个定制的Column Type
2014/01/21 面试题
营业员实习自我鉴定
2013/12/07 职场文书
博士生求职信
2014/07/06 职场文书
十佳家长事迹材料
2014/08/26 职场文书
传承焦裕禄精神思想汇报2014
2014/09/10 职场文书
销售开票员岗位职责
2015/04/15 职场文书
MySQL添加索引特点及优化问题
2022/07/23 MySQL