Проблема с созданием комнаты xmpp muc: код 503 (служба недоступна)

Мой код для создания комнаты:

XMPPRoomCoreDataStorage *rosterstorage = [[XMPPRoomCoreDataStorage alloc] init];
XMPPRoom *xmppRoom = [[XMPPRoom alloc] initWithRoomStorage:rosterstorage jid:[XMPPJID jidWithString:@"[email protected]/groupchat"] dispatchQueue:dispatch_get_main_queue()];

[xmppRoom activate:[[self appDelegate] xmppStream]];
if ([xmppRoom preJoinWithNickname:@"nameToCreateRoom"]) 
{
    NSLog(@"room created");
    [xmppRoom joinRoomUsingNickname:self.userName history:nil];
}
[xmppRoom fetchConfigurationForm];
[xmppRoom configureRoomUsingOptions:nil];
[xmppRoom addDelegate:[self appDelegate] delegateQueue:dispatch_get_main_queue()];

Отлаживать:

2012-08-03 07:46:29.204 iPhoneXMPP[9887:fb03] room created
2012-08-03 07:46:29:230 iPhoneXMPP[9887:15003] SEND: <iq type="get" to="[email protected]" id="B793062B-0E09-492F-BC0F-703503AAA664"><query xmlns="http://jabber.org/protocol/muc#owner"/></iq>
2012-08-03 07:46:29:237 iPhoneXMPP[9887:15003] SEND: <iq type="set" to="[email protected]" id="392D5BFC-707B-4F68-A829-56F949F4E96D"><query xmlns="http://jabber.org/protocol/muc#owner"><x xmlns="jabber:x:data" type="submit"/></query></iq>
2012-08-03 07:46:29:326 iPhoneXMPP[9887:14f03] SEND: <presence to="[email protected]"><x xmlns="http://jabber.org/protocol/muc"/><x xmlns="vcard-temp:x:update"><photo>91217a961321f8f6380ea2feefd0632353ad296c</photo></x><c xmlns="http://jabber.org/protocol/caps" hash="sha-1" node="http://code.google.com/p/xmppframework" ver="VyOFcFX6+YNmKssVXSBKGFP0BS4="/></presence>
2012-08-03 07:46:29:327 iPhoneXMPP[9887:14f03] RECV: <iq xmlns="jabber:client" from="[email protected]" to="[email protected]/41068195801343976386548353" type="error" id="B793062B-0E09-492F-BC0F-703503AAA664"><query xmlns="http://jabber.org/protocol/muc#owner"/><error code="503" type="cancel"><service-unavailable xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"/></error></iq>
2012-08-03 07:46:29:343 iPhoneXMPP[9887:fb03] iPhoneXMPPAppDelegate: xmppStream:didReceiveIQ:
2012-08-03 07:46:29:421 iPhoneXMPP[9887:15003] RECV: <iq xmlns="jabber:client" from="[email protected]" to="[email protected]/41068195801343976386548353" type="error" id="392D5BFC-707B-4F68-A829-56F949F4E96D"><query xmlns="http://jabber.org/protocol/muc#owner"><x xmlns="jabber:x:data" type="submit"/></query><error code="503" type="cancel"><service-unavailable xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"/></error></iq>
2012-08-03 07:46:29:440 iPhoneXMPP[9887:fb03] iPhoneXMPPAppDelegate: xmppStream:didReceiveIQ:

Я вижу, что он создает/присоединяется к [email protected], а не к [email protected]/groupchat, как я указал. Читал, что проблема скорее всего в этом. Однако я указал для полного джида, так что я потерялся.

Заранее спасибо всем, кто поможет.


person user1561639    schedule 03.08.2012    source источник


Ответы (1)


Во-первых, посмотрите здесь XEP-0045: многопользовательский чат.
Как видите, сначала вам нужно узнать, какие возможности имеет ваш пользователь (XMPPJID) на сервере Jabber.

