Skip to content

Commit 615352d

Browse files
authored
[Functions] Several functions improvements (#6796)
These improvements are both in the SDK itself, and in the code used for the actual functions running in the cloud during integration testing: A summary of the changes is: - [Backend functions] Update the dependencies and node runtime version - [Backend functions] Add logging in case the client does not support streaming - [SDK] Use a more robust handle of the media type - [SDK] Fix issue introduced in #6773 that missed a return in case of error - [SDK] Other minor code fixes b/404814020
1 parent f9d1802 commit 615352d

File tree

5 files changed

+33
-14
lines changed

5 files changed

+33
-14
lines changed

firebase-functions/CHANGELOG.md

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
# Unreleased
2-
2+
* [fixed] Fixed issue that caused the SDK to crash when trying to stream a function that does not exist.
33

44
# 21.2.0
55
* [feature] Streaming callable functions are now supported.
@@ -235,4 +235,3 @@ updates.
235235
optional region to override the default "us-central1".
236236
* [feature] New `useFunctionsEmulator` method allows testing against a local
237237
instance of the [Cloud Functions Emulator](https://firebase.google.com/docs/functions/local-emulator).
238-

firebase-functions/src/androidTest/backend/functions/index.js

+5
Original file line numberDiff line numberDiff line change
@@ -154,6 +154,9 @@ exports.genStream = functionsV2.https.onCall(async (request, response) => {
154154
response.sendChunk(chunk);
155155
}
156156
}
157+
else {
158+
console.log("CLIENT DOES NOT SUPPORT STEAMING");
159+
}
157160
return streamData.join(' ');
158161
});
159162

@@ -225,6 +228,8 @@ exports.genStreamLargeData = functionsV2.https.onCall(
225228
response.sendChunk(chunk);
226229
await sleep(100);
227230
}
231+
} else {
232+
console.log("CLIENT DOES NOT SUPPORT STEAMING")
228233
}
229234
return "Stream Completed";
230235
}

firebase-functions/src/androidTest/backend/functions/package.json

+3-3
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,11 @@
22
"name": "functions",
33
"description": "Cloud Functions for Firebase",
44
"dependencies": {
5-
"firebase-admin": "11.8.0",
6-
"firebase-functions": "4.4.0"
5+
"firebase-admin": "13.2.0",
6+
"firebase-functions": "6.3.2"
77
},
88
"private": true,
99
"engines": {
10-
"node": "18"
10+
"node": "22"
1111
}
1212
}

firebase-functions/src/androidTest/java/com/google/firebase/functions/StreamTests.kt

+2-1
Original file line numberDiff line numberDiff line change
@@ -116,11 +116,11 @@ class StreamTests {
116116
throwable = e
117117
}
118118

119+
assertThat(throwable).isNull()
119120
assertThat(messages.map { it.message.data.toString() })
120121
.containsExactly("hello", "world", "this", "is", "cool")
121122
assertThat(result).isNotNull()
122123
assertThat(result!!.result.data.toString()).isEqualTo("hello world this is cool")
123-
assertThat(throwable).isNull()
124124
assertThat(isComplete).isTrue()
125125
}
126126

@@ -196,6 +196,7 @@ class StreamTests {
196196
function.stream(mapOf("data" to "test")).subscribe(subscriber)
197197

198198
withTimeout(2000) { delay(500) }
199+
assertThat(subscriber.throwable).isNull()
199200
assertThat(subscriber.messages).isEmpty()
200201
assertThat(subscriber.result).isNull()
201202
}

firebase-functions/src/main/java/com/google/firebase/functions/PublisherStream.kt

+22-8
Original file line numberDiff line numberDiff line change
@@ -136,12 +136,20 @@ internal class PublisherStream(
136136
MediaType.parse("application/json"),
137137
JSONObject(mapOf("data" to serializer.encode(data))).toString()
138138
)
139-
val requestBuilder =
140-
Request.Builder().url(url).post(requestBody).header("Accept", "text/event-stream")
141-
context?.authToken?.let { requestBuilder.header("Authorization", "Bearer $it") }
142-
context?.instanceIdToken?.let { requestBuilder.header("Firebase-Instance-ID-Token", it) }
143-
context?.appCheckToken?.let { requestBuilder.header("X-Firebase-AppCheck", it) }
144-
val request = requestBuilder.build()
139+
val request =
140+
Request.Builder()
141+
.url(url)
142+
.post(requestBody)
143+
.apply {
144+
header("Accept", "text/event-stream")
145+
header("Content-Type", "application/json")
146+
context?.apply {
147+
authToken?.let { header("Authorization", "Bearer $it") }
148+
instanceIdToken?.let { header("Firebase-Instance-ID-Token", it) }
149+
appCheckToken?.let { header("X-Firebase-AppCheck", it) }
150+
}
151+
}
152+
.build()
145153
val call = configuredClient.newCall(request)
146154
activeCall = call
147155

@@ -206,6 +214,9 @@ internal class PublisherStream(
206214
eventBuffer.append(dataChunk.trim()).append("\n")
207215
}
208216
}
217+
if (eventBuffer.isNotEmpty()) {
218+
processEvent(eventBuffer.toString())
219+
}
209220
} catch (e: Exception) {
210221
notifyError(
211222
FirebaseFunctionsException(
@@ -296,9 +307,11 @@ internal class PublisherStream(
296307
private fun validateResponse(response: Response) {
297308
if (response.isSuccessful) return
298309

299-
val htmlContentType = "text/html; charset=utf-8"
300310
val errorMessage: String
301-
if (response.code() == 404 && response.header("Content-Type") == htmlContentType) {
311+
if (
312+
response.code() == 404 &&
313+
MediaType.parse(response.header("Content-Type") ?: "")?.subtype() == "html"
314+
) {
302315
errorMessage = """URL not found. Raw response: ${response.body()?.string()}""".trimMargin()
303316
notifyError(
304317
FirebaseFunctionsException(
@@ -307,6 +320,7 @@ internal class PublisherStream(
307320
null
308321
)
309322
)
323+
return
310324
}
311325

312326
val text = response.body()?.string() ?: ""

0 commit comments

Comments
 (0)