XMPP即时通讯

XMPP:XMPP是基于XML的点对点通讯协议,The Extensible Messaging and Presence Protocol(可扩展通讯和表示协议)。

XMPP可用于服务类实时通讯,表示和需求响应服务中的XML数据元流失传输。XMPP以Jabber协议为基础,而Jabber是即时通讯中常用的开放式协议。

基本结构。

XMPP是一个典型的C/S架构,而不是像大多数即时通讯软件一样,使用P2P客户端到客户端的架构,也就是说在大多数情况下,当两个客户端进行通讯时,他们的消息都是通过服务器传递的。采用这种架构,主要是为了简化客户端,将大多数工作放在服务器端进行。

XMPP中定义了三个角色,客户端,服务器,网关。通信能够在这三者的任意两个之间双向发生。

服务器同时承担了客户端信息记录,连接管理和信息的路由功能。网关承担着与异构即时通讯系统的互联互通,异构系统可以包括SMS(短信),MSN,ICQ等。

基本的网络形式是单客户端通过TCP/IP连接到单服务器,然后在之上传输XML流。

1.1XMPP中的常用对象

XMPPStream:xmpp基础服务类

XMPPRoster:好友列表类

XMPPRosterCoreDataStoreage:好友列表(用户账号)在core data中的操作类

XMPPvCardCoreDataStorage:好友名片(昵称、签名、性别、年龄等信息)在core data中的操作类

XMPPvCardTemp:好友名片实体类,从数据库里面取出来的都是它

xmppvCardAvatarModule:好友头像

XMPPReconnect:如果失去连接,自动重连

XMPPRoom:提供多用户聊天支持

XMPPPubSub:发布订阅

1.2登录操作,也就是连接xmpp服务器

XMPP的地址叫做JabberID(简写为JID),它用来标识XMPP网络中的各个XMPP实体,JID由三个部分组成:domain、node identifier和resource。JID中domain是必不可少的部分。注意:domain和user部分是不分大小写的,但是resource区别大小写。

domain:通常指网络中的网关或者服务器。

node identifier:通常表示一个向服务器或者网关请求和使用网络服务的实体(比如一个客户端),当然它也能够表示其他的实体(比如在多用户聊天系统中的一个房间)。

resource:通常表示一个特定的会话(与某个设备),连接(与某个地址),或者一个附属于某个节点ID实体相关实体的对象(比如多用户聊天室中的一个参加者)。

JID种类有:

bare JID:user@domain.tld

full JID:user@domain.tld/resource

如:

tom@jabber.org:表示服务器jabber.org上的用户tom

room@service:一个用来提供多用户聊天服务的特定的聊天室。这里“room”是聊天室的名字,“service”是多用户聊天服务的主机名。

room@service/nick:加入了聊天室的用户nick的地址。这里“room”是聊天室的名字,“service”是多用户聊天服务的主机名,“nick”是用户在聊天室的昵称。

为了表示JID,xmpp也有自己的URI,例如xmpp:stpeter.org,默认规则是在JID前加xmpp:。

代码操作:

1、登录操作,也就是连接XMPP服务器。

 1 - (void)connect {
 2     if (self.xmppStream == nil) {
 3         self.xmppStream = [[XMPPStream alloc] init];
 4         [self.xmppStream addDelegate:self delegateQueue:dispatch_get_main_queue()];
 5     }
 6     if (![self.xmppStream isConnected]) {
 7         NSString *username = [[NSUserDefaults standardUserDefaults] objectForKey:@"username"];
 8         XMPPJID *jid = [XMPPJID jidWithUser:username domain:@"lizhen" resource:@"Ework"];
 9         [self.xmppStream setMyJID:jid];
10         [self.xmppStream setHostName:@"10.4.125.113"];
11         NSError *error = nil;
12         if (![self.xmppStream connect:&error]) {
13             NSLog(@"Connect Error: %@", [[error userInfo] description]);
14         }
15     }
16 }

connect成功之后会依次调用XMPPStreamDelegate的方法,首先调用

- (void)xmppStream:(XMPPStream *)sender socketDidConnect:(GCDAsyncSocket *)socket
 
...

然后

- (void)xmppStreamDidConnect:(XMPPStream *)sender

