使用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命令,以保证机器人客户端正与服务器保持连接。
注意,目前我的测试是在之前连接断开间隔时间内,甚至更长一些,机器人始终保持连接。不保证全天候都能保持连接,但是其它的断开原因可能需要不同的处理方法。