无法使用 Gtalk 应用程序的 asmack api 获取离线消息

Posted

技术标签:

【中文标题】无法使用 Gtalk 应用程序的 asmack api 获取离线消息【英文标题】:Cant get offline messages using asmack api for Gtalk application 【发布时间】:2012-03-16 06:08:24 【问题描述】:

我正在使用 asmack api 为 Gtalk 创建一个聊天应用程序,我想在用户离线时获取消息,但由于 ServiceDiscoveryManager 中的一些问题以及它说该功能未实现(501)并试图实现了其他用户有同样问题的所有东西,但现在我得到了这个错误。我正在发布代码和 logcat。任何帮助表示赞赏。

 ConnectionConfiguration connConfig = new ConnectionConfiguration(
                    host, Integer.parseInt(port), service);
 connConfig.setSASLAuthenticationEnabled(true);
 connConfig.setSendPresence(false);
 connection = new XMPPConnection(connConfig);
 connection.connect();
 connection.login(username, password);
 ServiceDiscoveryManager sdm= ServiceDiscoveryManager.getInstanceFor(connection);
 mOfflineMessageManager = new OfflineMessageManager(connection);
 offlinemsgs = mOfflineMessageManager.getMessageCount(); 

这是我在登录后调用离线消息的代码,下面是 logcat 错误中的响应:

03-16 11:26:53.871: W/System.err(325): feature-not-implemented(501)
03-16 11:26:53.881: W/System.err(325):  at org.jivesoftware.smackx.OfflineMessageManager.getMessages(OfflineMessageManager.java:210)
03-16 11:26:53.881: W/System.err(325):  at com.apache.android.xmpp.MainScreen.getOfflinemessages(MainScreen.java:911)
03-16 11:26:53.881: W/System.err(325):  at com.apache.android.xmpp.MainScreen$LogIn.doInBackground(MainScreen.java:612)
03-16 11:26:53.881: W/System.err(325):  at com.apache.android.xmpp.MainScreen$LogIn.doInBackground(MainScreen.java:1)
03-16 11:26:53.881: W/System.err(325):  at android.os.AsyncTask$2.call(AsyncTask.java:185)
03-16 11:26:53.881: W/System.err(325):  at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
03-16 11:26:53.881: W/System.err(325):  at java.util.concurrent.FutureTask.run(FutureTask.java:137)
03-16 11:26:53.881: W/System.err(325):  at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1068)
03-16 11:26:53.881: W/System.err(325):  at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:561)
03-16 11:26:53.891: W/System.err(325):  at java.lang.Thread.run(Thread.java:1096)

请对此提供帮助,尝试了很多天的解决方案,但找不到任何解决方案。

【问题讨论】:

