我们到底能走多远系列(36)

扯淡:

  年关将至,总是会在一些时间节点上才感觉时光飞逝,在平时浑浑噩噩的岁月里都浪费掉了太多的宝贵。请珍惜!

  

主题:

     我们在编写http请求处理和响应的代码的时候,经常会处理到session,这里的session是指服务器和客户端交互时把一些信息存在服务器上,下一次请求是,可以在服务器上继续使用这些信息,我们都知道http是无状态的,在服务端维持一个session就是为了解决一些多个请求需要状态维持的问题。

     它的工作原理,我的理解是,在第一次http请求时,服务端在自己内存里创建出一个对应这个客户端的session,往这个session中放好信息后,把标识这个session的唯一字段在响应的时候带给客户端,客户端将这个字段放入cookie,下一次请求的时候客户端就会把这个cookie信息带上来,服务端就可以找出这个客户端对应的session了,也就可以重新使用原来保存的信息。
     这个工作是web容器完成的,像tomcat,jetty, weblogic 都实现了对session相关的接口。
     比如说遇到这样的问题:多个容器分布部署的时候,web容器中的session无法共享,这样可以不把session的信息部存在内存中,而是存在类似redis,memcache这样的数据库中。这样就需要重写处理session的逻辑。修改web容器的源码,或者自己实现以下存取session,和传输解析cookie的方法来模拟session。
java编码中的使用:
 HttpSession session = request. getSession();
session.setAttribute ("uid", 1) ;
session.getAttribute ("uid") ;
  
从原理上来看,实现的流程很清晰,在用node实现web应用的时候,现在流行用express.js。不使用框架的session接口的话,我们自己也可以粗糙的实现一下:
 
// 获得客户端的Cookie
var Cookies = {};
req.headers.cookie && req.headers.cookie.split(';').forEach(function( Cookie ) {
var parts = Cookie.split('=');
Cookies[ parts[ 0 ].trim() ] = ( parts[ 1 ] || '' ).trim();
});
console.info(Cookies); // 设置cookie
res.setHeader(
'Set-Cookie', 'myCookie=test'
);
  
  利用上面的获取cookie和设置cookie,还不能完全完成session维护的流程。还需要服务端存取session值。
     实际上我们在开发普通的web项目的时候,可以了解到,对每一个客户端,服务端都会维护一个session,这些session中存着一些键值对,就像前面jsva代码中看到的,"uid"对应1,然后来查的时候就利用cookies中带上来的id,找到session,再从这个session中查有没有'uid'对应的1。
  

  那么,下面就简单用js来实现一个登录进入一个下载页面,来模拟http session的机制。
 
流程类似如下:
 
首先,先来看一下SessionsManage这个核心模块
var Session = require('./Session');

var SessionsManage = function(expires, clean_time ){
this.def_expires = expires||100; // 过期时间
this.def_clean_time = clean_time||1000; // 执行清除操作间隔时间
this.sessions = {}; // session 放置处
//启动定时任务,就是说不停的会检查去除sessions中过期的的session
setInterval(this.cleanup, this.def_clean_time, this.sessions);
}; var init = function(expires, clean_time){
return new SessionsManage(expires, clean_time);
};
module.exports = SessionsManage;
/**
* 模拟取session
* @param req
* @param name
* @returns {null, session}
*/
SessionsManage.prototype.getSession = function(req, name){
var id = getIdFromCookies(req);
if(id){
// 现货器session对象
var session = this.sessions[id];
if(session && session[name]){
// 再从session对象中找对应的值
return session[name];
}else {
return null;
}
}else{
return null;
}
}; /**
* 模拟存session
* @param req
* @param res
* @param opts
* @returns {boolean}
*/
SessionsManage.prototype.setSession = function(req, res, opts){
if(!opts){
return false;
}else{
// 从cookie中获取id
var id = getIdFromCookies(req) || randomString(36);
var name = opts.name;
var value = opts.value;
var expires = opts.expires || this.def_expires;
if(id && value && name){
// 新创一个session
var session = new Session();
session[name] = value;
session["id"] = id;
session["overLifeTime"] = (+new Date) + expires*1000;
// 放置进sessions
this.sessions[id] = session;
// 写入返回客户端的cookie中
this.setCookieId(res, id, expires);
}
}
}; SessionsManage.prototype.setCookieId = function(res, id, expires){
// config cookie
var d = new Date();
d.setTime(d.getTime() + expires*1000); // in milliseconds
res.setHeader(
'Set-Cookie', 'D_SID='+ id +';expires='+d.toGMTString()+';'
);
}; SessionsManage.prototype.cleanup = function(sessions){
var now = new Date().getTime();
for(var id in sessions){
var session = sessions[id];
if(session.overLifeTime < now){
delete sessions[session.id];
}
}
}; var getIdFromCookies = function(req){
// client's Cookie
var Cookies = {};
req.headers.cookie && req.headers.cookie.split(';').forEach(function( Cookie ) {
var parts = Cookie.split('=');
Cookies[ parts[ 0 ].trim() ] = ( parts[ 1 ] || '' ).trim();
});
console.info(Cookies);
console.info(Cookies["D_SID"]);
if(Cookies["D_SID"]){
return Cookies["D_SID"];
}else{
return null;
}
}; function randomString(bits){
return new Date().getTime();
}
注:这里使用了时间戳作为相互传递的ID,也可以自己产生随机的ID来代替。
 
