@tuannvm/gemini-mcp-server 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (99) hide show
  1. package/LICENSE +25 -0
  2. package/README.md +381 -0
  3. package/dist/constants.d.ts +58 -0
  4. package/dist/constants.d.ts.map +1 -0
  5. package/dist/constants.js +64 -0
  6. package/dist/constants.js.map +1 -0
  7. package/dist/errors.d.ts +24 -0
  8. package/dist/errors.d.ts.map +1 -0
  9. package/dist/errors.js +57 -0
  10. package/dist/errors.js.map +1 -0
  11. package/dist/index.d.ts +8 -0
  12. package/dist/index.d.ts.map +1 -0
  13. package/dist/index.js +24 -0
  14. package/dist/index.js.map +1 -0
  15. package/dist/server.d.ts +19 -0
  16. package/dist/server.d.ts.map +1 -0
  17. package/dist/server.js +166 -0
  18. package/dist/server.js.map +1 -0
  19. package/dist/tools/analyze-media.tool.d.ts +3 -0
  20. package/dist/tools/analyze-media.tool.d.ts.map +1 -0
  21. package/dist/tools/analyze-media.tool.js +102 -0
  22. package/dist/tools/analyze-media.tool.js.map +1 -0
  23. package/dist/tools/ask-gemini.tool.d.ts +3 -0
  24. package/dist/tools/ask-gemini.tool.d.ts.map +1 -0
  25. package/dist/tools/ask-gemini.tool.js +86 -0
  26. package/dist/tools/ask-gemini.tool.js.map +1 -0
  27. package/dist/tools/brainstorm.tool.d.ts +3 -0
  28. package/dist/tools/brainstorm.tool.d.ts.map +1 -0
  29. package/dist/tools/brainstorm.tool.js +220 -0
  30. package/dist/tools/brainstorm.tool.js.map +1 -0
  31. package/dist/tools/fetch-chunk.tool.d.ts +3 -0
  32. package/dist/tools/fetch-chunk.tool.d.ts.map +1 -0
  33. package/dist/tools/fetch-chunk.tool.js +85 -0
  34. package/dist/tools/fetch-chunk.tool.js.map +1 -0
  35. package/dist/tools/index.d.ts +2 -0
  36. package/dist/tools/index.d.ts.map +1 -0
  37. package/dist/tools/index.js +19 -0
  38. package/dist/tools/index.js.map +1 -0
  39. package/dist/tools/registry.d.ts +26 -0
  40. package/dist/tools/registry.d.ts.map +1 -0
  41. package/dist/tools/registry.js +74 -0
  42. package/dist/tools/registry.js.map +1 -0
  43. package/dist/tools/search.tool.d.ts +3 -0
  44. package/dist/tools/search.tool.d.ts.map +1 -0
  45. package/dist/tools/search.tool.js +86 -0
  46. package/dist/tools/search.tool.js.map +1 -0
  47. package/dist/tools/shell.tool.d.ts +3 -0
  48. package/dist/tools/shell.tool.d.ts.map +1 -0
  49. package/dist/tools/shell.tool.js +106 -0
  50. package/dist/tools/shell.tool.js.map +1 -0
  51. package/dist/tools/simple-tools.d.ts +4 -0
  52. package/dist/tools/simple-tools.d.ts.map +1 -0
  53. package/dist/tools/simple-tools.js +48 -0
  54. package/dist/tools/simple-tools.js.map +1 -0
  55. package/dist/tools/test-tool.example.d.ts +13 -0
  56. package/dist/tools/test-tool.example.d.ts.map +1 -0
  57. package/dist/tools/test-tool.example.js +44 -0
  58. package/dist/tools/test-tool.example.js.map +1 -0
  59. package/dist/tools/timeout-test.tool.d.ts +3 -0
  60. package/dist/tools/timeout-test.tool.d.ts.map +1 -0
  61. package/dist/tools/timeout-test.tool.js +45 -0
  62. package/dist/tools/timeout-test.tool.js.map +1 -0
  63. package/dist/types.d.ts +43 -0
  64. package/dist/types.d.ts.map +1 -0
  65. package/dist/types.js +22 -0
  66. package/dist/types.js.map +1 -0
  67. package/dist/utils/changeModeChunker.d.ts +11 -0
  68. package/dist/utils/changeModeChunker.d.ts.map +1 -0
  69. package/dist/utils/changeModeChunker.js +95 -0
  70. package/dist/utils/changeModeChunker.js.map +1 -0
  71. package/dist/utils/changeModeParser.d.ts +15 -0
  72. package/dist/utils/changeModeParser.d.ts.map +1 -0
  73. package/dist/utils/changeModeParser.js +67 -0
  74. package/dist/utils/changeModeParser.js.map +1 -0
  75. package/dist/utils/changeModeTranslator.d.ts +8 -0
  76. package/dist/utils/changeModeTranslator.d.ts.map +1 -0
  77. package/dist/utils/changeModeTranslator.js +73 -0
  78. package/dist/utils/changeModeTranslator.js.map +1 -0
  79. package/dist/utils/chunkCache.d.ts +22 -0
  80. package/dist/utils/chunkCache.d.ts.map +1 -0
  81. package/dist/utils/chunkCache.js +163 -0
  82. package/dist/utils/chunkCache.js.map +1 -0
  83. package/dist/utils/commandExecutor.d.ts +2 -0
  84. package/dist/utils/commandExecutor.d.ts.map +1 -0
  85. package/dist/utils/commandExecutor.js +74 -0
  86. package/dist/utils/commandExecutor.js.map +1 -0
  87. package/dist/utils/geminiExecutor.d.ts +3 -0
  88. package/dist/utils/geminiExecutor.d.ts.map +1 -0
  89. package/dist/utils/geminiExecutor.js +182 -0
  90. package/dist/utils/geminiExecutor.js.map +1 -0
  91. package/dist/utils/logger.d.ts +13 -0
  92. package/dist/utils/logger.d.ts.map +1 -0
  93. package/dist/utils/logger.js +42 -0
  94. package/dist/utils/logger.js.map +1 -0
  95. package/dist/utils/timeoutManager.d.ts +1 -0
  96. package/dist/utils/timeoutManager.d.ts.map +1 -0
  97. package/dist/utils/timeoutManager.js +2 -0
  98. package/dist/utils/timeoutManager.js.map +1 -0
  99. package/package.json +68 -0
