Bonnes pratiques de gestion des jetons d'enregistrement FCM

Si vous utilisez les API FCM pour créer des requêtes d'envoi par programmation, vous pouvez : constatez qu'au fil du temps, vous gaspillez des ressources en envoyant des messages à des comptes appareils avec des jetons d'enregistrement obsolètes. Cette situation peut affecter les données de diffusion des messages indiquées dans la console Firebase ou les données exportées vers BigQuery, ce qui se traduit par une baisse importante (mais non valide) des taux de diffusion. Ce guide présente certaines mesures que vous pouvez prendre pour vous assurer d'un ciblage efficace des messages et de rapports de diffusion valides.

Jetons d'enregistrement obsolètes et expirés

Les jetons d'enregistrement obsolètes sont des jetons associés à des appareils inactifs qui ne se sont pas connectés à FCM depuis plus d'un mois. Au fur et à mesure que le temps passe, et moins susceptible de se reconnecter à FCM. Envoyer un message Il est peu probable que les envois et les distributions de sujet pour ces jetons obsolètes livrés.

Un jeton peut devenir obsolète pour plusieurs raisons. Par exemple, l'appareil auquel le jeton est associé peuvent être perdus, détruits ou mis dans le stockage ou oubliés.

Lorsque les jetons obsolètes atteignent 270 jours d'inactivité, FCM les considère comme des jetons expirés. Une fois qu'un jeton expire, FCM le marque comme non valide et refuse les envois qui lui sont destinés. Cependant, FCM émet un nouveau jeton pour l'application. dans les rares cas où l'appareil se reconnecte et que l'application est ouverte.

Bonnes pratiques de base

Il existe quelques pratiques fondamentales que vous devez suivre dans toute application qui utilise Des API FCM permettant de créer des requêtes d'envoi de manière automatisée Le meilleur pratiques sont:

  • Récupérez les jetons d'enregistrement à partir de FCM et stockez-les sur votre serveur. Le serveur a un rôle important à jouer : il doit suivre le jeton de chaque client et conserver une liste à jour des jetons actifs. Nous vous recommandons vivement d'implémenter un code temporel de jeton dans votre code et vos serveurs, et de le mettre à jour à intervalles réguliers.
  • Maintenez la fraîcheur des jetons et supprimez les jetons obsolètes. En plus de supprimer les jetons que FCM ne considère plus comme valides, vous pouvez surveiller d'autres signes indiquant que les jetons sont obsolètes et les supprimer de manière proactive. Ce guide décrit certaines des options qui s'offrent à vous pour y parvenir.

Récupérer et stocker des jetons d'enregistrement

Au démarrage initial de votre application, le SDK FCM génère un jeton d'enregistrement pour l'instance de l'application cliente. Il s'agit du jeton que vous devez inclure dans des demandes d'envoi ciblées à partir de l'API ou ajouter des abonnés à des thèmes à des fins de ciblage des sujets.

Nous vous recommandons vivement de récupérer ce jeton au démarrage initial de votre application et de l'enregistrer sur votre serveur d'application avec un code temporel. Ce code temporel doit être implémentée par votre code et vos serveurs, car elle n'est pas fournie FCM.

Il est également important d'enregistrer le jeton sur le serveur et de mettre à jour le code temporel chaque fois qu'il change, par exemple :

  • L'appli est restaurée sur un nouvel appareil
  • L'utilisateur désinstalle ou réinstalle l'application
  • L'utilisateur efface les données de l'application.
  • L'application redevient active une fois que FCM a expiré son jeton existant.

Exemple : stocker des jetons et des codes temporels dans Cloud Firestore

Par exemple, vous pouvez utiliser Cloud Firestore pour stocker des jetons dans une collection appelée fcmTokens. Chaque ID de document de la collection correspond à un ID utilisateur, et le document stocke le jeton d'enregistrement actuel et son code temporel de la dernière mise à jour. Utilisez la fonction set comme indiqué dans cet exemple Kotlin :

    /**
     * Persist token to third-party servers.
     *
     * Modify this method to associate the user's FCM registration token with any server-side account
     * maintained by your application.
     *
     * @param token The new token.
     */
    private fun sendTokenToServer(token: String?) {
        // If you're running your own server, call API to send token and today's date for the user

        // Example shown below with Firestore
        // Add token and timestamp to Firestore for this user
        val deviceToken = hashMapOf(
            "token" to token,
            "timestamp" to FieldValue.serverTimestamp(),
        )
        // Get user ID from Firebase Auth or your own server
        Firebase.firestore.collection("fcmTokens").document("myuserid")
            .set(deviceToken)
    }

