Android XMPP通讯自定义Packet&Provider

摘要

在xmpp通信过程中,asmack中提供的Packet组件是IQ,Message,Presence三种: IQ用于查询 Message用于消息传递 Presence用于状态交互 他们都是Packet的子类,实质是用于将消息封装成响应的xml格式来进行数据交换,都有着良好的可扩展性。

简介

我们以开源项目androidpn为例:

androidpn (Android Push Notification)是一个基于XMPP协议的java开源Android push notification实现。它包含了完整的客户端和服务器端。

androidpn包括Server端和Client端,项目名称是androidpn-server和androidpn-client。

事实上,androidpn-server可以支持app运行于iOS,UWP,Windows,Linux等平台,不仅限于android,因此,希望项目的名称改为XPN(Xmpp Push Notification)似乎更加符合其实际场景,我们以后涉及到Android Push Notification统称为XPN。

XNP目前状态

项目自2014年1月就已经停止更新了,此外,asmack项目也停止更新了,作者建议使用openfire官方的smack4.0,不过这样的话引入的jar会特别多,特别大。当然,我们下载到了asmack8.10.0比较新的稳定版本,可以完全用于学习和扩展。

项目相关下载站点

asmack-github.com - asmack项目地址

asmack-asmack.freakempire.de - asmack镜像地址

androidpn(XPN)-github.com - androidpn下载地址

一.关于Packet数据包

Packet是IQ,Message,Presence的父类,用于实现消息组件类型。

消息语义学message

message是一种基本推送消息方法,它不要求响应。主要用于IM、groupChat、alert和notification之类的应用中。
主要 属性如下:

type属性,它主要有5种类型:
        normal:类似于email,主要特点是不要求响应;
        chat:类似于qq里的好友即时聊天,主要特点是实时通讯;
        groupchat:类似于聊天室里的群聊;
        headline:用于发送alert和notification;
        error:如果发送message出错,发现错误的实体会用这个类别来通知发送者出错了;

to属性:标识消息的接收方。

from属性:指发送方的名字或标示。为防止地址外泄,这个地址通常由发送者的server填写,而不是发送者。
载荷(payload):例如body,subject

<message to="lily@jabber.org/contact" type="chat" > <body> 你好,在忙吗</body> </message>

出席信息语义学presence

presence用来表明用户的状态,如:online、away、dnd(请勿打扰)等。当改变自己的状态时,就会在stream的上下文中插入一个Presence元素,来表明自身的状态。要想接受presence消息,必须经过一个叫做presence subscription的授权过程。
属性:

type属性,非必须。有以下类别
    subscribe:订阅其他用户的状态
    probe:请求获取其他用户的状态
    unavailable:不可用,离线(offline)状态

to属性:标识消息的接收方。

from属性:指发送方的名字或标示。

载荷(payload):
    show:
    chat:聊天中
    away:暂时离开
    xa:eXtend Away,长时间离开
    dnd:勿打扰
status:格式自由,可阅读的文本。也叫做rich presence或者extended presence,常用来表示用户当前心情,活动,听的歌曲,看的视频,所在的聊天室,访问的网页,玩的游戏等等。
priority:范围-128~127。高优先级的resource能接受发送到bare JID的消息,低优先级的resource不能。优先级为
<presence from="alice@wonderland.lit/pda">
  <show>xa</show>
  <status>down the rabbit hole!</status>
</presence>

IQ语义学

一种请求/响应机制,从一个实体从发送请求,另外一个实体接受请求,并进行响应。例如,client在stream的上下文中插入一个元素,向Server请求得到自己的好友列表,Server返回一个,里面是请求的结果。
主要的属性是type。包括:
    Get :获取当前域值。类似于http get方法。
    Set :设置或替换get查询的值。类似于http put方法。
    Result :说明成功的响应了先前的查询。类似于http状态码200。
    Error: 查询和响应中出现的错误。
<iq from="alice@wonderland.lit/pda" 
    id="rr82a1z7"
    to="alice@wonderland.lit" 
    type="get">
  <query xmlns="jabber:iq:roster"/>
</iq>

二.自定义Packet

由于服务器和客户端使用的Packet不同,但是他们交互的数据格式都是xml,因此,这个过程我们理解xml实现过程即可。

1.定义Packet封装对象

由于asmack标签解析的限制,我们不能自定义解析,除非修改源码,这里出于简单,这里只能继承现有标签之一。

