express如何使用session與cookie

無狀態(tài)的http

我們都知道http的請(qǐng)求和響應(yīng)式相互獨(dú)立的晤揣,服務(wù)器無法識(shí)別兩條http請(qǐng)求是否是同一個(gè)用戶發(fā)送的疑故。也就是說服務(wù)器端并沒有記錄通信狀態(tài)的能力踱启。我們通常使用cookie和session來確定會(huì)話雙方的身份隅要。

cookie

cookie 是從服務(wù)器端發(fā)送的蝴罪,服務(wù)器給不同的用戶發(fā)送不同的標(biāo)識(shí),這個(gè)標(biāo)識(shí)表示用戶的身份步清,服務(wù)器通過客戶端發(fā)送的這個(gè)標(biāo)識(shí)來識(shí)別用戶的身份要门,從而查詢服務(wù)器中的該用戶的相關(guān)數(shù)據(jù),然后發(fā)送到該用戶廓啊。

安裝express提供的cookie-parser中間件:

npm i -S cookie-parser

在我們使用的項(xiàng)目頁面模塊中引入 cookie-parser 插件欢搜,然后實(shí)例化它,如下:

var cookieParser = require('cookie-parser');

var cp = cookieParser(secret, options);

它有兩個(gè)參數(shù)谴轮,第一個(gè)參數(shù)secret炒瘟,用它可以對(duì)cookie進(jìn)行簽名,也就是我們常說的cookie加密第步。它可以是字符串也可以是數(shù)組疮装,如果熟悉加密原理的同學(xué)應(yīng)該知道,這個(gè)字符串就是服務(wù)器所擁有的密文粘都,第二個(gè)參數(shù)options包含如下可選參數(shù):

path:指定 cookie 影響到的路徑
expires: 指定時(shí)間格式
maxAge:指定 cookie 什么時(shí)候過期
secure:當(dāng) secure 值為 true 時(shí)廓推,在 HTTPS 中才有效;反之翩隧,cookie 在 HTTP 中是有效樊展。
httpOnly:瀏覽器不允許腳本操作 document.cookie 去更改 cookie。設(shè)置為true可以避免被 xss 攻擊拿到 cookie

參考cookie-parser中的例子,實(shí)現(xiàn)一個(gè)記住訪問路徑的demo专缠,代碼如下:

var path = require('path');
var express = require('express');
var cookieParser = require('cookie-parser');
var app = express();

// 使用 cookieParser 中間件;
app.use(cookieParser());

// 如果請(qǐng)求中的 cookie 存在 isFirst
// 否則雷酪,設(shè)置 cookie 字段 isFirst, 并設(shè)置過期時(shí)間為10秒
app.get('/', function(req, res) {
    if (req.cookies.isFirst) {
        res.send("再次歡迎訪問");
        console.log(req.cookies)
    } else {
        res.cookie('isFirst', 1, { maxAge: 60 * 1000});
        res.send("歡迎第一次訪問");
    }
});

app.listen(3030, function() {
    console.log('express start on: ' + 3030)
});

cookie-parser 還可以對(duì)Cookie數(shù)據(jù)進(jìn)行加密,也就是我們所說的signedCookies藤肢。

signedCookies

實(shí)現(xiàn)代碼如下:

var path = require('path');
var express = require('express');
var cookieParser = require('cookie-parser');
var app = express();

// 使用 cookieParser 中間件;
app.use(cookieParser('my_cookie_secret'));

// cookie
app.get('/', function(req, res) {
    if (req.signedCookies.isFirst) {
        res.send("歡迎再一次訪問");
        console.log(req.signedCookies)
    } else {
        res.cookie('isFirst', 1, { maxAge: 60 * 1000, signed: true});
        res.send("歡迎第一次訪問");
    }
});

