從應用程序服務器或受信任環境發送到 FCM 的請求必須經過授權。請注意已棄用的舊版 HTTP API 和 HTTP v1 API 授權之間的重要區別:
- FCM HTTP v1 API 使用短期 OAuth 2.0 訪問令牌授權請求。要創建此令牌,您可以使用 Google 應用程序默認憑據(在 Google 服務器環境中)和/或從為服務帳戶生成的 JSON 私鑰文件手動獲取所需的憑據。如果您使用 Firebase Admin SDK 發送消息,該庫會為您處理令牌。
- 已棄用的舊協議只能使用從 Firebase 控制台獲取的長期 API 密鑰。
授權 HTTP v1 發送請求
根據服務器環境的詳細信息,使用以下策略的組合來授權對 Firebase 服務的服務器請求:
- Google 應用程序默認憑據 (ADC)
- 服務帳戶 JSON 文件
- 從服務帳戶派生的短暫 OAuth 2.0 訪問令牌
如果您的應用程序在 Compute Engine、Google Kubernetes Engine、App Engine 或 Cloud Functions(包括 Cloud Functions for Firebase)上運行,請使用應用程序默認憑據 (ADC)。 ADC 使用您現有的默認服務帳戶來獲取憑據來授權請求,並且 ADC 通過環境變量GOOGLE_APPLICATION_CREDENTIALS啟用靈活的本地測試。為了實現授權流程的最完全自動化,請將 ADC 與 Admin SDK 服務器庫一起使用。
如果您的應用程序在非 Google 服務器環境上運行,則需要從 Firebase 項目下載服務帳戶 JSON 文件。只要您有權訪問包含私鑰文件的文件系統,就可以使用環境變量GOOGLE_APPLICATION_CREDENTIALS通過這些手動獲取的憑據來授權請求。如果您缺乏此類文件訪問權限,則必須在代碼中引用服務帳戶文件 - 由於存在暴露憑據的風險,因此應格外小心。
使用 ADC 提供憑據
Google 應用程序默認憑據 (ADC) 按以下順序檢查您的憑據:
ADC 檢查環境變量GOOGLE_APPLICATION_CREDENTIALS是否已設置。如果設置了該變量,ADC 將使用該變量指向的服務帳戶文件。
如果未設置環境變量,ADC 將使用 Compute Engine、Google Kubernetes Engine、App Engine 和 Cloud Functions 為在這些服務上運行的應用程序提供的默認服務帳戶。
如果 ADC 無法使用上述任一憑據,系統將引發錯誤。
以下 Admin SDK 代碼示例說明了此策略。該示例未明確指定應用程序憑據。但是,只要設置了環境變量,或者只要應用程序在 Compute Engine、Google Kubernetes Engine、App Engine 或 Cloud Functions 上運行,ADC 就能夠隱式找到憑據。
Node.js
admin.initializeApp({
credential: admin.credential.applicationDefault(),
});
爪哇
FirebaseOptions options = FirebaseOptions.builder()
.setCredentials(GoogleCredentials.getApplicationDefault())
.setDatabaseUrl("https://<DATABASE_NAME>.firebaseio.com/")
.build();
FirebaseApp.initializeApp(options);
Python
default_app = firebase_admin.initialize_app()
去
app, err := firebase.NewApp(context.Background(), nil)
if err != nil {
log.Fatalf("error initializing app: %v\n", err)
}
C#
FirebaseApp.Create(new AppOptions()
{
Credential = GoogleCredential.GetApplicationDefault(),
});
手動提供憑據
Firebase 項目支持 Google服務帳戶,您可以使用它從應用服務器或受信任的環境中調用 Firebase 服務器 API。如果您在本地開發代碼或在本地部署應用程序,則可以使用通過此服務帳戶獲取的憑據來授權服務器請求。
要對服務帳戶進行身份驗證並授權其訪問 Firebase 服務,您必須生成 JSON 格式的私鑰文件。
要為您的服務帳戶生成私鑰文件:
在 Firebase 控制台中,打開設置 >服務帳戶。
單擊“生成新私鑰” ,然後單擊“生成密鑰”進行確認。
安全地存儲包含密鑰的 JSON 文件。
通過服務帳戶授權時,您有兩種選擇向應用程序提供憑據。您可以設置GOOGLE_APPLICATION_CREDENTIALS環境變量,也可以在代碼中顯式傳遞服務帳戶密鑰的路徑。第一個選項更安全,強烈推薦。
設置環境變量:
將環境變量GOOGLE_APPLICATION_CREDENTIALS設置為包含服務帳戶密鑰的 JSON 文件的文件路徑。該變量僅適用於當前的 shell 會話,因此如果您打開新會話,請再次設置該變量。
Linux 或 macOS
export GOOGLE_APPLICATION_CREDENTIALS="/home/user/Downloads/service-account-file.json"
視窗
使用 PowerShell:
$env:GOOGLE_APPLICATION_CREDENTIALS="C:\Users\username\Downloads\service-account-file.json"
完成上述步驟後,應用程序默認憑據 (ADC) 能夠隱式確定您的憑據,從而允許您在非 Google 環境中測試或運行時使用服務帳戶憑據。
使用憑據創建訪問令牌
除非您使用自動處理授權的Admin SDK ,否則您需要創建訪問令牌並將其添加到發送請求。
將您的 Firebase 憑據與您的首選語言的Google Auth 庫一起使用,以檢索短期 OAuth 2.0 訪問令牌:
節點.js
function getAccessToken() {
return new Promise(function(resolve, reject) {
const key = require('../placeholders/service-account.json');
const jwtClient = new google.auth.JWT(
key.client_email,
null,
key.private_key,
SCOPES,
null
);
jwtClient.authorize(function(err, tokens) {
if (err) {
reject(err);
return;
}
resolve(tokens.access_token);
});
});
}
在此示例中,Google API 客戶端庫使用 JSON Web 令牌或 JWT 對請求進行身份驗證。有關更多信息,請參閱JSON Web 令牌。
Python
def _get_access_token():
"""Retrieve a valid access token that can be used to authorize requests.
:return: Access token.
"""
credentials = service_account.Credentials.from_service_account_file(
'service-account.json', scopes=SCOPES)
request = google.auth.transport.requests.Request()
credentials.refresh(request)
return credentials.token
爪哇
private static String getAccessToken() throws IOException {
GoogleCredentials googleCredentials = GoogleCredentials
.fromStream(new FileInputStream("service-account.json"))
.createScoped(Arrays.asList(SCOPES));
googleCredentials.refresh();
return googleCredentials.getAccessToken().getTokenValue();
}
訪問令牌過期後,會自動調用令牌刷新方法以檢索更新的訪問令牌。
要授權訪問 FCM,請請求範圍https://www.googleapis.com/auth/firebase.messaging
。
要將訪問令牌添加到 HTTP 請求標頭:
將令牌添加為Authorization
標頭的值,格式為Authorization: Bearer <access_token>
:
節點.js
headers: {
'Authorization': 'Bearer ' + accessToken
}
Python
headers = {
'Authorization': 'Bearer ' + _get_access_token(),
'Content-Type': 'application/json; UTF-8',
}
爪哇
URL url = new URL(BASE_URL + FCM_SEND_ENDPOINT);
HttpURLConnection httpURLConnection = (HttpURLConnection) url.openConnection();
httpURLConnection.setRequestProperty("Authorization", "Bearer " + getServiceAccountAccessToken());
httpURLConnection.setRequestProperty("Content-Type", "application/json; UTF-8");
return httpURLConnection;
授權舊協議發送請求
對於 HTTP 舊協議,每個請求都必須包含 Firebase 控制台“設置”窗格的“雲消息傳遞”選項卡中的服務器密鑰。對於 XMPP,您必須使用相同的服務器密鑰來建立連接。
遷移舊服務器密鑰
從 2020 年 3 月開始,FCM 停止創建舊服務器密鑰。現有的舊服務器密鑰將繼續有效,但我們建議您在Firebase 控制台中使用標記為服務器密鑰的較新版本的密鑰。
如果您想刪除現有的舊服務器密鑰,可以在Google Cloud Console中執行此操作。
授權 HTTP 請求
消息請求由兩部分組成:HTTP 標頭和 HTTP 正文。 HTTP 標頭必須包含以下標頭:
-
Authorization
:key=YOUR_SERVER_KEY
確保這是服務器密鑰,其值可在 Firebase 控制台“設置”窗格的“雲消息傳遞”選項卡中找到。 Android、Apple 平台和瀏覽器密鑰被 FCM 拒絕。 -
Content-Type
:JSON 的application/json
;application/x-www-form-urlencoded;charset=UTF-8
用於純文本。
如果省略Content-Type
,則假定格式為純文本。
例如:
Content-Type:application/json Authorization:key=AIzaSyZ-1u...0GBYzPu7Udno5aA { "to" : "bk3RNwTe3H0:CI2k_HHwgIpoDKCIZvvDMExUdFQ3P1...", "data" : { ... }, }
有關創建發送請求的完整詳細信息,請參閱構建發送請求。舊版 HTTP 協議參考提供了消息可以包含的所有參數的列表。
檢查服務器密鑰的有效性
如果您在發送消息時收到身份驗證錯誤,請檢查服務器密鑰的有效性。例如,在 Linux 上,運行以下命令:
api_key=YOUR_SERVER_KEY curl --header "Authorization: key=$api_key" \ --header Content-Type:"application/json" \ https://fcm.googleapis.com/fcm/send \ -d "{\"registration_ids\":[\"ABC\"]}"
如果您收到 401 HTTP 狀態代碼,則您的服務器密鑰無效。
授權 XMPP 連接
使用 XMPP,您可以維護與 FCM 服務器的持久、異步、雙向連接。該連接可用於在您的服務器和用戶的 FCM 連接設備之間發送和接收消息。
您可以使用大多數 XMPP 庫來管理與 FCM 的長期連接。 XMPP 端點在fcm-xmpp.googleapis.com:5235
運行。當使用非生產用戶測試功能時,您應該連接到fcm-xmpp.googleapis.com:5236
處的預生產服務器(請注意不同的端口)。
對預生產(運行最新 FCM 版本的較小環境)進行定期測試有利於將真實用戶與測試代碼隔離開來。連接到fcm-xmpp.googleapis.com:5236
的測試設備和測試代碼應使用不同的 FCM 發件人 ID,以避免向生產用戶發送測試消息或通過測試連接從生產流量發送上游消息的任何風險。
連接有兩個重要要求:
- 您必須啟動傳輸層安全 (TLS) 連接。請注意,FCM 目前不支持STARTTLS 擴展。
- FCM 需要 SASL PLAIN 身份驗證機制,使用
<your_FCM_Sender_Id>@fcm.googleapis.com
(FCM發件人 ID )和服務器密鑰作為密碼。這些值可在 Firebase 控制台“設置”窗格的“雲消息傳遞”選項卡中找到。
如果在任何時候連接失敗,您應該立即重新連接。身份驗證後發生斷開連接後無需退出。對於每個發送者 ID ,FCM 允許 2500 個並行連接。
以下片段說明瞭如何對 FCM 的 XMPP 連接執行身份驗證和授權。
XMPP服務器
XMPP 服務器請求連接到 FCM
<stream:stream to="fcm.googleapis.com" version="1.0" xmlns="jabber:client" xmlns:stream="http://etherx.jabber.org/streams">
流式細胞儀
FCM 打開連接並請求身份驗證機制,包括PLAIN
方法。
<stream:features> <mechanisms xmlns="urn:ietf:params:xml:ns:xmpp-sasl"> <mechanism>X-OAUTH2</mechanism> <mechanism>X-GOOGLE-TOKEN</mechanism> <mechanism>PLAIN</mechanism> </mechanisms> </stream:features>
XMPP服務器
XMPP 服務器必須使用PLAIN
身份驗證方法進行響應,並從 Firebase 控制台“設置”窗格的“雲消息傳遞”選項卡中提供服務器密鑰。
<auth mechanism="PLAIN" xmlns="urn:ietf:params:xml:ns:xmpp-sasl">MTI2MjAwMzQ3OTMzQHByb2plY3RzLmdjbS5hb mFTeUIzcmNaTmtmbnFLZEZiOW1oekNCaVlwT1JEQTJKV1d0dw==</auth>
流式細胞儀
<success xmlns="urn:ietf:params:xml:ns:xmpp-sasl"/>
XMPP服務器
<stream:stream to="fcm.googleapis.com" version="1.0" xmlns="jabber:client" xmlns:stream="http://etherx.jabber.org/streams">
流式細胞儀
<stream:features> <bind xmlns="urn:ietf:params:xml:ns:xmpp-bind"/> <session xmlns="urn:ietf:params:xml:ns:xmpp-session"/> </stream:features>
XMPP服務器
<iq type="set"> <bind xmlns="urn:ietf:params:xml:ns:xmpp-bind"></bind> </iq>
流式細胞儀
<iq type="result"> <bind xmlns="urn:ietf:params:xml:ns:xmpp-bind"> <jid>SENDER_ID@fcm.googleapis.com/RESOURCE</jid> </bind> </iq>
注意:FCM 在路由消息時不使用綁定資源。
有關創建發送請求的完整詳細信息,請參閱構建發送請求。舊版 XMPP 協議參考提供了消息可以包含的所有參數的列表。