向ios发送推送通知,以便与离线用户openfire xmpp进行聊天

时间:2013-04-19 04:07:37

标签: ios push-notification xmpp chat openfire

我有一个使用openfire的ios聊天应用程序,我需要做的是当消息(1)因任何原因无法发送时发送推送通知,(2)app处于暂停状态,即不能自己生成通知。

我已经在stackoverflow和其他地方阅读了大部分有关此问题的相关问题/建议,我已经为我的问题找到了一些解决方案。我不是ios开发人员,也不是几天之前我对openfire或xmpp一无所知,所以我担心我对事情的理解可能不完整,我的解决方案可能存在缺陷。

请确认我对它的理解并建议我是否遗漏了某些内容或是否有更好的方法。还请建议实施下面列出的特定解决方案的复杂程度。

这里的挑战是确定何时需要推送以及启动流程的位置,所以

1)一种方法是使用xmpp的xep-0184实现来检查消息是否已传递。要做到这一点,我们应该在ios数据库中有一些带有消息的传递标志,当从另一端收到传递的响应时更新。因此,我们需要在一段时间后检查此标志,如果传递的状态为false,则使用该消息启动推送过程。看起来是一个复杂的解决方案(等待响应..检查标志有一些时滞......不是很令人印象深刻)

2)更直接的方法是在openfire中做一些事情,当openfire无法传递它将其存储在离线表中的消息时,我们可以对该部分进行一些拦截并使用消息启动推送过程。这看起来是正确的方法,但我真的害怕在openfire中获得那么多东西并改变一些东西(也可能很容易,有人和openfire一起工作的人可以说出来吗?)

3)这是我的最后一招,这不是一个解决方案......但如果我无法在预期的时间范围内(从现在起一周内)正确地做到这一点,我们计划发送推送通知给所有人消息。 oppenfire将关注正常聊天,同时从我们的服务器发送每个消息的推送但是当应用程序在前台时,我们做一些事情来处理不需要显示的额外推送消息,否则只要有一个推送就会收到推送信息。你们怎么看待这种暂时的方式(我们当然必须尽快改变这一点),这是可行的(或者我在这里也遗漏了一些东西)。

P.S。任何人都可以告诉Whatsapp和其他流行的应用程序如何处理这个问题?

非常感谢你的帮助。

2 个答案:

答案 0 :(得分:6)

XMPP在整个XMPP会话期间需要持久套接字连接或“持久”BOSH连接。我认为您的挑战是iOS不允许您在后台运行您的应用和套接字。每当您的iOS应用程序进入后台时iOS都会终止您的套接字连接,并且您的Openfire服务器会终止您的XMPP会话。这意味着用户离线。这也是此用户的传入邮件进入离线存储的原因。

很抱歉这个回复,但你建议的所有3个解决方案都是糟糕的黑客;-)。如果你想提出一个好的解决方案,你必须深入到XMPP和iOS。 1周是非常短的时间。

  

有人能说出Whatsapp和其他热门应用是如何处理这个问题的吗?

他们保持XMPP会话活着。这仅适用于经过高度修改的XMPP服务器,其中一些“XMPP客户端代理”可以在您的应用处于后台或两者兼有时保持会话运行。

答案 1 :(得分:2)

我有一个解决方案。

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
    self.ary_UserStatus = [NSMutableArray array];
    NSMutableArray *ary_TempFromUserDefaults = [NSMutableArray array];
    ary_TempFromUserDefaults = [[NSUserDefaults standardUserDefaults] valueForKey:@"KejdoUserStatus"];

    if ([ary_TempFromUserDefaults count]>0)
    {
         self.ary_UserStatus = [[NSUserDefaults standardUserDefaults]    valueForKey:@"KejdoUserStatus"];
       }

      self.df_UserStatus = [[NSDateFormatter alloc] init];
      [self.df_UserStatus setDateFormat: @"hh:mm a MM/dd/yyyy"];
}

- (void)xmppStream:(XMPPStream *)sender didReceivePresence:(XMPPPresence *)presence
{
    DDLogVerbose(@"%@: %@ - %@", THIS_FILE, THIS_METHOD, [presence fromStr]);

    NSString *str_UserName = [[presence from] user];
    NSString *str_LastSeenDate = [self.df_UserStatus stringFromDate:[NSDate date]];
    NSMutableDictionary *mdic_UserPresence = [[NSMutableDictionary alloc] init];

    [mdic_UserPresence setValue:str_UserName forKey:@"Name"];
    [mdic_UserPresence setValue:str_LastSeenDate forKey:@"Date"];
    [mdic_UserPresence setValue:[presence type] forKey:@"Type"];

    if ([self.ary_UserStatus count]>0)
    {
        int index;
        BOOL IS_exist=FALSE;
        for (int i=0; i<[self.ary_UserStatus count]; i++)
        {
            NSString *str_UserFromArray = [[self.ary_UserStatus objectAtIndex:i] valueForKey:@"Name"];
            if ([str_UserName isEqualToString:str_UserFromArray])
            {
                IS_exist = TRUE;
                index = i;
                [[NSUserDefaults standardUserDefaults] setObject:str_UserName forKey:@"Status"];
            }
            else
            {
            }

        }
        if (IS_exist) {
            [self.ary_UserStatus replaceObjectAtIndex:index withObject:mdic_UserPresence];
        }
        else
        {
            [self.ary_UserStatus addObject:mdic_UserPresence];
        }
    }
    else
    {
        [self.ary_UserStatus addObject:mdic_UserPresence];
    }


    [[NSUserDefaults standardUserDefaults] setObject:self.ary_UserStatus forKey:@"KejdoUserStatus"];
    [[NSUserDefaults standardUserDefaults] synchronize];

    [[NSNotificationCenter defaultCenter] postNotificationName:@"UserStatusChangeNotification" object:self];

}

无论您在聊天中向其他用户发送消息的位置。这样做

 if(appDelegate.ary_UserStatus.count>0)
    {
     for (int i=0; i<[appDelegate.ary_UserStatus count]; i++)
     {
      if ([jid.user isEqualToString:[NSString stringWithFormat:@"%@",[[appDelegate.ary_UserStatus objectAtIndex:i] valueForKey:@"Name"]]])
        {
         if ([[[appDelegate.ary_UserStatus objectAtIndex:i] valueForKey:@"Type"] isEqualToString:@"available"])
           {
                                // Do something like table reload.
                                break;
             }
               else
                    [self sendPushNotification];
           }
         }
       }
        else
            [self sendPushNotification];