從上面的代碼中我們知道cooke-parser的第一個(gè)參數(shù)可以指定服務(wù)器端的提供的加密密匙太闺,然后我們使用options中的signed配置項(xiàng)可實(shí)現(xiàn)加密糯景。雖然這樣相對(duì)安全嘁圈,但是客戶端的Cookie有局限性,在客戶端發(fā)送請(qǐng)求時(shí)會(huì)增加請(qǐng)求頭部的數(shù)據(jù)量蟀淮,導(dǎo)致請(qǐng)求速度變慢最住;另外它不能實(shí)現(xiàn)數(shù)據(jù)的共享。

session

express-session 是expressjs的一個(gè)中間件用來創(chuàng)建session怠惶。服務(wù)器端生成了一個(gè)sessionn-id涨缚,客戶端使用了cookie保存了session-id這個(gè)加密的請(qǐng)求信息,而將用戶請(qǐng)求的數(shù)據(jù)保存在服務(wù)器端策治,但是它也可以實(shí)現(xiàn)將用戶的數(shù)據(jù)加密后保存在客戶端脓魏。

session記錄的是客戶端與服務(wù)端之間的會(huì)話狀態(tài),該狀態(tài)用來確定客戶端的身份通惫。

express-session支持session存放位置

可以存放在cookie中茂翔,也可以存放在內(nèi)存中,或者是redis履腋、mongodb等第三方服務(wù)器中珊燎。

session默認(rèn)存放在內(nèi)存中,存放在cookie中安全性太低遵湖,存放在非redis數(shù)據(jù)庫中查詢速度太慢悔政,一般項(xiàng)目開發(fā)中都是存放在redis中(緩存數(shù)據(jù)庫)。

在express提供的express-session中間件安裝命令:

npm i -S express-session

在我們使用的項(xiàng)目頁面模塊中引入 express-session 插件延旧,然后實(shí)例化它谋国,如下:

var session = require('express-session');

var se = session(options);

session()的參數(shù)options配置項(xiàng)主要有:

name: 設(shè)置cookie中,保存session的字段名稱迁沫,默認(rèn)為connect.sid
store: session的存儲(chǔ)方式烹卒,默認(rèn)為存放在內(nèi)存中,我們可以自定義redis等
genid: 生成一個(gè)新的session_id時(shí)弯洗,默認(rèn)為使用uid2這個(gè)npm包
rolling: 每個(gè)請(qǐng)求都重新設(shè)置一個(gè)cookie旅急,默認(rèn)為false
resave: 即使session沒有被修改,也保存session值牡整,默認(rèn)為true
saveUninitialized:強(qiáng)制未初始化的session保存到數(shù)據(jù)庫
secret: 通過設(shè)置的secret字符串藐吮,來計(jì)算hash值并放在cookie中,使產(chǎn)生的signedCookie防篡改
cookie : 設(shè)置存放sessionid的cookie的相關(guān)選項(xiàng)

那么,使用它我們都能做些什么呢谣辞?下面我們將一一介紹迫摔。

cookie session

cookie session 使用很簡(jiǎn)單就是我們?cè)谂渲庙?xiàng)中使用cookie配置項(xiàng),就可以將session數(shù)據(jù)保存在cookie中,它和signedCookies類似都是將數(shù)據(jù)保存在客戶端泥从,而且都對(duì)數(shù)據(jù)進(jìn)行了加密句占,但是加密后的請(qǐng)求得到的數(shù)據(jù)結(jié)構(gòu)不一樣。

cooke session 的結(jié)構(gòu)如下:

Session {
  cookie:
   { path: '/',
     _expires: 2018-01-29T17:58:49.950Z,
     originalMaxAge: 60000,
     httpOnly: true },
  isFirst: 1 }

signedCookie 結(jié)構(gòu)如下:

{ isFirst: '1' }

實(shí)現(xiàn)cookie session代碼如下:

var path = require('path');
var express = require('express');
var session = require('express-session');
var redisStore = require('connect-redis')(session);
var app = express();

