From 803071baf2603b7e90dc07ac9f9f6029c1b58ee7 Mon Sep 17 00:00:00 2001 From: Christina Holland Date: Fri, 12 Jul 2024 11:50:13 -0700 Subject: [PATCH] Add import comments to all samples (#207) --- samples/cache.js | 17 +++ samples/chat.js | 6 + samples/code_execution.js | 6 + samples/controlled_generation.js | 4 + samples/count_tokens.js | 19 +++ samples/embed.js | 4 + samples/files.js | 18 +++ samples/function_calling.js | 2 + samples/model_configuration.js | 2 + samples/package.json | 1 + samples/safety_settings.js | 4 + samples/system_instruction.js | 2 + samples/text_generation.js | 20 ++++ samples/utils/check-samples.js | 70 ++++++----- samples/utils/common.js | 64 ++++++++++ samples/utils/insert-import-comments.js | 148 ++++++++++++++++++++++++ 16 files changed, 351 insertions(+), 36 deletions(-) create mode 100644 samples/utils/common.js create mode 100644 samples/utils/insert-import-comments.js diff --git a/samples/cache.js b/samples/cache.js index 2e991f38..2e783aea 100644 --- a/samples/cache.js +++ b/samples/cache.js @@ -28,6 +28,9 @@ const mediaPath = __dirname + "/media"; async function cacheCreate() { // [START cache_create] + // Make sure to include these imports: + // import { GoogleAICacheManager, GoogleAIFileManager } from "@google/generative-ai/server"; + // import { GoogleGenerativeAI } from "@google/generative-ai"; const cacheManager = new GoogleAICacheManager(process.env.API_KEY); const fileManager = new GoogleAIFileManager(process.env.API_KEY); @@ -66,6 +69,9 @@ async function cacheCreate() { async function cacheCreateFromName() { // [START cache_create_from_name] + // Make sure to include these imports: + // import { GoogleAICacheManager, GoogleAIFileManager } from "@google/generative-ai/server"; + // import { GoogleGenerativeAI } from "@google/generative-ai"; const cacheManager = new GoogleAICacheManager(process.env.API_KEY); const fileManager = new GoogleAIFileManager(process.env.API_KEY); @@ -102,6 +108,9 @@ async function cacheCreateFromName() { async function cacheCreateFromChat() { // [START cache_create_from_chat] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; + // import { GoogleAICacheManager, GoogleAIFileManager } from "@google/generative-ai/server"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const cacheManager = new GoogleAICacheManager(process.env.API_KEY); const fileManager = new GoogleAIFileManager(process.env.API_KEY); @@ -147,6 +156,8 @@ async function cacheCreateFromChat() { async function cacheDelete() { // [START cache_delete] + // Make sure to include these imports: + // import { GoogleAICacheManager, GoogleAIFileManager } from "@google/generative-ai/server"; const cacheManager = new GoogleAICacheManager(process.env.API_KEY); const fileManager = new GoogleAIFileManager(process.env.API_KEY); @@ -176,6 +187,8 @@ async function cacheDelete() { async function cacheGet() { // [START cache_get] + // Make sure to include these imports: + // import { GoogleAICacheManager, GoogleAIFileManager } from "@google/generative-ai/server"; const cacheManager = new GoogleAICacheManager(process.env.API_KEY); const fileManager = new GoogleAIFileManager(process.env.API_KEY); @@ -207,6 +220,8 @@ async function cacheGet() { async function cacheList() { // [START cache_list] + // Make sure to include these imports: + // import { GoogleAICacheManager, GoogleAIFileManager } from "@google/generative-ai/server"; const cacheManager = new GoogleAICacheManager(process.env.API_KEY); const fileManager = new GoogleAIFileManager(process.env.API_KEY); @@ -241,6 +256,8 @@ async function cacheList() { async function cacheUpdate() { // [START cache_update] + // Make sure to include these imports: + // import { GoogleAICacheManager, GoogleAIFileManager } from "@google/generative-ai/server"; const cacheManager = new GoogleAICacheManager(process.env.API_KEY); const fileManager = new GoogleAIFileManager(process.env.API_KEY); diff --git a/samples/chat.js b/samples/chat.js index b3b8712d..6f0ca5ff 100644 --- a/samples/chat.js +++ b/samples/chat.js @@ -25,6 +25,8 @@ const mediaPath = __dirname + "/media"; async function chat() { // [START chat] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); const chat = model.startChat({ @@ -48,6 +50,8 @@ async function chat() { async function chatStreaming() { // [START chat_streaming] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); const chat = model.startChat({ @@ -77,6 +81,8 @@ async function chatStreaming() { async function chatStreamingWithImages() { // [START chat_streaming_with_images] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); const chat = model.startChat(); diff --git a/samples/code_execution.js b/samples/code_execution.js index cd3fdbbb..83a2b174 100644 --- a/samples/code_execution.js +++ b/samples/code_execution.js @@ -19,6 +19,8 @@ import { GoogleGenerativeAI } from "@google/generative-ai"; async function codeExecutionBasic() { // [START code_execution_basic] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", @@ -37,6 +39,8 @@ async function codeExecutionBasic() { async function codeExecutionRequestOverride() { // [START code_execution_request_override] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", @@ -65,6 +69,8 @@ async function codeExecutionRequestOverride() { async function codeExecutionChat() { // [START code_execution_chat] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", diff --git a/samples/controlled_generation.js b/samples/controlled_generation.js index 533f881f..f1f1ce75 100644 --- a/samples/controlled_generation.js +++ b/samples/controlled_generation.js @@ -22,6 +22,8 @@ import { async function jsonControlledGeneration() { // [START json_controlled_generation] + // Make sure to include these imports: + // import { GoogleGenerativeAI, FunctionDeclarationSchemaType } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const schema = { @@ -57,6 +59,8 @@ async function jsonControlledGeneration() { async function jsonNoSchema() { // [START json_no_schema] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ diff --git a/samples/count_tokens.js b/samples/count_tokens.js index ab83c91c..d93a4c2d 100644 --- a/samples/count_tokens.js +++ b/samples/count_tokens.js @@ -30,6 +30,8 @@ const mediaPath = __dirname + "/media"; async function tokensTextOnly() { // [START tokens_text_only] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", @@ -57,6 +59,8 @@ async function tokensTextOnly() { async function tokensChat() { // [START tokens_chat] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", @@ -70,6 +74,8 @@ async function tokensChat() { async function tokensMultimodalImageInline() { // [START tokens_multimodal_image_inline] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", @@ -99,6 +105,9 @@ async function tokensMultimodalImageInline() { async function tokensMultimodalImageFileApi() { // [START tokens_multimodal_image_file_api] + // Make sure to include these imports: + // import { GoogleAIFileManager } from "@google/generative-ai/server"; + // import { GoogleGenerativeAI } from "@google/generative-ai"; const fileManager = new GoogleAIFileManager(process.env.API_KEY); const uploadResult = await fileManager.uploadFile( @@ -129,6 +138,9 @@ async function tokensMultimodalImageFileApi() { async function tokensMultimodalVideoAudioFileApi() { // [START tokens_multimodal_video_audio_file_api] + // Make sure to include these imports: + // import { GoogleAIFileManager, FileState } from "@google/generative-ai/server"; + // import { GoogleGenerativeAI } from "@google/generative-ai"; const fileManager = new GoogleAIFileManager(process.env.API_KEY); function waitForProcessing(fileName) { @@ -195,6 +207,9 @@ async function tokensMultimodalVideoAudioFileApi() { async function tokensCachedContent() { // [START tokens_cached_content] + // Make sure to include these imports: + // import { GoogleAICacheManager } from "@google/generative-ai/server"; + // import { GoogleGenerativeAI } from "@google/generative-ai"; // Generate a very long string let longContentString = ""; for (let i = 0; i < 32001; i++) { @@ -235,6 +250,8 @@ async function tokensCachedContent() { async function tokensSystemInstruction() { // [START tokens_system_instruction] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "models/gemini-1.5-flash", @@ -266,6 +283,8 @@ async function tokensSystemInstruction() { async function tokensTools() { // [START tokens_tools] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "models/gemini-1.5-flash", diff --git a/samples/embed.js b/samples/embed.js index 0b39f88d..890465cb 100644 --- a/samples/embed.js +++ b/samples/embed.js @@ -19,6 +19,8 @@ import { GoogleGenerativeAI } from "@google/generative-ai"; async function embedContent() { // [START embed_content] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "text-embedding-004", @@ -32,6 +34,8 @@ async function embedContent() { async function batchEmbedContents() { // [START batch_embed_contents] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "text-embedding-004", diff --git a/samples/files.js b/samples/files.js index 8bf99b82..71a899e7 100644 --- a/samples/files.js +++ b/samples/files.js @@ -25,6 +25,9 @@ const mediaPath = __dirname + "/media"; async function filesCreateImage() { // [START files_create_image] + // Make sure to include these imports: + // import { GoogleAIFileManager } from "@google/generative-ai/server"; + // import { GoogleGenerativeAI } from "@google/generative-ai"; const fileManager = new GoogleAIFileManager(process.env.API_KEY); const uploadResult = await fileManager.uploadFile( @@ -56,6 +59,9 @@ async function filesCreateImage() { async function filesCreateAudio() { // [START files_create_audio] + // Make sure to include these imports: + // import { GoogleAIFileManager, FileState } from "@google/generative-ai/server"; + // import { GoogleGenerativeAI } from "@google/generative-ai"; const fileManager = new GoogleAIFileManager(process.env.API_KEY); const uploadResult = await fileManager.uploadFile( @@ -101,6 +107,9 @@ async function filesCreateAudio() { async function filesCreateText() { // [START files_create_text] + // Make sure to include these imports: + // import { GoogleAIFileManager } from "@google/generative-ai/server"; + // import { GoogleGenerativeAI } from "@google/generative-ai"; const fileManager = new GoogleAIFileManager(process.env.API_KEY); const uploadResult = await fileManager.uploadFile(`${mediaPath}/a11.txt`, { @@ -129,6 +138,9 @@ async function filesCreateText() { async function filesCreateVideo() { // [START files_create_video] + // Make sure to include these imports: + // import { GoogleAIFileManager, FileState } from "@google/generative-ai/server"; + // import { GoogleGenerativeAI } from "@google/generative-ai"; const fileManager = new GoogleAIFileManager(process.env.API_KEY); const uploadResult = await fileManager.uploadFile( @@ -174,6 +186,8 @@ async function filesCreateVideo() { async function filesList() { // [START files_list] + // Make sure to include these imports: + // import { GoogleAIFileManager } from "@google/generative-ai/server"; const fileManager = new GoogleAIFileManager(process.env.API_KEY); const listFilesResponse = await fileManager.listFiles(); @@ -187,6 +201,8 @@ async function filesList() { async function filesGet() { // [START files_get] + // Make sure to include these imports: + // import { GoogleAIFileManager } from "@google/generative-ai/server"; const fileManager = new GoogleAIFileManager(process.env.API_KEY); const uploadResponse = await fileManager.uploadFile( @@ -209,6 +225,8 @@ async function filesGet() { async function filesDelete() { // [START files_delete] + // Make sure to include these imports: + // import { GoogleAIFileManager } from "@google/generative-ai/server"; const fileManager = new GoogleAIFileManager(process.env.API_KEY); const uploadResult = await fileManager.uploadFile( diff --git a/samples/function_calling.js b/samples/function_calling.js index 98e34fb3..a864efb5 100644 --- a/samples/function_calling.js +++ b/samples/function_calling.js @@ -19,6 +19,8 @@ import { GoogleGenerativeAI } from "@google/generative-ai"; async function functionCalling() { // [START function_calling] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; async function setLightValues(brightness, colorTemperature) { // This mock API returns the requested lighting values return { diff --git a/samples/model_configuration.js b/samples/model_configuration.js index a2b76910..2146f902 100644 --- a/samples/model_configuration.js +++ b/samples/model_configuration.js @@ -19,6 +19,8 @@ import { GoogleGenerativeAI } from "@google/generative-ai"; async function configureModel() { // [START configure_model] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", diff --git a/samples/package.json b/samples/package.json index aadabcd0..6a79a2a3 100644 --- a/samples/package.json +++ b/samples/package.json @@ -5,6 +5,7 @@ }, "scripts": { "check-samples": "node ./utils/check-samples.js", + "import-comments": "node ./utils/insert-import-comments.js", "test": "yarn check-samples" } } diff --git a/samples/safety_settings.js b/samples/safety_settings.js index 6a6d2355..0a6070c9 100644 --- a/samples/safety_settings.js +++ b/samples/safety_settings.js @@ -23,6 +23,8 @@ import { async function safetySettings() { // [START safety_settings] + // Make sure to include these imports: + // import { GoogleGenerativeAI, HarmCategory, HarmBlockThreshold } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", @@ -52,6 +54,8 @@ async function safetySettings() { async function safetySettingsMulti() { // [START safety_settings_multi] + // Make sure to include these imports: + // import { GoogleGenerativeAI, HarmCategory, HarmBlockThreshold } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", diff --git a/samples/system_instruction.js b/samples/system_instruction.js index 8a7b682b..bbca5f75 100644 --- a/samples/system_instruction.js +++ b/samples/system_instruction.js @@ -19,6 +19,8 @@ import { GoogleGenerativeAI } from "@google/generative-ai"; async function systemInstruction() { // [START system_instruction] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash", diff --git a/samples/text_generation.js b/samples/text_generation.js index 1049a2b5..23fc52d5 100644 --- a/samples/text_generation.js +++ b/samples/text_generation.js @@ -26,6 +26,8 @@ const mediaPath = __dirname + "/media"; async function textGenTextOnlyPrompt() { // [START text_gen_text_only_prompt] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); @@ -38,6 +40,8 @@ async function textGenTextOnlyPrompt() { async function textGenTextOnlyPromptStreaming() { // [START text_gen_text_only_prompt_streaming] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); @@ -55,6 +59,8 @@ async function textGenTextOnlyPromptStreaming() { async function textGenMultimodalOneImagePrompt() { // [START text_gen_multimodal_one_image_prompt] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); @@ -81,6 +87,8 @@ async function textGenMultimodalOneImagePrompt() { async function textGenMultimodalOneImagePromptStreaming() { // [START text_gen_multimodal_one_image_prompt_streaming] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); @@ -112,6 +120,8 @@ async function textGenMultimodalOneImagePromptStreaming() { async function textGenMultimodalMultiImagePrompt() { // [START text_gen_multimodal_multi_image_prompt] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); @@ -142,6 +152,8 @@ async function textGenMultimodalMultiImagePrompt() { async function textGenMultimodalMultiImagePromptStreaming() { // [START text_gen_multimodal_multi_image_prompt_streaming] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); @@ -177,6 +189,8 @@ async function textGenMultimodalMultiImagePromptStreaming() { async function textGenMultimodalAudio() { // [START text_gen_multimodal_audio] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); @@ -203,6 +217,9 @@ async function textGenMultimodalAudio() { async function textGenMultimodalVideoPrompt() { // [START text_gen_multimodal_video_prompt] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; + // import { GoogleAIFileManager, FileState } from "@google/generative-ai/server"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); @@ -242,6 +259,9 @@ async function textGenMultimodalVideoPrompt() { async function textGenMultimodalVideoPromptStreaming() { // [START text_gen_multimodal_video_prompt_streaming] + // Make sure to include these imports: + // import { GoogleGenerativeAI } from "@google/generative-ai"; + // import { GoogleAIFileManager, FileState } from "@google/generative-ai/server"; const genAI = new GoogleGenerativeAI(process.env.API_KEY); const model = genAI.getGenerativeModel({ model: "gemini-1.5-flash" }); diff --git a/samples/utils/check-samples.js b/samples/utils/check-samples.js index 47b03448..efd3559f 100644 --- a/samples/utils/check-samples.js +++ b/samples/utils/check-samples.js @@ -15,48 +15,43 @@ * limitations under the License. */ +import { findFunctions, samplesDir } from "./common.js"; import fs from "fs"; -import { dirname, join } from "path"; -import { fileURLToPath } from "url"; - -const __dirname = dirname(fileURLToPath(import.meta.url)); -const samplesDir = join(__dirname, '../') +import { join } from "path"; +/** + * Checks samples to make sure they have region tags and the tags match the + * function name. + */ async function checkSamples() { const files = fs.readdirSync(samplesDir); for (const filename of files) { - if (filename.match(/.+\.js$/) && !filename.includes('-')) { - const file = fs.readFileSync(join(samplesDir, filename), 'utf-8'); - const lines = file.split('\n'); - let currentFunctionName = ''; - let currentStartTag = ''; - let tagsOk = false; - for (const line of lines) { - const functionStartParts = line.match(/^(async function|function) (.+)\(/); - if (functionStartParts) { - currentFunctionName = functionStartParts[2]; + if (filename.match(/.+\.js$/) && !filename.includes("-")) { + const file = fs.readFileSync(join(samplesDir, filename), "utf-8"); + const functions = findFunctions(file); + for (const sampleFn in functions) { + if (sampleFn === "runAll" || sampleFn === "run") { + continue; } - const tagStartParts = line.match(/\/\/ \[START (.+)\]/); - if (tagStartParts) { - currentStartTag = tagStartParts[1]; - if (camelCaseToUnderscore(currentFunctionName) !== currentStartTag) { - console.error(`[${filename}]: Region start tag ${currentStartTag} doesn't match function name ${currentFunctionName}`); - } + if (!functions[sampleFn].startTag || !functions[sampleFn].endTag) { + console.error( + `[${filename}]: Start and end tag not found or not correct in function ${sampleFn}`, + ); } - const tagEndParts = line.match(/\/\/ \[END (.+)\]/); - if (tagEndParts) { - if (tagEndParts[1] !== currentStartTag) { - console.error(`[${filename}]: Region end tag ${currentEndTag} doesn't match start tag ${currentStartTag}`); - } else { - tagsOk = true; - } + if ( + camelCaseToUnderscore(sampleFn) !== functions[sampleFn].startTag.tag + ) { + console.error( + `[${filename}]: Region start tag ${functions[sampleFn].startTag.tag} doesn't match function name ${sampleFn}`, + ); } - if (line.match(/^}$/)) { - if (!tagsOk && currentFunctionName !== 'runAll') { - console.error(`[${filename}]: Start and end tag not found or not correct in function ${currentFunctionName}`); - } - currentFunctionName = ''; - tagsOk = false; + if ( + functions[sampleFn].startTag.tag !== functions[sampleFn].endTag.tag || + functions[sampleFn].endTag.line <= functions[sampleFn].startTag.line + ) { + console.error( + `[${filename}]: Region end tag ${functions[sampleFn].endTag.tag} doesn't match start tag ${functions[sampleFn].startTag.tag}`, + ); } } } @@ -64,7 +59,10 @@ async function checkSamples() { } function camelCaseToUnderscore(camelCaseName) { - return camelCaseName.split(/\.?(?=[A-Z])/).join('_').toLowerCase(); + return camelCaseName + .split(/\.?(?=[A-Z])/) + .join("_") + .toLowerCase(); } -checkSamples(); \ No newline at end of file +checkSamples(); diff --git a/samples/utils/common.js b/samples/utils/common.js new file mode 100644 index 00000000..f7eeba99 --- /dev/null +++ b/samples/utils/common.js @@ -0,0 +1,64 @@ +/** + * @license + * Copyright 2023 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import { dirname, join } from "path"; +import { fileURLToPath } from "url"; + +const __dirname = dirname(fileURLToPath(import.meta.url)); +export const samplesDir = join(__dirname, "../"); + +/** + * Extracts individual function information, given the text of a samples file. + */ +export function findFunctions(fileText) { + const lines = fileText.split("\n"); + const functions = {}; + let currentFunctionName = ""; + for (const [index, line] of lines.entries()) { + const functionStartParts = line.match(/^(async function|function) (.+)\(/); + if (functionStartParts) { + currentFunctionName = functionStartParts[2]; + functions[currentFunctionName] = { body: [] }; + } else if (line.match(/^}$/)) { + currentFunctionName = ""; + } else if (currentFunctionName) { + const tagStartParts = line.match(/\/\/ \[START (.+)\]/); + const tagEndParts = line.match(/\/\/ \[END (.+)\]/); + const importHead = line.match(/\/\/ Make sure to include/); + const importComment = line.match(/\/\/ import /); + if (tagStartParts) { + functions[currentFunctionName].startTag = { + line: index, + tag: tagStartParts[1], + }; + } else if (tagEndParts) { + functions[currentFunctionName].endTag = { + line: index, + tag: tagEndParts[1], + }; + } else if (importHead || importComment) { + if (!functions[currentFunctionName].importComments) { + functions[currentFunctionName].importComments = []; + } + functions[currentFunctionName].importComments.push(line); + } else { + functions[currentFunctionName].body.push(line); + } + } + } + return functions; +} diff --git a/samples/utils/insert-import-comments.js b/samples/utils/insert-import-comments.js new file mode 100644 index 00000000..54fd1ea3 --- /dev/null +++ b/samples/utils/insert-import-comments.js @@ -0,0 +1,148 @@ +/** + * @license + * Copyright 2024 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import { findFunctions, samplesDir } from "./common.js"; +import fs from "fs"; +import ts from "typescript"; +import { dirname, join } from "path"; +import { createRequire } from "module"; + +const require = createRequire(import.meta.url); + +function getTopLevelSymbols(filePath) { + const typings = fs.readFileSync(filePath, "utf-8"); + + const sourceFile = ts.createSourceFile( + filePath, + typings, + ts.ScriptTarget.ES2015, + ); + let symbols = []; + ts.forEachChild(sourceFile, (node) => { + if (node.name) { + symbols.push(node.name.text); + } + }); + return symbols; +} + +export function getAvailableSymbols() { + let packagePath = require.resolve('@google/generative-ai/package.json'); + const pkg = require(packagePath); + const coreSymbols = getTopLevelSymbols(join(dirname(packagePath), pkg.exports["."].types)); + const serverSymbolsRaw = getTopLevelSymbols(join(dirname(packagePath), pkg.exports["./server"].types)); + const serverSymbols = serverSymbolsRaw.filter( + (serverSymbol) => !coreSymbols.includes(serverSymbol), + ); + return [ + { + importPath: "@google/generative-ai", + symbols: coreSymbols, + }, + { + importPath: "@google/generative-ai/server", + symbols: serverSymbols, + }, + ]; +} + +const requiredImports = getAvailableSymbols(); + +function listRequiredImports(line) { + const results = []; + for (const requiredImport of requiredImports) { + for (const symbol of requiredImport.symbols) { + if (line.match(new RegExp(`[^a-zA-Z0-9]${symbol}[^a-zA-Z0-9]`))) { + if (!results[requiredImport.importPath]) { + results[requiredImport.importPath] = []; + } + results[requiredImport.importPath].push(symbol); + results.push({ symbol, importPath: requiredImport.importPath }); + } + } + } + return results; +} + +/** + * Inserts comments describing the required imports for making the code + * sample work, since we cannot add actual import statements inside + * the samples. + */ +async function insertImportComments() { + const files = fs.readdirSync(samplesDir); + for (const filename of files) { + if (filename.match(/.+\.js$/) && !filename.includes("-")) { + const file = fs.readFileSync(join(samplesDir, filename), "utf-8"); + const functions = findFunctions(file); + for (const fnName in functions) { + const sampleFn = functions[fnName]; + let results = []; + for (const line of sampleFn.body) { + results = results.concat(listRequiredImports(line)); + } + if (results.length > 0) { + functions[fnName].requiredImports = {}; + for (const result of results) { + if (!functions[fnName].requiredImports[result.importPath]) { + functions[fnName].requiredImports[result.importPath] = new Set(); + } + functions[fnName].requiredImports[result.importPath].add( + result.symbol, + ); + } + } + } + const fileLines = file.split("\n"); + const newFileLines = []; + for (const fileLine of fileLines) { + const importHead = fileLine.match(/\/\/ Make sure to include/); + const importComment = fileLine.match(/\/\/ import /); + if (!importHead && !importComment) { + newFileLines.push(fileLine); + } + const tagStartParts = fileLine.match(/\/\/ \[START (.+)\]/); + if (tagStartParts) { + const fnName = underscoreToCamelCase(tagStartParts[1]); + if (functions[fnName].requiredImports) { + newFileLines.push(` // Make sure to include these imports:`); + for (const importPath in functions[fnName].requiredImports) { + const symbols = Array.from( + functions[fnName].requiredImports[importPath], + ); + newFileLines.push( + ` // import { ${symbols.join(", ")} } from "${importPath}";`, + ); + } + } + } + } + fs.writeFileSync(join(samplesDir, filename), newFileLines.join("\n")); + } + } +} + +function underscoreToCamelCase(underscoreName) { + return underscoreName + .split("_") + .map((part, i) => + i === 0 ? part : part.charAt(0).toUpperCase() + part.slice(1), + ) + .join(""); +} + +insertImportComments();