溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Angular.js與node.js項目里用cookie校驗賬戶登錄詳解

發布時間:2020-09-17 07:02:44 來源:腳本之家 閱讀:246 作者:xishiyi7 欄目:web開發

前言

最近的新項目中,用戶登錄需要采用cookie來記住用戶,校驗身份。所以本文就把實現的過程總結出來分享給大家,需要的朋友們可以參考學習。

在header中攜帶authId登錄

在之前老的項目里,沒有采用cookie來記錄用戶登錄狀態,而是在請求的header中攜帶一個身份標識來校驗,大致方案如下:

  1. 客戶端使用post請求提交user、password給服務端進行登錄操作;
  2. 服務端校驗用戶是否合法,如果合法將產生一個唯一的身份標識authId,返回給客戶端,客戶端將此authId存放本地(如localStorage);
  3. 客戶端在每次需要校驗身份的請求中,往header中加入這個authId;
  4. 服務端檢測當前的authId是否有效,有效則表示當前用戶合法,允許操作;
  5. 客戶端用戶登出的時候,發送一個delete請求,告訴服務端用戶注銷,同時刪除本地的authId信息;
  6. 服務端收到注銷請求后,刪除當前的authId數據。

上面的方案,如果其他客戶端知道了這個authId后,可以在其他客戶端模擬身份,不安全,因此棄用。

用cookie記住用戶

新項目中,將采用此文即將介紹的方案–利用cookie來記住用戶。主要流程是:

  1. 客戶端使用post請求提交user、password給服務端進行登錄操作;
  2. 服務端校驗用戶是否合法,如果合法將產生一個唯一的身份標識authId,以cookie的形式返回給客戶端;
  3. 客戶端再次請求服務端時,會攜帶此前已經拿到的cookie給服務端,服務端校驗是否合法,合法則可以繼續操作;
  4. 客戶端用戶登出的時候,發送一個delete請求,告訴服務端用戶注銷,服務端刪除登錄標識。

     整個過程可以用下面這張圖簡單表示:

Angular.js與node.js項目里用cookie校驗賬戶登錄詳解

前臺用angular搭建單頁客戶端應用,后臺用node搭建服務器,數據存放在mongodb中,這三個技術及cookie基礎知識本文不做介紹,感興趣的同學可以自行了解。

以下的代碼都是最簡單的get/post請求,但也是最核心的部分,其他有關登錄的繁瑣操作,感興趣的同學可以自行補充。

從開始–>結束,遇到的問題

首先,我用的是最基礎的post請求,服務端也只是簡單的返回數據,部分簡單但比較核心的代碼如下:

// client
$http({
 method   : 'POST',
 url   : 'http://127.0.0.1:8888/rest/user',
 data   : {name: 'xxx',password:'***'}
 }).success(function (data) {
 console.log('login success,data is:'+data);
}).error(function (data) {
 console.log('login error');
}).then(function () {
 console.log(arguments);
});

// server
var cookie = "authId=" + authId;
res.setHeader('Content-Type', 'application/json;charset=utf-8');
res.setHeader('Set-Cookie', cookie + ';Max-Age=3600;HttpOnly=false;Path=/;');
res.writeHead(200, {'Content-Type': 'text/plain'});
res.end();

查看chrome調試,發現雖然服務端的cookie推過來了,但整體出了問題,提示如下:

Angular.js與node.js項目里用cookie校驗賬戶登錄詳解

XMLHttpRequest cannot load http://127.0.0.1:8888/rest/user. Response to preflight request doesn't pass access control check: No 'Access-Control-Allow-Origin' header is present on the requested resource. Origin 'http://localhost:62427' is therefore not allowed access.

分析問題后,發現原因是來自客戶端的請求不能跨域訪問服務端的請求,請求的資源header中沒有攜帶允許跨越請求的信息。根據這個提示,我們把服務端的代碼稍加改進后,如下:

// server
var cookie = "authId=" + authId;
res.setHeader('Content-Type', 'application/json;charset=utf-8');
res.setHeader('Set-Cookie', cookie + ';Max-Age=3600;HttpOnly=false;Path=/;');

// 添加允許跨越的頭信息
res.setHeader('Access-Control-Allow-Origin', '*');
res.setHeader('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');

res.writeHead(200, {'Content-Type': 'text/plain'});
res.end();

解釋下上面代碼什么意思,第一句主要是允許來自任何域的請求訪問,第二句是允許哪些類型的請求訪問。加上后再次運行,提示如下:

Angular.js與node.js項目里用cookie校驗賬戶登錄詳解XMLHttpRequest

cannot load http://127.0.0.1:8888/rest/user. Request header field Content-Type is not allowed by Access-Control-Allow-Headers in preflight response.

