無憂無鹵

          鳥在籠中,恨關羽不能張飛;人處世上,要八戒更需悟空

          定制的Spring Security(Acegi)的并發會話過濾器(ConcurrentSessionFilter)的編碼過程(轉)

          定制的Spring Security(Acegi)的并發會話過濾器(ConcurrentSessionFilter)的編碼過程

          上一帖述及使用ConcurrentSessionFilter限制同帳號登錄多次的方法,同帳號多次登錄限制是運行系統必需的功能,所以作者對其深入測試,在上一帖中也列舉了Spring Security的ConcurrentSessionFilter和ConcurrentSessionControllerImpl類的幾個限制。做一下簡單的總結,下面假設同時使用DigestProcessingFilter和輔助類:

          • 如果exceptionIfMaximumExceeded = true,即第二個發起的會話被禁止,如果一個用戶重新啟動瀏覽器,再次登錄失敗,因為前一個會話沒有超時,被當成了多次登錄。
          • 如果exceptionIfMaximumExceeded = false,如果兩個人使用同一個帳號登錄,將出現交互將對方踢出去的現象,實際上并沒有禁止任何人登錄,只是每次要先將另一個人踢下去。


          需求

          我想使用exceptionIfMaximumExceeded = true,同時允許同一個用戶在同一臺機器上連續登錄多次,我采取了編寫定制的ConcurrentSessionController實現類的方法。



          原理

          ConcurrentSessionController是一個接口,有兩個需要實現的方法:checkAuthenticationAllowed()和registerSuccessfulAuthentication(),Spring Security提供了一個實現類ConcurrentSessionControllerImpl,經過分析缺省的實現類,發現方法allowableSessionsExceeded()處理多次并發會話,在SecurityRegistry中保存每個會話的信息,主要是用戶帳號對應的會話ID(sessionId)和最后發起時間,在并發發生時,從SecurityRegistry中取出關于某個用戶帳號的所有會話,如果exceptionIfMaximumExceeded = false,找到最早一個會話,將其釋放掉,騰出空間給新會話,如果exceptionIfMaximumExceeded = true,將發出一個異常。

          所以,需要改進allowableSessionsExceeded(),如果exceptionIfMaximumExceeded = true讓程序判斷客戶地址,如果同一個IP,則允許登錄,將最早的會話釋放掉,如果不是同一個IP在發出異常。

          在SecurityRegistry中,用戶帳號信息存在一個對象中,名字是principal,當前是一個Object對象,實際上只是存了一個字符串,所以需要擴展principal,寫一個定制的類(我的類含有兩個屬性:username和userip),里面保存客戶IP信息。allowableSessionsExceeded()只是使用SecurityRegistry,SecurityRegistry中的內容是由registerSuccessfulAuthentication()方法寫入的,所以,在該方法中需要將原來的pricipal對象替換成定制的Principal類的對象。同時checkAuthenticationAllowed()方法也要修改,因為這個方法要查詢SecurityRegistry,查詢條件替換成定制的Principal類的對象。



          注意事項

          定制的Principal類要實現equals()和hashCode()和toString()三個方法,在equals()方法中只要username相同就表示兩個對象相同,而在hashCode()中只需要將username的hashcode計算在內,因為SecurityRegistry是以principal為關鍵字的Map容器,這兩個方法決定了對Map的查詢。toString()方法可以根據自己的需要寫,我只是將username輸出。



          測試

          將定制的ConcurrentSessionController對象編制(wire)到應用系統中,經過測試,能夠達到預想目的。



          存在的問題

          原來想省點勁,只要繼承ConcurrentSessionControllerImpl并重載上述三個方法就行了,但是不知道為什么securityRegistery屬性一直注入不了,一氣之下,寫了一個直接實現ConcurrentSessionController接口的新類。實際上也不是從頭寫,將ConcurrentSessionControllerImpl代碼改吧該吧即可,用不了幾分鐘,這就是開源的好處。



              轉自 http://www.gooseeker.com/cn/node/517

          posted on 2009-05-08 10:33 beyond 閱讀(3318) 評論(0)  編輯  收藏


          只有注冊用戶登錄后才能發表評論。


          網站導航:
           

          導航

          <2009年5月>
          262728293012
          3456789
          10111213141516
          17181920212223
          24252627282930
          31123456

          統計

          常用鏈接

          留言簿(1)

          隨筆檔案

          相冊

          搜索

          最新評論

          • 1.?re: 近期任務
          • 大哥 等著你的drools5翻譯呢 drools4把我搞死了 如果你有關于drools4的心得的話請指教 感覺自己使用的不是很好 望賜教 QQ:53816565 謝謝
          • --sunnidy
          • 2.?re: Mysql遷移到Oracle
          • 不錯,看樣子支持大多數數據庫遷移到oracle,不知道是否支持從oracle轉出?
          • --FastUnit

          閱讀排行榜

          評論排行榜

          主站蜘蛛池模板: 张家港市| 洛南县| 乌兰县| 淮安市| 景德镇市| 景东| 龙岩市| 湄潭县| 伊宁县| 沧州市| 武夷山市| 沾化县| 阜南县| 舟山市| 双牌县| 惠来县| 河源市| 鱼台县| 柘荣县| 桐庐县| 马公市| 河池市| 宁都县| 博乐市| 全州县| 安图县| 社旗县| 图木舒克市| 阿拉善左旗| 靖宇县| 都匀市| 鹿泉市| 富顺县| 锡林浩特市| 富源县| 甘肃省| 阿坝| 深水埗区| 休宁县| 蓬莱市| 松溪县|