// session
app.use(session({
    name: 'session-name', // 這里是cookie的name躯嫉,默認(rèn)是connect.sid
    secret: 'my_session_secret', // 建議使用 128 個(gè)字符的隨機(jī)字符串
    resave: true,
    saveUninitialized: false,
    cookie: { maxAge: 60 * 1000, httpOnly: true }
}));

// route
app.get('/', function(req, res, next) {
    if(req.session.isFirst || req.cookies.isFirst) {
        res.send("歡迎再一次訪問");
    } else {
        req.session.isFirst = 1;
        res.cookie('isFirst', 1, { maxAge: 60 * 1000, singed: true});
        res.send("歡迎第一次訪問纱烘。");
    }
});

app.listen(3030, function() {
    console.log('express start on: ' + 3030)
});

signed-cookie vs cookie session

  • signedCookies 信息可見但不可修改,cookie session不可見也不可修改
  • signedCookies 信息長(zhǎng)期保存客戶端祈餐,后者客戶端關(guān)閉擂啥,信息消失

針對(duì)Cooke session增加了客戶端請(qǐng)求的數(shù)據(jù)規(guī)模,我們一般這樣使用帆阳,數(shù)據(jù)庫存儲(chǔ)session哺壶。

數(shù)據(jù)庫保存session

用數(shù)據(jù)庫保存session,我們一般使用redis蜒谤,因?yàn)樗蔷彺鏀?shù)據(jù)庫山宾,查詢速度相較于非緩存的速度更快。

express-session 的實(shí)例代碼如下:

var path = require('path');
var express = require('express');
var session = require('express-session');
var redisStore = require('connect-redis')(session);
var app = express();

// session
app.use(session({
    name: 'session-name', // 這里是cookie的name鳍徽,默認(rèn)是connect.sid
    secret: 'my_session_secret', // 建議使用 128 個(gè)字符的隨機(jī)字符串
    resave: true,
    saveUninitialized: false,
    store: new redisStore({
        host: '127.0.0.1',
        port: '6379',
        db: 0,
        pass: '',
    })
}));

// route
app.get('/', function(req, res) {
    if (req.session.isFirst) {
        res.send("歡迎再一次訪問资锰。");
        console.log(req.session)
    } else {
        req.session.isFirst = 1;
        res.send("歡迎第一次訪問。");
    }
});

app.listen(3030, function() {
    console.log('express start on: ' + 3030)
});

但有時(shí)我們也使用非redis數(shù)據(jù)庫保存session旬盯,這時(shí)我們就需要對(duì)項(xiàng)目結(jié)構(gòu)有深刻的認(rèn)識(shí)和理解台妆;否則,使用后反而會(huì)適得其反胖翰。

另外接剩,我們要注意使用數(shù)據(jù)庫保存session數(shù)據(jù),在瀏覽器端的session-id會(huì)隨著瀏覽器的關(guān)閉而消失萨咳,下次打開瀏覽器發(fā)送請(qǐng)求時(shí)懊缺,服務(wù)器依然不能識(shí)別請(qǐng)求者的身份。

cookie session 雖然能解決這個(gè)問題培他,但是它本身存在著安全風(fēng)險(xiǎn)鹃两,其實(shí)cookie session 和 signedCookies都面臨xss攻擊。

其實(shí)舀凛,使用signedCookies和session的結(jié)合會(huì)在一定程度上降低這樣的風(fēng)險(xiǎn)俊扳。

signedCookies(cookies) 和 session的結(jié)合

在開發(fā)中,我們往往需要signedCookies的長(zhǎng)期保存特性猛遍,又需要session的不可見不可修改的特性馋记。

var path = require('path');
var express = require('express');
var cookieParser = require('cookie-parser');
var session = require('express-session');
var redisStore = require('connect-redis')(session);
var app = express();

// 使用 cookieParser 中間件;
app.use(cookieParser());