原因是來自客戶端的請求中,Content-Type頭字段,在服務端的響應信息的頭中,沒有攜帶,再次修改代碼如下:

// server
var cookie = "authId=" + authId;
res.setHeader('Content-Type', 'application/json;charset=utf-8');
res.setHeader('Set-Cookie', cookie + ';Max-Age=3600;HttpOnly=false;Path=/;');

// 添加允許跨越的頭信息
res.setHeader('Access-Control-Allow-Origin', '*');
res.setHeader('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');

// 添加支持Content-Type允許的頭信息
res.setHeader('Access-Control-Allow-Headers', 'Content-Type');

res.writeHead(200, {'Content-Type': 'text/plain'});
res.end();

再次運行代碼,發現沒有錯誤提示,但是當我們再次請求服務器時,發現客戶端的請求并沒有攜帶cookie信息,這顯然不是我們想要的效果:

Angular.js與node.js項目里用cookie校驗賬戶登錄詳解

在查閱了一段時間后了解到,客戶端是會默認攜帶cookie給服務端的,但是當客戶端的請求是跨域請求時,由于跨域請求本身就有風險,而攜帶給cookie同樣有風險。

因此在進行跨域訪問時,客戶端不會將服務端返回的cookie攜帶。此時,我們需要同時在客戶端和服務端都設置“withCredentials”為true,代碼如下:

// client
$http({
 method   : 'POST',
 url   : 'http://127.0.0.1:8888/rest/user',
 withCredentials: true
 data   : {name: 'xxx',password:'***'}
 }).success(function (data) {
 console.log('login success,data is:'+data);
}).error(function (data) {
 console.log('login error');
}).then(function () {
 console.log(arguments);
});

// server
var cookie = "authId=" + authId;
res.setHeader('Content-Type', 'application/json;charset=utf-8');
res.setHeader('Set-Cookie', cookie + ';Max-Age=3600;HttpOnly=false;Path=/;');

// 添加允許跨越的頭信息
res.setHeader('Access-Control-Allow-Origin', '*');
res.setHeader('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');

// 添加支持Content-Type允許的頭信息
res.setHeader('Access-Control-Allow-Headers', 'Content-Type');

// 設置已攜帶憑證為true
//res.setHeader('Access-Control-Allow-Credentials', true);

res.writeHead(200, {'Content-Type': 'text/plain'});
res.end();

運行后,發現又有錯誤提示,如下:

Angular.js與node.js項目里用cookie校驗賬戶登錄詳解

XMLHttpRequest cannot load http://127.0.0.1:8888/rest/user. Response to preflight request doesn't pass access control check: A wildcard '*' cannot be used in the 'Access-Control-Allow-Origin' header when the credentials flag is true. Origin 'http://localhost:62427' is therefore not allowed access.

分析錯誤后發現,原因是當設置了已攜帶憑證參數為true時,允許跨域請求的源不能設置為泛型的“*”,因此我們再次修改代碼如下:(最終代碼)

// client
$http({
 method   : 'POST',
 url   : 'http://127.0.0.1:8888/rest/user',
 withCredentials: true
 data   : {name: 'xxx',password:'***'}
 }).success(function (data) {
 console.log('login success,data is:'+data);
}).error(function (data) {
 console.log('login error');
}).then(function () {
 console.log(arguments);
});

// server
var cookie = "authId=" + authId;
res.setHeader('Content-Type', 'application/json;charset=utf-8');
res.setHeader('Set-Cookie', cookie + ';Max-Age=3600;HttpOnly=false;Path=/;');

// 添加允許跨越的頭信息
// res.setHeader('Access-Control-Allow-Origin', '*');
// 用當前的客戶端origin來取代泛型的“*”
res.setHeader('Access-Control-Allow-Origin', 'http://localhost:62427');

res.setHeader('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');

// 添加支持Content-Type允許的頭信息
res.setHeader('Access-Control-Allow-Headers', 'Content-Type');

// 設置已攜帶憑證為true
res.setHeader('Access-Control-Allow-Credentials', true);

res.writeHead(200, {'Content-Type': 'text/plain'});
res.end();

此時,第一次請求服務端時,服務端返回cookie信息,以后每次客戶端請求服務端,客戶端的header中都會攜帶cookie信息,效果如下圖:

Angular.js與node.js項目里用cookie校驗賬戶登錄詳解

最后

以上就是在使用cookie記住用戶身份時遇到的一些問題及簡單解決方法,一般在angular應用中,可能使用較多的是resoure進行http通信,此時只要在GET/POST/PUT/DELETE等請求的參數中,將“withCredentials”設置為true即可。希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女