Для этого отправьте следующую команду на ваш сервер Jabber:

<iq from='[email protected]/resource' id='some_expression' to='jabber.server.com' type='get'>
    <query xmlns='http://jabber.org/protocol/disco#items'/>
</iq>

или написать в target-c с использованием библиотечных функций XMPP:

NSError *error = nil;
NSXMLElement *query = [[NSXMLElement alloc] initWithXMLString:@"<query xmlns='http://jabber.org/protocol/disco#items'/>" 
                                                        error:&error];
XMPPIQ *iq = [XMPPIQ iqWithType:@"get" 
                             to:[XMPPJID jidWithString:@"jabber.server.com"] 
                      elementID:[xmppStream generateUUID] child:query];
[xmppStream sendElement:iq];

Теперь прослушайте ответ сервера в делегате XMPPStream - (BOOL)xmppStream:(XMPPStream *)sender didReceiveIQ:(XMPPIQ *)iq, и ответ сервера должен быть примерно таким:

<iq from='jabber.server.com' id='some_expression' to='[email protected]/resource' type='result'>
    <query xmlns='http://jabber.org/protocol/disco#items'>
        <item jid='im.jabber.server.com' name='Instant Message Service'/>
        <item jid='conference.jabber.server.com' name='Chatroom Service'/>
    </query>
</iq>

или цель c:

- (BOOL)xmppStream:(XMPPStream *)sender didReceiveIQ:(XMPPIQ *)iq
{

    if([iq isResultIQ])
    {
        if([iq elementForName:@"query" xmlns:@"http://jabber.org/protocol/disco#items"])
        {
            NSLog(@"Jabber Server's Capabilities: %@", [iq XMLString]);
        }
    }
}

Теперь для каждого возвращенного элемента отправьте IQ на свой сервер, чтобы узнать его свойства и выяснить, какой из них является типом конференции, примерно так:

<iq from='[email protected]/resource' id='some_expression' to='conference.jabber.server.com' type='get'>
    <query xmlns='http://jabber.org/protocol/disco#info'/>
</iq>

или в цели c:

- (BOOL)xmppStream:(XMPPStream *)sender didReceiveIQ:(XMPPIQ *)iq
{

    if([iq isResultIQ])
    {
        if([iq elementForName:@"query" xmlns:@"http://jabber.org/protocol/disco#items"])
        {
            NSXMLElement *query = [iq childElement];
            NSArray *items = [query children];
            for(NSXMLElement *item in items)
            {
                NSError *error = nil;
                NSXMLElement *sendQuery = [[NSXMLElement alloc] initWithXMLString:@"<query xmlns='http://jabber.org/protocol/disco#info'/>" 
                                                                            error:&error];
                XMPPIQ *sendIQ = [XMPPIQ iqWithType:@"get" 
                                                 to:[XMPPJID jidWithString:[item attributeStringValueForName:@"jid"]] 
                                          elementID:[xmppStream generateUUID] 
                                              child:sendQuery];
                [xmppStream sendElement:sendIQ];
            }
        }
    }
}

Слушайте ответы от сервера:

<iq from='conference.jabber.server.com' id='some_expression' to='[email protected]/resource' type='result'>
    <query xmlns='http://jabber.org/protocol/disco#info'>
        <identity category='conference' name='Server Group Chat Service' type='text'/>
        <feature var='http://jabber.org/protocol/muc'/>
    </query>
</iq>

и взять домен группового чата от личности с category:conference

- (BOOL)xmppStream:(XMPPStream *)sender didReceiveIQ:(XMPPIQ *)iq
{

    if([iq isResultIQ])
    {
        if([iq elementForName:@"query" xmlns:@"http://jabber.org/protocol/disco#items"])
        {
            ...
        }
        else if([iq elementForName:@"query" xmlns:@"http://jabber.org/protocol/disco#info"])
        {
            NSXMLElement *query = [iq childElement];
            NSXMLElement *identity = [query elementForName:@"identity"];
            if([[identity attributeStringValueForName:@"category"] isEqualToString:@"conference"])
            {
                groupChatDomain = [iq fromStr];
            }
        }
    }
}