Chaque fois qu'un jeton est récupéré, il est stocké dans Cloud Firestore en appelant sendTokenToServer :

    /**
     * Called if the FCM registration token is updated. This may occur if the security of
     * the previous token had been compromised. Note that this is called when the
     * FCM registration token is initially generated so this is where you would retrieve the token.
     */
    override fun onNewToken(token: String) {
        Log.d(TAG, "Refreshed token: $token")

        // If you want to send messages to this application instance or
        // manage this apps subscriptions on the server side, send the
        // FCM registration token to your app server.
        sendTokenToServer(token)
    }
        var token = Firebase.messaging.token.await()

        // Check whether the retrieved token matches the one on your server for this user's device
        val preferences = this.getPreferences(Context.MODE_PRIVATE)
        val tokenStored = preferences.getString("deviceToken", "")
        lifecycleScope.launch {
            if (tokenStored == "" || tokenStored != token)
            {
                // If you have your own server, call API to send the above token and Date() for this user's device

                // Example shown below with Firestore
                // Add token and timestamp to Firestore for this user
                val deviceToken = hashMapOf(
                    "token" to token,
                    "timestamp" to FieldValue.serverTimestamp(),
                )

                // Get user ID from Firebase Auth or your own server
                Firebase.firestore.collection("fcmTokens").document("myuserid")
                    .set(deviceToken).await()
            }
        }

Maintenir la fraîcheur des jetons et supprimer les jetons obsolètes

Il n'est pas toujours simple de déterminer si un jeton est à jour ou obsolète. Pour couvrir tous les cas, vous devez définir un seuil à partir duquel vous considérerez les jetons comme obsolètes. Par défaut, FCM considère qu'un jeton est obsolète si son application ne s'est pas connectée depuis un mois. Tout jeton datant de plus d'un mois est probablement un appareil inactif. Sinon, un appareil actif aurait actualisé son jeton.

Selon votre cas d'utilisation, un mois peut être trop court ou trop long. C'est donc à vous de déterminer les critères qui vous conviennent.

Détecter les réponses de jetons non valides du backend FCM

Assurez-vous de détecter les réponses de jeton non valides de FCM et de supprimer de votre système tous les jetons d'enregistrement connus comme non valides ou expirés. Avec l'API HTTP v1, ces messages d'erreur peuvent indiquer que votre requête d'envoi ciblait des jetons non valides ou expirés :

  • UNREGISTERED (HTTP 404)
  • INVALID_ARGUMENT (HTTP 400)

Si vous êtes certain que la charge utile du message est valide et que vous recevez ces réponses pour un jeton ciblé, vous pouvez supprimer sans risque votre enregistrement de ce car il ne sera plus jamais valide. Par exemple, pour supprimer les jetons non valides À partir de Cloud Firestore, vous pouvez déployer et exécuter une fonction semblable à celle-ci:

    // Registration token comes from the client FCM SDKs
    const registrationToken = 'YOUR_REGISTRATION_TOKEN';

    const message = {
    data: {
        // Information you want to send inside of notification
    },
    token: registrationToken
    };

    // Send message to device with provided registration token
    getMessaging().send(message)
    .then((response) => {
        // Response is a message ID string.
    })
    .catch((error) => {
        // Delete token for user if error code is UNREGISTERED or INVALID_ARGUMENT.
        if (errorCode == "messaging/registration-token-not-registered") {
            // If you're running your own server, call API to delete the
            token for the user

            // Example shown below with Firestore
            // Get user ID from Firebase Auth or your own server
            Firebase.firestore.collection("fcmTokens").document(user.uid).delete()
        }
    });

FCM ne renverra une réponse de jeton non valide que si le jeton a expiré après 270 jours ou si un client s'est désinscrit explicitement. Si vous devez suivre plus précisément l'obsolescence en fonction de vos propres définitions, vous pouvez supprimer de manière proactive les jetons d'enregistrement obsolètes.

Mettre à jour les jetons régulièrement

