Nếu sử dụng API FCM để tạo yêu cầu gửi theo phương thức lập trình, bạn có thể thấy rằng theo thời gian, bạn đang lãng phí tài nguyên bằng cách gửi thông báo đến các thiết bị không hoạt động có mã thông báo đăng ký không còn hiệu lực. Tình huống này có thể ảnh hưởng đến dữ liệu phân phối thông báo được báo cáo trong bảng điều khiển Firebase hoặc dữ liệu được xuất sang BigQuery, cho thấy tỷ lệ phân phối giảm đáng kể (nhưng không thực sự hợp lệ). Hướng dẫn này thảo luận về một số biện pháp bạn có thể thực hiện để đảm bảo nhắm đến thông điệp một cách hiệu quả và báo cáo phân phối hợp lệ.
Mã thông báo đăng ký đã hết hạn và không còn hiệu lực
Mã thông báo đăng ký không hợp lệ là mã thông báo được liên kết với những thiết bị không hoạt động và chưa kết nối với FCM trong hơn một tháng. Theo thời gian, thiết bị ngày càng ít có khả năng kết nối lại với FCM. Thông báo gửi và fanout chủ đề cho các mã thông báo cũ này có thể sẽ không bao giờ được gửi.
Có một số lý do khiến mã thông báo có thể trở nên cũ. Ví dụ: thiết bị được liên kết với mã thông báo có thể bị mất, bị phá huỷ hoặc được cất vào kho và bị quên.
Khi mã thông báo không còn hiệu lực và không hoạt động trong 270 ngày, FCM sẽ coi đó là mã thông báo đã hết hạn. Sau khi mã thông báo hết hạn, FCM sẽ đánh dấu mã thông báo đó là không hợp lệ và từ chối gửi đến mã thông báo đó. Tuy nhiên, FCM sẽ phát hành một mã thông báo mới cho phiên bản ứng dụng trong trường hợp hiếm gặp là thiết bị kết nối lại và ứng dụng được mở.
Các phương pháp cơ bản hay nhất
Bạn nên tuân theo một số phương pháp cơ bản trong mọi ứng dụng sử dụng API FCM để tạo yêu cầu gửi theo phương thức lập trình. Các phương pháp hay nhất chính là:
- Truy xuất mã thông báo đăng ký từ FCM và lưu trữ mã thông báo đó trên máy chủ của bạn. Một vai trò quan trọng của máy chủ là theo dõi mã thông báo của từng ứng dụng và duy trì danh sách mã thông báo đang hoạt động được cập nhật. Bạn nên triển khai dấu thời gian của mã thông báo trong mã và máy chủ của mình, đồng thời cập nhật dấu thời gian này theo định kỳ.
- Duy trì trạng thái mới của mã thông báo và xoá các mã thông báo cũ. Ngoài việc xoá những mã thông báo mà FCM không còn coi là hợp lệ, bạn có thể muốn theo dõi các dấu hiệu khác cho thấy mã thông báo đã trở nên cũ và chủ động xoá chúng. Hướng dẫn này trình bày một số lựa chọn để bạn đạt được mục tiêu này.
Truy xuất và lưu trữ mã thông báo đăng ký
Khi khởi động ứng dụng lần đầu, SDK FCM sẽ tạo mã thông báo đăng ký cho phiên bản ứng dụng khách. Đây là mã thông báo mà bạn phải đưa vào các yêu cầu gửi có mục tiêu từ API hoặc thêm vào các lượt đăng ký chủ đề để nhắm đến các chủ đề.
Bạn nên để ứng dụng truy xuất mã thông báo này khi khởi động lần đầu và lưu mã thông báo vào máy chủ ứng dụng cùng với dấu thời gian. Mã và máy chủ của bạn phải triển khai dấu thời gian này vì FCM SDK không cung cấp dấu thời gian này cho bạn.
Ngoài ra, bạn cần lưu mã thông báo vào máy chủ và cập nhật dấu thời gian bất cứ khi nào mã thông báo thay đổi, chẳng hạn như khi:
- Ứng dụng được khôi phục trên thiết bị mới
- Người dùng gỡ cài đặt hoặc cài đặt lại ứng dụng
- Người dùng xoá dữ liệu ứng dụng
- Ứng dụng sẽ hoạt động trở lại sau khi FCM hết hạn mã thông báo hiện có
Ví dụ: lưu trữ mã thông báo và dấu thời gian trong Cloud Firestore
Ví dụ: bạn có thể dùng Cloud Firestore để lưu trữ mã thông báo trong một tập hợp có tên là fcmTokens
. Mỗi mã nhận dạng tài liệu trong bộ sưu tập tương ứng với một mã nhận dạng người dùng và tài liệu lưu trữ mã thông báo đăng ký hiện tại cùng dấu thời gian cập nhật gần đây nhất. Dùng hàm set
như trong ví dụ này về 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)
}
Bất cứ khi nào một mã thông báo được truy xuất, mã thông báo đó sẽ được lưu trữ trong Cloud Firestore bằng cách gọi 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()
}
}
Duy trì trạng thái mới của mã thông báo và xoá mã thông báo cũ
Việc xác định xem mã thông báo còn mới hay đã hết hạn không phải lúc nào cũng đơn giản. Để bao gồm tất cả các trường hợp, bạn nên áp dụng một ngưỡng để xem xét khi nào mã thông báo trở nên cũ. Theo mặc định, FCM coi mã thông báo là cũ nếu phiên bản ứng dụng của mã thông báo đó chưa kết nối trong một tháng. Mọi mã thông báo đã được tạo hơn một tháng trước đều có khả năng là của một thiết bị không hoạt động; nếu không, một thiết bị đang hoạt động sẽ làm mới mã thông báo của mình.
Tuỳ thuộc vào trường hợp sử dụng của bạn, một tháng có thể quá ngắn hoặc quá dài, vì vậy, bạn phải tự xác định tiêu chí phù hợp với mình.
Phát hiện các phản hồi mã thông báo không hợp lệ từ phần phụ trợ FCM
Hãy nhớ phát hiện các phản hồi mã thông báo không hợp lệ từ FCM và phản hồi bằng cách xoá mọi mã thông báo đăng ký không hợp lệ hoặc đã hết hạn khỏi hệ thống của bạn. Với HTTP v1 API, những thông báo lỗi này có thể cho biết rằng yêu cầu gửi của bạn nhắm đến các mã thông báo không hợp lệ hoặc đã hết hạn:
UNREGISTERED
(HTTP 404)INVALID_ARGUMENT
(HTTP 400)
Nếu bạn chắc chắn rằng tải trọng thông báo hợp lệ và bạn nhận được một trong hai phản hồi này cho một mã thông báo được nhắm đến, thì bạn có thể xoá bản ghi mã thông báo này vì mã thông báo đó sẽ không bao giờ hợp lệ nữa. Ví dụ: để xoá các mã thông báo không hợp lệ khỏi Cloud Firestore, bạn có thể triển khai và chạy một hàm như sau:
// 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 sẽ chỉ trả về một phản hồi mã thông báo không hợp lệ nếu mã thông báo hết hạn sau 270 ngày hoặc nếu một ứng dụng khách huỷ đăng ký một cách rõ ràng. Nếu cần theo dõi độ cũ chính xác hơn theo định nghĩa của riêng bạn, bạn có thể chủ động xoá mã thông báo đăng ký cũ.
Thường xuyên cập nhật mã thông báo
Bạn nên định kỳ truy xuất và cập nhật tất cả mã thông báo đăng ký trên máy chủ của mình. Bạn phải:
- Thêm logic ứng dụng vào ứng dụng khách để truy xuất mã thông báo hiện tại bằng lệnh gọi API thích hợp (chẳng hạn như
token(completion):
cho nền tảng Apple hoặcgetToken()
cho Android), sau đó gửi mã thông báo hiện tại đến máy chủ ứng dụng của bạn để lưu trữ (kèm theo dấu thời gian). Đây có thể là một công việc hằng tháng được định cấu hình để bao gồm tất cả các ứng dụng hoặc mã thông báo. - Thêm logic máy chủ để cập nhật dấu thời gian của mã thông báo theo các khoảng thời gian đều đặn, bất kể mã thông báo có thay đổi hay không.
Để xem ví dụ về logic Android để cập nhật mã thông báo bằng WorkManager, hãy xem bài viết Quản lý mã thông báo của dịch vụ Gửi thông báo qua đám mây trên blog Firebase.
Dù bạn tuân theo mẫu thời gian nào, hãy nhớ cập nhật mã thông báo định kỳ. Tần suất cập nhật là mỗi tháng một lần, giúp cân bằng giữa mức tiêu thụ pin và việc phát hiện các mã thông báo đăng ký không hoạt động. Bằng cách làm mới này, bạn cũng đảm bảo rằng mọi thiết bị chuyển sang trạng thái không hoạt động sẽ làm mới thông tin đăng ký khi hoạt động trở lại. Việc làm mới thường xuyên hơn hằng tuần không mang lại lợi ích gì.
Xoá mã thông báo đăng ký không còn hiệu lực
Trước khi gửi thông báo đến một thiết bị, hãy đảm bảo rằng dấu thời gian của mã thông báo đăng ký của thiết bị nằm trong khoảng thời gian cho phép mã thông báo không còn hiệu lực. Ví dụ: bạn có thể triển khai Cloud Functions for Firebase để chạy quy trình kiểm tra hằng ngày nhằm đảm bảo rằng dấu thời gian nằm trong khoảng thời gian cũ được xác định, chẳng hạn như const
EXPIRATION_TIME = 1000 * 60 * 60 * 24 * 30;
, rồi xoá các mã thông báo cũ:
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(); });
});
Huỷ đăng ký các mã thông báo không còn hiệu lực khỏi các chủ đề
Nếu sử dụng các chủ đề, bạn cũng có thể huỷ đăng ký mã thông báo cũ khỏi các chủ đề mà chúng đã đăng ký. Việc này bao gồm hai bước:
- Ứng dụng của bạn nên đăng ký lại các chủ đề mỗi tháng một lần và bất cứ khi nào mã thông báo đăng ký thay đổi. Điều này tạo thành một giải pháp tự khắc phục, trong đó các gói thuê bao sẽ tự động xuất hiện lại khi ứng dụng hoạt động trở lại.
- Nếu một phiên bản ứng dụng không hoạt động trong một tháng (hoặc khoảng thời gian không hoạt động của riêng bạn), bạn nên huỷ đăng ký phiên bản đó khỏi các chủ đề bằng Firebase Admin SDK để xoá mối liên kết giữa mã thông báo và chủ đề khỏi phần phụ trợ FCM.
Lợi ích của hai bước này là các lệnh truyền tin của bạn sẽ diễn ra nhanh hơn vì có ít mã thông báo không hợp lệ hơn để truyền tin, đồng thời các phiên bản ứng dụng không hợp lệ sẽ tự động đăng ký lại khi hoạt động trở lại.
Đo lường mức độ thành công khi phân phối
Để có được bức tranh chính xác nhất về việc gửi tin nhắn, tốt nhất là bạn chỉ nên gửi tin nhắn đến các phiên bản ứng dụng đang được sử dụng. Điều này đặc biệt quan trọng nếu bạn thường xuyên gửi thông báo đến các chủ đề có nhiều người đăng ký; nếu một phần trong số những người đăng ký đó thực sự không hoạt động, thì tác động đến số liệu thống kê về việc gửi thông báo có thể rất đáng kể theo thời gian.
Trước khi nhắm đến một mã thông báo, hãy cân nhắc:
- Google Analytics, dữ liệu được ghi lại trong BigQuery hoặc các tín hiệu theo dõi khác có cho biết mã thông báo đang hoạt động không?
- Các lần giao hàng trước đó có liên tục không thành công trong một khoảng thời gian không?
- Mã thông báo đăng ký có được cập nhật trên máy chủ của bạn trong tháng vừa qua không?
- Đối với các thiết bị Android, FCM Data API có báo cáo tỷ lệ thất bại cao khi gửi thông báo do
droppedDeviceInactive
không?
Để biết thêm thông tin về việc gửi, hãy xem bài viết Tìm hiểu về việc gửi tin nhắn.