Наконец, когда мы получили домен группового чата, мы можем создать чат-комнату примерно так:

XMPPJID *chatRoomJID = [XMPPJID jidWithUser:@"chat_room" 
                                     domain:groupChatDomain 
                                   resource:@"user"];
XMPPRoomMemoryStorage *roomMemoryStorage = [[XMPPRoomMemoryStorage alloc] init];
XMPPRoom *xmppRoom = [[XMPPRoom alloc] initWithRoomStorage:roomMemoryStorage
                                                       jid:roomChatJID
                                             dispatchQueue:dispatch_get_main_queue()];
[xmppRoom activate:xmppStream];
[xmppRoom addDelegate:self delegateQueue:dispatch_get_main_queue()];
[xmppRoom joinRoomUsingNickname:user history:nil];

и добавьте протокол <XMPPRoomDelegate> в ваш контроллер представления и его делегатов:

- (void)xmppRoomDidCreate:(XMPPRoom *)sender
- (void)xmppRoomDidDestroy:(XMPPRoom *)sender
- (void)xmppRoom:(XMPPRoom *)sender didConfigure:(XMPPIQ *)iqResult
- (void)xmppRoom:(XMPPRoom *)sender didNotConfigure:(XMPPIQ *)iqResult
- (void)xmppRoomDidJoin:(XMPPRoom *)sender
- (void)xmppRoomDidLeave:(XMPPRoom *)sender
- (void)xmppRoom:(XMPPRoom *)sender occupantDidJoin:(XMPPJID *)occupantJID withPresence:(XMPPPresence *)presence
- (void)xmppRoom:(XMPPRoom *)sender occupantDidLeave:(XMPPJID *)occupantJID withPresence:(XMPPPresence *)presence
- (void)xmppRoom:(XMPPRoom *)sender didReceiveMessage:(XMPPMessage *)message fromOccupant:(XMPPJID *)occupantJID

Примечание. Перед тем, как пригласить других пользователей в комнату чата, вы должны отправить и подтвердить настройки комнаты (можно пригласить других пользователей, но нельзя отправлять сообщения).
Поэтому вы можете сделать это после создания комнаты (вызывается делегат - (void)xmppRoomDidCreate:(XMPPRoom *)sender) или когда ваш пользователь присоединился (вызывается делегат - (void)xmppRoomDidJoin:(XMPPRoom *)sender) к комнате чата.

Чтобы отправить и подтвердить конфигурацию комнаты, выполните одно из следующих действий:

- (void)xmppRoomDidCreate:(XMPPRoom *)sender
{
    [sender configureRoomUsingOptions:nil];
}

or

- (void)xmppRoomDidJoin:(XMPPRoom *)sender
{    
    [sender configureRoomUsingOptions:nil];
}

Отправьте nil, чтобы принять параметры по умолчанию, или вы можете отправить IQ с синтаксисом, как показано ниже, на свой сервер:

<iq type='set' from='[email protected]/resource' id='some_expression' to='[email protected]'>
    <query xmlns='http://jabber.org/protocol/muc#owner'>
        <x xmlns='jabber:x:data' type='submit'>
            <field var='FORM_TYPE'>
                <value>http://jabber.org/protocol/muc#roomconfig</value>
            </field>
            <field var='muc#roomconfig_roomname'>
                <value>My Chat Room</value>
            </field>
              .
              .
              .
        <x>
</query>
</iq>

или объективный код c:

NSError *error = nil;
NSXMLElement *query = [[NSXMLElement alloc] initWithXMLString:@"<query xmlns='http://jabber.org/protocol/muc#owner'/>" 
                                                        error:&error];
