单点登录(java-Discuz)

首先,需要明确自身需要的需求,本帖子是java 语言开发的网站和Discuz论坛实现的单点登录。

一 Discuz论坛搭建步骤

1:服务器环境配置

     服务器要支持php语言+支持mysql 5.0以上的数据库 + Apache服务器

     本人在本地通过搭建XAMPP Control Panel部署的Discuz,方便调试。

     最终的服务器是配置在阿里云的ACE.这个根据个人的服务器,不做赘述。

2:安装Discuz

(1)下载Discuz源码包(Discuz论坛去下载,本人下载的是Discuz3.2 GBK版本)

(2)解压源码包Discuz_X3.2_SC_GBK.zip(这里以Discuz X3/2为例)有三个文件,把 其中的upload(27个文件)文件夹里面的内容放 在网站的根目录下。

注意:

是upload文件夹下面的所有内容放入网站根目录下,如果把upload整个文件夹放在网站根目录访问的地址就是( http://localhost/upload ) 

(3)访问域名地址就开始安装,按照步骤完成。

        配置数据库信息:

        提前创建好mysql数据库,用户/密码,这里进行配置。同时给设置一个管理员账号和密码:点击下一步


       注意:一定是mysql 5.0以上的数据库版本

       如果没有问题的话就安装成功了。它会自动创建对应的数据库表等相关信息。

       然后访问域名就直接可以访问论坛了。

3:Discuz的简单配置介绍

(1)前台登录及管理员功能 (这里不一一介绍)

(2)后台管理心中(这里不一一介绍)

二 Discuz论坛与java无缝对接(单点登录)

1: Discuz和java对接的原理

Discuz和java对接需要一个中间件,它就是Ucenter。Comsenz(康盛)的UCenter当前在国内的单点登录领域占据绝对份额,其完整的产品线令UCenter成为了账号集成方面事实上的标准。基于UCenter,可以将Comsenz旗下的Discuz!(社区论坛系统)、SupeSite(门户CMS系统)、X-Space(博客系统)从用户资源层面进行无缝整合,使得账号实现统一管理,在任何一个系统中进行注册、登录、注销等操作时,该账号在其他系统中的会话状态也将同步更新,最终实现一号通的单点登录模式。

使用ucenter进行同步操作,主要依托于ucenter  server 和ucenter client之间的api接口进行通讯。要使得通讯成为可能,首先要通过ucenter管理所用应用(通过配置使得当前应用和ucenter产生联系,也就是通讯成功)。其他应用通过挂接到ucenter的接口上,从而使得某些数据可以进行同步操作。

2 : Discuz和Ucenter通道的配置

(1)Ucenter用户管理中心的简单介绍

Discuz和Ucenter都属于Comsenz(康盛)公司的产品,所以Discuz和Ucenter之间的通信通道非常容易建立。

Discuz中附带的有Ucenter应用程序例如:

之前Discuz论坛安装之后访问地址是http://localhost/ upload,那么Ucenter的访问地址是http://localhost/ upload/uc_server,登录界面如下:

输入之前的管理员密码即可登录成功!

后台管理页面如下

(2)Ucenter于Discuz通道建立。

我们如果用的是Discuz自带的Ucenter,他与Discuz就已经默认是互通的,我可以直接来用也可以重新配置。建立成功如下:

注释:这里有系统默认的我们就直接用,下面Ucenter于java建立通道的时候会详细说明怎么建立通道,建立通信通道的方式都是一样的。

3 : Java和Ucenter通道的配置

(1)Ucenter端添java web网站通道

登录成功之后

应用管理》添加应用

标红的为主要填写的。其他的可以不填写。

安装方式:自定义

应用类型:其他

应用名称:JAVA项目名称,可随意

应用主URL:JAVA项目的访问地址,例如:http://localhost:8080/javaTest

通信密钥:任意字符,例如:123456

应用接口文件名称: uc.php ,这个不要改,前面也不要加/api/,UCenter在与其通信时会自动转换为: http://localhost:8080/javaTest/api/uc.php

是否开启同步登录:是

是否接受通知:是

以上的配置项必填,其他保持为空即可 。

点击“提交”,保存成功后, 记下该 JAVA 应用的 APPID ,下面的配置中会需要。

此时返回应用列表,UCenter会自动进行通信验证,不出意外肯定会出现“ 通信失败 ”的字样,先不必管它

因为java端也需要一些配置

(2)java web网站端的配置

1、从“ discuz-ucenter-api-for-java ”的开源项目网站中下载相关的API文件,也可以从这里下载:JAVA_UCenter.zip

这里给一个比较好的下载地址,包括源码,及jar包文件:

http://code.google.com/p/discuz-ucenter-api-for-java

2、将JAVA_UCenter.zip解压、覆盖到你的JAVA项目中,其中的新文件有:

src/config.properties:本地的JAVA项目与UCenter的接口配置文件( 需要根据实际环境进行配置 )

src/api/ucenter/Base64.java

src/api/ucenter/Client.java: 将常用的 UCenter 操作封装成的客户端对象,我们在项目中主要用它来与 UCenter 打交道

src/api/ucenter/PHPFunctions.java

src/api/ucenter/UC.java: 本地的 JAVA 项目用来接收 UCenter 同步命令的 Servlet 接口,其访问地址必须为: /api/uc.php


src/api/ucenter/XMLHelper.java

WebRoot/WEB-INF/web.xml: 主要就是将 src/api/ucenter/UC.java 定义为 Servlet

注意:

(1) src/config.properties中的代码如下:

# ================================================

# Ucenter API for JAVA

# ================================================

UC_API = http://localhost:9201

UC_IP = 

UC_KEY = 123456

UC_APPID = 2

UC_CONNECT = 

其中,UC_API表示UCenter的访问地址,UC_KEY就是在UCenter中添加新应用时定义的通信密钥,UC_APPID表示新应用的APPID,就是刚才记下的那个。

除这三项外,其他的配置保持为空即可。

(2) WebRoot/WEB-INF/web.xml中的代码如下:

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.4" 
xmlns="http://java.sun.com/xml/ns/j2ee" 
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee 
http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
<servlet>
<display-name>同步UC Server发出的操作指令</display-name>
<servlet-name>UC</servlet-name>
<servlet-class>api.ucenter.UC</servlet-class>
<load-on-startup>2</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>UC</servlet-name>
<url-pattern>/api/uc.php</url-pattern>
</servlet-mapping>
</web-app>

别的无所谓,只需注意这一行:<url-pattern>/api/uc.php</url-pattern>

3、JAVA端配置完成,在eclipse中启动调试模式

4、再次返回到UCenter的控制台,点击应用管理,此时应该会看到绿色的“ 通信成功 ”字样

到这里Discuz和Ucenter之间的通信通道与java和Ucenter之间的通信通道都已经建立成功,下面就可以在2.3.4看具体如何开发

2.3.4  Java端具体代码实现

(1)java网站web 网站注册同步Discuz论坛注册

1> 步骤说明:

注册java网站web网站 > 调用Client里面的UC注册Discuz论坛的接口(uc_user_register) > 判断注册结果


2> 具体代码实现(仅供参考)

//java网站网站的注册
boolean falg = tempUserService.insert(tempUserVO);
//同步论坛的注册
if (falg){
Client uc = new Client();
String $returns = uc.uc_user_register(tempUserVO.getUsername(), tempUserVO.getPass(), tempUserVO.getEamil());
int $uid = Integer. parseInt ($returns); 
if ($uid <= 0) { 
if ($uid == -1) { 
System. out .print("用户名不合法"); 
} else if ($uid == -2) { 
System. out .print("包含要允许注册的词语"); 
} else if ($uid == -3) { 
System. out .print("用户名已经存在"); 
} else if ($uid == -4) { 
System. out .print("Email 格式有误"); 
} else if ($uid == -5) { 
System. out .print("Email 不允许注册"); 
} else if ($uid == -6) { 
System. out .print("该 Email 已经被注册"); 
} else { 
System. out .print("未定义"); 
}
}
System. out .println("id:"+$uid);
System. out .println("添加成功!");
} else {
System. out .println("添加失败!");
}

(2)java网站web 网站登录,同步Discuz论坛登录

1> 步骤说明:

登录民信web网站 > 调用Client里面的UC登录接口获取用户信息 > 接受返回的用户XML信息并解析 > 调用Client登录DIscuz论坛的登录接口 > 接受返回过来的js代码 > 输出到java网站web网站 > 跳转到论坛地址(此时已经你会发现用户已经登录)

2> 具体代码实现(仅供参考)

//网站的登录
TempUser tempUser =  tempUserService.queryByName(tempUserVO.getUsername());
System. out .println("结果:"+tempUser.getAddress()+"  =="+tempUser.getEamil());
request.getSession().setAttribute("tempUser", tempUser);
// Discuz 论坛的登录
Client uc = new Client();
String $result = uc.uc_user_login(tempUser.getUsername(), tempUser.getPass());
String $ucsynlogin = "";
LinkedList<String> rs = XMLHelper. uc_unserialize ($result);
if (rs.size()>0){
int $uid = Integer. parseInt (rs.get(0));
String $username = rs.get(1);
String $password = rs.get(2);
String $email = rs.get(3);
if ($uid > 0) {
$ucsynlogin = uc.uc_user_synlogin($uid);
} else if ($uid == -1) {
System. out .println("用户不存在,或者被删除");
} else if ($uid == -2) {
System. out .println("密码错");
} else {
System. out .println("未定义");
}
//设置本地 Discuz 登录的cookie信息,cookie存活时间
//直接访问论坛就有用户信息了
try {
response.setCharacterEncoding("UTF-8");
//同步Cookie信息
//                response.addHeader("P3P"," CP=\"CURa ADMa DEVa PSAo PSDo OUR BUS UNI PUR INT DEM STA PRE COM NAV OTC NOI DSP COR\"");
//              Cookie auth = new Cookie(" auth ", uc.uc_authcode($password+"\t"+$ uid , "ENCODE"));
//              auth.setMaxAge(31536000);
//              //auth.setDomain(" localhost ");//设置本地cookie
//              response.addCookie( auth );
//              Cookie user = new Cookie("Discuz_loginuser", $ username );
//              response.addCookie(user);
//把返回过来的 js 文件直接输出在页面上,然后跳转到论坛网站首页就已经是登录之后的
PrintWriter out = response.getWriter();
out.write($ucsynlogin);
out.flush();         
  } catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} else {
System. out .println("Login failed");
}
//    System.out.println("登录的结果:"+$result);
(3)java网站退出,论坛同步退出
1> 步骤说明:
java网站网站注销 > 调用Client里面的uc_user_synlogout同时注销论坛用户
2> 具体代码实现(仅供参考)
Client uc = new Client();
String $ucsynlogout = uc.uc_user_synlogout();
System.out.println("退出成功"+$ucsynlogout);
return $ucsynlogout;

(4)java网站修改用户信息同步论坛资料信息

1> 步骤说明:

修改用户保存 > 获取之前论坛用户信息 > 调用接口进行修改uc_user_edit

2> 具体代码实现(仅供参考)

Client uc = new Client();
uc.uc_user_edit($username, $oldpw, $newpw, $email, $ignoreoldpw, $questionid, $answer)

主要代码片段

(5)论坛登录同步登录java网站web


我们来看看discuz那边登录了,java这边怎么实现同步登录。在uc.java里有一个doanswer的方法。所有同步操作都在这个方法里完成。其逻辑是这样的,uc这个类继承了HttpServlet。当ucenter发出通知时,会访问这个servlet.并执行他的doGet()方法。doGet()调用doAnswer().


String $code = request.getParameter("code");会传一个code进来(加密了的)。然后调用$code = new Client().uc_authcode($code, "DECODE");进行解密操作。新建一个Map,Map<String,String> $get = new HashMap<String, String>();然后根据解密出来的code的内容封装这个Map.调用这个方法,parse_str($code, $get); 经过以上操作后,所有同步操作需要的信息都在map里,如action。action代表的是什么操作,如同步登录,同步退出,修改密码等。还有其他信息,如username.登录的用户名。改密码时还会传新密码过来。所有的这些都包含在这个叫$get的Map里。然后我们就可以进行我们自己的逻辑操作了。(以上逻辑不想理解没关系,你只要知道当你在discuz登录之后,uc.java的doAnswer()方法就会被调用,你所需要的一切信息都包含在$get这个Map里)


操作到这里,discuz那边登录了,java这边完全没反应啊!!doAnswer方法没有被调用!是的,你还需要到discuz后台做一些设置。管理中心->站长->ucenter设置。把ucenter连接方式改为接口方式。然后,你登录discuz时,doAnswer()方法就会被调用了。(其他操作如退出也一样)


注意乱码问题:

Discuz登录同步java网站web网站登录,中文乱码问题

到这里,我们又遇到了中文的问题了,discuz用中文登录的时候不行啊!是的,这确实是个比较麻烦的问题。笔者也弄了好久。原因是这样的,上面提到过,传进来的$code是需要uc_authcode()方法进行解密的,当中文的时候解密出来的$code是"".解决方法如下:在Client.java的uc_authcode()方法里找到这句String $result = $result1.substring(0, $result1.length());并把它改成:

 String $result= $result1.toString();
         try{
              $result=new String($result.getBytes("iso-8859-1"),"GBK");
              }catch (Exception e) {
              $result = $result1.substring(0, $result1.length());
              // TODO: handle exception
              }

这样还不行,解密出来的code中文部分仍是乱码。所以在uc.java的$code = new Client().uc_authcode($code, "DECODE");这句下面还要加上$code = new String($code.getBytes("GBK"),"UTF-8");可能有人会说直接转成utf-8不行吗?为什么要先转成gbk呢?如果直接转成utf-8的话,解密出来的$code是""。到此中文的问题就得到解决了。笔者下载的是GBK版本的discuz,其他情况自己改一下编码应该没问题。


你可能感兴趣的:(单点登录(java-Discuz))