编写你自己的单点登录(SSO)服务(转)
生活随笔
收集整理的這篇文章主要介紹了
编写你自己的单点登录(SSO)服务(转)
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
編寫你自己的單點登錄(SSO)服務
分類: 文章 2006-05-24 10:48 48305人閱讀 評論(87) 收藏 舉報 王昱 yuwang881@gmail.com?? 博客地址http://yuwang881.blog.sohu.com 摘要:單點登錄(SSO)的技術被越來越廣泛地運用到各個領域的軟件系統當中。本文從業務的角度分析了單點登錄的需求和應用領域;從技術本身的角度分析了單點登錄技術的內部機制和實現手段,并且給出Web-SSO和桌面SSO的實現、源代碼和詳細講解;還從安全和性能的角度對現有的實現技術進行進一步分析,指出相應的風險和需要改進的方面。本文除了從多個方面和角度給出了對單點登錄(SSO)的全面分析,還并且討論了如何將現有的應用和SSO服務結合起來,能夠幫助應用架構師和系統分析人員從本質上認識單點登錄,從而更好地設計出符合需要的安全架構。 關鍵字:SSO, Java, J2EE, JAAS 1 什么是單點登陸 單點登錄(Single Sign On),簡稱為 SSO,是目前比較流行的企業業務整合的解決方案之一。SSO的定義是在多個應用系統中,用戶只需要登錄一次就可以訪問所有相互信任的應用系統。 較大的企業內部,一般都有很多的業務支持系統為其提供相應的管理和IT服 務。例如財務系統為財務人員提供財務的管理、計算和報表服務;人事系統為人事部門提供全公司人員的維護服務;各種業務系統為公司內部不同的業務提供不同的 服務等等。這些系統的目的都是讓計算機來進行復雜繁瑣的計算工作,來替代人力的手工勞動,提高工作效率和質量。這些不同的系統往往是在不同的時期建設起來 的,運行在不同的平臺上;也許是由不同廠商開發,使用了各種不同的技術和標準。如果舉例說國內一著名的IT公司(名字隱去),內部共有60多個業務系統,這些系統包括兩個不同版本的SAP的ERP系統,12個不同類型和版本的數據庫系統,8個不同類型和版本的操作系統,以及使用了3種不同的防火墻技術,還有數十種互相不能兼容的協議和標準,你相信嗎?不要懷疑,這種情況其實非常普遍。每一個應用系統在運行了數年以后,都會成為不可替換的企業IT架構的一部分,如下圖所示。 隨 著企業的發展,業務系統的數量在不斷的增加,老的系統卻不能輕易的替換,這會帶來很多的開銷。其一是管理上的開銷,需要維護的系統越來越多。很多系統的數 據是相互冗余和重復的,數據的不一致性會給管理工作帶來很大的壓力。業務和業務之間的相關性也越來越大,例如公司的計費系統和財務系統,財務系統和人事系 統之間都不可避免的有著密切的關系。 為了降低管理的消耗,最大限度的重用已有投資的系統,很多企業都在進行著企業應用集成(EAI)。 企業應用集成可以在不同層面上進行:例如在數據存儲層面上的“數據大集中”,在傳輸層面上的“通用數據交換平臺”,在應用層面上的“業務流程整合”,和用 戶界面上的“通用企業門戶”等等。事實上,還用一個層面上的集成變得越來越重要,那就是“身份認證”的整合,也就是“單點登錄”。 通常來說,每個單獨的系統都會有自己的安全體系和身份認證系統。整合以前,進入每個系統都需要進行登錄,這樣的局面不僅給管理上帶來了很大的困難,在安全方面也埋下了重大的隱患。下面是一些著名的調查公司顯示的統計數據:- 用戶每天平均 16 分鐘花在身份驗證任務上 - 資料來源: IDS
- 頻繁的 IT 用戶平均有 21 個密碼 - 資料來源: NTA Monitor Password Survey
- 49% 的人寫下了其密碼,而 67% 的人很少改變它們
- 每 79 秒出現一起身份被竊事件 - 資料來源:National Small Business Travel Assoc
- 全球欺騙損失每年約 12B - 資料來源:Comm Fraud Control Assoc
- 到 2007 年,身份管理市場將成倍增長至 $4.5B - 資料來源:IDS
- 提高 IT 效率:對于每 1000 個受管用戶,每用戶可節省$70K
- 幫助臺呼叫減少至少1/3,對于 10K 員工的公司,每年可以節省每用戶 $75,或者合計 $648K
- 生產力提高:每個新員工可節省 $1K,每個老員工可節省 $350 �資料來源:Giga
- ROI 回報:7.5 到 13 個月 �資料來源:Gartner
- 所有應用系統共享一個身份認證系統。
統一的認證系統是SSO的前提之一。認證系統的主要功能是將用戶的登錄信息和用戶信息庫相比較,對用戶進行登錄認證;認證成功后,認證系統應該生成統一的認證標志(ticket),返還給用戶。另外,認證系統還應該對ticket進行效驗,判斷其有效性。 - 所有應用系統能夠識別和提取ticket信息
要實現SSO的功能,讓用戶只登錄一次,就必須讓應用系統能夠識別已經登錄過的用戶。應用系統應該能對ticket進行識別和提取,通過與認證系統的通訊,能自動判斷當前用戶是否登錄過,從而完成單點登錄的功能。
- 單一的用戶信息數據庫并不是必須的,有許多系統不能將所有的用戶信息都集中存儲,應該允許用戶信息放置在不同的存儲中,如下圖所示。事實上,只要統一認證系統,統一ticket的產生和效驗,無論用戶信息存儲在什么地方,都能實現單點登錄。
- 統一的認證系統并不是說只有單個的認證服務器,如下圖所示,整個系統可以存在兩個以上的認證服務器,這些服務器甚至可以是不同的產品。認證服務器之間要通過標準的通訊協議,互相交換認證信息,就能完成更高級別的單點登錄。如下圖,當用戶在訪問應用系統1時,由第一個認證服務器進行認證后,得到由此服務器產生的ticket。當他訪問應用系統4的時候,認證服務器2能夠識別此ticket是由第一個服務器產生的,通過認證服務器之間標準的通訊協議(例如SAML)來交換認證信息,仍然能夠完成SSO的功能。
- Web-SSO的樣例是由三個標準Web應用組成,壓縮成三個zip文件,從http://gceclub.sun.com.cn/wangyu/web-sso/中下載。其中SSOAuth(http://gceclub.sun.com.cn/wangyu/web-sso/SSOAuth.zip)是身份認證服務;SSOWebDemo1(http://gceclub.sun.com.cn/wangyu/web-sso/SSOWebDemo1.zip)和SSOWebDemo2(http://gceclub.sun.com.cn/wangyu/web-sso/SSOWebDemo2.zip)是兩個用來演示單點登錄的Web應用。這三個Web應用之所以沒有打成war包,是因為它們不能直接部署,根據讀者的部署環境需要作出小小的修改。樣例部署和運行的環境有一定的要求,需要符合Servlet2.3以上標準的J2EE容器才能運行(例如Tomcat5,Sun Application Server 8, Jboss 4等)。另外,身份認證服務需要JDK1.5的運行環境。之所以要用JDK1.5是因為筆者使用了一個線程安全的高性能的Java集合類“ConcurrentMap”,只有在JDK1.5中才有。
- 這三個Web應用完全可以單獨部署,它們可以分別部署在不同的機器,不同的操作系統和不同的J2EE的產品上,它們完全是標準的和平臺無關的應用。但是有一個限制,那兩臺部署應用(demo1、demo2)的機器的域名需要相同,這在后面的章節中會解釋到cookie和domain的關系以及如何制作跨域的WEB-SSO
- 解壓縮SSOAuth.zip文件,在/WEB-INF/下的web.xml中請修改“domainname”的屬性以反映實際的應用部署情況,domainname需要設置為兩個單點登錄的應用(demo1和demo2)所屬的域名。這個domainname和當前SSOAuth服務部署的機器的域名沒有關系。我缺省設置的是“.sun.com”。如果你部署demo1和demo2的機器沒有域名,請輸入IP地址或主機名(如localhost),但是如果使用IP地址或主機名也就意味著demo1和demo2需要部署到一臺機器上了。設置完后,根據你所選擇的J2EE容器,可能需要將SSOAuth這個目錄壓縮打包成war文件。用“jar -cvf SSOAuth.war SSOAuth/”就可以完成這個功能。
- 解壓縮SSOWebDemo1和SSOWebDemo2文件,分別在它們/WEB-INF/下找到web.xml文件,請修改其中的幾個初始化參數
<init-param>
<param-name>SSOServiceURL</param-name>
<param-value>http://wangyu.prc.sun.com:8080/SSOAuth/SSOAuth</param-value>
</init-param>
<init-param>
<param-name>SSOLoginPage</param-name>
<param-value>http://wangyu.prc.sun.com:8080/SSOAuth/login.jsp</param-value>
</init-param>
將其中的SSOServiceURL和SSOLoginPage修改成部署SSOAuth應用的機器名、端口號以及根路徑(缺省是SSOAuth)以反映實際的部署情況。設置完后,根據你所選擇的J2EE容器,可能需要將SSOWebDemo1和SSOWebDemo2這兩個目錄壓縮打包成兩個war文件。用“jar -cvf SSOWebDemo1.war SSOWebDemo1/”就可以完成這個功能。 - 請輸入第一個web應用的測試URL(test.jsp),例如http://wangyu.prc.sun.com:8080/ SSOWebDemo1/test.jsp,如果是第一次訪問,便會自動跳轉到登錄界面,如下圖
- 使用系統自帶的三個帳號之一登錄(例如,用戶名:wangyu,密碼:wangyu),便能成功的看到test.jsp的內容:顯示當前用戶名和歡迎信息。
- 請接著在同一個瀏覽器中輸入第二個web應用的測試URL(test.jsp),例如http://wangyu.prc.sun.com:8080/ SSOWebDemo2/test.jsp。你會發現,不需要再次登錄就能看到test.jsp的內容,同樣是顯示當前用戶名和歡迎信息,而且歡迎信息中明確的顯示當前的應用名稱(demo2)。
| package DesktopSSO; import java.io.*; import java.net.*; import java.text.*; import java.util.*; import java.util.concurrent.*; import javax.servlet.*; import javax.servlet.http.*; public class SSOAuth extends HttpServlet { ??? ??? static private ConcurrentMap accounts; ??? static private ConcurrentMap SSOIDs; ??? String cookiename="WangYuDesktopSSOID"; ??? String domainname; ??? ??? public void init(ServletConfig config) throws ServletException { ??????? super.init(config); ??????? domainname= config.getInitParameter("domainname"); ??????? cookiename = config.getInitParameter("cookiename"); ??????? SSOIDs = new ConcurrentHashMap(); ??????? accounts=new ConcurrentHashMap(); ??????? accounts.put("wangyu", "wangyu"); ??????? accounts.put("paul", "paul"); ??????? accounts.put("carol", "carol"); ??? } ??? protected void processRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { ??????? PrintWriter out = response.getWriter(); ??????? String action = request.getParameter("action"); ??????? String result="failed"; ??????? if (action==null) { ??????????? handlerFromLogin(request,response); ??????? } else if (action.equals("authcookie")){ ??????????? String myCookie = request.getParameter("cookiename"); ??????????? if (myCookie != null)?result = authCookie(myCookie); ??????????? out.print(result); ??????????? out.close(); ??????? } else if (action.equals("authuser")) { ???????????result=authNameAndPasswd(request,response); ??????????? out.print(result); ??????????? out.close(); ??????? }?else if (action.equals("logout")) { ??????????? String myCookie = request.getParameter("cookiename"); ??????????? logout(myCookie); ?? ?????????out.close(); ??????? } ??? } ..... } |
| ?private void handlerFromLogin(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { ??????? String username = request.getParameter("username"); ??????? String password = request.getParameter("password"); ??????? String pass = (String)accounts.get(username); ??????? if ((pass==null)||(!pass.equals(password))) ??????????? getServletContext().getRequestDispatcher("/failed.html").forward(request, response); ??????? else { ??????????? String gotoURL = request.getParameter("goto"); ??????????? String newID = createUID(); ??????????? SSOIDs.put(newID, username); ??????????? Cookie wangyu = new Cookie(cookiename, newID); ??????????? wangyu.setDomain(domainname); ??????????? wangyu.setMaxAge(60000); ??????????? wangyu.setValue(newID); ??????????? wangyu.setPath("/"); ??????????? response.addCookie(wangyu); ??????????? System.out.println("login success, goto back url:" + gotoURL); ??????????? if (gotoURL != null) { ??????????????? PrintWriter out = response.getWriter(); ???? ???????????response.sendRedirect(gotoURL); ??????????????? out.close(); ??????????? } ??????? }?? ??? } |
- Web應用中每一個需要安全保護的URL在訪問以前,都需要進行安全檢查,如果發現沒有登錄(沒有發現認證之后所帶的cookie),就重新定向到SSOAuth中的login.jsp進行登錄。
- 登錄成功后,系統會自動給你的瀏覽器設置cookie,證明你已經登錄過了。
- 當你再訪問這個應用的需要保護的URL的時候,系統還是要進行安全檢查的,但是這次系統能夠發現相應的cookie。
- 有了這個cookie,還不能證明你就一定有權限訪問。因為有可能你已經logout,或者cookie已經過期了,或者身份認證服務重起過,這些情況下,你的cookie都可能無效。應用系統拿到這個cookie,還需要調用身份認證的服務,來判斷cookie時候真的有效,以及當前的cookie對應的用戶是誰。
- 如果cookie效驗成功,就允許用戶訪問當前請求的資源。
- 在每個被訪問的資源中(JSP或Servlet)中都加入身份認證的服務,來獲得cookie,并且判斷當前用戶是否登錄過。不過這個笨方法沒有人會用:-)。
- 可以通過一個controller,將所有的功能都寫到一個servlet中,然后在URL映射的時候,映射到所有需要保護的URL集合中(例如*.jsp,/security/*等)。這個方法可以使用,不過,它的缺點是不能重用。在每個應用中都要部署一個相同的servlet。
- Filter是比較好的方法。符合Servlet2.3以上的J2EE容器就具有部署filter的功能。(Filter的使用可以參考JavaWolrd的文章http://www.javaworld.com/javaworld/jw-06-2001/jw-0622-filters.html)Filter是一個具有很好的模塊化,可重用的編程API,用在SSO正合適不過。本樣例就是使用一個filter來完成以上的功能。
| package SSO; import java.io.*; import java.net.*; import java.util.*; import java.text.*; import javax.servlet.*; import javax.servlet.http.*; import javax.servlet.*; import org.apache.commons.httpclient.*; import org.apache.commons.httpclient.methods.GetMethod; public class SSOFilter implements Filter { ??? private FilterConfig filterConfig = null; ??? private String cookieName="WangYuDesktopSSOID"; ??? private String SSOServiceURL=?"http://wangyu.prc.sun.com:8080/SSOAuth/SSOAuth"; ??? private String SSOLoginPage= "http://wangyu.prc.sun.com:8080/SSOAuth/login.jsp"; ??? ??? public void init(FilterConfig filterConfig) { ??????? this.filterConfig = filterConfig; ??????? if (filterConfig != null) { ??????????? if (debug) { ??????????????? log("SSOFilter:Initializing filter"); ??????????? } ??????? }??????? ??????? cookieName = filterConfig.getInitParameter("cookieName"); ??????? SSOServiceURL = filterConfig.getInitParameter("SSOServiceURL"); ??????? SSOLoginPage = filterConfig.getInitParameter("SSOLoginPage"); ??? }? ..... ..... } |
| public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException { ??????? if (debug) log("SSOFilter:doFilter()"); ??????? HttpServletRequest request = (HttpServletRequest) req; ??????? HttpServletResponse response = (HttpServletResponse) res; ??????? String result="failed"; ??????? String url = request.getRequestURL().toString(); ??????? String qstring = request.getQueryString(); ??????? if (qstring == null) qstring =""; ??????? //檢查http請求的head是否有需要的cookie ??????? String cookieValue =""; ??????? javax.servlet.http.Cookie[] diskCookies = request.getCookies(); ??????? if (diskCookies != null) { ??????????? for (int i = 0; i < diskCookies.length; i++) { ??????????????? if(diskCookies[i].getName().equals(cookieName)){ ??????????????????? cookieValue = diskCookies[i].getValue(); ??????????????????? //如果找到了相應的cookie則效驗其有效性 ??????????????????? result = SSOService(cookieValue); ??????????????????? if (debug) log("found cookies!"); ???????? ???????} ??????????? } ??????? } ??????? if (result.equals("failed")) { //效驗失敗或沒有找到cookie,則需要登錄 ??????????? response.sendRedirect(SSOLoginPage+"?goto="+url); ??????? } else if (qstring.indexOf("logout") > 1) {//logout服務 ??????????? if (debug) log("logout action!"); ??????????? logoutService(cookieValue); ??????????? response.sendRedirect(SSOLoginPage+"?goto="+url); ??????? }?else {//效驗成功 ??????????? request.setAttribute("SSOUser",result); ??????????? Throwable problem = null; ??????????? try { ??????????? ????chain.doFilter(req, res); ??????????? } catch(Throwable t) { ??????????????? problem = t; ??????????????? t.printStackTrace(); ??????????? }?????? ??????????? if (problem != null) { ??????????????? if (problem instanceof ServletException) throw (ServletException)problem; ??????????????? if (problem instanceof IOException) throw (IOException)problem; ??????????????? sendProcessingError(problem, res); ??????????? } ??????? }?? ??? } |
<filter-name>SSOFilter</filter-name>
<url-pattern>*.jsp</url-pattern>
</filter-mapping> 下面還有幾個主要的函數需要說明:
| ??? private String SSOService(String cookievalue) throws IOException { ??????? String authAction = "?action=authcookie&cookiename="; ??????? HttpClient httpclient = new HttpClient(); ??????? GetMethod httpget = new GetMethod(SSOServiceURL+authAction+cookievalue); ??????? try {? ??????????? httpclient.executeMethod(httpget); ??????????? String result = httpget.getResponseBodyAsString(); ??????????? return result; ??????? } finally { ??????????? httpget.releaseConnection(); ??????? } ??? } ??? ??? private void logoutService(String cookievalue) throws IOException { ??????? String authAction = "?action=logout&cookiename="; ??????? HttpClient httpclient = new HttpClient(); ??????? GetMethod httpget = new GetMethod(SSOServiceURL+authAction+cookievalue); ??????? try { ??????????? httpclient.executeMethod(httpget); ??????????? httpget.getResponseBodyAsString(); ??????? } finally { ??????????? httpget.releaseConnection(); ??????? } ??? } |
- cookie的長度和復雜度
在本方案中,cookie是有一個固定的字符串(我的姓名)加上當前的時間戳。這樣的cookie很容易被偽造和猜測。懷有惡意的用戶如果猜測到合法的cookie就可以被當作已經登錄的用戶,任意訪問權限范圍內的資源 - cookie的效驗和保護
在本方案中,雖然密碼只要傳輸一次就夠了,可cookie在網絡中是經常傳來傳去。一些網絡探測工具(如sniff, snoop,tcpdump等)可以很容易捕獲到cookie的數值。在本方案中,并沒有考慮cookie在傳輸時候的保護。另外對cookie的效驗也過于簡單,并不去檢查發送cookie的來源究竟是不是cookie最初的擁有者,也就是說無法區分正常的用戶和仿造cookie的用戶。 - 當其中一個應用的安全性不好,其他所有的應用都會受到安全威脅
因為有SSO,所以當某個處于 SSO的應用被黒客攻破,那么很容易攻破其他處于同一個SSO保護的應用。
- 當前所提供的登錄認證模式只有一種:用戶名和密碼,而且為了簡單,將用戶名和密碼放在內存當中。事實上,用戶身份信息的來源應該是多種多樣的,可以是來自數據庫中,LDAP中,甚至于來自操作系統自身的用戶列表。還有很多其他的認證模式都是商務應用不可缺少的,因此SSO的解決方案應該包括各種認證的模式,包括數字證書,Radius, SafeWord ,MemberShip,SecurID等多種方式。最為靈活的方式應該允許可插入的JAAS框架來擴展身份認證的接口
- 我們編寫的Filter只能用于J2EE的應用,而對于大量非Java的Web應用,卻無法提供SSO服務。
- 在將Filter應用到Web應用的時候,需要對容器上的每一個應用都要做相應的修改,重新部署。而更加流行的做法是Agent機制:為每一個應用服務器安裝一個agent,就可以將SSO功能應用到這個應用服務器中的所有應用。
- 當前的方案不能支持分別位于不同domain的Web應用進行SSO。這是因為瀏覽器在訪問Web服務器的時候,僅僅會帶上和當前web服務器具有相同domain名稱的那些cookie。要提供跨域的SSO的解決方案有很多其他的方法,在這里就不多說了。Sun的Access Manager就具有跨域的SSO的功能。
- 另外,Filter的性能問題也是需要重視的方面。因為Filter會截獲每一個符合URL映射規則的請求,獲得cookie,驗證其有效性。這一系列任務是比較消耗資源的,特別是驗證cookie有效性是一個遠程的http的調用,來訪問SSOAuth的認證服務,有一定的延時。因此在性能上需要做進一步的提高。例如在本樣例中,如果將URL映射從“.jsp”改成“/*”,也就是說filter對所有的請求都起作用,整個應用會變得非常慢。這是因為,頁面當中包含了各種靜態元素如gif圖片,css樣式文件,和其他html靜態頁面,這些頁面的訪問都要通過filter去驗證。而事實上,這些靜態元素沒有什么安全上的需求,應該在filter中進行判斷,不去效驗這些請求,性能會好很多。另外,如果在filter中加上一定的cache,而不需要每一個cookie效驗請求都去遠端的身份認證服務中執行,性能也能大幅度提高。
- 另外系統還需要很多其他的服務,如在內存中定時刪除無用的cookie映射等等,都是一個嚴肅的解決方案需要考慮的問題。
a) WEB-SSO的身份認證應用應該正在運行,因為我們在桌面SSO當中需要用到統一的認證服務
b) 當前桌面需要運行Mozilla或Netscape瀏覽器,因為我們將ticket保存到mozilla的cookie文件中
c) 必須在JDK1.4以上運行。(WEB-SSO需要JDK1.5以上)
a) SSOServiceURL要指向WebSSO部署的身份認證的URL
b) SSOLoginPage要指向WebSSO部署的身份認證的登錄頁面URL
c) cookiefilepath要指向當前用戶的mozilla所存放cookie的文件
a) 如果是在unix下,各個jar文件需要用“:”來隔開,而不是“;”
b) java 運行程序需要放置在當前運行的路徑下,否則需要加上java的路徑全名。
| ??? DesktopSSO { ?? desktopsso.share.PasswordLoginModule required; ?? desktopsso.share.DesktopSSOLoginModule required; }; |
| ?public class DesktopSSOLoginModule implements LoginModule { ?? .......... ?? private String SSOServiceURL = ""; ?? private String SSOLoginPage = ""; ?? private static String cookiefilepath = "";?? ?? ......... |
| SSOServiceURL=http://wangyu.prc.sun.com:8080/SSOAuth/SSOAuth SSOLoginPage=http://wangyu.prc.sun.com:8080/SSOAuth/login.jsp cookiefilepath=C://Documents and Settings//yw137672//Application Data//Mozilla//Profiles//default//hog6z1ji.slt//cookies.txt |
| ??? public boolean login() throws LoginException{ ??????? try { ??????????? if (Cookielogin()) return true; ??????? } catch (IOException ex) { ??????????? ex.printStackTrace(); ??????? } ????? if (passwordlogin()) return true; ????? throw new FailedLoginException(); ?} |
| ??? public boolean Cookielogin() throws LoginException,IOException { ????? String?cookieValue=""; ????? int cookieIndex =foundCookie(); ????? if (cookieIndex<0) ??????????? return false; ????? else ??????????? cookieValue = getCookieValue(cookieIndex); ???? username = cookieAuth(cookieValue); ???? if (! username.equals("failed")) { ???????? loginSuccess =?true; ???????? return true; ???? } ???? return false; ?} |
| ?? public boolean passwordlogin() throws LoginException { ??? // ??? // Since we need input from a user, we need a callback handler ??? if (callbackHandler == null) { ?????? throw new LoginException("No CallbackHandler defined"); ??? } ??? Callback[] callbacks = new Callback[2]; ??? callbacks[0] = new NameCallback("Username"); ??? callbacks[1] = new PasswordCallback("Password", false); ??? // ??? // Call the callback handler to get the username and password ??? try { ????? callbackHandler.handle(callbacks); ????? username = ((NameCallback)callbacks[0]).getName(); ????? char[] temp = ((PasswordCallback)callbacks[1]).getPassword(); ????? password = new char[temp.length]; ????? System.arraycopy(temp, 0, password, 0, temp.length); ????? ((PasswordCallback)callbacks[1]).clearPassword(); ??? } catch (IOException ioe) { ????? throw new LoginException(ioe.toString()); ??? } catch (UnsupportedCallbackException uce) { ????? throw new LoginException(uce.toString()); ??? } ??? ??? System.out.println(); ??? String authresult =""; ??? try { ??????? authresult = userAuth(username, password); ??? } catch (IOException ex) { ??????? ex.printStackTrace(); ??? } ??? if (! authresult.equals("failed")) { ??????? loginSuccess= true; ??????? clearPassword(); ??????? try { ??????????? updateCookie(authresult); ??????? } catch (IOException ex) { ??????????? ex.printStackTrace(); ??????? } ??????? return true; ??? } ?? ? ??? loginSuccess = false; ??? username = null; ??? clearPassword(); ??? System.out.println( "Login: PasswordLoginModule FAIL" ); ??? throw new FailedLoginException(); ?} ? |
| ??????? private String cookieAuth(String cookievalue) throws IOException{ ??????? String result = "failed"; ??????? ??????? HttpClient httpclient = new HttpClient();?????? ??????? GetMethod httpget = new GetMethod(SSOServiceURL+Action1+cookievalue); ??? ??????? try { ??????????? httpclient.executeMethod(httpget); ??????????? result = httpget.getResponseBodyAsString(); ??????? } finally { ??????????? httpget.releaseConnection(); ??????? } ??????? return result; ??? } private String userAuth(String username, char[] password) throws IOException{ ??????? String result = "failed"; ??????? String passwd= new String(password); ??????? HttpClient httpclient = new HttpClient();?????? ??????? GetMethod httpget = new GetMethod(SSOServiceURL+Action2+username+"&password="+passwd); ??????? passwd = null; ??? ??????? try { ??????????? httpclient.executeMethod(httpget); ??????????? result = httpget.getResponseBodyAsString(); ??????? } finally { ??????????? httpget.releaseConnection(); ??????? } ??????? return result; ??????? ??? } |
- OpenSSO 的網站: https://opensso.dev.java.net
- 在java應用中使用掩碼的密碼提示:http://java.sun.com/developer/technicalArticles/Security/pwordmask/
- Kerberos的資源網站:http://web.mit.edu/kerberos/
- Sun的Java&Kerberos教程:http://java.sun.com/j2se/1.5.0/docs/guide/security/jgss/tutorials/index.html
- Apache社區提供的HttpClient工具包網址:http://jakarta.apache.org/commons/index.html)
- Filter的使用教程文章:http://www.javaworld.com/javaworld/jw-06-2001/jw-0622-filters.html
- 我的博客http://yuwang881.blog.sohu.com/3184816.html
轉載于:https://www.cnblogs.com/tonykan/archive/2013/01/18/2866778.html
創作挑戰賽新人創作獎勵來咯,堅持創作打卡瓜分現金大獎總結
以上是生活随笔為你收集整理的编写你自己的单点登录(SSO)服务(转)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: The Distinguish of t
- 下一篇: iOS Xcode4.2代码管理工具的使