在该方法下面需要使用xmppStream的authenticcateWithPassword方法进行密码验证,成功的话就会响应delegate的方法,就是下面这个:

1 - (void)xmppStreamDidAuthenticate:(XMPPStream *)sender

2、上线

实现- (void)xmppStreamDidAuthenticate:(XMPPStream *)sender 委托方法

- (void)xmppStreamDidAuthenticate:(XMPPStream *)sender {
    XMPPPresence *presence = [XMPPPresence presenceWithType:@"available"];
    [self.xmppStream sendElement:presence];
}

3、退出并断开连接

1 - (void)disconnect {
2     XMPPPresence *presence = [XMPPPresence presenceWithType:@"unavailable"];
3     [self.xmppStream sendElement:presence];
4       
5     [self.xmppStream disconnect];
6 }

4、好友状态

获取好友状态,通过实现

1 - (void)xmppStream:(XMPPStream *)sender didReceivePresence:(XMPPPresence *)presence
2  
3 ...

方法,当接收到presence标签的内容时,XMPPFramework框架回调该方法

一个presence标签的格式一般如下:

presence的状态:

available上线

away离开

do not disturb忙碌

unavailable 下线

 1 - (void)xmppStream:(XMPPStream *)sender didReceivePresence:(XMPPPresence *)presence {
 2     NSString *presenceType = [presence type];
 3     NSString *presenceFromUser = [[presence from] user];
 4     if (![presenceFromUser isEqualToString:[[sender myJID] user]]) {
 5         if ([presenceType isEqualToString:@"available"]) {
 6             //
 7         } else if ([presenceType isEqualToString:@"unavailable"]) {
 8             //
 9         }
10     }
11 }

5、接收消息

通过实现

- (void)xmppStream:(XMPPStream *)sender didReceiveMessage:(XMPPMessage *)message;

方法,当接收到message标签的内容时,XMPPFramework框架回调该方法,根据XMPP协议,消息体的内容存储在标签body内

1 - (void)xmppStream:(XMPPStream *)sender didReceiveMessage:(XMPPMessage *)message {
2     NSString *messageBody = [[message elementForName:@"body"] stringValue];
3 }

6、发送消息

发送消息,我们需要根据XMPP协议,将数据放到标签内,例如:

 1 - (void)sendMessage:(NSString *) message toUser:(NSString *) user {
 2     NSXMLElement *body = [NSXMLElement elementWithName:@"body"];
 3     [body setStringValue:message];
 4     NSXMLElement *message = [NSXMLElement elementWithName:@"message"];
 5     [message addAttributeWithName:@"type" stringValue:@"chat"];
 6     NSString *to = [NSString stringWithFormat:@"%@@example.com", user];
 7     [message addAttributeWithName:@"to" stringValue:to];
 8     [message addChild:body];
 9     [self.xmppStream sendElement:message];
10 }

7、获取好友信息

 1 [_xmppRoster fetchRoster];//获取好友列表
 2 //获取到一个好友节点
 3 - (void)xmppRoster:(XMPPRoster *)sender didRecieveRosterItem:(NSXMLElement *)item
 4 //获取完好友列表
 5 - (void)xmppRosterDidEndPopulating:(XMPPRoster *)sender
 6 //到服务器上请求联系人名片信息
 7 - (void)fetchvCardTempForJID:(XMPPJID *)jid;
 8 //请求联系人的名片,如果数据库有就不请求,没有就发送名片请求
 9 - (void)fetchvCardTempForJID:(XMPPJID *)jid ignoreStorage:(BOOL)ignoreStorage;
10 //获取联系人的名片,如果数据库有就返回,没有返回空,并到服务器上抓取
11 - (XMPPvCardTemp *)vCardTempForJID:(XMPPJID *)jid shouldFetch:(BOOL)shouldFetch;
12 //更新自己的名片信息
13 - (void)updateMyvCardTemp:(XMPPvCardTemp *)vCardTemp;
14 //获取到一盒联系人的名片信息的回调
15 - (void)xmppvCardTempModule:(XMPPvCardTempModule *)vCardTempModule 
16         didReceivevCardTemp:(XMPPvCardTemp *)vCardTemp 
17                      forJID:(XMPPJID *)jid

