我正在使用 PushKit 和 CallKit 开发一个 VoIP 应用程序。我知道类似的问题在不同的论坛上经常被问到,但不幸的是从来没有真正得到回答。此外,这些帖子中的大部分都来自 2015 年左右,所以我认为很可能已经发生了很多变化,因为有人可能同时想到了如何让它发挥作用。
什么是工作:
当应用程序在前台运行或发送到后台时,它会使用沙盒服务器以及生产服务器接收推送通知。
我在 iOS 11 和 Xcode 9 上,需要在“功能”中启用常规推送通知,并手动将 voip 后台模式添加到 info.plist 以便调用委托(delegate)方法。单独链接 CallKit 和 PushKit 不足以接收通知。我还启用了后台音频、后台提取和远程通知。我还在进行推送注册表注册之前创建了一个后台任务,并在收到推送 token 后结束该任务。
什么不起作用:
重启后唤醒应用程序或强制退出来电。
虽然我在 Instruments 事件监视器中看到进程在传入调用时启动,但看起来有些地方不起作用,因为我没有收到传入调用 ui。
问题:
- 当应用程序未运行时,传入通知的入口点是什么?
-pushRegistry:didReceiveIncomingPushWithPayload:forType:
或可能(仅)-application:didFinishLaunchingWithOptions:
? - 我将我的凭据存储在钥匙串(keychain)中以向服务器注册。是否有可能在应用程序启动期间查询钥匙串(keychain)以获取某些东西为时过早?
更新:
我自己想出了一个解决方案。这更多是客户端和服务器之间的时间问题。所以这对我有用。
- 从启动应用程序的服务器发送推送
- 在收到推送后,给应用一些时间启动并在 sip 服务器上重新注册
- 使用最新的注册信息向客户发送INVITE
同样重要的是不要过早调用推送通知委托(delegate)的完成处理程序,这会使应用程序重新进入休眠状态,并且永远不会收到 INVITE。
也可能有帮助,并且可以使场景更宽容一些。如果您的 sip 服务器多次发送邀请,请确保每次都使用最新的注册信息。意思是.. 假设第一个 INVITE 过早发送到一些过时的注册信息,并且客户端仍然启动尝试更新服务器上的信息......确保下一个使用更新的信息。一些 sip 服务器简单地使用每次迭代可用的第一个信息。因此,您将永远等待,而不会在客户端收到任何信息。
最佳答案
每次您的应用程序启动时,它都应该实例化一个 PKPushRegistry
实例并设置其委托(delegate)。这应该响应 UIApplication
-application:applicationDidFinishLaunching…
回调之一,而不是(例如)在 View Controller 的 -viewWillAppear/-viewDidAppear
回调,因为当您的应用程序在后台启动时,只会调用前者。一些开发人员遇到了一些错误,在这些错误中,他们只在 View Controller 或其他代码路径中实例化了一个 PKPushRegistry
,这些代码路径仅在应用程序在 前台 中启动时才相关(即在显示时可见 UI),因此请注意始终实例化此对象,无论它是背景还是前景。
一旦您创建了一个 PKPushRegistry
并分配了它的委托(delegate),它将有资格接收传入推送通知有效负载的委托(delegate)回调,并且您的应用程序应该能够处理这些有效负载并处理来电。
关于ios - VoIP、PushKit 和应用程序唤醒,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47369658/