diff --git a/bom/pom.xml b/bom/pom.xml index 08278900..43153c80 100644 --- a/bom/pom.xml +++ b/bom/pom.xml @@ -154,6 +154,16 @@ quarkus-google-cloud-firebase-devservices-deployment ${project.version} + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-realtime-database + ${project.version} + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-realtime-database-deployment + ${project.version} + io.quarkiverse.googlecloudservices quarkus-google-cloud-firestore diff --git a/docs/modules/ROOT/nav.adoc b/docs/modules/ROOT/nav.adoc index 3d528b84..68baf6fd 100644 --- a/docs/modules/ROOT/nav.adoc +++ b/docs/modules/ROOT/nav.adoc @@ -4,7 +4,8 @@ ** xref:index.adoc#examples[Example applications] * xref:bigquery.adoc[BigQuery] * xref:bigtable.adoc[BigTable] -* xref:firebase-devservices.adoc[Firebase Admin] +* xref:firebase-devservices.adoc[Firebase DevServices] +* xref:firebase-realtime-database.adoc[Firebase Realtime Database] * xref:firebase-admin.adoc[Firebase Admin] * xref:firestore.adoc[Firestore] * xref:logging.adoc[Logging] diff --git a/docs/modules/ROOT/pages/firebase-realtime-database.adoc b/docs/modules/ROOT/pages/firebase-realtime-database.adoc new file mode 100644 index 00000000..2d7f803a --- /dev/null +++ b/docs/modules/ROOT/pages/firebase-realtime-database.adoc @@ -0,0 +1,76 @@ += Google Cloud Services - Firebase Realtime Database + +This extension allows to inject a `com.google.firebase.database.FirebaseDatabase` object inside your Quarkus application. + +This extension will pickup on any available `com.google.firebase.FirebaseApp` (see https://quarkiverse.github.io/quarkiverse-docs/quarkus-google-cloud-services/main/firebase-admin.html[Firebase Admin extension] for more info) to configure the Realtime Database. + +Be sure to have read the https://quarkiverse.github.io/quarkiverse-docs/quarkus-google-cloud-services/main/index.html[Google Cloud Services extension pack global documentation] before this one, it contains general configuration and information. + +== Bootstrapping the project + +First, we need a new project. Create a new project with the following command (replace the version placeholder with the correct one): + +[source, shell script] +---- +mvn io.quarkus:quarkus-maven-plugin:${quarkusVersion}:create \ + -DprojectGroupId=org.acme \ + -DprojectArtifactId=firebase-realtime-database-quickstart \ + -Dextensions="resteasy-reactive-jackson,quarkus-google-cloud-firebase-realtime-database" +cd firebase-admin-quickstart +---- + +This command generates a Maven project, importing the Google Cloud Firebase Realtime Database extension.) + +If you already have your Quarkus project configured, you can add the `quarkus-google-cloud-firebase-realtime-database` extension to your project by running the following command in your project base directory: + +[source, shell script] +---- +./mvnw quarkus:add-extension -Dextensions="quarkus-google-cloud-firebase-realtime-database" +---- + +This will add the following to your pom.xml: + +[source, xml] +---- + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-realtime-database + +---- + +== Some example + +This is an example usage of the extension: we create a REST resource with a single endpoint that creates a 'persons' collection, inserts three persons in it, then search for persons with last name Doe and returns them. + +[source,java] +---- +import javax.inject.Inject; +import javax.ws.rs.POST; +import javax.ws.rs.Path; +import javax.ws.rs.Consumes; +import javax.ws.rs.core.MediaType; + +import com.google.firebase.database.FirebaseDatabase; + +@Path("/database") +public class RealtimeDatabaseResource { + + @Inject + FirebaseDatabase database; // Inject database + + @POST + @Consumes(MediaType.TEXT_PLAIN) + public void database(String data) { + var dbRef = firebaseDatabase.getReference("test"); + dbRef.setValueAsync(fields); + } +} +---- + +== Dev Service + +This extension uses the https://quarkiverse.github.io/quarkiverse-docs/quarkus-google-cloud-services/main/firebase-devservices.html[Firebase Devservices] extension to provide a Dev Service. Refer the documentation of this extension for more info. + +== Configuration Reference + +include::./includes/quarkus-google-cloud-firebase-realtime-database.adoc[] \ No newline at end of file diff --git a/docs/modules/ROOT/pages/includes/quarkus-google-cloud-firebase-realtime-database.adoc b/docs/modules/ROOT/pages/includes/quarkus-google-cloud-firebase-realtime-database.adoc new file mode 100644 index 00000000..f331d2d6 --- /dev/null +++ b/docs/modules/ROOT/pages/includes/quarkus-google-cloud-firebase-realtime-database.adoc @@ -0,0 +1,28 @@ +[.configuration-legend] +icon:lock[title=Fixed at build time] Configuration property fixed at build time - All other configuration properties are overridable at runtime +[.configuration-reference.searchable, cols="80,.^10,.^10"] +|=== + +h|[.header-title]##Configuration property## +h|Type +h|Default + +a| [[quarkus-google-cloud-firebase-realtime-database_quarkus-google-cloud-firebase-database-host-override]] [.property-path]##link:#quarkus-google-cloud-firebase-realtime-database_quarkus-google-cloud-firebase-database-host-override[`quarkus.google.cloud.firebase.database.host-override`]## + +[.description] +-- +Overrides the default service host. This is most commonly used for development or testing activities with a local Firebase Realtime Database emulator instance. + + +ifdef::add-copy-button-to-env-var[] +Environment variable: env_var_with_copy_button:+++QUARKUS_GOOGLE_CLOUD_FIREBASE_DATABASE_HOST_OVERRIDE+++[] +endif::add-copy-button-to-env-var[] +ifndef::add-copy-button-to-env-var[] +Environment variable: `+++QUARKUS_GOOGLE_CLOUD_FIREBASE_DATABASE_HOST_OVERRIDE+++` +endif::add-copy-button-to-env-var[] +-- +|string +| + +|=== + diff --git a/docs/modules/ROOT/pages/includes/quarkus-google-cloud-firebase-realtime-database_quarkus.google.adoc b/docs/modules/ROOT/pages/includes/quarkus-google-cloud-firebase-realtime-database_quarkus.google.adoc new file mode 100644 index 00000000..f331d2d6 --- /dev/null +++ b/docs/modules/ROOT/pages/includes/quarkus-google-cloud-firebase-realtime-database_quarkus.google.adoc @@ -0,0 +1,28 @@ +[.configuration-legend] +icon:lock[title=Fixed at build time] Configuration property fixed at build time - All other configuration properties are overridable at runtime +[.configuration-reference.searchable, cols="80,.^10,.^10"] +|=== + +h|[.header-title]##Configuration property## +h|Type +h|Default + +a| [[quarkus-google-cloud-firebase-realtime-database_quarkus-google-cloud-firebase-database-host-override]] [.property-path]##link:#quarkus-google-cloud-firebase-realtime-database_quarkus-google-cloud-firebase-database-host-override[`quarkus.google.cloud.firebase.database.host-override`]## + +[.description] +-- +Overrides the default service host. This is most commonly used for development or testing activities with a local Firebase Realtime Database emulator instance. + + +ifdef::add-copy-button-to-env-var[] +Environment variable: env_var_with_copy_button:+++QUARKUS_GOOGLE_CLOUD_FIREBASE_DATABASE_HOST_OVERRIDE+++[] +endif::add-copy-button-to-env-var[] +ifndef::add-copy-button-to-env-var[] +Environment variable: `+++QUARKUS_GOOGLE_CLOUD_FIREBASE_DATABASE_HOST_OVERRIDE+++` +endif::add-copy-button-to-env-var[] +-- +|string +| + +|=== + diff --git a/firebase-devservices/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/deployment/FirebaseDevServiceProcessor.java b/firebase-devservices/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/deployment/FirebaseDevServiceProcessor.java index eaae2547..52b75811 100644 --- a/firebase-devservices/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/deployment/FirebaseDevServiceProcessor.java +++ b/firebase-devservices/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/deployment/FirebaseDevServiceProcessor.java @@ -37,7 +37,7 @@ public class FirebaseDevServiceProcessor { FirebaseEmulatorContainer.Emulator.FIREBASE_HOSTING, "quarkus.google.cloud.firebase.hosting.emulator-host", FirebaseEmulatorContainer.Emulator.CLOUD_FUNCTIONS, "quarkus.google.cloud.functions.emulator-host", FirebaseEmulatorContainer.Emulator.EVENT_ARC, "quarkus.google.cloud.eventarc.emulator-host", - FirebaseEmulatorContainer.Emulator.REALTIME_DATABASE, "quarkus.google.cloud.database.emulator-host", + FirebaseEmulatorContainer.Emulator.REALTIME_DATABASE, "quarkus.google.cloud.firebase.database.host-override", FirebaseEmulatorContainer.Emulator.CLOUD_FIRESTORE, "quarkus.google.cloud.firestore.host-override", FirebaseEmulatorContainer.Emulator.CLOUD_STORAGE, "quarkus.google.cloud.storage.host-override", FirebaseEmulatorContainer.Emulator.PUB_SUB, "quarkus.google.cloud.pubsub.emulator-host"); diff --git a/firebase-devservices/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/deployment/testcontainers/FirebaseEmulatorContainer.java b/firebase-devservices/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/deployment/testcontainers/FirebaseEmulatorContainer.java index bde36472..61d13180 100644 --- a/firebase-devservices/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/deployment/testcontainers/FirebaseEmulatorContainer.java +++ b/firebase-devservices/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/deployment/testcontainers/FirebaseEmulatorContainer.java @@ -1494,10 +1494,10 @@ private void writeOutputFrame(OutputFrame frame, Level level) { } private String getEmulatorEndpoint(Emulator emulator) { - if (emulator.equals(Emulator.CLOUD_STORAGE)) { - return "http://" + this.getHost() + ":" + emulatorPort(emulator); + var endpoint = this.getHost() + ":" + emulatorPort(emulator); + if (emulator.equals(Emulator.REALTIME_DATABASE) || emulator.equals(Emulator.CLOUD_STORAGE)) { + endpoint = "http://" + endpoint; } - - return this.getHost() + ":" + emulatorPort(emulator); + return endpoint; } } diff --git a/firebase-realtime-database/deployment/pom.xml b/firebase-realtime-database/deployment/pom.xml new file mode 100644 index 00000000..22fb9879 --- /dev/null +++ b/firebase-realtime-database/deployment/pom.xml @@ -0,0 +1,71 @@ + + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-devservices-parent + 2.14.0-SNAPSHOT + ../pom.xml + + 4.0.0 + + quarkus-google-cloud-firebase-realtime-database-deployment + Quarkus - Google Cloud Services - Firebase Realtime Database - Deployment + + + + io.quarkus + quarkus-core-deployment + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-common-deployment + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-realtime-database + + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-admin + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-admin-deployment + + + + + org.testcontainers + junit-jupiter + test + + + io.quarkus + quarkus-grpc-common + test + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-common-grpc + test + + + + + + maven-compiler-plugin + + + + io.quarkus + quarkus-extension-processor + ${quarkus.version} + + + + + + + + \ No newline at end of file diff --git a/firebase-realtime-database/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/database/deployment/FirebaseDatabaseBuildSteps.java b/firebase-realtime-database/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/database/deployment/FirebaseDatabaseBuildSteps.java new file mode 100644 index 00000000..1a6ea44c --- /dev/null +++ b/firebase-realtime-database/deployment/src/main/java/io/quarkiverse/googlecloudservices/firebase/database/deployment/FirebaseDatabaseBuildSteps.java @@ -0,0 +1,22 @@ +package io.quarkiverse.googlecloudservices.firebase.database.deployment; + +import io.quarkiverse.googlecloudservices.firebase.database.FirebaseDatabaseProducer; +import io.quarkus.arc.deployment.AdditionalBeanBuildItem; +import io.quarkus.deployment.annotations.BuildStep; +import io.quarkus.deployment.builditem.FeatureBuildItem; + +public class FirebaseDatabaseBuildSteps { + + private static final String FEATURE = "google-cloud-firebase-realtime-database"; + + @BuildStep + public FeatureBuildItem feature() { + return new FeatureBuildItem(FEATURE); + } + + @BuildStep + public AdditionalBeanBuildItem producer() { + return new AdditionalBeanBuildItem(FirebaseDatabaseProducer.class); + } + +} diff --git a/firebase-realtime-database/pom.xml b/firebase-realtime-database/pom.xml new file mode 100644 index 00000000..0e7b2abc --- /dev/null +++ b/firebase-realtime-database/pom.xml @@ -0,0 +1,20 @@ + + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-services-parent + 2.14.0-SNAPSHOT + ../pom.xml + + 4.0.0 + + quarkus-google-cloud-firebase-realtime-database-parent + Quarkus - Google Cloud Services - Firebas Realtime Database + pom + + + runtime + deployment + + + \ No newline at end of file diff --git a/firebase-realtime-database/runtime/pom.xml b/firebase-realtime-database/runtime/pom.xml new file mode 100644 index 00000000..44889148 --- /dev/null +++ b/firebase-realtime-database/runtime/pom.xml @@ -0,0 +1,60 @@ + + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-realtime-database-parent + 2.14.0-SNAPSHOT + ../pom.xml + + 4.0.0 + + quarkus-google-cloud-firebase-realtime-database + Quarkus - Google Cloud Services - Firebase Realtime Database - Runtime + Use Google Cloud Firebase + + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-common + + + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-admin + + + + + + + io.quarkus + quarkus-extension-maven-plugin + ${quarkus.version} + + + + extension-descriptor + + compile + + ${project.groupId}:${project.artifactId}-deployment:${project.version} + + + + + + + maven-compiler-plugin + + + + io.quarkus + quarkus-extension-processor + ${quarkus.version} + + + + + + + \ No newline at end of file diff --git a/firebase-realtime-database/runtime/src/main/java/io/quarkiverse/googlecloudservices/firebase/database/FirebaseDatabaseConfig.java b/firebase-realtime-database/runtime/src/main/java/io/quarkiverse/googlecloudservices/firebase/database/FirebaseDatabaseConfig.java new file mode 100644 index 00000000..f27d8142 --- /dev/null +++ b/firebase-realtime-database/runtime/src/main/java/io/quarkiverse/googlecloudservices/firebase/database/FirebaseDatabaseConfig.java @@ -0,0 +1,24 @@ +package io.quarkiverse.googlecloudservices.firebase.database; + +import java.util.Optional; + +import io.quarkus.runtime.annotations.ConfigPhase; +import io.quarkus.runtime.annotations.ConfigRoot; +import io.smallrye.config.ConfigMapping; + +/** + * Root configuration class for Google Cloud Firebase Realtime database setup. + * + */ +@ConfigMapping(prefix = "quarkus.google.cloud.firebase.database") +@ConfigRoot(phase = ConfigPhase.RUN_TIME) +public interface FirebaseDatabaseConfig { + + /** + * Overrides the default service host. + * This is most commonly used for development or testing activities with a local Firebase Realtime Database emulator + * instance. + */ + Optional hostOverride(); + +} diff --git a/firebase-realtime-database/runtime/src/main/java/io/quarkiverse/googlecloudservices/firebase/database/FirebaseDatabaseProducer.java b/firebase-realtime-database/runtime/src/main/java/io/quarkiverse/googlecloudservices/firebase/database/FirebaseDatabaseProducer.java new file mode 100644 index 00000000..8091c234 --- /dev/null +++ b/firebase-realtime-database/runtime/src/main/java/io/quarkiverse/googlecloudservices/firebase/database/FirebaseDatabaseProducer.java @@ -0,0 +1,34 @@ +package io.quarkiverse.googlecloudservices.firebase.database; + +import jakarta.enterprise.context.ApplicationScoped; +import jakarta.enterprise.inject.Default; +import jakarta.enterprise.inject.Produces; +import jakarta.inject.Inject; +import jakarta.inject.Singleton; + +import com.google.firebase.FirebaseApp; +import com.google.firebase.database.FirebaseDatabase; + +/** + * Producer for the {@link FirebaseDatabase}. + */ +@ApplicationScoped +public class FirebaseDatabaseProducer { + + @Inject + FirebaseApp firebaseApp; + + @Inject + FirebaseDatabaseConfig config; + + @Produces + @Singleton + @Default + public FirebaseDatabase firebaseDatabase() { + if (config.hostOverride().isPresent()) { + return FirebaseDatabase.getInstance(firebaseApp, config.hostOverride().get()); + } else { + return FirebaseDatabase.getInstance(firebaseApp); + } + } +} diff --git a/firebase-realtime-database/runtime/src/main/resources/META-INF/quarkus-extension.yaml b/firebase-realtime-database/runtime/src/main/resources/META-INF/quarkus-extension.yaml new file mode 100644 index 00000000..cb7250b8 --- /dev/null +++ b/firebase-realtime-database/runtime/src/main/resources/META-INF/quarkus-extension.yaml @@ -0,0 +1,17 @@ +--- +name: "Google Cloud Firebase Realtime Database" +artifact: ${project.groupId}:${project.artifactId}:${project.version} +metadata: + keywords: + - "realtime database" + - "firebase" + - "google cloud" + - "gcloud" + - "gcp" + categories: + - "cloud" + - "data" + guide: "https://quarkiverse.github.io/quarkiverse-docs/quarkus-google-cloud-services/main/firebase-realtime-database.html" + status: "experimental" + config: + - "quarkus.google.cloud.firebase.database" diff --git a/integration-tests/firebase/pom.xml b/integration-tests/firebase/pom.xml index 0833face..095d97d4 100644 --- a/integration-tests/firebase/pom.xml +++ b/integration-tests/firebase/pom.xml @@ -26,6 +26,10 @@ io.quarkiverse.googlecloudservices quarkus-google-cloud-firebase-devservices + + io.quarkiverse.googlecloudservices + quarkus-google-cloud-firebase-realtime-database + io.quarkiverse.googlecloudservices quarkus-google-cloud-firestore diff --git a/integration-tests/firebase/src/main/java/io/quarkiverse/googlecloudservices/it/firebase/FirebaseResource.java b/integration-tests/firebase/src/main/java/io/quarkiverse/googlecloudservices/it/firebase/FirebaseResource.java index 3d084f66..207d9bf0 100644 --- a/integration-tests/firebase/src/main/java/io/quarkiverse/googlecloudservices/it/firebase/FirebaseResource.java +++ b/integration-tests/firebase/src/main/java/io/quarkiverse/googlecloudservices/it/firebase/FirebaseResource.java @@ -14,6 +14,7 @@ import com.google.cloud.firestore.Firestore; import com.google.cloud.pubsub.v1.Publisher; +import com.google.firebase.database.FirebaseDatabase; import com.google.protobuf.ByteString; import com.google.pubsub.v1.PubsubMessage; @@ -29,6 +30,9 @@ public class FirebaseResource { @Inject QuarkusPubSub quarkusPubSub; + @Inject + FirebaseDatabase firebaseDatabase; + Publisher publisher; public void init(@Observes StartupEvent event) throws IOException { @@ -48,15 +52,23 @@ public void createData(String data) throws InterruptedException, ExecutionExcept var fields = Map.of("test", msgData); col.document("test").create(fields).get(); + + var dbRef = firebaseDatabase.getReference("test"); + dbRef.setValue(fields, (err, ref) -> { + if (err == null) { + // Not pretty, but we just let the consumer timeout. + consumer.ack(); + } + + synchronized (monitor) { + monitor.notify(); + } + }); + } catch (InterruptedException | ExecutionException e) { throw new RuntimeException(e); } - consumer.ack(); - - synchronized (monitor) { - monitor.notify(); - } }); subscriber.startAsync().awaitRunning(); @@ -80,6 +92,7 @@ public Response getData() throws ExecutionException, InterruptedException { if (iter.hasNext()) { var docRef = iter.next(); var snapshot = docRef.get().get(); + return Response.ok(snapshot.getString("test")).build(); } else { return Response.status(Response.Status.NOT_FOUND).build(); diff --git a/integration-tests/firebase/src/main/resources/application.properties b/integration-tests/firebase/src/main/resources/application.properties index 67e2656b..c93b2805 100644 --- a/integration-tests/firebase/src/main/resources/application.properties +++ b/integration-tests/firebase/src/main/resources/application.properties @@ -5,5 +5,6 @@ quarkus.google.cloud.devservices.project-id=demo-test-project-id quarkus.google.cloud.access-token-enabled=false quarkus.google.cloud.devservices.firebase.auth.enabled=true quarkus.google.cloud.devservices.firebase.firestore.enabled=true +quarkus.google.cloud.devservices.firebase.database.enabled=true quarkus.google.cloud.devservices.pubsub.enabled=true quarkus.google.cloud.devservices.firebase.emulator.cli.experiments=webframeworks diff --git a/pom.xml b/pom.xml index f6f42c6e..83111766 100644 --- a/pom.xml +++ b/pom.xml @@ -32,6 +32,7 @@ storage firebase-devservices firebase-admin + firebase-realtime-database firestore bigtable secret-manager