我么按照项目代码NotificationIQ为例,这里没有继承Packet,而是继承了IQ

import org.jivesoftware.smack.packet.IQ; /** * This class represents a notifcatin IQ packet. * * @author Sehwan Noh (devnoh@gmail.com) */ public class NotificationIQ extends IQ { private String id; private String apiKey; private String title; private String message; private String uri; public NotificationIQ() { } @Override public String getChildElementXML() { StringBuilder buf = new StringBuilder(); buf.append("<").append("notification").append(" xmlns=\"").append( "androidpn:iq:notification").append("\">"); if (id != null) { buf.append("<id>").append(id).append("</id>"); } buf.append("</").append("notification").append("> "); return buf.toString(); } public String getId() { return id; } public void setId(String id) { this.id = id; } public String getApiKey() { return apiKey; } public void setApiKey(String apiKey) { this.apiKey = apiKey; } public String getTitle() { return title; } public void setTitle(String title) { this.title = title; } public String getMessage() { return message; } public void setMessage(String message) { this.message = message; } public String getUri() { return uri; } public void setUri(String url) { this.uri = url; } }

其中,getChildElementXml()是IQ的子类,用来拼接成<iq>下的直接点。

public abstract class IQ extends Packet { private Type type = Type.GET; public IQ() { super(); } public IQ(IQ iq) { super(iq); type = iq.getType(); } /** * Returns the type of the IQ packet. * * @return the type of the IQ packet. */ public Type getType() { return type; } /** * Sets the type of the IQ packet. * * @param type the type of the IQ packet. */ public void setType(Type type) { if (type == null) { this.type = Type.GET; } else { this.type = type; } } public String toXML() { StringBuilder buf = new StringBuilder(); buf.append("<iq "); if (getPacketID() != null) { buf.append("id=\"" + getPacketID() + "\" "); } if (getTo() != null) { buf.append("to=\"").append(StringUtils.escapeForXML(getTo())).append("\" "); } if (getFrom() != null) { buf.append("from=\"").append(StringUtils.escapeForXML(getFrom())).append("\" "); } if (type == null) { buf.append("type=\"get\">"); } else { buf.append("type=\"").append(getType()).append("\">"); } // Add the query section if there is one. String queryXML = getChildElementXML(); if (queryXML != null) { buf.append(queryXML); } // Add the error sub-packet, if there is one. XMPPError error = getError(); if (error != null) { buf.append(error.toXML()); } buf.append("</iq>"); return buf.toString(); } /** * Returns the sub-element XML section of the IQ packet, or <tt>null</tt> if there * isn't one. Packet extensions <b>must</b> be included, if any are defined.<p> * * Extensions of this class must override this method. * * @return the child element section of the IQ XML. */ public abstract String getChildElementXML(); /** * Convenience method to create a new empty {@link Type#RESULT IQ.Type.RESULT} * IQ based on a {@link Type#GET IQ.Type.GET} or {@link Type#SET IQ.Type.SET} * IQ. The new packet will be initialized with:<ul> * <li>The sender set to the recipient of the originating IQ. * <li>The recipient set to the sender of the originating IQ. * <li>The type set to {@link Type#RESULT IQ.Type.RESULT}. * <li>The id set to the id of the originating IQ. * <li>No child element of the IQ element. * </ul> * * @param iq the {@link Type#GET IQ.Type.GET} or {@link Type#SET IQ.Type.SET} IQ packet. * @throws IllegalArgumentException if the IQ packet does not have a type of * {@link Type#GET IQ.Type.GET} or {@link Type#SET IQ.Type.SET}. * @return a new {@link Type#RESULT IQ.Type.RESULT} IQ based on the originating IQ. */ public static IQ createResultIQ(final IQ request) { if (!(request.getType() == Type.GET || request.getType() == Type.SET)) { throw new IllegalArgumentException( "IQ must be of type 'set' or 'get'. Original IQ: " + request.toXML()); } final IQ result = new IQ() { public String getChildElementXML() { return null; } }; result.setType(Type.RESULT); result.setPacketID(request.getPacketID()); result.setFrom(request.getTo()); result.setTo(request.getFrom()); return result; } /** * Convenience method to create a new {@link Type#ERROR IQ.Type.ERROR} IQ * based on a {@link Type#GET IQ.Type.GET} or {@link Type#SET IQ.Type.SET} * IQ. The new packet will be initialized with:<ul> * <li>The sender set to the recipient of the originating IQ. * <li>The recipient set to the sender of the originating IQ. * <li>The type set to {@link Type#ERROR IQ.Type.ERROR}. * <li>The id set to the id of the originating IQ. * <li>The child element contained in the associated originating IQ. * <li>The provided {@link XMPPError XMPPError}. * </ul> * * @param iq the {@link Type#GET IQ.Type.GET} or {@link Type#SET IQ.Type.SET} IQ packet. * @param error the error to associate with the created IQ packet. * @throws IllegalArgumentException if the IQ packet does not have a type of * {@link Type#GET IQ.Type.GET} or {@link Type#SET IQ.Type.SET}. * @return a new {@link Type#ERROR IQ.Type.ERROR} IQ based on the originating IQ. */ public static IQ createErrorResponse(final IQ request, final XMPPError error) { if (!(request.getType() == Type.GET || request.getType() == Type.SET)) { throw new IllegalArgumentException( "IQ must be of type 'set' or 'get'. Original IQ: " + request.toXML()); } final IQ result = new IQ() { public String getChildElementXML() { return request.getChildElementXML(); } }; result.setType(Type.ERROR); result.setPacketID(request.getPacketID()); result.setFrom(request.getTo()); result.setTo(request.getFrom()); result.setError(error); return result; } /** * A class to represent the type of the IQ packet. The types are: * * <ul> * <li>IQ.Type.GET * <li>IQ.Type.SET * <li>IQ.Type.RESULT * <li>IQ.Type.ERROR * </ul> */ public static class Type { public static final Type GET = new Type("get"); public static final Type SET = new Type("set"); public static final Type RESULT = new Type("result"); public static final Type ERROR = new Type("error"); /** * Converts a String into the corresponding types. Valid String values * that can be converted to types are: "get", "set", "result", and "error". * * @param type the String value to covert. * @return the corresponding Type. */ public static Type fromString(String type) { if (type == null) { return null; } type = type.toLowerCase(); if (GET.toString().equals(type)) { return GET; } else if (SET.toString().equals(type)) { return SET; } else if (ERROR.toString().equals(type)) { return ERROR; } else if (RESULT.toString().equals(type)) { return RESULT; } else { return null; } } private String value; private Type(String value) { this.value = value; } public String toString() { return value; } } }

最终可生成如下结构的数据

<iq from=""> <nofitication xlns=""> <iq>

我们在项目中的使用很简单

xmppManager.getConnection().sendPacket(<NotificationIQ>niq)

当然,上面只是实现了object->xml,接下来我们实现xml->data

2.实现IQProvider

先来看看IQProvider源码

public interface IQProvider { /** * Parse the IQ sub-document and create an IQ instance. Each IQ must have a * single child element. At the beginning of the method call, the xml parser * will be positioned at the opening tag of the IQ child element. At the end * of the method call, the parser <b>must</b> be positioned on the closing tag * of the child element. * * @param parser an XML parser. * @return a new IQ instance. * @throws Exception if an error occurs parsing the XML. */ public IQ parseIQ(XmlPullParser parser) throws Exception; }

实现自定义的解析工具

public class NotificationIQProvider implements IQProvider { public NotificationIQProvider() { } @Override public IQ parseIQ(XmlPullParser parser) throws Exception { NotificationIQ notification = new NotificationIQ(); for (boolean done = false; !done;) { int eventType = parser.next(); if (eventType == 2) { if ("id".equals(parser.getName())) { notification.setId(parser.nextText()); } if ("apiKey".equals(parser.getName())) { notification.setApiKey(parser.nextText()); } if ("title".equals(parser.getName())) { notification.setTitle(parser.nextText()); } if ("message".equals(parser.getName())) { notification.setMessage(parser.nextText()); } if ("uri".equals(parser.getName())) { notification.setUri(parser.nextText()); } } else if (eventType == 3 && "notification".equals(parser.getName())) { done = true; } } return notification; } }

项目中使用方法

ProviderManager.getInstance().addIQProvider("notification", "androidpn:iq:notification", new NotificationIQProvider());

在asmack中PacketParserUtils类中会进行如下调用

Object provider = ProviderManager.getInstance().getIQProvider(elementName, namespace); if (provider != null) { if (provider instanceof IQProvider) { iqPacket = ((IQProvider)provider).parseIQ(parser); } else if (provider instanceof Class) { iqPacket = (IQ)PacketParserUtils.parseWithIntrospection(elementName, (Class<?>)provider, parser); } }

时间: 2024-09-15 08:57:09

Android XMPP通讯自定义Packet&Provider的相关文章

Android xmpp 资源列表

Beem-0.1.7.rar http://dldx.csdn.net/fd.php?i=720645490112095&s=6325e80b23e8c5863336d463bd8b589e 在文件夹中显示从列表中删除 IMLoveSong_18619 (1).rar http://dldx.csdn.net/fd.php?i=134138952482005&s=748f3e38b8899f9ea6a167e0195c95aa 在文件夹中显示从列表中删除 XMPP协议--即时通讯工具实现原

Android 中TabLayout自定义选择背景滑块的实例代码_Android

