转载http://blog.sina.com.cn/s/blog_8cb05a6b0101a904.html
一、前提:
在Openfire 中,客户端登出的状态分为正常离线和断线,正常离线会发送presence消息通知。非正常离线的时候,openfire自带的心跳包会检测客户端的响 应,如果客户端长时间无响应,Openfire无法接收到客户端的presence消息的时候,mina框架会在OF的 ConnectionHandler的sessionClosed()方法中进行后续的处理。二、离线检测:
MINA框架本身提供了idle检测功能,这项功能可检测客户端建立的TCP/IP连接,却不发送任何消息的状况。
在 ClientConnectionHandler的sessionidle方法中判断当前的idle次数大于1时将关闭客户端的连接。如果设置了idle time之后在这个idle的检测发生在达到一半时间和达到指定的时间,每次检测都会将idle的次数加1,也就是我们一旦设定了这个时间长,mina框 架就会在这个时长的一半时间内,客户端仍未发送消息时触发一次sessionidle事件,然后在到达指定的时长,客户端仍未发送消息时再触发一次。
触发做法:第一次触发sessionidle时发送一次ping,强迫客户端进行响应。
三、业务需求:
既然明白了OF的离线做法,我们何不使用包的过滤机制,来过滤presnece的消息通知包。那么就让我们来实践下这种机制:
1、创建包过滤器,使用包过滤器。可以将presence的状态拦下,之后对其做处理,代码如下:
public class PresenceInterceptorimplements PacketInterceptor {
@Override
publicvoid interceptPacket(Packet packet, Session session, boolean incoming,
booleanprocessed) throws PacketRejectedException {
if(!processed && packet instanceofPresence && incoming){
PresencemyPresence =(Presence)packet;
System.out.println("presenceunavlilate"+packet.toXML());
}
}
}
我们使用了Spark测试,当用户下线的时候,发送了一次presence,当用户上线的时候,发送一次presence,结果呈现如图1-1所示:
图1-1 所示 Presence状态处理
按照以上方式,我们可以为其匹配些业务触发器。现在让我来介绍下各个触发器的实现:
1、创建触发器:
package org.jivesoftware.openfire.tigger;
import java.util.ArrayList;
import java.util.List;
public abstract class Tigger{
private static ClassLoader loader=null;
public static TiggerManager tiggerMangerGlobal=null;
//创建触发器
List<TiggerManager> tiggerList = newArrayList<TiggerManager>();
boolean changed = false;
public Tigger(){
super();
}
public void addTigger(String tiggerManagerModule){
Class<?> isTiggerManageClass=null;;
ObjecttiggerManager=null;
loader =Thread.currentThread().getContextClassLoader();
try {
isTiggerManageClass = loader.loadClass(tiggerManagerModule);
tiggerManager=(TiggerManager)isTiggerManageClass.newInstance();
System.out.println("实例化对象="+tiggerManager);
tiggerMangerGlobal=(TiggerManager) tiggerManager;
if(tiggerManager == null){
System.out.println("触发器为空");
}synchronized(this){
//如果触发器和容器中的触发器不相等,则
if(!tiggerList.contains(tiggerManager)){
tiggerList.add((TiggerManager) tiggerManager);
}
}
} catch(ClassNotFoundException e) {
e.printStackTrace();
} catch(InstantiationException e) {
e.printStackTrace();
} catch(IllegalAccessException e) {
e.printStackTrace();
}
}
protected void clearChanged(){
changed=false;
}
protected int countTigger(){
returntiggerList.size();
}
protected synchronized void deleteTigger(){
tiggerList.clear();
}
protected boolean hasChanged(){
return changed;
}
protected void setChanged(){
changed=true;
}
public void notifyTigger(Object data) {
int size = 0;
TiggerManager[] arrays =null;
synchronized (this) {
if (hasChanged()) {
clearChanged();
size =tiggerList.size();
arrays = newTiggerManager[size];
tiggerList.toArray(arrays);
}
}
if (arrays != null) {
for (TiggerManager tigger : arrays) {
tigger.update(this,data);
}
}
}
public void notifyTiggerKey(String key,String value) {
for(int i=0;i<tiggerList.size(); i++){
TiggerManager tigger = tiggerList.get(i);
tigger.update(key,value);
}
}
public void notifyTigger() {
for(int i=0;i<tiggerList.size(); i++){
TiggerManager tigger = tiggerList.get(i);
tigger.update();
}
}
public void notifyTiggerKey(Object obj) {
for(int i=0;i<tiggerList.size(); i++){
TiggerManager tigger = tiggerList.get(i);
tigger.update(obj);
}
}
}
2、
2、触发器管理类:
package org.jivesoftware.openfire.tigger;
public interface TiggerManager {
boolean update(Tigger tigger, Object data);
boolean update(String key,String value);
boolean update();
boolean update(Object obj);
}
3、基础触发器:
package org.jivesoftware.openfire.tigger;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public abstract class TiggerSolutionManager implementsTiggerManager {
private staticClassLoader loader=null;
private staticTiggerManager initlize(String classResouceTigger,String tiggerMethodModule){
TiggerManager isTiggerManage=null;
loader = Thread.currentThread().getContextClassLoader();
Object tigger = null;
try {
Class<?> tiggerCls= loader.loadClass(classResouceTigger);
tigger=(Tigger)tiggerCls.newInstance();
Method method =tiggerCls.getDeclaredMethod("addTigger",new Class[]{String.class});
method.setAccessible(true);
String[] argments = newString[1];
argments[0]=tiggerMethodModule ;
method.invoke(tigger,argments);
isTiggerManage=Tigger.tiggerMangerGlobal;
} catch (SecurityException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
return isTiggerManage;
}
public static voidcreateTiggerMethodKey(String classResouceTigger,StringtiggerMethodModule,String key,String value){
TiggerManager isTiggerSolution=null;
try{
if(isTiggerSolution ==null){
isTiggerSolution= initlize(classResouceTigger,tiggerMethodModule);
//即时响应
isTiggerSolution.update(key,value);
}
}catch(Exception e){
e.printStackTrace();
}
}
public static voidcreateTiggerMethod(String classResouceTigger,String tiggerMethodModule){
TiggerManager isTiggerSolution=null;
try{
if(isTiggerSolution ==null){
isTiggerSolution=initlize(classResouceTigger,tiggerMethodModule);
//即时响应
isTiggerSolution.update();
}
}catch(Exception e){
e.printStackTrace();
}
}
public static voidcreateTiggerMethod(String classResouceTigger,String tiggerMethodModule,Stringvalue){
TiggerManager isTiggerSolution=null;
try{
if(isTiggerSolution ==null){
isTiggerSolution=initlize(classResouceTigger,tiggerMethodModule);
//即时响应
isTiggerSolution.update(value);
}
}catch(Exception e){
e.printStackTrace();
}
}
public abstract booleanupdate(Tigger tigger, Object data);
public abstract booleanupdate(String key, String value);
public abstract booleanupdate();
}
4、触发器响应:
package org.jivesoftware.openfire.tigger;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public abstract class TiggerSolutionManager implementsTiggerManager {
private staticClassLoader loader=null;
private staticTiggerManager initlize(String classResouceTigger,String tiggerMethodModule){
TiggerManager isTiggerManage=null;
loader = Thread.currentThread().getContextClassLoader();
Object tigger = null;
try {
Class<?> tiggerCls= loader.loadClass(classResouceTigger);
tigger=(Tigger)tiggerCls.newInstance();
Method method =tiggerCls.getDeclaredMethod("addTigger",new Class[]{String.class});
method.setAccessible(true);
String[] argments = newString[1];
argments[0]=tiggerMethodModule ;
method.invoke(tigger,argments);
isTiggerManage=Tigger.tiggerMangerGlobal;
} catch (SecurityException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
return isTiggerManage;
}
public static voidcreateTiggerMethodKey(String classResouceTigger,StringtiggerMethodModule,String key,String value){
TiggerManager isTiggerSolution=null;
try{
if(isTiggerSolution ==null){
isTiggerSolution=initlize(classResouceTigger,tiggerMethodModule);
//即时响应
isTiggerSolution.update(key,value);
}
}catch(Exception e){
e.printStackTrace();
}
}
public static voidcreateTiggerMethod(String classResouceTigger,String tiggerMethodModule){
TiggerManager isTiggerSolution=null;
try{
if(isTiggerSolution ==null){
isTiggerSolution=initlize(classResouceTigger,tiggerMethodModule);
//即时响应
isTiggerSolution.update();
}
}catch(Exception e){
e.printStackTrace();
}
}
public static voidcreateTiggerMethod(String classResouceTigger,String tiggerMethodModule,Stringvalue){
TiggerManager isTiggerSolution=null;
try{
if(isTiggerSolution ==null){
isTiggerSolution= initlize(classResouceTigger,tiggerMethodModule);
//即时响应
isTiggerSolution.update(value);
}
}catch(Exception e){
e.printStackTrace();
}
}
public abstract booleanupdate(Tigger tigger, Object data);
public abstract booleanupdate(String key, String value);
public abstract booleanupdate();
}
好了,以上是触发器的功能,现在我们可以将这些触发器匹配各自的业务:
1、RosterTigger:用于匹配当好友下线或在线时的触发事件
2、TranslateTigger:用于匹配发送文件时,对方下线时的触发事件
注意:
Openfire中判断当前用户是否在线,则使用以下方式:
User user = XMPPServer.getInstance().getUserManager().getUser(route.bar seID());
XMPPServer.getInstance().getPresenceManager().isAvailable(user)