Как заставить push-уведомления firebase работать

Я потратил около 2 дней на изучение документации firebase, задавая вопросы и просматривая другие, следуя руководствам YouTube/medium и многому другому. Вот проект github с моим полным делегатом приложения, так что весь мой код виден.

Но мне еще предстоит выяснить, как отправить уведомление из облачного обмена сообщениями Firebase. Я настроил службы Apple push, получил и ввел ключ аутентификации в firebase и написал код в своем делегате приложения, увиденном здесь

Может ли кто-нибудь дать мне знать, что мне нужно сделать, чтобы заставить его работать? Что я пропустил?

Вот мой didRegisterForRemoteNotificationsWithDeviceToken метод:

    func application(_ application: UIApplication, didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data) {
    Messaging.messaging().apnsToken = deviceToken
    let tokenParts = deviceToken.map { data in String(format: "%02.2hhx", data) }
    let token = tokenParts.joined()
    print("Device Token: \(token)")

}

Обновлять:

Я снова просмотрел всю эту документацию по firebase, и по-прежнему не работает отправка уведомлений из обмена сообщениями в огненном облаке.

Однако ясно, что код работает, учитывая, что у меня есть 2 строки, которые сохраняют токен в firebase, успешно сохраняя токен.

ты реализовал didRegisterForRemoteNotificationsWithDeviceToken? Не могли бы вы предоставить этот фрагмент кода?

Vyacheslav 28.05.2019 23:46

проверить обновление...

user11182249 28.05.2019 23:55

не забудьте проверить режим производства/разработки

Vyacheslav 28.05.2019 23:59

Где мне это найти? @Вячеслав

user11182249 29.05.2019 00:02

проверить сертификаты и панель администратора firebase

Vyacheslav 29.05.2019 00:05

не могу найти что вы имеете в виду я думаю вы имеете в виду служебные аккаунты но там нет ничего продакшна /dev @Vyacheslav

user11182249 29.05.2019 00:15

Вы проверили сертификаты в консоли разработчика Apple. У вас есть платная подписка для разработчиков?

Aakash Dave 29.05.2019 00:16

Да, я деловой человек, @AakashDave. У меня есть два сертификата, Apple push и обычный

user11182249 29.05.2019 00:19

Отлично, так что технически есть 3 сертификата. [1. Сертификат разработки, 2. Сертификат производства, 3. Ключ аутентификации APNS]. У вас есть все?

Aakash Dave 29.05.2019 00:21

@AakashDave под всем этим написано 2 сертификата, но под ключами я . иметь ключ авторизации APNs

user11182249 29.05.2019 00:29

И если вы подключили все правильно, это должно быть хорошо. С какой проблемой вы столкнулись и в чем ошибка, если она есть

Aakash Dave 29.05.2019 00:35

Я не получаю никаких ошибок, проблема в том, что в консоли обмена облачными сообщениями firebase, когда я делаю и отправляю уведомление, оно не отображается на моем устройстве. @АакашДэйв

user11182249 29.05.2019 00:36

Вы распечатали результат для каждого шага вашего кода nodejs, чтобы быть уверенным, что все проходит?

Adrian Murray 29.05.2019 01:17

Я прошу об этом, чтобы выяснить, является ли это проблемой с настройкой облачной функции или с настройкой вашего устройства. Обеспечение того, чтобы консоль выдавала журнал в конце функции, означало бы, что это определенно проблема на стороне устройства. Кроме того, не все проблемы с облачными функциями будут отображаться как ошибки.

Adrian Murray 29.05.2019 01:24

Хотя у меня есть серверная часть nodejs, с которой я намерен иметь возможность обрабатывать уведомления, до этого момента все, что я пытаюсь сделать, — это иметь возможность отправлять уведомления через консоль обмена облачными сообщениями firebase. @Адриан Мюррей

user11182249 29.05.2019 01:26

Ах, так что определенно проблема на вашем устройстве. Вы хотя бы получаете запрос на получение уведомлений на устройстве?

Adrian Murray 29.05.2019 01:29

При первой загрузке приложения (удаление и повторный запуск) да. @Адриан Мюррей