 TabLayout是Android 的Material Design包中的一个控件,可以和V4包中的ViewPager搭配产生一个联动的效果.这里我自定义了一个滑块能够跟随TabLayout进行滑动选择的SliderLayout.效果见下图(白色方框): 下面是SliderLayout的源码: import android.content.Context; import android.content.res.TypedArray; import android.graphics.drawab

Android编程实现自定义手势的方法详解_Android

本文实例讲述了Android编程实现自定义手势的方法.分享给大家供大家参考,具体如下: 之前介绍过如何在Android程序中使用手势,主要是系统默认提供的几个手势,这次介绍一下如何自定义手势,以及如何对其进行管理. 先介绍一下Android系统对手势的管理,Android系统允许应用程序把用户的手势以文件的形式保存以前,以后要使用这些手势只需要加载这个手势库文件即可,同时Android系统还提供了诸如手势识别.查找及删除等的函数接口,具体如下: 一.加载手势库文件: staticGestureL

Android不使用自定义布局情况下实现自定义通知栏图标的方法_Android

本文实例讲述了Android不使用自定义布局情况下实现自定义通知栏图标的方法.分享给大家供大家参考,具体如下: 自定义通知栏图标?不是很简单么.自定义布局都不在话下! 是的,有xml布局文件当然一切都很简单,如果不给你布局文件用呢? 听我慢慢道来! 首先怎么创建一个通知呢? 1.new 一个 复制代码 代码如下: Notification n = new Notification(android.R.drawable.ic_menu_share, null, System.currentTime

Android百度地图自定义公交路线导航_Android

一.问题描述 基于百度地图实现检索指定城市指定公交的交通路线图,效果如图所示 二.通用组件Application类,主要创建并初始化BMapManager public class App extends Application { static App mDemoApp; //百度MapAPI的管理类 public BMapManager mBMapMan = null; // 授权Key // 申请地址:http://dev.baidu.com/wiki/static/imap/key/ p

Android中制作自定义dialog对话框的实例分享_Android

自定义dialog基础版很多时候,我们在使用android sdk提供的alerdialog的时候,会因为你的系统的不同而产生不同的效果,就好比如你刷的是MIUI的系统,弹出框都会在顶部显示!这里简单的介绍自定义弹出框的应用. 首先创建布局文件dialog: 代码: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.andr

Android如何在自定义view中发送消息给主线程

问题描述 Android如何在自定义view中发送消息给主线程 求求各位大神支招,小弟在自己开发一个东西,遇到了一个小问题. ![ 图片说明](http://img.ask.csdn.net/upload/201603/15/1458054689_616589.jpg) 我在自定义view的onDraw里,当条件达到后就开启一个子线程发送消息给主线程,让主线程中的handler接收消息并执行相应的动作,可是在Log的输出下只能输出"子线程,"而没有"handlemessage

系统-android usb通讯 大神。。。。来吧

问题描述 android usb通讯 大神....来吧 android 系统是怎样判断不同设备连接的 ? 解决方案 1,USB存储设备(如:U盘,移动硬盘): //USB存储设备 插拔监听与 SD卡插拔监听一致. private USBBroadCastReceiver mBroadcastReceiver; IntentFilter iFilter = new IntentFilter(); iFilter.addAction(Intent.ACTION_MEDIA_EJECT); iFilt

android网络通讯Xutils问题

问题描述 android网络通讯Xutils问题 String url = getResources().getString(R.string.url); RequestParams params = new RequestParams(); params.addBodyParameter("cmd", "301"); HttpUtils httpUtils = new HttpUtils(); httpUtils.send(HttpMethod.POST, url