8、添加好友

1 //name为用户账号
2     - (void)XMPPAddFriendSubscribe:(NSString *)name
3     {
4         //XMPPHOST 就是服务器名,  主机名
5         XMPPJID *jid = [XMPPJID jidWithString:[NSString stringWithFormat:@"%@@%@",name,XMPPHOST]];
6         //[presence addAttributeWithName:@"subscription" stringValue:@"好友"];
7         [xmppRoster subscribePresenceToUser:jid];
8           
9     }

9、收到添加好友的请求

 - (void)xmppRoster:(XMPPRoster *)sender didReceivePresenceSubscriptionRequest:(XMPPPresence *)presence
    {
        //取得好友状态
        NSString *presenceType = [NSString stringWithFormat:@"%@", [presence type]]; //online/offline
        //请求的用户
        NSString *presenceFromUser =[NSString stringWithFormat:@"%@", [[presence from] user]];
        NSLog(@"presenceType:%@",presenceType);
          
        NSLog(@"presence2:%@  sender2:%@",presence,sender);
          
        XMPPJID *jid = [XMPPJID jidWithString:presenceFromUser];
        //接收添加好友请求
        [xmppRoster acceptPresenceSubscriptionRequestFrom:jid andAddToRoster:YES];
          
    }

10、删除好友

1 //删除好友,name为好友账号
2 - (void)removeBuddy:(NSString *)name  
3 {  
4     XMPPJID *jid = [XMPPJID jidWithString:[NSString stringWithFormat:@"%@@%@",name,XMPPHOST]];  
5         
6     [self xmppRoster] removeUser:jid];  
7 }

11、聊天室

1 XMPPJID *roomJID = [XMPPJID jidWithString:ROOM_JID];
2       
3     xmppRoom = [[XMPPRoom alloc] initWithRoomStorage:self jid:roomJID];
4       
5     [xmppRoom activate:xmppStream];
6     [xmppRoom addDelegate:self delegateQueue:dispatch_get_main_queue()];

12、创建聊天室成功

1 - (void)xmppRoomDidCreate:(XMPPRoom *)sender
2     {
3         DDLogInfo(@"%@: %@", THIS_FILE, THIS_METHOD);
4     }

13、加入聊天室,使用昵称

1 [xmppRoom joinRoomUsingNickname:@"quack" history:nil];

14、获取聊天室信息

1 - (void)xmppRoomDidJoin:(XMPPRoom *)sender
2     {
3         [xmppRoom fetchConfigurationForm];
4         [xmppRoom fetchBanList];
5         [xmppRoom fetchMembersList];
6         [xmppRoom fetchModeratorsList];
7     }

15、如果房间存在,会调用委托

1 // 收到禁止名单列表
2     - (void)xmppRoom:(XMPPRoom *)sender didFetchBanList:(NSArray *)items;
3     // 收到好友名单列表
4     - (void)xmppRoom:(XMPPRoom *)sender didFetchMembersList:(NSArray *)items;
5     // 收到主持人名单列表
6     - (void)xmppRoom:(XMPPRoom *)sender didFetchModeratorsList:(NSArray *)items;

如果房间不存在,调用委托

1 - (void)xmppRoom:(XMPPRoom *)sender didNotFetchBanList:(XMPPIQ *)iqError;
2     - (void)xmppRoom:(XMPPRoom *)sender didNotFetchMembersList:(XMPPIQ *)iqError;

16、离开房间

1 [xmppRoom deactivate:xmppStream];

17、离开聊天室

1 - (void)xmppRoomDidLeave:(XMPPRoom *)sender
2     {
3         DDLogVerbose(@"%@: %@", THIS_FILE, THIS_METHOD);
4     }

18、新人加入群聊

1 - (void)xmppRoom:(XMPPRoom *)sender occupantDidJoin:(XMPPJID *)occupantJID
2     {
3         DDLogVerbose(@"%@: %@", THIS_FILE, THIS_METHOD);
4     }

19、有人退出群聊

1 - (void)xmppRoom:(XMPPRoom *)sender occupantDidLeave:(XMPPJID *)occupantJID
2     {
3         DDLogVerbose(@"%@: %@", THIS_FILE, THIS_METHOD);
4     }

