Mit Cloud Functions können Sie Ereignisse in der Firebase Realtime Database verarbeiten, ohne Clientcode aktualisieren zu müssen. Mit Cloud Functions können Sie Realtime Database-Vorgänge mit vollen Administratorberechtigungen ausführen. Außerdem wird sichergestellt, dass jede Änderung an Realtime Database einzeln verarbeitet wird. Sie können Firebase Realtime Database-Änderungen über den Daten-Snapshot oder das Admin SDK vornehmen.
Der typische Lebenszyklus einer Firebase Realtime Database-Funktion sieht so aus:
- Sie wartet auf Änderungen an einem bestimmten Realtime Database-Pfad.
- Sie wird ausgelöst, wenn ein Ereignis eintritt, und führt dessen Aufgaben aus.
- Sie erhält ein Datenobjekt, das einen Snapshot der an diesem Pfad gespeicherten Daten enthält.
Sie können eine Funktion als Reaktion auf das Schreiben, Erstellen, Aktualisieren oder Löschen von Datenbankknoten in Firebase Realtime Database auslösen. Wenn Sie festlegen möchten, wann die Funktion ausgelöst wird, geben Sie einen der Ereignishandler und den Realtime Database-Pfad an, an dem nach Ereignissen gesucht wird.
Funktionsstandort festlegen
Die Entfernung zwischen dem Standort einer Realtime Database-Instanz und dem Standort der Funktion kann zu einer erheblichen Netzwerklatenz führen. Außerdem kann eine Abweichung zwischen Regionen zu einem Bereitstellungsfehler führen. Um diese Situationen zu vermeiden, geben Sie den Funktionsspeicherort so an, dass er mit dem Speicherort der Datenbankinstanz übereinstimmt.
Realtime Database-Ereignisse verarbeiten
Mit Cloud Functions können Sie Realtime Database-Ereignisse mit zwei Spezifitätsgraden verarbeiten: Sie können gezielt nur nach Schreib-, Erstellungs-, Aktualisierungs- oder Löschereignissen oder nach beliebigen Änderungen jeglicher Art an einer Referenz Ausschau halten.
Die folgenden Handler für die Reaktion auf Realtime Database-Ereignisse sind verfügbar:
onValueWritten()
Wird ausgelöst, wenn Daten in Realtime Database erstellt, aktualisiert oder gelöscht werden.onValueCreated()
Wird nur ausgelöst, wenn Daten in Realtime Database erstellt werden.onValueUpdated()
Wird nur ausgelöst, wenn Daten in Realtime Database aktualisiert werden.onValueDeleted()
Wird nur ausgelöst, wenn Daten in Realtime Database gelöscht werden.
on_value_written()
Wird ausgelöst, wenn Daten in Realtime Database erstellt, aktualisiert oder gelöscht werden.on_value_created()
Wird nur ausgelöst, wenn Daten in Realtime Database erstellt werden.on_value_updated()
Wird nur ausgelöst, wenn Daten in Realtime Database aktualisiert werden.on_value_deleted()
Wird nur ausgelöst, wenn Daten in Realtime Database gelöscht werden.
Erforderliche Module importieren
Im Funktionsquellcode müssen Sie die SDK-Module importieren, die Sie verwenden möchten. Für dieses Beispiel müssen die HTTP- und Realtime Database-Module sowie das Firebase Admin SDK-Modul zum Schreiben in Realtime Database importiert werden.
// The Cloud Functions for Firebase SDK to setup triggers and logging.
const {onRequest} = require("firebase-functions/v2/https");
const {onValueCreated} = require("firebase-functions/v2/database");
const {logger} = require("firebase-functions");
// The Firebase Admin SDK to access the Firebase Realtime Database.
const admin = require("firebase-admin");
admin.initializeApp();
# The Cloud Functions for Firebase SDK to create Cloud Functions and set up triggers.
from firebase_functions import db_fn, https_fn
# The Firebase Admin SDK to access the Firebase Realtime Database.
from firebase_admin import initialize_app, db
app = initialize_app()
Instanz und Pfad angeben
Wenn Sie festlegen möchten, wann und wo Ihre Funktion ausgelöst werden soll, konfigurieren Sie sie mit einem Pfad und optional einer Realtime Database-Instanz. Wenn Sie keine Instanz angeben, wird die Funktion für alle Realtime Database-Instanzen in der Funktionsregion ausgeführt. Sie können auch ein Realtime Database-Instanzmuster angeben, um die Bereitstellung auf eine ausgewählte Teilmenge von Instanzen in derselben Region zu beschränken.
Beispiel:
// All Realtime Database instances in default function region us-central1 at path "/user/{uid}" // There must be at least one Realtime Database present in us-central1. const onWrittenFunctionDefault = onValueWritten("/user/{uid}", (event) => { // … }); // Instance named "my-app-db-2", at path "/user/{uid}". // The "my-app-db-2" instance must exist in this region. const OnWrittenFunctionInstance = onValueWritten( { ref: "/user/{uid}", instance: "my-app-db-2" // This example assumes us-central1, but to set location: // region: "europe-west1" }, (event) => { // … } ); // Instance with "my-app-db-" prefix, at path "/user/{uid}", where uid ends with @gmail.com. // There must be at least one Realtime Database with "my-app-db-*" prefix in this region. const onWrittenFunctionInstance = onValueWritten( { ref: "/user/{uid=*@gmail.com}", instance: "my-app-db-*" // This example assumes us-central1, but to set location: // region: "europe-west1" }, (event) => { // … } );
# All Realtime Database instances in default function region us-central1 at path "/user/{uid}"
# There must be at least one Realtime Database present in us-central1.
@db_fn.on_value_written(r"/user/{uid}")
def onwrittenfunctiondefault(event: db_fn.Event[db_fn.Change]):
# ...
pass
# Instance named "my-app-db-2", at path "/user/{uid}".
# The "my-app-db-2" instance must exist in this region.
@db_fn.on_value_written(
reference=r"/user/{uid}",
instance="my-app-db-2",
# This example assumes us-central1, but to set location:
# region="europe-west1",
)
def on_written_function_instance(event: db_fn.Event[db_fn.Change]):
# ...
pass
# Instance with "my-app-db-" prefix, at path "/user/{uid}", where uid ends with @gmail.com.
# There must be at least one Realtime Database with "my-app-db-*" prefix in this region.
@db_fn.on_value_written(
reference=r"/user/{uid=*@gmail.com}",
instance="my-app-db-*",
# This example assumes us-central1, but to set location:
# region="europe-west1",
)
def on_written_function_instance(event: db_fn.Event[db_fn.Change]):
# ...
pass
Mit diesen Parametern wird Ihre Funktion angewiesen, Schreibvorgänge an einem bestimmten Pfad in der Realtime Database-Instanz zu verarbeiten.
Die Pfadspezifikationen gleichen alle Schreibvorgänge ab, die einen Pfad tangieren, einschließlich Schreibvorgängen, die an einem beliebigen untergeordneten Punkt auftreten. Wenn Sie als Pfad für Ihre Funktion /foo/bar
festlegen, werden die Ereignisse an diesen beiden Speicherorten abgeglichen:
/foo/bar
/foo/bar/baz/really/deep/path
In beiden Fällen geht Firebase davon aus, dass das Ereignis unter /foo/bar
auftritt, und die Ereignisdaten umfassen sowohl die alten als auch die neuen Daten unter /foo/bar
. Wenn die Ereignisdaten sehr umfangreich sein können, sollten Sie erwägen, mehrere Funktionen auf tieferen Pfadebenen anstatt lediglich einer Funktion im Bereich des Datenbankstamms zu verwenden. Für optimale Leistung sollten Sie Daten nur auf der untersten Ebene anfordern.
Platzhalter und Erfassung
Sie können {key}
, {key=*}
, {key=prefix*}
und {key=*suffix}
für die Erfassung verwenden. *
, prefix*
, *suffix
für Wildcards mit einem Segment.
Hinweis: **
steht für Platzhalter mit mehreren Segmenten, die von Realtime Database nicht unterstützt werden.
Weitere Informationen finden Sie unter Informationen zu Pfadmustern.
Platzhalter für Pfade: Sie können eine Pfadkomponente als Platzhalter angeben:
- Mit Sternchen (
*
).foo/*
stimmt beispielsweise mit allen untergeordneten Elementen überein, die sich eine Ebene unterfoo/
in der Knotenstruktur befinden. - Sie verwenden ein Segment, das genau ein Sternchen (
*
) enthält.foo/app*-us
stimmt beispielsweise mit allen untergeordneten Segmenten unterfoo/
mit dem Präfixapp
und dem Suffix-us
überein.
Pfade mit Platzhaltern können mit mehreren Ereignissen aus beispielsweise einem einzigen Schreibvorgang übereinstimmen. Die Verwendung von
{
"foo": {
"hello": "world",
"firebase": "functions"
}
}
ergibt zwei Übereinstimmungen mit dem Pfad "/foo/*"
: einmal "hello": "world"
und einmal "firebase": "functions"
.
Pfadaufzeichnung: Sie können Pfadübereinstimmungen in benannten Variablen erfassen, die in Ihrem Funktionscode verwendet werden sollen (z.B. /user/{uid}
, /user/{uid=*-us}
).
Die Werte der Erfassungsvariablen sind im database.DatabaseEvent.params-Objekt Ihrer Funktion verfügbar.
Platzhalter für Instanzen Sie können auch eine Instanzkomponente mit Platzhaltern angeben. Ein Instanz-Platzhalter kann ein Präfix, ein Suffix oder beides haben (z.B. my-app-*-prod
).
Platzhalter und Erfassung
Bei Cloud Functions (2. Generation) und Realtime Database kann ein Muster verwendet werden, wenn ref
und instance
angegeben werden. Für jede Trigger-Schnittstelle sind die folgenden Optionen zum Festlegen des Bereichs einer Funktion verfügbar:
ref angeben |
instance angeben |
Verhalten |
---|---|---|
Einzeln (/foo/bar ) |
Nicht angegeben | Bereichshandler für alle Instanzen in der Funktionsregion. |
Einzeln (/foo/bar ) |
Einzeln (‘my-new-db' ) |
Bereichshandler für die spezifische Instanz in der Funktionsregion. |
Einzeln (/foo/bar ) |
Muster (‘inst-prefix*' ) |
Bereichshandler für alle Instanzen, die dem Muster in der Funktionsregion entsprechen. |
Muster (/foo/{bar} ) |
Nicht angegeben | Bereichshandler für alle Instanzen in der Funktionsregion. |
Muster (/foo/{bar} ) |
Einzeln (‘my-new-db' ) |
Bereichshandler für die spezifische Instanz in der Funktionsregion. |
Muster (/foo/{bar} ) |
Muster (‘inst-prefix*' ) |
Bereichshandler für alle Instanzen, die dem Muster in der Funktionsregion entsprechen. |
Ereignisdaten verarbeiten
Wenn ein Realtime Database-Ereignis ausgelöst wird, wird ein Event
-Objekt an Ihre Handler-Funktion übergeben.
Dieses Objekt hat die Eigenschaft data
, die bei Erstellungs- und Löschereignissen einen Snapshot der erstellten oder gelöschten Daten enthält.
In diesem Beispiel ruft die Funktion die Daten für den referenzierten Pfad ab, wandelt den String an dieser Stelle in Großbuchstaben um und schreibt den geänderten String in die Datenbank:
// Listens for new messages added to /messages/:pushId/original and creates an
// uppercase version of the message to /messages/:pushId/uppercase
// for all databases in 'us-central1'
exports.makeuppercase = onValueCreated(
"/messages/{pushId}/original",
(event) => {
// Grab the current value of what was written to the Realtime Database.
const original = event.data.val();
logger.log("Uppercasing", event.params.pushId, original);
const uppercase = original.toUpperCase();
// You must return a Promise when performing
// asynchronous tasks inside a function, such as
// writing to the Firebase Realtime Database.
// Setting an "uppercase" sibling in the
// Realtime Database returns a Promise.
return event.data.ref.parent.child("uppercase").set(uppercase);
},
);
@db_fn.on_value_created(reference="/messages/{pushId}/original")
def makeuppercase(event: db_fn.Event[Any]) -> None:
"""Listens for new messages added to /messages/{pushId}/original and
creates an uppercase version of the message to /messages/{pushId}/uppercase
"""
# Grab the value that was written to the Realtime Database.
original = event.data
if not isinstance(original, str):
print(f"Not a string: {event.reference}")
return
# Use the Admin SDK to set an "uppercase" sibling.
print(f"Uppercasing {event.params['pushId']}: {original}")
upper = original.upper()
parent = db.reference(event.reference).parent
if parent is None:
print("Message can't be root node.")
return
parent.child("uppercase").set(upper)
Vorherigen Wert lesen
Für write
- oder update
-Ereignisse ist die Property data
ein Change
-Objekt, das zwei Snapshots enthält, die den Datenstatus vor und nach dem auslösenden Ereignis darstellen.
Das Change
-Objekt hat eine before
-Property, mit der Sie prüfen können, was vor dem Ereignis in Realtime Database gespeichert wurde, und eine after
-Property, die den Status der Daten nach dem Ereignis darstellt.
Mit dem Attribut before
kann beispielsweise dafür gesorgt werden, dass Text nur beim Erstellen in Großbuchstaben umgewandelt wird:
exports makeUppercase = onValueWritten("/messages/{pushId}/original", (event) => { // Only edit data when it is first created. if (event.data.before.exists()) { return null; } // Exit when the data is deleted. if (!event.data.after.exists()) { return null; } // Grab the current value of what was written to the Realtime Database. const original = event.data.after.val(); console.log('Uppercasing', event.params.pushId, original); const uppercase = original.toUpperCase(); // You must return a Promise when performing asynchronous tasks inside a Functions such as // writing to the Firebase Realtime Database. // Setting an "uppercase" sibling in the Realtime Database returns a Promise. return event.data.after.ref.parent.child('uppercase').set(uppercase); });
@db_fn.on_value_written(reference="/messages/{pushId}/original")
def makeuppercase2(event: db_fn.Event[db_fn.Change]) -> None:
"""Listens for new messages added to /messages/{pushId}/original and
creates an uppercase version of the message to /messages/{pushId}/uppercase
"""
# Only edit data when it is first created.
if event.data.before is not None:
return
# Exit when the data is deleted.
if event.data.after is None:
return
# Grab the value that was written to the Realtime Database.
original = event.data.after
if not hasattr(original, "upper"):
print(f"Not a string: {event.reference}")
return
# Use the Admin SDK to set an "uppercase" sibling.
print(f"Uppercasing {event.params['pushId']}: {original}")
upper = original.upper()
parent = db.reference(event.reference).parent
if parent is None:
print("Message can't be root node.")
return
parent.child("uppercase").set(upper)