Skip to content

Commit 703c035

Browse files
authored
Merge pull request #1135 from firebase/harveyjen-rc-server-vertex
Add function sample for server-side Remote Config + Vertex AI Gemini API
2 parents 071ac15 + 547306d commit 703c035

File tree

16 files changed

+2161
-52
lines changed

16 files changed

+2161
-52
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
Call the Vertex AI Gemini API with Remote Config and App Check
2+
==============================================================
3+
4+
Introduction
5+
------------
6+
7+
This is a sample callable function that authenticates clients with App
8+
Check and then sends queries to Gemini using the Vertex AI Gemini API. Vertex
9+
AI model parameters (including the model itself) are controlled by
10+
Remote Config server features included in the Firebase Admin SDK for
11+
Node.js.
12+
13+
Use the web client provided in `client/` to test the function.
14+
15+
- [Read more about Remote Config for servers](https://firebase.google.com/docs/remote-config/server).
16+
- [Read more about App Check](https://firebase.google.com/docs/app-check).
17+
- [Read more about the Vertex AI Node.js Client library](https://cloud.google.com/nodejs/docs/reference/aiplatform/latest).
18+
19+
Important: Vertex AI and Cloud Functions require a billing account. Review
20+
[Vertex AI pricing](https://cloud.google.com/vertex-ai/pricing) and
21+
[Firebase pricing](https://firebase.google.com/pricing) before running
22+
this function. If you're new to Firebase and Google Cloud, check to see if
23+
you're eligible for a
24+
[$300 credit](https://firebase.google.com/support/faq#pricing-free-trial) and
25+
a Free Trial Cloud Billing account.
26+
27+
Get Started
28+
---------------
29+
30+
1. Follow the instructions in client/README.md to create a Firebase project,
31+
enable ReCAPTCHA Enterprise, enable and enforce Firebase App Check, and add
32+
your Firebase config and ReCAPTCHA Enterprise key to the client config.
33+
34+
2. Enable [recommended Vertex AI APIs](https://console.cloud.google.com/vertex-ai).
35+
36+
3. Configure a Remote Config server template on the Firebase console. Use the template
37+
described in
38+
[Use server side Remote Config with Cloud Functions and Vertex
39+
AI](https://firebase.google.com/docs/remote-config/solution-server#implementation-create-template),
40+
which contains all of the parameters used in this function sample.
41+
42+
4. Install dependencies: `cd functions && npm install`
43+
44+
5. If you haven't already done so, install firebase-tools:
45+
46+
`npm i firebase-tools@latest`
47+
48+
6. Log into Firebase:
49+
50+
`firebase login`
51+
52+
7. Deploy the function. We recommend testing in the
53+
[Firebase emulator](https://firebase.google.com/docs/remote-config/solution-server#implementation-deploy-and-test-in-emulator):
54+
55+
`firebase emulators:start`
56+
57+
8. If testing in the emulator, verify that `testMode` is set to `true` in
58+
`client/main.ts`, then start the client:
59+
60+
`cd client && npm run dev`
61+
62+
TIP: If you're using the emulator, you can deploy both the function and hosting
63+
to the emulator. From the `client` directory, run `npm run build`.
64+
Then, from the parent directory, run `firebase server --only functions,hosting`.
65+
Open http://localhost:5000 to access and test the web client's connection
66+
to the `callVertexWithRC` function.
67+
68+
0. Open the [client app in a browser](http://localhost:5173) and enter a
69+
prompt. To access the Vertex AI Gemini API, make sure that you have
70+
set the `is_vertex_enabled` boolean parameter in your Remote Config
71+
server template to `true`.
72+
73+
Support
74+
-------
75+
76+
- [Firebase Support](https://firebase.google.com/support/)
77+
78+
License
79+
-------
80+
81+
© Google, 2024. Licensed under an [Apache-2](../../LICENSE) license.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
Test client for call-vertex-remote-config-server
2+
================================================
3+
4+
Introduction
5+
------------
6+
7+
This is a basic web app that calls the `callVertexWithRC` function. The
8+
function uses values stored in Remote Config server templates with
9+
the Firebase Admin SDK to dynamically update Vertex AI Gemini API
10+
parameters. Access is controlled using Firebase App Check.
11+
12+
- [Read more about Remote Config for servers](https://firebase.google.com/docs/remote-config/server).
13+
- [Read more about App Check](https://firebase.google.com/docs/app-check).
14+
- [Read more about the Vertex AI Node.js Client library](https://cloud.google.com/nodejs/docs/reference/aiplatform/latest).
15+
16+
Important: Vertex AI and Cloud Functions require a billing account. Review
17+
[Vertex AI pricing](https://cloud.google.com/vertex-ai/pricing) and
18+
[Firebase pricing](https://firebase.google.com/pricing) before running
19+
this function. If you're new to Firebase and Google Cloud, check to see if
20+
you're eligible for a
21+
[$300 credit](https://firebase.google.com/support/faq#pricing-free-trial) and
22+
a Free Trial Cloud Billing account.
23+
24+
Get started
25+
---------------
26+
27+
1. Create a [Firebase project and register a web app](https://firebase.google.com/docs/web/setup#create-firebase-project-and-app).
28+
2. [Create a ReCAPTCHA Enterprise key](https://firebase.google.com/docs/app-check/web/recaptcha-enterprise-provider#project-setup)
29+
in the same project.
30+
3. [Enable App Check](https://firebase.google.com/docs/app-check/web/recaptcha-enterprise-provider)
31+
in the Firebase console with the ReCAPTCHA Enterprise site key you created.
32+
4. Copy your Firebase project config and your ReCAPTCHA Enterprise site key
33+
into the appropriate places in `config.ts` in this directory.
34+
5. In this directory, run `npm install`.
35+
6. Set up and deploy the function as described in [../README.md](../README.md).
36+
7. In this directory, run `npm run dev` to run the client.
37+
38+
To run this app against the `callVertexWithRC` function running in the Firebase
39+
emulator, ensure that `testMode`in `main.ts` is set to `true`. Before testing
40+
with a deployed (i.e., not emulated) function, set `testMode` to `false`.
41+
42+
TIP: You can build the client and deploy to Firebase Hosting by running
43+
`npm run build` from the `client` directory. Hosting deliverables are
44+
generated and saved in `client/dist` and you can then deploy to
45+
the emulator or Firebase Hosting from the parent directory. We recommend
46+
deploying to the emulator first--you can use the following command to
47+
deploy the function and web client simultaneously:
48+
49+
firebase serve --only functions,hosting
50+
51+
Support
52+
-------
53+
54+
- [Firebase Support](https://firebase.google.com/support/)
55+
56+
License
57+
-------
58+
59+
© Google, 2024. Licensed under an [Apache-2](../../../LICENSE) license.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
export const firebaseConfig = {
2+
YOUR_FIREBASE_CONFIG
3+
};
4+
5+
// Your ReCAPTCHA Enterprise site key (must be from the same project
6+
// as the Firebase config above).
7+
export const RECAPTCHA_ENTERPRISE_SITE_KEY =
8+
"YOUR_RECAPTCHA_KEY";
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
<!DOCTYPE html>
2+
<html lang="en">
3+
4+
<head>
5+
<meta charset="UTF-8">
6+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
7+
<title>Client app for Remote Config server function with Vertex AI and App Check</title>
8+
<style>
9+
body {
10+
font-family: Arial, sans-serif;
11+
background-color: #f2f2f2;
12+
justify-content: center;
13+
align-items: center;
14+
height: 100vh;
15+
margin: 70px;
16+
}
17+
18+
h1 {
19+
margin-bottom: 20px;
20+
}
21+
22+
.container {
23+
background-color: #ffffff;
24+
border-radius: 8px;
25+
box-shadow: 0px 0px 10px rgba(0, 0, 0, 0.1);
26+
padding: 20px;
27+
max-width: 400px;
28+
width: 100%;
29+
text-align: center;
30+
}
31+
32+
input[type="text"],
33+
input[type="submit"] {
34+
width: 35%;
35+
padding: 10px;
36+
margin-bottom: 10px;
37+
border: 1px solid #ccc;
38+
border-radius: 4px;
39+
box-sizing: border-box;
40+
}
41+
42+
input[type="submit"] {
43+
background-color: #FFA500;
44+
color: white;
45+
border: none;
46+
cursor: pointer;
47+
font-size: 16px;
48+
font-weight: bold;
49+
}
50+
51+
input[type="submit"]:hover {
52+
background-color: #45a049;
53+
}
54+
55+
#waitingMessage,
56+
#generatedText,
57+
#errorMessage {
58+
margin-top: 20px;
59+
text-align: left;
60+
}
61+
</style>
62+
</head>
63+
64+
<body>
65+
<div>
66+
<h1>Client app for Remote Config server function with Vertex AI and App Check</h1>
67+
<br />
68+
</div>
69+
<script type="module" src="main.ts"></script>
70+
</body>
71+
72+
</html>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,117 @@
1+
/**
2+
* @license
3+
* Copyright 2024 Google LLC
4+
*
5+
* Licensed under the Apache License, Version 2.0 (the "License");
6+
* you may not use this file except in compliance with the License.
7+
* You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*/
17+
18+
import { initializeApp } from "firebase/app";
19+
import { firebaseConfig, RECAPTCHA_ENTERPRISE_SITE_KEY } from "./config";
20+
import {
21+
initializeAppCheck,
22+
ReCaptchaEnterpriseProvider,
23+
} from "firebase/app-check";
24+
import {
25+
getFunctions,
26+
httpsCallable,
27+
connectFunctionsEmulator,
28+
} from "firebase/functions";
29+
30+
// Set to true to test in emulator.
31+
const testMode = true;
32+
33+
// Use showdown to convert Gemini-provided Markdown to HTML
34+
import { Converter } from "showdown";
35+
const converter = new Converter();
36+
37+
// Set up output elements.
38+
const outputDiv = document.createElement("div");
39+
document.body.appendChild(outputDiv);
40+
41+
// Initialize Firebase app.
42+
const app = initializeApp(firebaseConfig);
43+
44+
// Initialize App Check.
45+
initializeAppCheck(app, {
46+
provider: new ReCaptchaEnterpriseProvider(RECAPTCHA_ENTERPRISE_SITE_KEY),
47+
});
48+
49+
// Define callVertexWithRC as a call to the callVertexWithRC function.
50+
const callVertexWithRC = httpsCallable(getFunctions(), "callVertexWithRC", {
51+
limitedUseAppCheckTokens: true,
52+
});
53+
54+
// Enable emulator so that it can be used in test mode.
55+
const functions = getFunctions(app, "us-central1"); // Replace with your region
56+
57+
if (testMode) {
58+
connectFunctionsEmulator(functions, "localhost", 5001);
59+
}
60+
61+
// Generate body for index.html.
62+
document.body.innerHTML += `
63+
64+
<div id="waitingMessage"></div>
65+
<div id="generatedText"></div>
66+
<div id="errorMessage"></div>
67+
<br/>
68+
<form id="promptForm">
69+
<label for="promptInput">Ask Gemini a question!</label><br>
70+
<input type="text" id="promptInput" name="prompt"><br><br>
71+
<input type="submit" value="Submit">
72+
</form>
73+
`;
74+
75+
const promptForm = document.getElementById("promptForm") as HTMLFormElement;
76+
77+
promptForm.addEventListener("submit", async (event) => {
78+
event.preventDefault();
79+
const promptInput = document.getElementById(
80+
"promptInput"
81+
) as HTMLInputElement;
82+
const prompt = promptInput.value;
83+
84+
const waitingMessageElement = document.getElementById("waitingMessage");
85+
86+
// Define a variable to keep track of the number of dots
87+
let dotCount = 0;
88+
89+
// Set interval to add dots every second
90+
const intervalId = setInterval(() => {
91+
// Increment dotCount
92+
dotCount = (dotCount + 1) % 7;
93+
const dots = ".".repeat(dotCount);
94+
waitingMessageElement.textContent = "Waiting for response" + dots;
95+
}, 1000);
96+
97+
const errorMessageElement = document.getElementById("errorMessage");
98+
errorMessageElement.textContent = "";
99+
100+
try {
101+
const { data } = await callVertexWithRC({ prompt });
102+
const generatedTextElement = document.getElementById("generatedText"); // Access the element
103+
const htmlContent = converter.makeHtml(data);
104+
if (!generatedTextElement) {
105+
throw new Error("Missing generated text.");
106+
}
107+
generatedTextElement.innerHTML = htmlContent; // Set the element's content
108+
waitingMessageElement.textContent = "";
109+
errorMessageElement.textContent = "";
110+
111+
} catch (error) {
112+
errorMessageElement.textContent = "Error calling function: " + error.message;
113+
waitingMessageElement.textContent = "";
114+
}
115+
// Clear welcome dots.
116+
clearInterval(intervalId);
117+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
{
2+
"name": "call-vertex-remote-config-server-client",
3+
"version": "1.0.0",
4+
"description": "JavaScript quickstart for Vertex AI, Firebase Remote Config server, and App Check.",
5+
"repository": {
6+
"type": "git",
7+
"url": "git+https://github.com/firebase/functions-samples.git"
8+
},
9+
"author": "",
10+
"license": "Apache-2.0",
11+
"bugs": {
12+
"url": "https://github.com/firebase/functions-samples/issues"
13+
},
14+
"engines": {
15+
"npm": ">=9.0.0 <10.0.0",
16+
"node": ">=18.0.0 <=20.0.0"
17+
},
18+
"homepage": "https://github.com/firebase/functions-samples#readme",
19+
"devDependencies": {
20+
"typescript": "^5.1.6",
21+
"vite": "^4.4.9"
22+
},
23+
"scripts": {
24+
"dev": "vite",
25+
"build": "vite build",
26+
"format": "prettier --write ."
27+
},
28+
"dependencies": {
29+
"@firebase/functions": "^0.11.5",
30+
"firebase": "^10.12.1",
31+
"showdown": "^2.1.0"
32+
}
33+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
/// <reference types="vite/client" />
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
import { defineConfig } from 'vite';
2+
3+
export default defineConfig({
4+
base: '',
5+
build: {
6+
rollupOptions: {
7+
input: ['index.html','main.ts'],
8+
},
9+
},
10+
logLevel: 'info',
11+
});

0 commit comments

Comments
 (0)