20、有人在群里发言

1 - (void)xmppRoom:(XMPPRoom *)sender didReceiveMessage:(XMPPMessage *)message fromOccupant:(XMPPJID *)occupantJID
2     {
3         DDLogVerbose(@"%@: %@", THIS_FILE, THIS_METHOD);
4     }

21、消息回执

这个是XEP - 0184协议的内容

代码实现

1 NSString *siID = [XMPPStream generateUUID];
2     //发送消息
3     XMPPMessage *message = [XMPPMessage messageWithType:@"chat" to:jid elementID:siID];
4     NSXMLElement *receipt = [NSXMLElement elementWithName:@"request" xmlns:@"urn:xmpp:receipts"];
5     [message addChild:receipt];
6     [message addBody:@"测试"];
7     [self.xmppStream sendElement:message];

收到回执请求的消息,发送回执

代码实现

 1 - (void)xmppStream:(XMPPStream *)sender didReceiveMessage:(XMPPMessage *)message
 2     {
 3         //回执判断
 4         NSXMLElement *request = [message elementForName:@"request"];
 5         if (request)
 6         {
 7             if ([request.xmlns isEqualToString:@"urn:xmpp:receipts"])//消息回执
 8             {
 9                 //组装消息回执
10                 XMPPMessage *msg = [XMPPMessage messageWithType:[message attributeStringValueForName:@"type"] to:message.from elementID:[message attributeStringValueForName:@"id"]];
11                 NSXMLElement *recieved = [NSXMLElement elementWithName:@"received" xmlns:@"urn:xmpp:receipts"];
12                 [msg addChild:recieved];
13                   
14                 //发送回执
15                 [self.xmppStream sendElement:msg];
16             }
17         }else
18         {
19             NSXMLElement *received = [message elementForName:@"received"];
20             if (received)
21             {
22                 if ([received.xmlns isEqualToString:@"urn:xmpp:receipts"])//消息回执
23                 {
24                     //发送成功
25                     NSLog(@"message send success!");
26                 }  
27             }  
28         }  
29           
30         //消息处理  
31         //...  
32     }

22、添加AutoPing

为了监听服务器是否有效,增加心跳监听。用XEP-0199协议,在XMPPFraneWork框架下,封装了XMPPAutoPing和XMPPPing两个类都可以使用,因为XMPPAutoPing已经组合进了XMPPPing类,所以XMPPAutoPing使用起来更加方便。

//初始化并启动ping
-(void)autoPingProxyServer:(NSString*)strProxyServer
{
    _xmppAutoPing = [[XMPPAutoPingalloc] init];
    [_xmppAutoPingactivate:_xmppStream];
    [_xmppAutoPingaddDelegate:selfdelegateQueue:  dispatch_get_main_queue()];
    _xmppAutoPing.respondsToQueries = YES;
    _xmppAutoPing.pingInterval=2;//ping 间隔时间
    if (nil != strProxyServer)
    {
       _xmppAutoPing.targetJID = [XMPPJID jidWithString: strProxyServer ];//设置ping目标服务器,如果为nil,则监听socketstream当前连接上的那个服务器
    }
}
//卸载监听
 [_xmppAutoPing   deactivate];
  [_xmppAutoPing   removeDelegate:self];
   _xmppAutoPing = nil;
//ping XMPPAutoPingDelegate的委托方法:
- (void)xmppAutoPingDidSendPing:(XMPPAutoPing *)sender
{
    NSLog(@"- (void)xmppAutoPingDidSendPing:(XMPPAutoPing *)sender");
}
- (void)xmppAutoPingDidReceivePong:(XMPPAutoPing *)sender
{
    NSLog(@"- (void)xmppAutoPingDidReceivePong:(XMPPAutoPing *)sender");
}
   
- (void)xmppAutoPingDidTimeout:(XMPPAutoPing *)sender
{
    NSLog(@"- (void)xmppAutoPingDidTimeout:(XMPPAutoPing *)sender");
}

demo下载:http://download.csdn.net/detail/shaoting19910730/9178475

原文地址:https://www.cnblogs.com/shaoting/p/4875917.html