嘿,您找到解决方案了吗?我遇到了同样的异常。 no.:( 仍在寻找解决方案。如果你有任何解决方案,请告诉我。!! ***.com/questions/7454995/… 我知道了 亲爱的不工作,看看logcat它的那种奇怪的错误,认为gtalk服务或权限有问题。 好吧,我试过了,它对我有用,只需设置监听器以获取消息,它就会捕获离线消息。 【参考方案1】:

试试这个:

 ConnectionConfiguration connConfig = new ConnectionConfiguration(
                    host, Integer.parseInt(port), service);
 connConfig.setSASLAuthenticationEnabled(true);
 connConfig.setSendPresence(false);
 connection = new XMPPConnection(connConfig);
 connection.connect();
 connection.login(username, password);
 ServiceDiscoveryManager sdm= ServiceDiscoveryManager.getInstanceFor(connection);

////////////////////////////

    OfflineMessageManager offlineManager = new OfflineMessageManager(  
                    Client.getConnection());  
            try   
                Iterator<org.jivesoftware.smack.packet.Message> it = offlineManager  
                        .getMessages();  
                System.out.println(offlineManager.supportsFlexibleRetrieval());  
                System.out.println("Number of offline messages:: " + offlineManager.getMessageCount());   
                Map<String,ArrayList<Message>> offlineMsgs = new HashMap<String,ArrayList<Message>>();    
                while (it.hasNext())   
                    org.jivesoftware.smack.packet.Message message = it.next();  
                    System.out  
                            .println("receive offline messages, the Received from [" + message.getFrom()  
                                    + "] the message:" + message.getBody());  
                    String fromUser = message.getFrom().split("/")[0];  

                    if(offlineMsgs.containsKey(fromUser))  
                      
                        offlineMsgs.get(fromUser).add(message);  
                    else  
                        ArrayList<Message> temp = new ArrayList<Message>();  
                        temp.add(message);  
                        offlineMsgs.put(fromUser, temp);  
                      
                  
                / / Deal with a collection of offline messages ...  
                Set<String> keys = offlineMsgs.keySet();  
                Iterator<String> offIt = keys.iterator();  
                while(offIt.hasNext())  
                  
                    String key = offIt.next();  
                    ArrayList<Message> ms = offlineMsgs.get(key);  
                    TelFrame tel = new TelFrame(key);  
                    ChatFrameThread cft = new ChatFrameThread(key, null);  
                    cft.setTel(tel);  
                    cft.start();  
                    for (int i = 0; i < ms.size(); i++)   
                        tel.messageReceiveHandler(ms.get(i));  
                      
                  
                offlineManager.deleteMessages();  
             catch (Exception e)   
                e.printStackTrace();  
              

看到这个喜欢:http://community.igniterealtime.org

【讨论】:

因为没有任何其他答案,而且您的答案似乎几乎是正确的,我只是在谷歌服务中遇到问题,而不是来自 asmack api。我应该感谢您的回答。 @imran khan 什么是 xmpp 客户端的最佳 Api 必须同时支持 facebook 和 gtalk 聊天。如果我对 gtalk 使用 asmack-issue5 面临问题,对 gtalk 使用 smack 面临问题(获取好友列表).. 谢谢,但为什么我们需要将存在发送为假。这意味着我处于离线状态(我在服务器中的可用状态为离线)。但是当我在线时我们需要获取离线消息,对吗?这意味着我必须在下一次登录会话中获取离线消息。为此我该怎么办? @pρяσѕρєяK @prospec 你能告诉我有关 Client.getconnection 的信息吗?新的 OfflineMessageManager(Client.getConnection()); Client.getConnection() 找不到客户端? 这工作正常...我的问题是设置 connConfig.setSendPresence(true);【参考方案2】:

为什么你不只是使用 PacketListener,我正在使用它并获取离线消息。这是我的代码,我也将消息保存到 Sqlite db 中,以便在他离开此聊天并与另一个朋友聊天然后去的时候获取用户消息返回。

PacketFilter filter = new MessageTypeFilter(Message.Type.chat);
            Main.conn.addPacketListener(new PacketListener() 

                public void processPacket(Packet p) 

                    message = (Message) p;// this was the problem

                    if (message.getBody() != null) 
                        messages.add(message.getBody());
                        Log.i("XMPPClient",
                                "Reciveing text [" + message.getBody() + "] ");

                        time.add(TimeDate());

                        // Add the incoming message to the list view
                        mHandler.post(new Runnable() 
                            public void run() 
                                // / saving convertsations into SQLite db
                                try 

                                    save.open();
                                    save.InsertIntoDatabase(jid, messages, time);
                                    messagesDB = save.GetAllValues(Table_Name,
                                            new String[]  "messages" ,
                                            friendJid);
                                    timeDB = save.GetAllValues(Table_Name,
                                            new String[]  "time" , friendJid);
                                    adapter = new ConversationListAdapter(
                                            getApplicationContext(),
                                            messagesDB, timeDB);
                                    list.setAdapter(adapter);
                                    adapter.notifyDataSetChanged();
                                    save.close();

                                 catch (SQLException e) 
                                    // TODO Auto-generated catch block
                                    e.printStackTrace();
                                


                            
                        );
                    
                

            , filter);
        

【讨论】:

谢谢你:)。但是我正在寻找启动应用程序时没有收到的离线消息,应该有一些东西可以从中检索离线消息,我已经尝试通过这种方法获取它们,但没有任何效果。如果您知道如何获取离线消息,请指导我。 @mak_just4anything 嗨,你找到解决方案了吗 :( 我还在找 :(【参考方案3】:

我在开发带有连接到 Openfire 服务器的 aSmack 的 Android 客户端时遇到了同样的问题:我根本无法检索离线消息。所以阅读[XEP-0013]:http://www.xmpp.org/extensions/xep-0013.html我明白了原因:

“在收到发往“http://jabber.org/protocol/offline”节点的服务发现请求(本用例中的 disco#info 请求或下一个用例中的 disco#items 请求)时,服务器不得“

所以,我学到了什么:

    检查您的服务器是否支持灵活离线消息检索 XEP-0013; 您必须初始化 SmackAndroid.init(Context) - 请参阅 aSmack README; 您必须使用 AndroidConnectionConfiguration 而不是 ConnectionConfiguration; 如果需要,您可以将配置设置为 setSendPresence(true)。

为了使您必须连接的东西正常工作,请将初始状态发送为“不可用”以检索您的消息,然后将状态发送为“可用”。

那么您的代码应该如下所示(省略 try/catch 块):

SmackAndroid.init(context);
AndroidConnectionConfiguration connConfig = null;
connConfig = new AndroidConnectionConfiguration(domain, PORT);
connConfig.setSendPresence(true);
connConfig.setReconnectionAllowed(true);
connection = new XMPPConnection(connConfig);
connection.connect();
connection.login("username", "password");

PacketFilter filter = new MessageTypeFilter(Message.Type.chat);
//RECEIVER
connection.addPacketListener(getPacketListener(), filter);
//in order to retrieve offline messages: XEP-0013
connection.sendPacket(new Presence(Presence.Type.unavailable));
connection.sendPacket(new Presence(Presence.Type.available));

然后发送 Presence.Type.unavailable 数据包后,所有离线消息都自动发送!

【讨论】:

以上是关于无法使用 Gtalk 应用程序的 asmack api 获取离线消息的主要内容,如果未能解决你的问题,请参考以下文章

启用两步验证后无法在 Pidgin 中使用 GTalk [关闭]

无法使用 aSmack 连接到本地 XMPP 服务器

无法使用 Beem-Asmack 在 XMPP FileTransfer 中处理上传

asmack android,无法传输文件

aSmack 错误:XMPPConnection 是抽象的;无法实例化

asmack 中的 MessageListener 以集成无法绑定到任何小部件的 facebook 聊天