NSXMLElement *x = [NSXMLElement elementWithName:@"x" 
                                          xmlns:@"jabber:x:data"];
[x addAttributeWithName:@"type" stringValue:@"submit"];
NSXMLElement *field1 = [NSXMLElement elementWithName:@"field"];
[field1 addAttributeWithName:@"var" stringValue:@"FORM_TYPE"];
NSXMLElement *value1 = [NSXMLElement elementWithName:@"value" 
                                         stringValue:@"http://jabber.org/protocol/muc#roomconfig"];
[field1 addChild:value1];

NSXMLElement *field2 = [NSXMLElement elementWithName:@"field"];
[field2 addAttributeWithName:@"var" stringValue:@"muc#roomconfig_roomname"];
NSXMLElement *value2 = [NSXMLElement elementWithName:@"value" 
                                         stringValue:@"My Chat Room"];
[field2 addChild:value2];

//Add other fields you need, just like field1 and field2

[x addChild:field1];
[x addChild:field2];

[query addChild:x];

NSXMLElement *roomOptions = [NSXMLElement elementWithName:@"iq"];
[roomOptions addAttributeWithName:@"type" stringValue:@"set"];
[roomOptions addAttributeWithName:@"id" stringValue:[xmppStream generateUUID];
[roomOptions addAttributeWithName:@"to" stringValue:@"[email protected]"];

[roomOptions addChild:query];

[sender configureRoomUsingOptions:roomOptions];

а список всех возможных полей формы конфигурации находится здесь

person Sihad Begovic    schedule 05.02.2013
comment
Спасибо, Сихад. Это очень подробный и полезный пост! - person CodeBrew; 02.12.2013
comment
Привет Sihad, спасибо за подробный код! у меня есть пара вопросов, я пытался создать xmpproom, но мои делегаты xmpproom по какой-то причине не срабатывают, у меня есть мой код здесь stackoverflow.com/questions/20765641/ - person DevCali; 25.12.2013
comment
В файле .h вашего контроллера представления вы добавили ‹XMPPRoomDelegate›, он должен быть примерно таким: @interface YourViewController : UIViewController ‹..., XMPPRoomDelegate› - person Sihad Begovic; 25.12.2013
comment
Привет, я успешно реализовал комнату, но почему-то возникли следующие проблемы. Может быть, я ищу подобное решение, например WhatsApp. 1) пользователь все еще не открывает приложение, как позволить ему добавиться в группу и отправить push-уведомление, как создатель добавил вас в группу 2) даже когда пользователь не открывает приложение, сообщения начинают приходить к тому времени, когда пользователь был добавлен 3 ) автономные сообщения для всех пользователей, которые покинули группу, но все еще связаны с группой 4) как выйти из группы и выйти из группы так же, как выход, но получить список участников по-прежнему дает этот пользователь. - person Vicky Dhas; 26.11.2015
comment
5) не может автоматически принимать пользователя на сервере для приглашений, поскольку предполагается, что необходимо отправить доступное присутствие, но это приведет к сбросу всех его автономных сообщений, хранящихся на сервере, с потерей его данных. Любая помощь очень ценится по этим проблемам, оставшимся в моем приложении. - person Vicky Dhas; 26.11.2015
comment
Я успешно все реализовал: 1) создать комнату 2) настроить 3) присоединиться к комнате 4) пригласить 5) отправить сообщение 6) при выходе из сети и возвращении присоединиться к комнате автоматически, используя логику закладок, планирую попробовать это. - person Vicky Dhas; 26.11.2015
comment
Очень полезное решение. - person KSR; 17.02.2017
comment
@SihadBegovic... Спасибо за это полезное решение. На моей стороне группа была создана, но другие участники не получили никаких указаний относительно новой группы, к которой им нужно присоединиться. Я что-то упускаю? - person Kiran Jasvanee; 16.09.2020