user11182249 29.05.2019 01:29

Давайте продолжить обсуждение в чате.

Adrian Murray 29.05.2019 01:37

На самом деле, у вас есть права проверки возможностей?? Push-уведомления включены, фоновый режим и удаленные уведомления? Кроме того, у вас есть последняя версия GoogleService-Info.plist? Можете ли вы проверить файл находится в нужном месте.

m1sh0 02.06.2019 09:14

@ m1sh0 да ко всему, что вы упомянули. За исключением: нужно ли загружать новый google.plist после установки облачного обмена сообщениями?

user11182249 02.06.2019 17:31

Что вы имеете в виду под установкой облачных сообщений, как в модулях? Я думаю, вам это нужно, если вы меняете конфигурации в консоли firebase. Вы ничего не потеряете, если замените его последним из консоли. И насколько я понимаю у вас есть все остальное. Я не помню, был ли у вас ключ apns или сертификаты (пожалуйста, настройте свой ключ apns, я думаю, это лучше).

m1sh0 02.06.2019 17:42

@ m1sh0 Я перезагрузил его, и он все равно не работает.

user11182249 02.06.2019 19:20
Стоит ли изучать PHP в 2023-2024 годах?
Стоит ли изучать PHP в 2023-2024 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
2
22
1 398
2
Перейти к ответу Данный вопрос помечен как решенный

Ответы 2

Предполагая, что у вас правильно настроен ключ аутентификации APNS в консоли firebase, и учитывая предоставленную вами реализацию appDelegate, единственный фрагмент кода, который я вижу отсутствующим, — это создание FIRInstanceId.

Согласно документации Firebase здесь :

Instance ID provides a unique identifier for each app instance and a mechanism >to authenticate and authorize actions (for example, sending an FCM message).

Once an InstanceID is generated, the library periodically sends information >about the application and the device where it’s running to the Firebase >backend.

И здесь

To send or receive messages, the app must get a registration token from >FIRInstanceID. This token authorizes an app server to send messages to an app >instance.

Непосредственно перед тем, как вы зарегистрируетесь для удаленного уведомления в своем приложение сделалоконецзапуск..., добавьте это перед application.registerForRemoteNotifications()

InstanceID.instanceID().instanceID { (result, error) in
    if let error = error {
        print("Error fetching remote instange ID: \(error)")
    } else if let result = result {
        print("Remote instance ID token: \(result.token)")
    }
}

Позвольте мне знать, если это помогает :)

Ответ принят как подходящий

У меня была такая проблема в прошлом, и для меня это исправило проверку того, что bundle ID одинаков как для ключа авторизации, так и для проекта огня, а также в Xcode.

Также, чтобы убедиться, что код делегата вашего приложения верен, просто используйте Пример Firebase:

import UIKit
import UserNotifications
import Firebase

