With Cloud Functions (2nd gen), you can trigger functions in response to custom events. These are events provided by special or additional event providers, as opposed to the Firebase events natively supported by the Firebase SDK for Cloud Functions. Via custom event triggers, your app can respond to events provided by Firebase Extensions, or you can publish your own custom events and trigger functions in response to them.
All custom events conform to the CloudEvents JSON event format and are published to Eventarc. Eventarc usage fees apply.
Trigger functions with custom events
You can publish custom events (or obtain events from Firebase extensions) and trigger functions in response to those events by implementing this basic flow:
- Publish the desired events to an Eventarc channel, or identify available events provided by an extension that you have installed.
- In your function code, subscribe to events on the Eventarc channel with an event handler.
- In the function, parse the payload returned in the CloudEvent object and perform whatever custom logic your app requires.
For example, a game app might want to send notifications to users as they enter or leave the leaderboard of top ten competitors. This app could publish leaderboard events to the default channel, and then handle the event in a function that sends targeted push notifications to users.
In another example, an extension designed to help apps process large images might emit an event on the completion of image resizing. Apps with this extension installed could handle the completion event by updating links in the app to point to resized versions of the image.
Publish an event to a channel
Eventarc events are published into
channels.
Channels are a way to group related events and manage access
permissions. When you install an extension or deploy a function that consumes
custom events, Firebase automatically creates a default channel named
firebase
in the us-central1
region. The Firebase Admin SDK provides
an eventarc
subpackage for publishing to channels.
To publish an event from a trusted server (or another function) using the default channel:
import {getEventarc} from 'firebase-admin/eventarc';
getEventarc().channel().publish({
type: 'achieved-leaderboard',
subject: 'Welcome to the top 10',
data: {
message: 'You have achieved the nth position in our leaderboard! To see . . .'
}
});
In addition to automatically creating the default channel, Firebase sets the
environment variable EVENTARC_CLOUD_EVENT_SOURCE
, which specifies the source
of the event. If you are publishing events outside of Cloud Functions for Firebase,
you'll need to explicitly add the source
field in your event payload.
Handle custom events
You can handle all custom events, including extensions events, with the
onCustomEventPublished
or
on_custom_event_published
handlers. First, import this handler from the Eventarc SDK along with the
Firebase Admin SDK:
Node.js
const {onCustomEventPublished} = require("firebase-functions/v2/eventarc");
const logger = require("firebase-functions/logger");
const {initializeApp} = require("firebase-admin/app");
const {getFirestore} = require("firebase-admin/firestore");
Python
from firebase_admin import firestore, initialize_app
from firebase_functions import eventarc_fn
In your function code, pass in the event name as shown for the example function:
Node.js
exports.onimageresized = onCustomEventPublished(
"firebase.extensions.storage-resize-images.v1.complete",
(event) => {
logger.info("Received image resize completed event", event);
// For example, write resized image details into Firestore.
return getFirestore()
.collection("images")
.doc(event.subject.replace("/", "_")) // original file path
.set(event.data); // resized images paths and sizes
});
Python
@eventarc_fn.on_custom_event_published(
event_type="firebase.extensions.storage-resize-images.v1.complete")
def onimageresized(event: eventarc_fn.CloudEvent) -> None:
print("Received image resize completed event: ", event.type)
if not isinstance(event.subject, str):
print("No 'subject' data.")
return
# For example, write resized image details into Firestore.
firestore_client: google.cloud.firestore.Client = firestore.client()
collection = firestore_client.collection("images")
doc = collection.document(event.subject.replace("/", "_")) # original file path
doc.set(event.data) # resized images paths and sizes
For each particular extension, the payload returned in the event object provides
data you can use to perform custom logic for your application flow. In this
case, the function uses the Admin SDK to copy metadata about the resized
image to a collection in Cloud Firestore, obtaining the filename from the
subject
provided by the event, and saving metadata from the data
provided
by the event.
Publish and handle events on non-default channels
Custom channels can be useful for cases where you have special permission needs or other requirements, and don't want the same level of visibility and access for all events. You can create your own channels using the Google Cloud console. Publishing and subscribing for events must be done on the same channel.
In cases where a custom event is published on a non-default channel,
you'll need to specify the channel in your function code. For example, if you
want to handle events that are published in a non-default channel for the
us-west1
location, you need to specify the channel as shown:
Node.js
import { onCustomEventPublished } from "firebase-functions/v2/eventarc";
export const func = onCustomEventPublished(
{
eventType: "firebase.extensions.storage-resize-images.v1.complete",
channel: "locations/us-west1/channels/firebase",
region: "us-west1",
},
(event) => { ... });
Python
@eventarc_fn.on_custom_event_published(
event_type="firebase.extensions.storage-resize-images.v1.complete",
channel="locations/us-west1/channels/firebase",
region="us-west1")
def onimageresizedwest(event: eventarc_fn.CloudEvent) -> None:
print("Received image resize completed event: ", event.type)
# ...