Nous vous recommandons de récupérer et de mettre à jour régulièrement tous les jetons d'enregistrement sur votre serveur. Pour cela, vous devez:

  • Ajoutez une logique d'application à votre application cliente pour récupérer le jeton actuel à l'aide de la méthode l'appel d'API approprié (comme token(completion): pour les plates-formes Apple getToken() pour Android), puis envoyer le jeton actuel à votre serveur d'applications pour le stocker (avec code temporel). Il peut s'agir d'une tâche mensuelle configurée pour couvrir tous les clients ou jetons.
  • Ajouter une logique de serveur pour mettre à jour l'horodatage du jeton à intervalles réguliers que le jeton ait changé ou non.

Voici un exemple de logique Android pour la mise à jour de jetons à l'aide de WorkManager voir Gérer les jetons Cloud Messaging sur le blog Firebase.

Quel que soit le schéma de synchronisation que vous suivez, veillez à mettre à jour les jetons régulièrement. Une fréquence de mise à jour d'une fois par mois offre un bon équilibre entre l'impact sur la batterie et la détection des jetons d'enregistrement inactifs. En effectuant cette actualisation, vous vous assurez également que tout appareil qui devient inactif actualisera son enregistrement lorsqu'il redeviendra actif. L'actualisation plus fréquente ne présente aucun avantage. qu'une fois par semaine.

Supprimer les jetons d'enregistrement obsolètes

Avant d'envoyer des messages à un appareil, assurez-vous que le code temporel du jeton d'enregistrement de l'appareil se trouve dans la période de validité. Par exemple : pourrait implémenter Cloud Functions for Firebase pour exécuter une vérification quotidienne afin de s'assurer que se situe dans une période d'obsolescence définie, telle que const EXPIRATION_TIME = 1000 * 60 * 60 * 24 * 30;, puis supprimez les jetons obsolètes:

exports.pruneTokens = functions.pubsub.schedule('every 24 hours').onRun(async (context) => {
  // Get all documents where the timestamp exceeds is not within the past month
  const staleTokensResult = await admin.firestore().collection('fcmTokens')
      .where("timestamp", "<", Date.now() - EXPIRATION_TIME)
      .get();
  // Delete devices with stale tokens
  staleTokensResult.forEach(function(doc) { doc.ref.delete(); });
});

Se désabonner des jetons obsolètes des sujets

Si vous utilisez des sujets, vous pouvez également désenregistrer les jetons obsolètes des sujets auxquels ils sont abonnés. Pour cela, deux étapes sont nécessaires:

  1. Votre application doit se réabonner aux thèmes une fois par mois et dès que le les modifications de jetons d'enregistrement. Cela forme une solution d'auto-réparation, où le Les abonnements réapparaissent automatiquement lorsqu'une application est réactivée.
  2. Si une instance d'application est inactive pendant un mois (ou pendant votre propre période d'inactivité), vous devez la désabonner des sujets à l'aide du SDK Admin Firebase pour supprimer le mappage du jeton sur le sujet du backend FCM.

L'avantage de ces deux étapes est que vos fanouts se produisent plus rapidement, car il y a moins de jetons obsolètes à diffuser, et vos instances d'application obsolètes se réabonnent automatiquement une fois qu'elles sont à nouveau actives.

Mesurer la réussite des diffusions

Pour obtenir une image la plus précise possible de la distribution des messages, il est préférable de n'envoyer des messages qu'aux instances d'application utilisées activement. Ceci est particulièrement important si vous envoyer régulièrement des messages à des sujets comptant un grand nombre d'abonnés ; si un qui sont inactifs, ce qui affecte la diffusion les statistiques peuvent être significatives au fil du temps.

Avant de cibler des messages sur un jeton, tenez compte des points suivants :

  • Google Analytics, les données capturées dans BigQuery ou d'autres signaux de suivi indiquent-ils que le jeton est actif ?
  • Les tentatives d'envoi précédentes ont-elles échoué régulièrement sur une période donnée ?
  • Le jeton d'enregistrement a-t-il été mis à jour sur vos serveurs au cours du mois dernier ?
  • Pour les appareils Android, l'API FCM Data est-elle disponible ? signalent un pourcentage élevé d'échecs de distribution des messages dus à droppedDeviceInactive?

Pour en savoir plus sur la diffusion, consultez Comprendre la distribution des messages