@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {

    var window: UIWindow?
    let gcmMessageIDKey = "gcm.message_id"

    func application(_ application: UIApplication,
                     didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {

        FirebaseApp.configure()

        // [START set_messaging_delegate]
        Messaging.messaging().delegate = self
        // [END set_messaging_delegate]

        // Register for remote notifications. This shows a permission dialog on first run, to
        // show the dialog at a more appropriate time move this registration accordingly.
        // [START register_for_notifications]
        if #available(iOS 10.0, *) {
            // For iOS 10 display notification (sent via APNS)
            UNUserNotificationCenter.current().delegate = self

            let authOptions: UNAuthorizationOptions = [.alert, .badge, .sound]
            UNUserNotificationCenter.current().requestAuthorization(
                options: authOptions,
                completionHandler: {_, _ in })
        } else {
            let settings: UIUserNotificationSettings =
                UIUserNotificationSettings(types: [.alert, .badge, .sound], categories: nil)
            application.registerUserNotificationSettings(settings)
        }

        application.registerForRemoteNotifications()

        // [END register_for_notifications]

        return true
    }

    // [START receive_message]
    func application(_ application: UIApplication, didReceiveRemoteNotification userInfo: [AnyHashable: Any]) {
        // If you are receiving a notification message while your app is in the background,
        // this callback will not be fired till the user taps on the notification launching the application.
        // TODO: Handle data of notification

        // With swizzling disabled you must let Messaging know about the message, for Analytics
        // Messaging.messaging().appDidReceiveMessage(userInfo)

        // Print message ID.
        if let messageID = userInfo[gcmMessageIDKey] {
            print("Message ID: \(messageID)")
        }

        // Print full message.
        print(userInfo)
    }

    func application(_ application: UIApplication, didReceiveRemoteNotification userInfo: [AnyHashable: Any],
                     fetchCompletionHandler completionHandler: @escaping (UIBackgroundFetchResult) -> Void) {
        // If you are receiving a notification message while your app is in the background,
        // this callback will not be fired till the user taps on the notification launching the application.
        // TODO: Handle data of notification

        // With swizzling disabled you must let Messaging know about the message, for Analytics
        // Messaging.messaging().appDidReceiveMessage(userInfo)

        // Print message ID.
        if let messageID = userInfo[gcmMessageIDKey] {
            print("Message ID: \(messageID)")
        }

        // Print full message.
        print(userInfo)

        completionHandler(UIBackgroundFetchResult.newData)
    }
    // [END receive_message]

    func application(_ application: UIApplication, didFailToRegisterForRemoteNotificationsWithError error: Error) {
        print("Unable to register for remote notifications: \(error.localizedDescription)")
    }

    // This function is added here only for debugging purposes, and can be removed if swizzling is enabled.
    // If swizzling is disabled then this function must be implemented so that the APNs token can be paired to
    // the FCM registration token.
    func application(_ application: UIApplication, didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data) {
        print("APNs token retrieved: \(deviceToken)")

        // With swizzling disabled you must set the APNs token here.
        // Messaging.messaging().apnsToken = deviceToken
    }
}

// [START ios_10_message_handling]
@available(iOS 10, *)
extension AppDelegate : UNUserNotificationCenterDelegate {

    // Receive displayed notifications for iOS 10 devices.
    func userNotificationCenter(_ center: UNUserNotificationCenter,
                                willPresent notification: UNNotification,
                                withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void) {
        let userInfo = notification.request.content.userInfo

        // With swizzling disabled you must let Messaging know about the message, for Analytics
        // Messaging.messaging().appDidReceiveMessage(userInfo)

        // Print message ID.
        if let messageID = userInfo[gcmMessageIDKey] {
            print("Message ID: \(messageID)")
        }

        // Print full message.
        print(userInfo)

        // Change this to your preferred presentation option
        completionHandler([])
    }

    func userNotificationCenter(_ center: UNUserNotificationCenter,
                                didReceive response: UNNotificationResponse,
                                withCompletionHandler completionHandler: @escaping () -> Void) {
        let userInfo = response.notification.request.content.userInfo
        // Print message ID.
        if let messageID = userInfo[gcmMessageIDKey] {
            print("Message ID: \(messageID)")
        }

        // Print full message.
        print(userInfo)

        completionHandler()
    }
}
// [END ios_10_message_handling]


extension AppDelegate : MessagingDelegate {
    // [START refresh_token]
    func messaging(_ messaging: Messaging, didReceiveRegistrationToken fcmToken: String) {
        print("Firebase registration token: \(fcmToken)")

        let dataDict:[String: String] = ["token": fcmToken]
        NotificationCenter.default.post(name: Notification.Name("FCMToken"), object: nil, userInfo: dataDict)
        // TODO: If necessary send token to application server.
        // Note: This callback is fired at each app startup and whenever a new token is generated.
    }
    // [END refresh_token]

    // [START ios_10_data_message]
    // Receive data messages on iOS 10+ directly from FCM (bypassing APNs) when the app is in the foreground.
    // To enable direct data messages, you can set Messaging.messaging().shouldEstablishDirectChannel to true.
    func messaging(_ messaging: Messaging, didReceive remoteMessage: MessagingRemoteMessage) {
        print("Received data message: \(remoteMessage.appData)")
    }
    // [END ios_10_data_message]
}

Другие вопросы по теме