210202
안드로이드 FCM HTTP v1으로 마이그레이션 (+ OAtuh 2.0 구글 로그인 없이 발급) 본문
안드로이드 기기(A)에서 다른 기기(B)로 메시지를 보내려면
보내는 클라이언트(A) -> FCM 백엔드 -> 받는 클라이언트(B)
이 순서로 통신이 이루어진다.
보내는 클라이언트 -> FCM 백엔드에서 상호작용 하는 방식은 다음과 같은 방식이 있다.
- Firebase Admin SDK
- FCM HTTP v1 API
- 기존 HTTP
- 기존 XMPP
이중에서 구글링했을 때 대부분 기존 HTTP 방식을 대부분 사용하여 설명되어있고, 나도 이 방식을 사용해왔다.
그러나 기존 HTTP 방식은 2023년 6월에 지원이 중단되어 2024년 6월까지 FCM HTTP v1 API로 마이그레이션이 필요하다.
변경점: HTTP v1 부터는 기존에 서버키를 사용하던 방식 대신 OAuth 2.0 AccessToken을 사용하고, 보내는 메시지가 변경되었다.
그래서 먼저 OAuth 2.0을 사용하여 AccessToken을 발급받기 위해 방법을 찾아봤고, Google Login API를 사용해서 구현한 내용들이 다수였다. 나의 프로젝트에서는 로그인을 사용하지 않아서 로그인 없이 발급받을 수 있는 방법을 적용한 과정을 적어서 나와 같은 문제를 가지는 개발자들이 참고할 수 있도록 글을 적어본다.
1. OAuth 2.0 AccessToken 발급
OAuth 2.0 AccessToken을 발급받을 수 있는 방식들 중 사용자 인증 정보를 사용한 방식을 사용한다.
private static String getAccessToken() throws IOException {
GoogleCredentials googleCredentials = GoogleCredentials.fromStream(new FileInputStream("service-account.json")) .createScoped(Arrays.asList(SCOPES));
googleCredentials.refreshAccessToken();
return googleCredentials.getAccessToken().getTokenValue();
}
여기서 service-account.json
, SCOPES
를 채워넣어야한다.
json 파일
json 파일은 서비스 비공개 키로, 다음 과정을 통해 발급받을 수 있다.
- 사용할 Firebase 프로젝트에서 설정 > 프로젝트 설정
- 새 비공개 키 생성을 클릭하고 키 생성을 클릭하여 확인.
- 키가 들어있는 JSON 파일 저장.
SCOPES
SCOPES는 공식문서 링크에 나와있는https://www.googleapis.com/auth/firebase.messaging
를 추가한다.
위의 예시 코드를 Kotlin으로 변환하고, json파일을 asset을 사용해서 추가하고, SCOPES에 코드를 추가한 결과는 다음과 같다. (asset 가이드)
GoogleCredential이 import가 안된다면 build.gradle의 아래 dependencies 부분의 코드를 추가
'2 files found with path 'META-INF/DEPENDENCIES' from ~'이라고 에러가 발생하면 build.gradle의 위 android 부분의 코드를 추가
fun getAccessToken(){
val asset = resources.assets.open("파일명.json")
val googleCredential = GoogleCredential.fromStream(asset)
.createScoped(listOf("https://www.googleapis.com/auth/firebase.messaging"))
googleCredential.refreshToken()
val accessToekn = googleCredential.accessToken
}
// bulid.gradle
android {
....
packagingOptions {
resources.excludes.add("META-INF/*")
}
}
dependencies {
....
// OAuth 2.0
implementation 'com.google.android.gms:play-services-auth:20.7.0'
implementation 'com.google.api-client:google-api-client:1.32.2'
}
2. FCM 백엔드에 요청 보내기
이제 FCM 백엔드에 AccessToken을 사용해서 요청을 보낸다.
링크를 참고하여 마이그레이션을 진행한다.
1. 서버 엔드포인트 변경
val api = "https://fcm.googleapis.com/v1/projects/(projectname)/messages:send"
Firebase 프로젝트 ID를 (projectname) 부분에 입력한다.
2. 전송 요청 업데이트
기존에 사용하던 서버키 대신 위에서 발급받은 AccessToken을 사용한다.
val request = Request.Builder()
.url(url)
.post(body)
.addHeader("Authorization", "Bearer $accessToken")
.addHeader("Content-Type", "application/json")
.build()
3. 메시지 페이로드 업데이트
링크에서 설명된 변경점에 맞춰 jsonObject를 생성한다.
val json = JSONObject().apply {
put("message", JSONObject().apply {
put("token", token)
put("notification", JSONObject().apply {
put("title", title)
put("body", content)
})
})
}
실제 구현한 전체 코드
fun sendNotifications(
accessToken: String,
token: String,
title: String,
content: String
) {
val api =
"https://fcm.googleapis.com/v1/projects/(projectname)/messages:send"
val url = api.toHttpUrlOrNull()!!.newBuilder().build()
val client = OkHttpClient()
val json = JSONObject().apply {
put("message", JSONObject().apply {
put("token", token)
put("notification", JSONObject().apply {
put("title", title)
put("body", content)
})
})
}
val body =
json.toString().toRequestBody("application/json; charset=utf-8".toMediaTypeOrNull())
val request = Request.Builder()
.url(url)
.post(body)
.addHeader("Authorization", "Bearer $accessToken")
.addHeader("Content-Type", "application/json")
.build()
client.newCall(request).enqueue(object : Callback {
override fun onFailure(call: Call, e: IOException) {
e.printStackTrace()
}
override fun onResponse(call: Call, response: Response) {
// Handle success
val responseBody = response.body?.string()
println("FCM Response: $responseBody")
}
})
}
기타
주제로 여러개의 기기에 메시지 보내기
참고
<받는 쪽>
Firebase.messaging.subscribeToTopic(topic)
<보내는 쪽>
FCM 백엔드에 보낼 메시지에 token대신 topic을 사용한다.
val json = JSONObject().apply {
put("message", JSONObject().apply {
put("topic", topic)
put("notification", JSONObject().apply {
put("title", title)
put("body", content)
})
})
}
메시지 보내기 유형
'Android' 카테고리의 다른 글
Material CalendarView XML Preview 오류 (0) | 2023.11.27 |
---|---|
안드로이드 크기 단위(dp,sp 등) 비교 (0) | 2023.09.24 |
BottomNavigation + Jetpack Navigation 버그findNavController().navigate()로 이동한 뒤 BottomNavigation으로 이동시 원래 Fragment로 돌아오지 않음 (0) | 2022.07.19 |
[LiveData] Fragment간 ViewModel 공유 시 LiveData의 값이 안바뀌는 현상 (0) | 2021.04.04 |
[ViewModel] ViewModel 사용이유 , ViewModelProvider 작동방식 (0) | 2021.03.24 |