阿不

潜水

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::

使用DotMSN来开发MSN机器人,机器人在一段时间里如果没有接收到任何会话请求时,经常会自动断线。而且这个连接断线是没有响应任何事件的,而且Connected是True,Owner.Status也是正常的,这不得不让人头痛啊。

经过查阅一些资料,发现只要定时向MSN服务器发送Ping命令,以保持客户端与服务器的连接,即可解决这个问题。但是问题是DotMSN里提到的SendPing()方法它并没有提供啊!?可以查找到MSNP协议里头,发送Ping命令是通过PNG\r\n。但是如何来发送是一个问题了。

最初的想法是构建一个NSMessage对象,由NameserverProcessor.SendMessage方法来发送。使用代码如下:

   1: void SendPing()
   2:         {
   3:             NSMessage message = new NSMessage("PNG\r\n", new string[] {  });
   4:             messenger.NameserverProcessor.SendMessage(message);
   5:         }

但是每当我发送一个Ping命令后,虽然PingAnswer事件会被触发,也就是服务器有响应了,Ping命令发送成功。但是同时我的登录也会被服务器踢出,SignedOff事件被触发了。在仔细查阅了NSMessage信息对象的定义后发现,这个对象构建的发送字节除了命令本身还会有空格,TransactionID,还有参数,最后才会加上\r\n。而Ping命令本身是不需要任何参数的,整个命令就只有PNG\r\n。解决办法是不是可以构建一个专门的PingMessage对象来表示Ping命令,这个对象很简单:

   1: public class PingMessage : XihSolutions.DotMSN.Core.NSMessage
   2: {
   3:     public override byte[] GetBytes()
   4:     {
   5:         return System.Text.Encoding.UTF8.GetBytes("PNG\r\n");
   6:     }
   7: }

重新修改一下SendPing方法:

   1: void SendPing()
   2: {
   3:    messenger.NameserverProcessor.SendMessage(new PingMessage());
   4: }

登录后,发送一个Ping命令试试,PingAnswer事件触发了,用户不会被踢出了。这表示Ping命令发送成功了。接下来所要做的就是创建一个Timer对象,由Timer对象来控制定时向服务器发送Ping命令,以保证机器人客户端正与服务器保持连接。

注意,目前我的测试是在之前连接断开间隔时间内,甚至更长一些,机器人始终保持连接。不保证全天候都能保持连接,但是其它的断开原因可能需要不同的处理方法。

posted on 2008-01-10 09:31  阿不  阅读(3993)  评论(23编辑  收藏  举报