Session对象:
var Session = function(opt){
if(opt){
this.id = opt.id;
this.overLifeTime = opt.expires;
}
};
module.exports = Session;
  看了SessionManage的代码,只要在登录的时候调用一下setSession,然后再过滤器上调用下getSession,就可以完成上面的流程了。使用express,虽然它自带了session机制,没有使用,模拟过滤器的代码实现在app.js中的:
 
// 正则匹配全部请求
app.get(/^\/*/,function(req, res, next){
if(req.path == "/upload" || req.path == "/doupload"){
var user = SessionsManage.getSession(req,"user");
if(!user){
res.render('login', {});
return;
}
}
next();
});
登录的时候:
 
exports.dologin = function(db, sessions){
return function(req, res) {
console.log("dologin");
var username = req.body.username;
var password = req.body.password;
console.log("username:" + username + "password:" + password);
var collection = db.get('usercollection');
collection.find({},{'username':username,'password':password},function(e,docs){
if(docs){
console.log("username and password is valid");
var opts = {
name : "user",
value : username,
expires : 500
};
// 调用了setSession
sessions.setSession(req,res,opts);
res.render('upload', {});
}else{
console.log("username and password is invalid");
}
});
};
};

  

  以上就基本走完了流程,只要使用express进行一些请求上的配置,就可以了。主要是sessionManage中的实现,用提供出去的几个方法来模拟了整个机制。
  另外,还有一些不足的地方,和遗留的漏洞。
 
 
 
 
 

让我们继续前行

----------------------------------------------------------------------

努力不一定成功,但不努力肯定不会成功。

node模拟http服务器session机制-我们到底能走多远系列(36)的更多相关文章

  1. node实现http上传文件进度条 -我们到底能走多远系列(37)

    我们到底能走多远系列(37) 扯淡: 又到了一年一度的跳槽季,相信你一定准备好了,每每跳槽,总有好多的路让你选,我们的未来也正是这一个个选择机会组合起来的结果,所以尽可能的找出自己想要的是什么再做决定 ...

  2. ThreadPoolExecutor机制探索-我们到底能走多远系列(41)

    我们到底能走多远系列(41) 扯淡: 这一年过的不匆忙,也颇多感受,成长的路上难免弯路,这个世界上没人关心你有没有变强,只有自己时刻提醒自己,不要忘记最初出发的原因. 其实这个世界上比我们聪明的人无数 ...

  3. js中this和回调方法循环-我们到底能走多远系列(35)

    我们到底能走多远系列(35) 扯淡: 13年最后一个月了,你们在13年初的计划实现了吗?还来得及吗? 请加油~ 主题: 最近一直在写js,遇到了几个问题,可能初入门的时候都会遇到吧,总结下. 例子: ...

  4. Spring mvc源码url路由-我们到底能走多远系列(38)

    我们到底能走多远系列38 扯淡: 马航的事,挺震惊的.还是多多珍惜身边的人吧. 主题: Spring mvc 作为表现层的框架,整个流程是比较好理解的,毕竟我们做web开发的,最早也经常接触的就是一个 ...

  5. 服务调用方案(Spring Http Invoker) - 我们到底能走多远系列(40)

    我们到底能走多远系列(40) 扯淡:  判断是否加可以效力于这家公司,一个很好的判断是,接触下这公司工作几年的员工,了解下生活工作状态,这就是你几年后的状态,如果满意就可以考虑加入了. 主题: 场景: ...

  6. Sharded实现学习-我们到底能走多远系列(32)

    我们到底能走多远系列(32) 扯淡: 工作是容易的赚钱是困难的 恋爱是容易的成家是困难的 相爱是容易的相处是困难的 决定是容易的可是等待是困难的 主题: 1,Sharded的实现    Sharded ...

  7. JMS生产者+单线程发送-我们到底能走多远系列(29)

    我们到底能走多远系列(29) 扯淡: “然后我俩各自一端/望着大河弯弯/终于敢放胆/嘻皮笑脸/面对/人生的难”      --- <山丘> “迎着风/迎向远方的天空/路上也有艰难/也有那解 ...

  8. ArrayBlockingQueue-我们到底能走多远系列(42)

    我们到底能走多远系列(42) 扯淡: 乘着有空,读些juc的源码学习下.后续把juc大致走一边,反正以后肯定要再来. 主题: BlockingQueue 是什么 A java.util.Queue t ...

  9. 初始化IoC容器(Spring源码阅读)-我们到底能走多远系列(31)

    我们到底能走多远系列(31) 扯淡: 有个问题一直想问:各位你们的工资剩下来会怎么处理?已婚的,我知道工资永远都是不够的.未婚的你们,你们是怎么分配工资的? 毕竟,对自己的收入的分配差不多体现了自己的 ...

随机推荐

  1. eclipse 连接 mysql

    1.下载驱动. 2.eclipse->add extend jars -> 添加驱动. 3.测试: 在mysql 建立数据库和表,在eclipse 里对数据库进行操作. 代码: mysql ...

  2. js 日期时间控制器

    /////////////////////////调用实例 // <div> // <span>交易查询:</span> <span>从 // < ...

  3. qt--- vs

    qt with vs 1.安装vs2012: 2.下载Qt 5.2.0 for Windows 32-bit (VS 2012, 579 MB) 和 Visual Studio Add-in 1.2. ...

  4. GL10控制图形旋转

    GL10提供了glRotatef(float  angle , float  x ,  float  y , float  z)方法,该方法用于控制旋转,该方法种angle控制旋转角度:而x.y.z参 ...

  5. 告别硬编码-发个获取未导出函数地址的Dll及源码

    还在为找内核未导出函数地址而苦恼嘛? 还在为硬编码通用性差而不爽吗? 还在为暴搜内核老蓝屏而痛苦吗? 请看这里: 最近老要用到内核未导出的函数及一些结构,不想再找特征码了,准备到网上找点符号文件解析的 ...

  6. 《剑指offer-名企面试官精讲典型编程题》读后感

    首先,不得不说这是一本好书!!! 我接触这本书是在学长的推荐下去看的,而且口碑还是挺好的一本书,豆瓣的评分也比较高,当我刚看了它,我就深深的爱上了这本书,到现在为止,我已经看了三遍这本书了,平时无聊时 ...

  7. Centos 6 安装 epel yum库

    1.获得epel库安装rpm包 wget https://dl.fedoraproject.org/pub/epel/epel-release-latest-6.noarch.rpm 2.安装获得的r ...

  8. php的数据访问

    方法一:过去时方法 $定义一个变量 = $mysql_connect("要连接的服务器,默认是 localhost","登录所使用的用户名,默认是 root", ...

  9. poj1125(Floyd最短路)

    //Accepted 164 KB 0 ms //floyd #include <cstdio> #include <cstring> #include <iostrea ...

  10. struts2DMI(动态方法调用)

    DMI(Dynamic Method Invoke)即动态,是strus2的一个特性,我们知道,在最开始学习strus2时,往往一个action中只有一个excute方法,比如说add,delete, ...