forked from Ivasoft/mattermost-mobile
113 lines
4.0 KiB
Swift
113 lines
4.0 KiB
Swift
import UserNotifications
|
|
import UploadAttachments
|
|
|
|
class NotificationService: UNNotificationServiceExtension {
|
|
|
|
var contentHandler: ((UNNotificationContent) -> Void)?
|
|
var bestAttemptContent: UNMutableNotificationContent?
|
|
|
|
var retryIndex = 0
|
|
|
|
override func didReceive(_ request: UNNotificationRequest, withContentHandler contentHandler: @escaping (UNNotificationContent) -> Void) {
|
|
self.contentHandler = contentHandler
|
|
|
|
let fibonacciBackoffsInSeconds = [1.0, 2.0, 3.0, 5.0, 8.0]
|
|
|
|
func fetchReceipt(notificationId: String, receivedAt: Int, type: String, postId: String, idLoaded: Bool ) -> Void {
|
|
if (self.retryIndex >= fibonacciBackoffsInSeconds.count) {
|
|
contentHandler(self.bestAttemptContent!)
|
|
return
|
|
}
|
|
|
|
UploadSession.shared.notificationReceipt(
|
|
notificationId: notificationId,
|
|
receivedAt: receivedAt,
|
|
type: type,
|
|
postId: postId,
|
|
idLoaded: idLoaded) { data, response, error in
|
|
if let httpResponse = response as? HTTPURLResponse, httpResponse.statusCode != 200 {
|
|
contentHandler(self.bestAttemptContent!)
|
|
return
|
|
}
|
|
|
|
guard let data = data, error == nil else {
|
|
if (idLoaded) {
|
|
// Receipt retrieval failed. Kick off retries.
|
|
let backoffInSeconds = fibonacciBackoffsInSeconds[self.retryIndex]
|
|
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + backoffInSeconds, execute: {
|
|
fetchReceipt(
|
|
notificationId: notificationId,
|
|
receivedAt: Date().millisecondsSince1970,
|
|
type: type,
|
|
postId: postId,
|
|
idLoaded: idLoaded
|
|
)
|
|
})
|
|
|
|
self.retryIndex += 1
|
|
}
|
|
return
|
|
}
|
|
self.processResponse(data: data, bestAttemptContent: self.bestAttemptContent!, contentHandler: contentHandler)
|
|
}
|
|
}
|
|
|
|
bestAttemptContent = (request.content.mutableCopy() as? UNMutableNotificationContent)
|
|
if let bestAttemptContent = bestAttemptContent {
|
|
let ackId = (bestAttemptContent.userInfo["ack_id"] ?? "") as! String
|
|
let type = (bestAttemptContent.userInfo["type"] ?? "") as! String
|
|
let postId = (bestAttemptContent.userInfo["post_id"] ?? "") as! String
|
|
let idLoaded = (bestAttemptContent.userInfo["id_loaded"] ?? false) as! Bool
|
|
|
|
fetchReceipt(
|
|
notificationId: ackId,
|
|
receivedAt: Date().millisecondsSince1970,
|
|
type: type,
|
|
postId: postId,
|
|
idLoaded: idLoaded
|
|
)
|
|
}
|
|
}
|
|
|
|
func processResponse(data: Data, bestAttemptContent: UNMutableNotificationContent, contentHandler: ((UNNotificationContent) -> Void)?) {
|
|
let json = try? JSONSerialization.jsonObject(with: data) as! [String: Any]
|
|
if let json = json {
|
|
if let message = json["message"] as? String {
|
|
bestAttemptContent.body = message
|
|
}
|
|
if let channelName = json["channel_name"] as? String {
|
|
bestAttemptContent.title = channelName
|
|
}
|
|
|
|
let userInfoKeys = ["channel_name", "team_id", "sender_id", "root_id", "override_username", "override_icon_url", "from_webhook"]
|
|
for key in userInfoKeys {
|
|
if let value = json[key] as? String {
|
|
bestAttemptContent.userInfo[key] = value
|
|
}
|
|
}
|
|
}
|
|
if let contentHandler = contentHandler {
|
|
contentHandler(bestAttemptContent)
|
|
}
|
|
}
|
|
|
|
override func serviceExtensionTimeWillExpire() {
|
|
// Called just before the extension will be terminated by the system.
|
|
// Use this as an opportunity to deliver your "best attempt" at modified content, otherwise the original push payload will be used.
|
|
if let contentHandler = contentHandler, let bestAttemptContent = bestAttemptContent {
|
|
contentHandler(bestAttemptContent)
|
|
}
|
|
}
|
|
}
|
|
|
|
extension Date {
|
|
var millisecondsSince1970: Int {
|
|
return Int((self.timeIntervalSince1970 * 1000.0).rounded())
|
|
}
|
|
|
|
init(milliseconds: Int) {
|
|
self = Date(timeIntervalSince1970: TimeInterval(milliseconds) / 1000)
|
|
}
|
|
}
|