在 UNNotificationServiceExtension 中下载大图

Download larger images in UNNotificationServiceExtension

我正在使用 UNNotificationServiceExtension 在 iOS 10 上创建带有图像附件的丰富通知。除非要下载的图像稍大 (e.g. 7MB),否则一切都适用于较小的图像。

在这种情况下,下载开始但从未完成,"best attempt" 通知几乎在下载开始后立即显示。

根据 Apple's docs,图像最大可达 10MB,因此大小无关紧要:)

我确实实施了 serviceExtensionTimeWillExpire,其中 iOS 应该通知我必须尽快交付内容,但这种方法 不是 打电话所以我想知道发生了什么事。

更新 1 我还注意到,将同一图像存储在本地时效果很好。所以看起来问题出在下载图像上。但是,如前所述,下载几乎立即被终止。

实施很简单

- (BOOL)handleNotificationRequest:(UNNotificationRequest *)request withContentHandler:(void (^)(UNNotificationContent *_Nonnull))contentHandler
{

    self.bestAttemptContent = (UNMutableNotificationContent *) [request.content mutableCopy];

    UNMutableNotificationContent *content = [self.bestAttemptContent mutableCopy];

    NSString *urlString = [content.userInfo valueForKeyPath:@"attachment-url"];

    NSLog(@"Downloading notification attachment completed with: %@", url.absoluteString);
    NSURLSessionDataTask *task = [[NSURLSession sharedSession] dataTaskWithURL:url completionHandler:^(NSData *_Nullable data, NSURLResponse *_Nullable response, NSError *_Nullable error) {

        NSLog(@"Downloading notification attachment %@", error == nil ? @"success" : [NSString stringWithFormat:@"error: %@", error]);

        NSError *fileError;
        UNNotificationAttachment *attachment = [UNNotificationAttachment d360_createWithFileName:url.lastPathComponent identifier:url.absoluteString data:data options:nil error:&fileError];

        if (!attachment) {
            NSLog(@"Could not create local attachment file: %@", fileError);
            contentHandler(content);
            return;
        }

        NSLog(@"Adding attachment: %@", attachment);

        NSMutableArray *attachments = content.attachments ? [content.attachments mutableCopy] : [NSMutableArray array];

        [attachments addObject:attachment];

        content.attachments = attachments;
        contentHandler(content);


    }];
    [task resume];


    return YES;
}

- (void)serviceExtensionTimeWillExpire
{
    NSLog(@"Service extension expired. Using best attempt content  %@", self.bestAttemptContent);
    self.contentHandler(self.bestAttemptContent);
}

当我在 XCode 中调试时,我看到以下日志:

2017-03-30 14:51:43.723669 D360TestAppNotificationExtension[3393:398992] [D360Extension]: Handling notification request content
2017-03-30 14:51:43.724103 D360TestAppNotificationExtension[3393:398992] [D360Extension]: Downloading notification attachment: https://upload.wikimedia.org/wikipedia/commons/b/b2/Bled_Castle_05.jpg
Program ended with exit code: 0

Program ended with exit code: 0 可疑。知道发生了什么事吗?

谢谢 一月

所以解决方案是使用 NSURLSessionDownloadTask 而不是 NSURLSessionDataTask。在这种情况下,下载内容将保存到一个临时位置并使用较少的内存。然后可以直接从临时文件创建 UNNotificationAttachment

请注意 UNNotificationAttachment 需要读取具有支持的文件扩展名的文件,因此我只是将远程文件名附加到本地临时文件 URL

NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]];
NSURLSessionDownloadTask *task = [session downloadTaskWithURL:url completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {

    NSLog(@"Downloading notification attachment completed with %@", error == nil ? @"success" : [NSString stringWithFormat:@"error: %@", error]);

    NSError *fileError;
    // create a local URL with extension
    NSURL *urlWithExtension = [NSURL fileURLWithPath:[location.path stringByAppendingString:url.lastPathComponent]];

    if (![[NSFileManager defaultManager] moveItemAtURL:location toURL:urlWithExtension error:&fileError]) {
        NSLog(@"Could not append  local attachment file name: %@", fileError);
        contentHandler(content);
        return;
    }

    UNNotificationAttachment *attachment = [UNNotificationAttachment attachmentWithIdentifier:url.absoluteString
                                                                                          URL:urlWithExtension options:nil
                                                                                        error:&fileError];

    if (!attachment) {
        NSLog(@"Could not create local attachment file: %@", fileError);
        contentHandler(content);
        return;
    }

    NSLog(@"Adding attachment: %@", attachment);

    NSMutableArray *attachments = content.attachments ? [content.attachments mutableCopy] : [NSMutableArray array];

    [attachments addObject:attachment];

    content.attachments = attachments;

    contentHandler(content);


}];
[task resume];