@@ -0,0 +1,74 @@
1
+ import { spawn } from 'child_process';
2
+ import { Logger } from './logger.js';
3
+ export async function executeCommand(command, args, onProgress) {
4
+ return new Promise((resolve, reject) => {
5
+ const startTime = Date.now();
6
+ Logger.commandExecution(command, args, startTime);
7
+ const childProcess = spawn(command, args, {
8
+ env: process.env,
9
+ shell: false,
10
+ stdio: ['ignore', 'pipe', 'pipe'],
11
+ });
12
+ let stdout = '';
13
+ let stderr = '';
14
+ let isResolved = false;
15
+ let lastReportedLength = 0;
16
+ childProcess.stdout.on('data', (data) => {
17
+ stdout += data.toString();
18
+ // Report new content if callback provided
19
+ if (onProgress && stdout.length > lastReportedLength) {
20
+ const newContent = stdout.substring(lastReportedLength);
21
+ lastReportedLength = stdout.length;
22
+ onProgress(newContent);
23
+ }
24
+ });
25
+ // CLI level errors
26
+ childProcess.stderr.on('data', (data) => {
27
+ stderr += data.toString();
28
+ // find RESOURCE_EXHAUSTED when gemini-3-pro quota is exceeded
29
+ if (stderr.includes('RESOURCE_EXHAUSTED')) {
30
+ const modelMatch = stderr.match(/Quota exceeded for quota metric '([^']+)'/);
31
+ const statusMatch = stderr.match(/status["\s]*[:=]\s*(\d+)/);
32
+ const reasonMatch = stderr.match(/"reason":\s*"([^"]+)"/);
33
+ const model = modelMatch ? modelMatch[1] : 'Unknown Model';
34
+ const status = statusMatch ? statusMatch[1] : '429';
35
+ const reason = reasonMatch ? reasonMatch[1] : 'rateLimitExceeded';
36
+ const errorJson = {
37
+ error: {
38
+ code: parseInt(status),
39
+ message: `GMCPT: --> Quota exceeded for ${model}`,
40
+ details: {
41
+ model: model,
42
+ reason: reason,
43
+ statusText: 'Too Many Requests -- > try using gemini-3-flash-preview by asking',
44
+ },
45
+ },
46
+ };
47
+ Logger.error(`Gemini Quota Error: ${JSON.stringify(errorJson, null, 2)}`);
48
+ }
49
+ });
50
+ childProcess.on('error', (error) => {
51
+ if (!isResolved) {
52
+ isResolved = true;
53
+ Logger.error(`Process error:`, error);
54
+ reject(new Error(`Failed to spawn command: ${error.message}`));
55
+ }
56
+ });
57
+ childProcess.on('close', (code) => {
58
+ if (!isResolved) {
59
+ isResolved = true;
60
+ if (code === 0) {
61
+ Logger.commandComplete(startTime, code, stdout.length);
62
+ resolve(stdout.trim());
63
+ }
64
+ else {
65
+ Logger.commandComplete(startTime, code);
66
+ Logger.error(`Failed with exit code ${code}`);
67
+ const errorMessage = stderr.trim() || 'Unknown error';
68
+ reject(new Error(`Command failed with exit code ${code}: ${errorMessage}`));
69
+ }
70
+ }
71
+ });
72
+ });
73
+ }
74
+ //# sourceMappingURL=commandExecutor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"commandExecutor.js","sourceRoot":"","sources":["../../src/utils/commandExecutor.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ export declare function executeGeminiCLI(prompt: string, model?: string, sandbox?: boolean, changeMode?: boolean, onProgress?: (newOutput: string) => void): Promise<string>;
2
+ export declare function processChangeModeOutput(rawResult: string, chunkIndex?: number, chunkCacheKey?: string, prompt?: string): Promise<string>;
3
+ //# sourceMappingURL=geminiExecutor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"geminiExecutor.d.ts","sourceRoot":"","sources":["../../src/utils/geminiExecutor.ts"],"names":[],"mappings":"AAeA,wBAAsB,gBAAgB,CACpC,MAAM,EAAE,MAAM,EACd,KAAK,CAAC,EAAE,MAAM,EACd,OAAO,CAAC,EAAE,OAAO,EACjB,UAAU,CAAC,EAAE,OAAO,EACpB,UAAU,CAAC,EAAE,CAAC,SAAS,EAAE,MAAM,KAAK,IAAI,GACvC,OAAO,CAAC,MAAM,CAAC,CAoIjB;AAED,wBAAsB,uBAAuB,CAC3C,SAAS,EAAE,MAAM,EACjB,UAAU,CAAC,EAAE,MAAM,EACnB,aAAa,CAAC,EAAE,MAAM,EACtB,MAAM,CAAC,EAAE,MAAM,GACd,OAAO,CAAC,MAAM,CAAC,CAyEjB"}
@@ -0,0 +1,182 @@
1
+ import { executeCommand } from './commandExecutor.js';
2
+ import { Logger } from './logger.js';
3
+ import { ERROR_MESSAGES, STATUS_MESSAGES, MODELS, CLI } from '../constants.js';
4
+ import { parseChangeModeOutput, validateChangeModeEdits, } from './changeModeParser.js';
5
+ import { formatChangeModeResponse, summarizeChangeModeEdits, } from './changeModeTranslator.js';
6
+ import { chunkChangeModeEdits } from './changeModeChunker.js';
7
+ import { cacheChunks, getChunks } from './chunkCache.js';
8
+ export async function executeGeminiCLI(prompt, model, sandbox, changeMode, onProgress) {
9
+ let prompt_processed = prompt;
10
+ if (changeMode) {
11
+ prompt_processed = prompt.replace(/file:(\S+)/g, '@$1');
12
+ const changeModeInstructions = `
13
+ [CHANGEMODE INSTRUCTIONS]
14
+ You are generating code modifications that will be processed by an automated system. The output format is critical because it enables programmatic application of changes without human intervention.
15
+
16
+ INSTRUCTIONS:
17
+ 1. Analyze each provided file thoroughly
18
+ 2. Identify locations requiring changes based on the user request
19
+ 3. For each change, output in the exact format specified
20
+ 4. The OLD section must be EXACTLY what appears in the file (copy-paste exact match)
21
+ 5. Provide complete, directly replacing code blocks
22
+ 6. Verify line numbers are accurate
23
+
24
+ CRITICAL REQUIREMENTS:
25
+ 1. Output edits in the EXACT format specified below - no deviations
26
+ 2. The OLD string MUST be findable with Ctrl+F - it must be a unique, exact match
27
+ 3. Include enough surrounding lines to make the OLD string unique
28
+ 4. If a string appears multiple times (like </div>), include enough context lines above and below to make it unique
29
+ 5. Copy the OLD content EXACTLY as it appears - including all whitespace, indentation, line breaks
30
+ 6. Never use partial lines - always include complete lines from start to finish
31
+
32
+ OUTPUT FORMAT (follow exactly):
33
+ **FILE: [filename]:[line_number]**
34
+ \`\`\`
35
+ OLD:
36
+ [exact code to be replaced - must match file content precisely]
37
+ NEW:
38
+ [new code to insert - complete and functional]
39
+ \`\`\`
40
+
41
+ EXAMPLE 1 - Simple unique match:
42
+ **FILE: src/utils/helper.js:100**
43
+ \`\`\`
44
+ OLD:
45
+ function getMessage() {
46
+ return "Hello World";
47
+ }
48
+ NEW:
49
+ function getMessage() {
50
+ return "Hello Universe!";
51
+ }
52
+ \`\`\`
53
+
54
+ EXAMPLE 2 - Common tag needing context:
55
+ **FILE: index.html:245**
56
+ \`\`\`
57
+ OLD:
58
+ </div>
59
+ </div>
60
+ </section>
61
+ NEW:
62
+ </div>
63
+ </footer>
64
+ </section>
65
+ \`\`\`
66
+
67
+ IMPORTANT: The OLD section must be an EXACT copy from the file that can be found with Ctrl+F!
68
+
69
+ USER REQUEST:
70
+ ${prompt_processed}
71
+ `;
72
+ prompt_processed = changeModeInstructions;
73
+ }
74
+ const args = [];
75
+ if (model) {
76
+ args.push(CLI.FLAGS.MODEL, model);
77
+ }
78
+ if (sandbox) {
79
+ args.push(CLI.FLAGS.SANDBOX);
80
+ }
81
+ // Ensure @ symbols work cross-platform by wrapping in quotes if needed
82
+ const finalPrompt = prompt_processed.includes('@') && !prompt_processed.startsWith('"')
83
+ ? `"${prompt_processed}"`
84
+ : prompt_processed;
85
+ args.push(CLI.FLAGS.PROMPT, finalPrompt);
86
+ try {
87
+ return await executeCommand(CLI.COMMANDS.GEMINI, args, onProgress);
88
+ }
89
+ catch (error) {
90
+ const errorMessage = error instanceof Error ? error.message : String(error);
91
+ if (errorMessage.includes(ERROR_MESSAGES.QUOTA_EXCEEDED) &&
92
+ model !== MODELS.FLASH) {
93
+ Logger.warn(`${ERROR_MESSAGES.QUOTA_EXCEEDED}. Falling back to ${MODELS.FLASH}.`);
94
+ await sendStatusMessage(STATUS_MESSAGES.FLASH_RETRY);
95
+ const fallbackArgs = [];
96
+ fallbackArgs.push(CLI.FLAGS.MODEL, MODELS.FLASH);
97
+ if (sandbox) {
98
+ fallbackArgs.push(CLI.FLAGS.SANDBOX);
99
+ }
100
+ // Same @ symbol handling for fallback
101
+ const fallbackPrompt = prompt_processed.includes('@') && !prompt_processed.startsWith('"')
102
+ ? `"${prompt_processed}"`
103
+ : prompt_processed;
104
+ fallbackArgs.push(CLI.FLAGS.PROMPT, fallbackPrompt);
105
+ try {
106
+ const result = await executeCommand(CLI.COMMANDS.GEMINI, fallbackArgs, onProgress);
107
+ Logger.warn(`Successfully executed with ${MODELS.FLASH} fallback.`);
108
+ await sendStatusMessage(STATUS_MESSAGES.FLASH_SUCCESS);
109
+ return result;
110
+ }
111
+ catch (fallbackError) {
112
+ const fallbackErrorMessage = fallbackError instanceof Error
113
+ ? fallbackError.message
114
+ : String(fallbackError);
115
+ throw new Error(`${MODELS.PRO} quota exceeded, ${MODELS.FLASH} fallback also failed: ${fallbackErrorMessage}`);
116
+ }
117
+ }
118
+ else {
119
+ throw error;
120
+ }
121
+ }
122
+ }
123
+ export async function processChangeModeOutput(rawResult, chunkIndex, chunkCacheKey, prompt) {
124
+ // Check for cached chunks first
125
+ if (chunkIndex && chunkCacheKey) {
126
+ const cachedChunks = getChunks(chunkCacheKey);
127
+ if (cachedChunks && chunkIndex > 0 && chunkIndex <= cachedChunks.length) {
128
+ Logger.debug(`Using cached chunk ${chunkIndex} of ${cachedChunks.length}`);
129
+ const chunk = cachedChunks[chunkIndex - 1];
130
+ let result = formatChangeModeResponse(chunk.edits, {
131
+ current: chunkIndex,
132
+ total: cachedChunks.length,
133
+ cacheKey: chunkCacheKey,
134
+ });
135
+ // Add summary for first chunk only
136
+ if (chunkIndex === 1 && chunk.edits.length > 5) {
137
+ const allEdits = cachedChunks.flatMap((c) => c.edits);
138
+ result = summarizeChangeModeEdits(allEdits) + '\n\n' + result;
139
+ }
140
+ return result;
141
+ }
142
+ Logger.debug(`Cache miss or invalid chunk index, processing new result`);
143
+ }
144
+ // Parse OLD/NEW format
145
+ const edits = parseChangeModeOutput(rawResult);
146
+ if (edits.length === 0) {
147
+ return `No edits found in Gemini's response. Please ensure Gemini uses the OLD/NEW format. \n\n+ ${rawResult}`;
148
+ }
149
+ // Validate edits
150
+ const validation = validateChangeModeEdits(edits);
151
+ if (!validation.valid) {
152
+ return `Edit validation failed:\n${validation.errors.join('\n')}`;
153
+ }
154
+ const chunks = chunkChangeModeEdits(edits);
155
+ // Cache if multiple chunks and we have the original prompt
156
+ let cacheKey;
157
+ if (chunks.length > 1 && prompt) {
158
+ cacheKey = cacheChunks(prompt, chunks);
159
+ Logger.debug(`Cached ${chunks.length} chunks with key: ${cacheKey}`);
160
+ }
161
+ // Return requested chunk or first chunk
162
+ const returnChunkIndex = chunkIndex && chunkIndex > 0 && chunkIndex <= chunks.length
163
+ ? chunkIndex
164
+ : 1;
165
+ const returnChunk = chunks[returnChunkIndex - 1];
166
+ // Format the response
167
+ let result = formatChangeModeResponse(returnChunk.edits, chunks.length > 1
168
+ ? { current: returnChunkIndex, total: chunks.length, cacheKey }
169
+ : undefined);
170
+ // Add summary if helpful (only for first chunk)
171
+ if (returnChunkIndex === 1 && edits.length > 5) {
172
+ result =
173
+ summarizeChangeModeEdits(edits, chunks.length > 1) + '\n\n' + result;
174
+ }
175
+ Logger.debug(`ChangeMode: Parsed ${edits.length} edits, ${chunks.length} chunks, returning chunk ${returnChunkIndex}`);
176
+ return result;
177
+ }
178
+ // Placeholder
179
+ async function sendStatusMessage(message) {
180
+ Logger.debug(`Status: ${message}`);
181
+ }
182
+ //# sourceMappingURL=geminiExecutor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"geminiExecutor.js","sourceRoot":"","sources":["../../src/utils/geminiExecutor.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ export declare class Logger {
2
+ private static formatMessage;
3
+ static log(message: string, ...args: any[]): void;
4
+ static warn(message: string, ...args: any[]): void;
5
+ static error(message: string, ...args: any[]): void;
6
+ static debug(message: string, ...args: any[]): void;
7
+ static toolInvocation(toolName: string, args: any): void;
8
+ static toolParsedArgs(prompt: string, model?: string, sandbox?: boolean, changeMode?: boolean): void;
9
+ static commandExecution(command: string, args: string[], startTime: number): void;
10
+ private static _commandStartTimes;
11
+ static commandComplete(startTime: number, exitCode: number | null, outputLength?: number): void;
12
+ }
13
+ //# sourceMappingURL=logger.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../../src/utils/logger.ts"],"names":[],"mappings":"AAGA,qBAAa,MAAM;IACjB,OAAO,CAAC,MAAM,CAAC,aAAa;IAI5B,MAAM,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE,GAAG,IAAI;IAIjD,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE,GAAG,IAAI;IAIlD,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE,GAAG,IAAI;IAInD,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE,GAAG,IAAI;IAInD,MAAM,CAAC,cAAc,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAE,GAAG,GAAG,IAAI;IAIxD,MAAM,CAAC,cAAc,CACnB,MAAM,EAAE,MAAM,EACd,KAAK,CAAC,EAAE,MAAM,EACd,OAAO,CAAC,EAAE,OAAO,EACjB,UAAU,CAAC,EAAE,OAAO,GACnB,IAAI;IAIP,MAAM,CAAC,gBAAgB,CACrB,OAAO,EAAE,MAAM,EACf,IAAI,EAAE,MAAM,EAAE,EACd,SAAS,EAAE,MAAM,GAChB,IAAI;IAUP,OAAO,CAAC,MAAM,CAAC,kBAAkB,CAG7B;IAEJ,MAAM,CAAC,eAAe,CACpB,SAAS,EAAE,MAAM,EACjB,QAAQ,EAAE,MAAM,GAAG,IAAI,EACvB,YAAY,CAAC,EAAE,MAAM,GACpB,IAAI;CAUR"}
@@ -0,0 +1,42 @@
1
+ // WIP
2
+ import { LOG_PREFIX } from '../constants.js';
3
+ export class Logger {
4
+ static formatMessage(message) {
5
+ return `${LOG_PREFIX} ${message}` + '\n';
6
+ }
7
+ static log(message, ...args) {
8
+ console.warn(this.formatMessage(message), ...args);
9
+ }
10
+ static warn(message, ...args) {
11
+ console.warn(this.formatMessage(message), ...args);
12
+ }
13
+ static error(message, ...args) {
14
+ console.error(this.formatMessage(message), ...args);
15
+ }
16
+ static debug(message, ...args) {
17
+ console.warn(this.formatMessage(message), ...args);
18
+ }
19
+ static toolInvocation(toolName, args) {
20
+ this.warn('Raw:', JSON.stringify(args, null, 2));
21
+ }
22
+ static toolParsedArgs(prompt, model, sandbox, changeMode) {
23
+ this.warn(`Parsed prompt: "${prompt}"\nchangeMode: ${changeMode || false}`);
24
+ }
25
+ static commandExecution(command, args, startTime) {
26
+ this.warn(`[${startTime}] Starting: ${command} ${args.map((arg) => `"${arg}"`).join(' ')}`);
27
+ // Store command execution start for timing analysis
28
+ this._commandStartTimes.set(startTime, { command, args, startTime });
29
+ }
30
+ // Track command start times for duration calculation
31
+ static _commandStartTimes = new Map();
32
+ static commandComplete(startTime, exitCode, outputLength) {
33
+ const elapsed = ((Date.now() - startTime) / 1000).toFixed(1);
34
+ this.warn(`[${elapsed}s] Process finished with exit code: ${exitCode}`);
35
+ if (outputLength !== undefined) {
36
+ this.warn(`Response: ${outputLength} chars`);
37
+ }
38
+ // Clean up command tracking
39
+ this._commandStartTimes.delete(startTime);
40
+ }
41
+ }
42
+ //# sourceMappingURL=logger.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/utils/logger.ts"],"names":[],"mappings":"AAAA,MAAM;AACN,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAE7C,MAAM,OAAO,MAAM;IACT,MAAM,CAAC,aAAa,CAAC,OAAe;QAC1C,OAAO,GAAG,UAAU,IAAI,OAAO,EAAE,GAAG,IAAI,CAAC;IAC3C,CAAC;IAED,MAAM,CAAC,GAAG,CAAC,OAAe,EAAE,GAAG,IAAW;QACxC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE,GAAG,IAAI,CAAC,CAAC;IACrD,CAAC;IAED,MAAM,CAAC,IAAI,CAAC,OAAe,EAAE,GAAG,IAAW;QACzC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE,GAAG,IAAI,CAAC,CAAC;IACrD,CAAC;IAED,MAAM,CAAC,KAAK,CAAC,OAAe,EAAE,GAAG,IAAW;QAC1C,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE,GAAG,IAAI,CAAC,CAAC;IACtD,CAAC;IAED,MAAM,CAAC,KAAK,CAAC,OAAe,EAAE,GAAG,IAAW;QAC1C,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE,GAAG,IAAI,CAAC,CAAC;IACrD,CAAC;IAED,MAAM,CAAC,cAAc,CAAC,QAAgB,EAAE,IAAS;QAC/C,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IACnD,CAAC;IAED,MAAM,CAAC,cAAc,CACnB,MAAc,EACd,KAAc,EACd,OAAiB,EACjB,UAAoB;QAEpB,IAAI,CAAC,IAAI,CAAC,mBAAmB,MAAM,kBAAkB,UAAU,IAAI,KAAK,EAAE,CAAC,CAAC;IAC9E,CAAC;IAED,MAAM,CAAC,gBAAgB,CACrB,OAAe,EACf,IAAc,EACd,SAAiB;QAEjB,IAAI,CAAC,IAAI,CACP,IAAI,SAAS,eAAe,OAAO,IAAI,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,GAAG,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CACjF,CAAC;QAEF,oDAAoD;QACpD,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,SAAS,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC,CAAC;IACvE,CAAC;IAED,qDAAqD;IAC7C,MAAM,CAAC,kBAAkB,GAAG,IAAI,GAAG,EAGxC,CAAC;IAEJ,MAAM,CAAC,eAAe,CACpB,SAAiB,EACjB,QAAuB,EACvB,YAAqB;QAErB,MAAM,OAAO,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;QAC7D,IAAI,CAAC,IAAI,CAAC,IAAI,OAAO,uCAAuC,QAAQ,EAAE,CAAC,CAAC;QACxE,IAAI,YAAY,KAAK,SAAS,EAAE,CAAC;YAC/B,IAAI,CAAC,IAAI,CAAC,aAAa,YAAY,QAAQ,CAAC,CAAC;QAC/C,CAAC;QAED,4BAA4B;QAC5B,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IAC5C,CAAC"}
@@ -0,0 +1 @@
1
+ //# sourceMappingURL=timeoutManager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timeoutManager.d.ts","sourceRoot":"","sources":["../../src/utils/timeoutManager.ts"],"names":[],"mappings":""}
@@ -0,0 +1,2 @@
1
+ "use strict";
2
+ //# sourceMappingURL=timeoutManager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timeoutManager.js","sourceRoot":"","sources":["../../src/utils/timeoutManager.ts"],"names":[],"mappings":""}
package/package.json ADDED
@@ -0,0 +1,68 @@
1
+ {
2
+ "name": "@tuannvm/gemini-mcp-server",
3
+ "version": "1.0.0",
4
+ "description": "MCP server for Gemini CLI integration",
5
+ "type": "module",
6
+ "main": "dist/index.js",
7
+ "bin": {
8
+ "gemini-mcp": "dist/index.js"
9
+ },
10
+ "files": [
11
+ "dist"
12
+ ],
13
+ "scripts": {
14
+ "build": "tsc",
15
+ "start": "node dist/index.js",
16
+ "dev": "tsx src/index.ts",
17
+ "test": "jest",
18
+ "test:watch": "jest --watch",
19
+ "test:coverage": "jest --coverage",
20
+ "lint": "eslint 'src/**/*.ts' --ignore-pattern '**/__tests__/**' --ignore-pattern '**/*.test.ts'",
21
+ "lint:fix": "eslint 'src/**/*.ts' --ignore-pattern '**/__tests__/**' --ignore-pattern '**/*.test.ts' --fix",
22
+ "format": "prettier --write src/**/*.ts",
23
+ "format:check": "prettier --check src/**/*.ts",
24
+ "clean": "rm -rf dist",
25
+ "prepublishOnly": "npm run build"
26
+ },
27
+ "keywords": [
28
+ "mcp",
29
+ "modelcontextprotocol",
30
+ "gemini",
31
+ "cli",
32
+ "ai",
33
+ "llm"
34
+ ],
35
+ "author": "tuannvm",
36
+ "license": "MIT",
37
+ "repository": {
38
+ "type": "git",
39
+ "url": "git+https://github.com/tuannvm/gemini-mcp-server.git"
40
+ },
41
+ "bugs": {
42
+ "url": "https://github.com/tuannvm/gemini-mcp-server/issues"
43
+ },
44
+ "homepage": "https://github.com/tuannvm/gemini-mcp-server#readme",
45
+ "engines": {
46
+ "node": ">=18.0.0"
47
+ },
48
+ "dependencies": {
49
+ "@modelcontextprotocol/sdk": "^1.24.0",
50
+ "chalk": "^5.6.0",
51
+ "zod": "^4.2.1"
52
+ },
53
+ "devDependencies": {
54
+ "@types/jest": "^30.0.0",
55
+ "@types/node": "^24.3.0",
56
+ "@typescript-eslint/eslint-plugin": "^8.39.1",
57
+ "@typescript-eslint/parser": "^8.39.1",
58
+ "eslint": "^9.33.0",
59
+ "jest": "^30.0.5",
60
+ "prettier": "^3.6.2",
61
+ "ts-jest": "^29.4.1",
62
+ "tsx": "^4.20.4",
63
+ "typescript": "^5.9.2"
64
+ },
65
+ "publishConfig": {
66
+ "access": "public"
67
+ }
68
+ }