mirror of
https://github.com/QwenLM/qwen-code.git
synced 2025-12-19 09:33:53 +00:00
208 lines
5.7 KiB
TypeScript
208 lines
5.7 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/**
|
|
* This test verifies we can provide MCP tools with recursive input schemas
|
|
* (in JSON, using the $ref keyword) and both the GenAI SDK and the Gemini
|
|
* API calls succeed. Note that prior to
|
|
* https://github.com/googleapis/js-genai/commit/36f6350705ecafc47eaea3f3eecbcc69512edab7#diff-fdde9372aec859322b7c5a5efe467e0ad25a57210c7229724586ee90ea4f5a30
|
|
* the Gemini API call would fail for such tools because the schema was
|
|
* passed not as a JSON string but using the Gemini API's tool parameter
|
|
* schema object which has stricter typing and recursion restrictions.
|
|
* If this test fails, it's likely because either the GenAI SDK or Gemini API
|
|
* has become more restrictive about the type of tool parameter schemas that
|
|
* are accepted. If this occurs: Gemini CLI previously attempted to detect
|
|
* such tools and proactively remove them from the set of tools provided in
|
|
* the Gemini API call (as FunctionDeclaration objects). It may be appropriate
|
|
* to resurrect that behavior but note that it's difficult to keep the
|
|
* GCLI filters in sync with the Gemini API restrictions and behavior.
|
|
*/
|
|
|
|
import { writeFileSync } from 'node:fs';
|
|
import { join } from 'node:path';
|
|
import { beforeAll, describe, expect, it } from 'vitest';
|
|
import { TestRig } from './test-helper.js';
|
|
|
|
// Create a minimal MCP server that doesn't require external dependencies
|
|
// This implements the MCP protocol directly using Node.js built-ins
|
|
const serverScript = `#!/usr/bin/env node
|
|
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
const readline = require('readline');
|
|
const fs = require('fs');
|
|
|
|
// Debug logging to stderr (only when MCP_DEBUG or VERBOSE is set)
|
|
const debugEnabled = process.env['MCP_DEBUG'] === 'true' || process.env['VERBOSE'] === 'true';
|
|
function debug(msg) {
|
|
if (debugEnabled) {
|
|
fs.writeSync(2, \`[MCP-DEBUG] \${msg}\\n\`);
|
|
}
|
|
}
|
|
|
|
debug('MCP server starting...');
|
|
|
|
// Simple JSON-RPC implementation for MCP
|
|
class SimpleJSONRPC {
|
|
constructor() {
|
|
this.handlers = new Map();
|
|
this.rl = readline.createInterface({
|
|
input: process.stdin,
|
|
output: process.stdout,
|
|
terminal: false
|
|
});
|
|
|
|
this.rl.on('line', (line) => {
|
|
debug(\`Received line: \${line}\`);
|
|
try {
|
|
const message = JSON.parse(line);
|
|
debug(\`Parsed message: \${JSON.stringify(message)}\`);
|
|
this.handleMessage(message);
|
|
} catch (e) {
|
|
debug(\`Parse error: \${e.message}\`);
|
|
}
|
|
});
|
|
}
|
|
|
|
send(message) {
|
|
const msgStr = JSON.stringify(message);
|
|
debug(\`Sending message: \${msgStr}\`);
|
|
process.stdout.write(msgStr + '\\n');
|
|
}
|
|
|
|
async handleMessage(message) {
|
|
if (message.method && this.handlers.has(message.method)) {
|
|
try {
|
|
const result = await this.handlers.get(message.method)(message.params || {});
|
|
if (message.id !== undefined) {
|
|
this.send({
|
|
jsonrpc: '2.0',
|
|
id: message.id,
|
|
result
|
|
});
|
|
}
|
|
} catch (error) {
|
|
if (message.id !== undefined) {
|
|
this.send({
|
|
jsonrpc: '2.0',
|
|
id: message.id,
|
|
error: {
|
|
code: -32603,
|
|
message: error.message
|
|
}
|
|
});
|
|
}
|
|
}
|
|
} else if (message.id !== undefined) {
|
|
this.send({
|
|
jsonrpc: '2.0',
|
|
id: message.id,
|
|
error: {
|
|
code: -32601,
|
|
message: 'Method not found'
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
on(method, handler) {
|
|
this.handlers.set(method, handler);
|
|
}
|
|
}
|
|
|
|
// Create MCP server
|
|
const rpc = new SimpleJSONRPC();
|
|
|
|
// Handle initialize
|
|
rpc.on('initialize', async (params) => {
|
|
debug('Handling initialize request');
|
|
return {
|
|
protocolVersion: '2024-11-05',
|
|
capabilities: {
|
|
tools: {}
|
|
},
|
|
serverInfo: {
|
|
name: 'cyclic-schema-server',
|
|
version: '1.0.0'
|
|
}
|
|
};
|
|
});
|
|
|
|
// Handle tools/list
|
|
rpc.on('tools/list', async () => {
|
|
debug('Handling tools/list request');
|
|
return {
|
|
tools: [{
|
|
name: 'tool_with_cyclic_schema',
|
|
inputSchema: {
|
|
type: 'object',
|
|
properties: {
|
|
data: {
|
|
type: 'array',
|
|
items: {
|
|
type: 'object',
|
|
properties: {
|
|
child: { $ref: '#/properties/data/items' },
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
}]
|
|
};
|
|
});
|
|
|
|
// Send initialization notification
|
|
rpc.send({
|
|
jsonrpc: '2.0',
|
|
method: 'initialized'
|
|
});
|
|
`;
|
|
|
|
describe('mcp server with cyclic tool schema is detected', () => {
|
|
const rig = new TestRig();
|
|
|
|
beforeAll(async () => {
|
|
// Setup test directory with MCP server configuration
|
|
await rig.setup('cyclic-schema-mcp-server', {
|
|
settings: {
|
|
mcpServers: {
|
|
'cyclic-schema-server': {
|
|
command: 'node',
|
|
args: ['mcp-server.cjs'],
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
// Create server script in the test directory
|
|
const testServerPath = join(rig.testDir!, 'mcp-server.cjs');
|
|
writeFileSync(testServerPath, serverScript);
|
|
|
|
// Make the script executable (though running with 'node' should work anyway)
|
|
if (process.platform !== 'win32') {
|
|
const { chmodSync } = await import('node:fs');
|
|
chmodSync(testServerPath, 0o755);
|
|
}
|
|
});
|
|
|
|
it('mcp tool with cyclic schema should be accessible', async () => {
|
|
const mcp_list_output = await rig.runCommand(['mcp', 'list']);
|
|
|
|
// Verify the cyclic schema server is configured
|
|
expect(mcp_list_output).toContain('cyclic-schema-server');
|
|
});
|
|
|
|
it('gemini api call should be successful with cyclic mcp tool schema', async () => {
|
|
// Run any command and verify that we get a non-error response from
|
|
// the Gemini API.
|
|
await rig.run('hello');
|
|
});
|
|
});
|