iOS13:替代 VOIP 推送通知(企业),现在 iOS13 已将其终止,可在后台静默与应用程序通信
iOS13: Alternative to VOIP push notifications (enterprise) to silently communicate with an app in background now that iOS13 killed it
因此,在我们的企业环境中,大约 4 年以来,我们一直愉快地使用 VOIP 推送通知来远程访问用户的平板电脑,以进行维护和远程数据修复。与常规 APNS 不同,VOIP 推送通知即使不是 运行 也会访问该应用程序。我早该想到苹果会在某个时候杀死它。
有人知道私人 API 绕过 pushkit 调用 reportNewIncomingCallWithUUID 的要求,这会导致全屏调用 UI,或者我无法想到的另一种即使被杀死也无法在后台访问应用程序的机制 - Notification Service Extensions 不会工作(我相信)因为它只会工作用于屏幕消息。谢谢
如果你不打算将它发布到 Apple store,并且不关心私有 API 的使用(它可以随时更改,破坏你的代码),你可以使用 method swizzling 来更改系统调用的使应用程序崩溃的函数的实现。
就我而言,我有一个具有 swift 和 objc 互操作性的项目。我是这样做的:
- 使用 gist 的内容创建一个名为
PKPushRegistry+PKPushFix_m.h
的文件。
- 将其包含在您的 swift 桥接头中。
其他选项(也不能在 Apple Store 应用程序上使用)正在使用 Xcode 10 构建它,或者使用 Xcode 11 手动覆盖 iOS SDK 13 iOS SDK 12.
副本
这里是要点的内容,以防以后无法使用:
#import <objc/runtime.h>
#import <PushKit/PushKit.h>
@interface PKPushRegistry (Fix)
@end
@implementation PKPushRegistry (Fix)
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Class class = [self class];
SEL originalSelector = @selector(_terminateAppIfThereAreUnhandledVoIPPushes);
SEL swizzledSelector = @selector(doNotCrash);
Method originalMethod = class_getInstanceMethod(class, originalSelector);
Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
// When swizzling a class method, use the following:
// Class class = object_getClass((id)self);
// ...
// Method originalMethod = class_getClassMethod(class, originalSelector);
// Method swizzledMethod = class_getClassMethod(class, swizzledSelector);
BOOL didAddMethod =
class_addMethod(class,
originalSelector,
method_getImplementation(swizzledMethod),
method_getTypeEncoding(swizzledMethod));
if (didAddMethod) {
class_replaceMethod(class,
swizzledSelector,
method_getImplementation(originalMethod),
method_getTypeEncoding(originalMethod));
} else {
method_exchangeImplementations(originalMethod, swizzledMethod);
}
[super load];
});
}
#pragma mark - Method Swizzling
- (void)doNotCrash {
NSLog(@"Unhandled VoIP Push");
}
@end
因此,在我们的企业环境中,大约 4 年以来,我们一直愉快地使用 VOIP 推送通知来远程访问用户的平板电脑,以进行维护和远程数据修复。与常规 APNS 不同,VOIP 推送通知即使不是 运行 也会访问该应用程序。我早该想到苹果会在某个时候杀死它。
有人知道私人 API 绕过 pushkit 调用 reportNewIncomingCallWithUUID 的要求,这会导致全屏调用 UI,或者我无法想到的另一种即使被杀死也无法在后台访问应用程序的机制 - Notification Service Extensions 不会工作(我相信)因为它只会工作用于屏幕消息。谢谢
如果你不打算将它发布到 Apple store,并且不关心私有 API 的使用(它可以随时更改,破坏你的代码),你可以使用 method swizzling 来更改系统调用的使应用程序崩溃的函数的实现。
就我而言,我有一个具有 swift 和 objc 互操作性的项目。我是这样做的:
- 使用 gist 的内容创建一个名为
PKPushRegistry+PKPushFix_m.h
的文件。 - 将其包含在您的 swift 桥接头中。
其他选项(也不能在 Apple Store 应用程序上使用)正在使用 Xcode 10 构建它,或者使用 Xcode 11 手动覆盖 iOS SDK 13 iOS SDK 12.
副本这里是要点的内容,以防以后无法使用:
#import <objc/runtime.h>
#import <PushKit/PushKit.h>
@interface PKPushRegistry (Fix)
@end
@implementation PKPushRegistry (Fix)
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Class class = [self class];
SEL originalSelector = @selector(_terminateAppIfThereAreUnhandledVoIPPushes);
SEL swizzledSelector = @selector(doNotCrash);
Method originalMethod = class_getInstanceMethod(class, originalSelector);
Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
// When swizzling a class method, use the following:
// Class class = object_getClass((id)self);
// ...
// Method originalMethod = class_getClassMethod(class, originalSelector);
// Method swizzledMethod = class_getClassMethod(class, swizzledSelector);
BOOL didAddMethod =
class_addMethod(class,
originalSelector,
method_getImplementation(swizzledMethod),
method_getTypeEncoding(swizzledMethod));
if (didAddMethod) {
class_replaceMethod(class,
swizzledSelector,
method_getImplementation(originalMethod),
method_getTypeEncoding(originalMethod));
} else {
method_exchangeImplementations(originalMethod, swizzledMethod);
}
[super load];
});
}
#pragma mark - Method Swizzling
- (void)doNotCrash {
NSLog(@"Unhandled VoIP Push");
}
@end