// session
app.use(session({
    name: 'session-name', // 這里是cookie的name号坡,默認(rèn)是connect.sid
    secret: 'my_session_secret', // 建議使用 128 個(gè)字符的隨機(jī)字符串
    resave: true,
    saveUninitialized: false,
    // cookie: { maxAge: 60 * 1000, httpOnly: true },
    store: new redisStore({
        host: '127.0.0.1',
        port: '6379',
        db: 0,
        pass: '',
    })
}));

app.get('/', function(req, res, next) {
    if(req.session.isFirst || req.cookies.isFirst) {
        res.send("歡迎再一次訪問");
    } else {
        req.session.isFirst = 1;
        res.cookie('isFirst', 1, { maxAge: 60 * 1000, singed: true});
        res.send("歡迎第一次訪問。");
    }
});

app.listen(3030, function() {
    console.log('express start on: ' + 3030)
});

這樣我們將session保存在redis中的信息梯醒,保存在了session_id所標(biāo)示的客戶端cooke中一份宽堆,這樣我們就不用擔(dān)心,瀏覽器關(guān)閉茸习,cookie中的session_id字段就會(huì)消失的情況畜隶,因?yàn)闉g覽器中還有它的備份cookie,如果沒有備份的cookie信息号胚,下次客戶端再次發(fā)出請(qǐng)求瀏覽就無法確定用戶的身份籽慢。

參考源碼

nodejs 快速上手

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市涕刚,隨后出現(xiàn)的幾起案子嗡综,更是在濱河造成了極大的恐慌乙帮,老刑警劉巖杜漠,帶你破解...
    沈念sama閱讀 211,884評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異察净,居然都是意外死亡驾茴,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,347評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門氢卡,熙熙樓的掌柜王于貴愁眉苦臉地迎上來锈至,“玉大人,你說我怎么就攤上這事译秦∠考瘢” “怎么了?”我有些...
    開封第一講書人閱讀 157,435評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵筑悴,是天一觀的道長(zhǎng)们拙。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,509評(píng)論 1 284
  • 正文 為了忘掉前任透且,我火速辦了婚禮痴施,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘件舵。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,611評(píng)論 6 386
  • 文/花漫 我一把揭開白布埂奈。 她就那樣靜靜地躺著,像睡著了一般定躏。 火紅的嫁衣襯著肌膚如雪账磺。 梳的紋絲不亂的頭發(fā)上海蔽,一...
    開封第一講書人閱讀 49,837評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音绑谣,去河邊找鬼党窜。 笑死,一個(gè)胖子當(dāng)著我的面吹牛借宵,可吹牛的內(nèi)容都是我干的幌衣。 我是一名探鬼主播,決...
    沈念sama閱讀 38,987評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼壤玫,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼豁护!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起欲间,我...
    開封第一講書人閱讀 37,730評(píng)論 0 267
  • 序言:老撾萬榮一對(duì)情侶失蹤楚里,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后猎贴,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體班缎,經(jīng)...
    沈念sama閱讀 44,194評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,525評(píng)論 2 327
  • 正文 我和宋清朗相戀三年她渴,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了达址。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,664評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡趁耗,死狀恐怖沉唠,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情苛败,我是刑警寧澤满葛,帶...
    沈念sama閱讀 34,334評(píng)論 4 330
  • 正文 年R本政府宣布,位于F島的核電站罢屈,受9級(jí)特大地震影響嘀韧,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜儡遮,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,944評(píng)論 3 313
  • 文/蒙蒙 一乳蛾、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧鄙币,春花似錦肃叶、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,764評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至绩衷,卻和暖如春蹦魔,著一層夾襖步出監(jiān)牢的瞬間激率,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,997評(píng)論 1 266
  • 我被黑心中介騙來泰國打工勿决, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留乒躺,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,389評(píng)論 2 360
  • 正文 我出身青樓低缩,卻偏偏與公主長(zhǎng)得像嘉冒,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子咆繁,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,554評(píng)論 2 349

推薦閱讀更多精彩內(nèi)容