aSmack - 使用数据包侦听器输出可空元素的数据包到 XML

Posted

技术标签:

【中文标题】aSmack - 使用数据包侦听器输出可空元素的数据包到 XML【英文标题】:aSmack - Packet to XML using Packet Listener outputs nullable elements 【发布时间】:2013-06-13 13:24:30 【问题描述】:

我在我的 android 应用程序中使用 aSmack 来与我的 XMPP 服务器进行通信,并且我打开了 Smack 的调试,这样我就可以看到所有 XML 的来/去。 我的问题是我正在使用 PacketListener 来获取服务器对我发送的包的响应,但是当我调用 Packet 的 toXML() 方法时,我得到了一个奇怪的输出。 下面的课程和更多详细信息。

我的服务器有 XEP-0136 实现,因为 aSmack 还没有它的代码,所以我正在制作 IQ 数据包并发送它,这按预期工作,正如你在我的 XML 数据包下面看到的那样,在 XEP-0136 中设计,检索集合列表(http://xmpp.org/extensions/xep-0136.html#manage-list)

Xml 包发送到服务器

06-13 14:11:21.769: D/SMACK(3018): 02:11:21 PM SENT (1079273464): 
<iq id="john@company.com/Smack/Conversations" type="get">
  <list with="john@company.com" xmlns="urn:xmpp:archive">
    <set xmlns="http://jabber.org/protocol/rsm">
      <max>30</max>
    </set>
  </list>
</iq>

为了创建这个 XML,我得到了 XEP-0136 中设计的 XML Schema 文件,在 SimpleXML 库的帮助下,我映射了所有元素,这是我用来创建和发送包的代码:

XMPPService.java

private static final int MAX_LIST = 30;

public void getConversations(String email, BaseActivity activity)

    if (isAuthenticated())
    
        String packetId = connection.getUser() + "/Conversations";
        Set set = new Set();
        set.setMax(MAX_LIST);

        List list = new List();
        list.setWith(email);
        list.setSet(set);

        final IQ iq = new IQ();
        iq.setList(list);
        iq.setType(IQType.get);
        iq.setId(packetId);

        PacketIDFilter filter = new PacketIDFilter(packetId);
        connection.addPacketListener(new ChatListListener(activity), filter);
        sendPacket(iq);
    


public void sendPacket(IQ iq)

    if (isAuthenticated())
    
        connection.sendPacket(new IQPacket(iq));
    

IQPacket.java

public class IQPacket extends Packet 

 private IQ iq;

 public IQPacket(IQ iq)
 
    this.iq = iq;
 

 public IQPacket(Packet packet, IQ iq)
 
    super(packet);
    this.iq = iq;
 

 @Override
 public String toXML()
 
    StringWriter writer = new StringWriter();
    Serializer serializer = new Persister();
    try
    
        serializer.write(iq, writer);
        return writer.getBuffer().toString();
     catch (Exception e)
    
        Log.e("COMPANY", "Error serializing xml", e);
    
    return null;
 


正如我所说,这部分有效,我的问题是监听器,当我为收到的包调用 toXML() 方法时,我无法获取有关聊天的重要信息,但 Smack 调试输出打印到告诉我所有我期待的信息,如下所示:

接收到的 XML 的 Smack 调试

06-13 14:11:21.989: D/SMACK(3018): 02:11:21 PM RCV  (1079273464): 
<iq type="result" id="john@company.com/Smack/Conversations" to="john@company.com/Smack">
  <list xmlns="urn:xmpp:archive">
    <chat with="anotheruser@company.com" start="2013-06-10T13:19:25.000Z"/>
    <chat with="yetanotheruser@company.com" start="2013-06-10T13:36:50.876Z"/>
    <set xmlns="http://jabber.org/protocol/rsm">
      <first index="0">2</first>
      <last>3</last>
      <count>9</count>
    </set>
  </list>
</iq>

这个 XML 也是预期的答案,因为我将所有这些元素都映射为 JavaBeans,但这是我在 ChatListener 上收到数据包并调用 toXML() 方法时得到的结果:

06-13 14:11:22.009: I/System.out(3018): 
<iq id="john@company.com/Smack/Conversations" to="john@company.com/Smack" type="result">nullnullnullnullnullnull2nullnull3nullnull9nullnull</iq>

ChatListListener.java

public class ChatListListener implements PacketListener 

 private BaseActivity activity;

 public ChatListListener(BaseActivity activity)
 
    this.activity = activity;
 

 @Override
 public void processPacket(Packet packet)
 
    activity.notifyPacketReceived();
    System.out.println(packet.toXML());
 

Packet 来自 org.jivesoftware.smack.packet.Packet,因此它是来自 aSmack 库的默认 Packet。

所以我的问题是,我与 Smack 调试器有何不同?我查看了它的代码,我看到它还从数据包中调用了 toXML() 方法并添加了一个 ReceiveListener。 我的想法是在调用 toXML() 之后,我可以使用 SimpleXML 将其转换为我映射的 IQ.java 并开始使用它的信息。

编辑

添加更多信息。因此,在查找 Smack 代码以及它如何处理收到的包之后,我发现也许我应该使用 IQProvider。 所以我注册了我的 IQProvider

ProviderManager.getInstance().addIQProvider("list", "urn:xmpp:archive", new ListIQProvider());

然后,我在我的 IQProvider 的方法 parseIQ(XmlPullParser arg0) 上设置了一个断点,并且包实际上正在发送到我的提供者,但它仍然包含所有这些 null 元素。我现在有点迷茫,因为我需要它来继续工作,我会继续研究 Smack 源代码。

【问题讨论】:

【参考方案1】:

经过大量研究和查找 Smack 资源后,我找到了解决方案。 步骤是,添加 IQProvider,为您的数据包添加 PacketListener,发送数据包。在 IQProvider 上等待数据包,对其进行解析,然后在 Listener 上获得响应。

所以 addIQProvideraddPacketListener 都是正确的,问题是,我需要对我的 ListIQProvider 上的 XML 进行完整解析,这听起来很简单,但花了我需要一些时间来解决这个问题。

public class ListIQProvider implements IQProvider 

 public ListIQProvider()
 
 

 @Override
 public IQ parseIQ(XmlPullParser parser) throws Exception
 
    Logger.d(String.format("Received iq packet, namespace[%s], name[%s]", parser.getNamespace(), parser.getName()));
    ListIQ iq = new ListIQ();
    ListIQ.Set set = new Set();
    boolean done = false;

    String with = "", start = "";
    while (!done)
    
        int eventType = parser.next();
        if (eventType == XmlPullParser.START_TAG)
        
            if (parser.getName().equals("chat"))
            
                with = parser.getAttributeValue("", "with");
                start = parser.getAttributeValue("", "start");
                iq.addChat(new Chat(with, start));
            
            else if (parser.getName().equals("first"))
            
                int index = parseInt(parser.getAttributeValue("", "index"));
                set.setIndexAtt(index);
                int first = parseInt(parser.nextText());
                set.setFirst(first);
            
            else if (parser.getName().equals("last"))
            
                int last = parseInt(parser.nextText());
                set.setLast(last);
            
            else if (parser.getName().equals("count"))
            
                int count = parseInt(parser.nextText());
                set.setCount(count);
            
        
        else if (eventType == XmlPullParser.END_TAG)
        
            if (parser.getName().equals("list"))
            
                iq.setSet(set);
                done = true;
            
        
    

    return iq;
 

 private int parseInt(String integer)
 
    return Integer.parseInt((integer != null ? integer : "0"));
 

之后,我在 ChatListListener 上要做的就是将 Packet 投射到我的 ListIQ 类中。就是这样。 这里的问题是,在我的 ChatListListener 上收到的 Packet 与在我的 上的 parseIQ 方法上返回的 Packet 相同ListIQProvider。 所以在这个问题/答案中,我们几乎拥有 XEP-0136 所需的一切,或者至少开始使用它。 由于我没有在网络上找到任何好的和简单的资源来帮助我,我在这里分享我的。下面是 ListIQ 类:

public class ListIQ extends IQ 

 private List<Chat> chats;

 private Set set;

 public ListIQ()
 
    this.chats = new ArrayList<ListIQ.Chat>();
 

 public Set getSet()
 
    return set;
 

 public void setSet(Set set)
 
    this.set = set;
 

 public void addChat(Chat chat)
 
    chats.add(chat);
 

 public List<Chat> getChats()
 
    return chats;
 

 @Override
 public String getChildElementXML()
 
    StringBuilder builder = new StringBuilder("<list xmlns=\"urn:xmpp:archive\">");
    for (Chat chat : chats)
    
        builder.append(chat.toXml());
    
    builder.append(set.toXml());
    builder.append("</list>");
    return builder.toString();
 

 public static class Chat 
    private String with;
    private String start;

    public Chat()
    
    

    public Chat(String with, String start)
    
        this.with = with;
        this.start = start;
    

    public String getWith()
    
        return with;
    

    public void setWith(String with)
    
        this.with = with;
    

    public String getStart()
    
        return start;
    

    public void setStart(String start)
    
        this.start = start;
    

    public String toXml()
    
        StringBuilder builder = new StringBuilder("<chat with=\"");
        builder.append(with).append("\"");
        builder.append(" start=\"");
        builder.append(start);
        builder.append("\"/>");
        return builder.toString();
    

 

 public static class Set 
    private int last;
    private int count;
    private int indexAtt;
    private int first;

    public Set()
    
    

    public int getLast()
    
        return last;
    

    public void setLast(int last)
    
        this.last = last;
    

    public int getCount()
    
        return count;
    

    public void setCount(int count)
    
        this.count = count;
    

    public int getIndexAtt()
    
        return indexAtt;
    

    public void setIndexAtt(int indexAtt)
    
        this.indexAtt = indexAtt;
    

    public int getFirst()
    
        return first;
    

    public void setFirst(int first)
    
        this.first = first;
    

    public String toXml()
    
        StringBuilder builder = new StringBuilder("<set xmlns=\"http://jabber.org/protocol/rsm\">");
        builder.append("<first index=\"").append(indexAtt).append("\">").append(first).append("</first>");
        builder.append("<last>").append(last).append("</last>");
        builder.append("<count>").append(count).append("</count>");
        builder.append("</set>");
        return builder.toString();
    
 


【讨论】:

你能回答***.com/questions/22566411/…我也面临同样的问题 感谢您提供完整的源代码,这正是我所需要的。这就是人们现在更喜欢 JSON 的原因还是这只是糟糕的设计? 没问题。其实XEP的设计不错,就是自己实现起来比较麻烦。我认为 JSON 不符合 XMPP 要求,我猜使用 XML 更有意义,而且协议比 JSON 更旧。【参考方案2】:

您好,我也遇到了同样的问题,我可以说这段代码有效,我只是改变您为服务器构建列表查询的方式。 (至少在我的服务器上)

final IQ iq = new IQ()
    

        @Override public String getChildElementXML()
        

            return "<list xmlns='urn:xmpp:archive'/>";

        
    ;

    iq.setType(IQ.Type.GET);

    PacketTypeFilter filter = new PacketTypeFilter(Packet.class);


    connection.addPacketListener(new PacketListener()
    
        @Override public void processPacket(Packet packet)
        
            Log.i(TAG, packet.toXML());
        
    , filter);

    connection.sendPacket(iq);

正如您之前所说,这似乎很容易,但文档并不清楚如何构建 IQ 对象。希望对你有帮助。

【讨论】:

但是您仍然无法解析响应。您只是可以发送检索聊天的请求。 当然不是,要处理答案,您需要像以前一样注册提供者。我只是试图让你的帖子更好一点,但你的代码仍然是重要的部分,因此我开发了从 IQProvider 扩展的类来解析完全不同的聊天(和消息)。不过我还是给了你+1,因为你的帖子帮了我很多。 嘿,我试过了,但我得到错误 503 服务不可用。 @Hardik ,听起来服务器不工作或消息存档不可用?

以上是关于aSmack - 使用数据包侦听器输出可空元素的数据包到 XML的主要内容,如果未能解决你的问题,请参考以下文章

asmak 数据包侦听器和自定义 IQProvider 未触发/调用

Smack 数据包侦听器未运行

如何使用 asmack 库发送和收听自定义 xmpp 存在数据包

用户存在值从 asmack 变为空

为 MultiUserChat 添加监听器 (aSmack)

Android with Asmack - 当我连接到 XMPP 时如何获取当前状态信息?