MCP Terminal Server

/** * 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 { GenerateContentCandidate } from '@google/generative-ai'; import * as assert from 'assert'; import { genkit, z } from 'genkit'; import { MessageData, ModelInfo } from 'genkit/model'; import { toJsonSchema } from 'genkit/schema'; import { afterEach, beforeEach, describe, it } from 'node:test'; import { GENERIC_GEMINI_MODEL, cleanSchema, fromGeminiCandidate, gemini, gemini15Flash, gemini15Pro, toGeminiMessage, toGeminiSystemInstruction, toGeminiTool, } from '../src/gemini.js'; import { googleAI } from '../src/index.js'; describe('toGeminiMessages', () => { const testCases = [ { should: 'should transform genkit message (text content) correctly', inputMessage: { role: 'user', content: [{ text: 'Tell a joke about dogs.' }], }, expectedOutput: { role: 'user', parts: [{ text: 'Tell a joke about dogs.' }], }, }, { should: 'should transform genkit message (tool request content) correctly', inputMessage: { role: 'model', content: [ { toolRequest: { name: 'tellAFunnyJoke', input: { topic: 'dogs' } } }, ], }, expectedOutput: { role: 'model', parts: [ { functionCall: { name: 'tellAFunnyJoke', args: { topic: 'dogs' } } }, ], }, }, { should: 'should transform genkit message (tool response content) correctly', inputMessage: { role: 'tool', content: [ { toolResponse: { name: 'tellAFunnyJoke', output: 'Why did the dogs cross the road?', }, }, ], }, expectedOutput: { role: 'function', parts: [ { functionResponse: { name: 'tellAFunnyJoke', response: { name: 'tellAFunnyJoke', content: 'Why did the dogs cross the road?', }, }, }, ], }, }, { should: 'should transform genkit message (inline base64 image content) correctly', inputMessage: { role: 'user', content: [ { text: 'describe the following image:' }, { media: { contentType: 'image/jpeg', url: '', }, }, ], }, expectedOutput: { role: 'user', parts: [ { text: 'describe the following image:' }, { inlineData: { mimeType: 'image/jpeg', data: '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', }, }, ], }, }, ]; for (const test of testCases) { it(test.should, () => { assert.deepEqual( toGeminiMessage(test.inputMessage as MessageData), test.expectedOutput ); }); } }); describe('toGeminiSystemInstruction', () => { const testCases = [ { should: 'should transform from system to user', inputMessage: { role: 'system', content: [{ text: 'You are an expert in all things cats.' }], }, expectedOutput: { role: 'user', parts: [{ text: 'You are an expert in all things cats.' }], }, }, { should: 'should transform from system to user with multiple parts', inputMessage: { role: 'system', content: [ { text: 'You are an expert in all things animals.' }, { text: 'You love cats.' }, ], }, expectedOutput: { role: 'user', parts: [ { text: 'You are an expert in all things animals.' }, { text: 'You love cats.' }, ], }, }, ]; for (const test of testCases) { it(test.should, () => { assert.deepEqual( toGeminiSystemInstruction(test.inputMessage as MessageData), test.expectedOutput ); }); } }); describe('fromGeminiCandidate', () => { const testCases = [ { should: 'should transform gemini candidate to genkit candidate (text parts) correctly', // had to delete the probabilityScore, severity, severityScore for the HARM_CATEGORY_SEXUALLY_EXPLICIT safety rating category geminiCandidate: { content: { role: 'model', parts: [ { text: 'Why did the dog go to the bank?\n\nTo get his bones cashed!', }, ], }, finishReason: 'STOP', safetyRatings: [ { category: 'HARM_CATEGORY_HATE_SPEECH', probability: 'NEGLIGIBLE', probabilityScore: 0.12074952, severity: 'HARM_SEVERITY_NEGLIGIBLE', severityScore: 0.18388656, }, { category: 'HARM_CATEGORY_DANGEROUS_CONTENT', probability: 'NEGLIGIBLE', probabilityScore: 0.37874627, severity: 'HARM_SEVERITY_LOW', severityScore: 0.37227696, }, { category: 'HARM_CATEGORY_HARASSMENT', probability: 'NEGLIGIBLE', probabilityScore: 0.3983479, severity: 'HARM_SEVERITY_LOW', severityScore: 0.22270013, }, { category: 'HARM_CATEGORY_SEXUALLY_EXPLICIT', probability: 'NEGLIGIBLE', }, ], }, expectedOutput: { index: 0, message: { role: 'model', content: [ { text: 'Why did the dog go to the bank?\n\nTo get his bones cashed!', }, ], }, finishReason: 'stop', finishMessage: undefined, custom: { citationMetadata: undefined, safetyRatings: [ { category: 'HARM_CATEGORY_HATE_SPEECH', probability: 'NEGLIGIBLE', probabilityScore: 0.12074952, severity: 'HARM_SEVERITY_NEGLIGIBLE', severityScore: 0.18388656, }, { category: 'HARM_CATEGORY_DANGEROUS_CONTENT', probability: 'NEGLIGIBLE', probabilityScore: 0.37874627, severity: 'HARM_SEVERITY_LOW', severityScore: 0.37227696, }, { category: 'HARM_CATEGORY_HARASSMENT', probability: 'NEGLIGIBLE', probabilityScore: 0.3983479, severity: 'HARM_SEVERITY_LOW', severityScore: 0.22270013, }, { category: 'HARM_CATEGORY_SEXUALLY_EXPLICIT', probability: 'NEGLIGIBLE', }, ], }, }, }, { should: 'should transform gemini candidate to genkit candidate (function call parts) correctly', geminiCandidate: { content: { role: 'model', parts: [ { functionCall: { name: 'tellAFunnyJoke', args: { topic: 'dog' } }, }, ], }, finishReason: 'STOP', safetyRatings: [ { category: 'HARM_CATEGORY_HATE_SPEECH', probability: 'NEGLIGIBLE', probabilityScore: 0.11858909, severity: 'HARM_SEVERITY_NEGLIGIBLE', severityScore: 0.11456649, }, { category: 'HARM_CATEGORY_DANGEROUS_CONTENT', probability: 'NEGLIGIBLE', probabilityScore: 0.13857833, severity: 'HARM_SEVERITY_NEGLIGIBLE', severityScore: 0.11417085, }, { category: 'HARM_CATEGORY_HARASSMENT', probability: 'NEGLIGIBLE', probabilityScore: 0.28012377, severity: 'HARM_SEVERITY_NEGLIGIBLE', severityScore: 0.112405084, }, { category: 'HARM_CATEGORY_SEXUALLY_EXPLICIT', probability: 'NEGLIGIBLE', }, ], }, expectedOutput: { index: 0, message: { role: 'model', content: [ { toolRequest: { name: 'tellAFunnyJoke', input: { topic: 'dog' } }, }, ], }, finishReason: 'stop', finishMessage: undefined, custom: { citationMetadata: undefined, safetyRatings: [ { category: 'HARM_CATEGORY_HATE_SPEECH', probability: 'NEGLIGIBLE', probabilityScore: 0.11858909, severity: 'HARM_SEVERITY_NEGLIGIBLE', severityScore: 0.11456649, }, { category: 'HARM_CATEGORY_DANGEROUS_CONTENT', probability: 'NEGLIGIBLE', probabilityScore: 0.13857833, severity: 'HARM_SEVERITY_NEGLIGIBLE', severityScore: 0.11417085, }, { category: 'HARM_CATEGORY_HARASSMENT', probability: 'NEGLIGIBLE', probabilityScore: 0.28012377, severity: 'HARM_SEVERITY_NEGLIGIBLE', severityScore: 0.112405084, }, { category: 'HARM_CATEGORY_SEXUALLY_EXPLICIT', probability: 'NEGLIGIBLE', }, ], }, }, }, ]; for (const test of testCases) { it(test.should, () => { assert.deepEqual( fromGeminiCandidate(test.geminiCandidate as GenerateContentCandidate), test.expectedOutput ); }); } }); describe('cleanSchema', () => { it('strips nulls from type', () => { const cleaned = cleanSchema({ type: 'object', properties: { title: { type: 'string', }, subtitle: { type: ['string', 'null'], }, }, required: ['title'], additionalProperties: true, $schema: 'http://json-schema.org/draft-07/schema#', }); assert.deepStrictEqual(cleaned, { type: 'object', properties: { title: { type: 'string', }, subtitle: { type: 'string', }, }, required: ['title'], }); }); }); describe('plugin', () => { it('should init the plugin without requiring the api key', async () => { const ai = genkit({ plugins: [googleAI()], }); assert.ok(ai); }); describe('plugin', () => { beforeEach(() => { process.env.GOOGLE_GENAI_API_KEY = 'testApiKey'; }); afterEach(() => { delete process.env.GOOGLE_GENAI_API_KEY; }); it('should pre-register a few flagship models', async () => { const ai = genkit({ plugins: [googleAI()], }); assert.ok(await ai.registry.lookupAction(`/model/${gemini15Flash.name}`)); assert.ok(await ai.registry.lookupAction(`/model/${gemini15Pro.name}`)); }); it('allow referencing models using `gemini` helper', async () => { const ai = genkit({ plugins: [googleAI()], }); const pro = await ai.registry.lookupAction( `/model/${gemini('gemini-1.5-pro').name}` ); assert.ok(pro); assert.strictEqual(pro.__action.name, 'googleai/gemini-1.5-pro'); const flash = await ai.registry.lookupAction( `/model/${gemini('gemini-1.5-flash').name}` ); assert.ok(flash); assert.strictEqual(flash.__action.name, 'googleai/gemini-1.5-flash'); }); it('references explicitly registered models', async () => { const flash002Ref = gemini('gemini-1.5-flash-002'); const ai = genkit({ plugins: [ googleAI({ models: ['gemini-1.5-pro-002', flash002Ref, 'gemini-4.0-banana'], }), ], }); const pro002Ref = gemini('gemini-1.5-pro-002'); assert.strictEqual(pro002Ref.name, 'googleai/gemini-1.5-pro-002'); assertEqualModelInfo( pro002Ref.info!, 'Google AI - gemini-1.5-pro-002', gemini15Pro.info! ); const pro002 = await ai.registry.lookupAction(`/model/${pro002Ref.name}`); assert.ok(pro002); assert.strictEqual(pro002.__action.name, 'googleai/gemini-1.5-pro-002'); assertEqualModelInfo( pro002.__action.metadata?.model, 'Google AI - gemini-1.5-pro-002', gemini15Pro.info! ); assert.strictEqual(flash002Ref.name, 'googleai/gemini-1.5-flash-002'); assertEqualModelInfo( flash002Ref.info!, 'Google AI - gemini-1.5-flash-002', gemini15Flash.info! ); const flash002 = await ai.registry.lookupAction( `/model/${flash002Ref.name}` ); assert.ok(flash002); assert.strictEqual( flash002.__action.name, 'googleai/gemini-1.5-flash-002' ); assertEqualModelInfo( flash002.__action.metadata?.model, 'Google AI - gemini-1.5-flash-002', gemini15Flash.info! ); const bananaRef = gemini('gemini-4.0-banana'); assert.strictEqual(bananaRef.name, 'googleai/gemini-4.0-banana'); assertEqualModelInfo( bananaRef.info!, 'Google AI - gemini-4.0-banana', GENERIC_GEMINI_MODEL.info! // <---- generic model fallback ); const banana = await ai.registry.lookupAction(`/model/${bananaRef.name}`); assert.ok(banana); assert.strictEqual(banana.__action.name, 'googleai/gemini-4.0-banana'); assertEqualModelInfo( banana.__action.metadata?.model, 'Google AI - gemini-4.0-banana', GENERIC_GEMINI_MODEL.info! // <---- generic model fallback ); // this one is not registered const flash003Ref = gemini('gemini-1.5-flash-003'); assert.strictEqual(flash003Ref.name, 'googleai/gemini-1.5-flash-003'); const flash003 = await ai.registry.lookupAction( `/model/${flash003Ref.name}` ); assert.ok(flash003 === undefined); }); }); }); describe('toGeminiTool', () => { it('', async () => { const got = toGeminiTool({ name: 'foo', description: 'tool foo', inputSchema: toJsonSchema({ schema: z.object({ simpleString: z.string().describe('a string').nullable(), simpleNumber: z.number().describe('a number'), simpleBoolean: z.boolean().describe('a boolean').optional(), simpleArray: z.array(z.string()).describe('an array').optional(), simpleEnum: z .enum(['choice_a', 'choice_b']) .describe('an enum') .optional(), }), }), }); const want = { description: 'tool foo', name: 'foo', parameters: { properties: { simpleArray: { description: 'an array', items: { type: 'string', }, type: 'array', }, simpleBoolean: { description: 'a boolean', type: 'boolean', }, simpleEnum: { description: 'an enum', enum: ['choice_a', 'choice_b'], type: 'string', }, simpleNumber: { description: 'a number', type: 'number', }, simpleString: { description: 'a string', nullable: true, type: 'string', }, }, required: ['simpleString', 'simpleNumber'], type: 'object', }, }; assert.deepStrictEqual(got, want); }); }); function assertEqualModelInfo( modelAction: ModelInfo, expectedLabel: string, expectedInfo: ModelInfo ) { assert.strictEqual(modelAction.label, expectedLabel); assert.deepStrictEqual(modelAction.supports, expectedInfo.supports); assert.deepStrictEqual(modelAction.versions, expectedInfo.versions); }