summaryrefslogtreecommitdiff
path: root/packages/cli/src/config/config.test.ts
diff options
context:
space:
mode:
authorJacob MacDonald <[email protected]>2025-08-07 14:19:06 -0700
committerGitHub <[email protected]>2025-08-07 21:19:06 +0000
commit19491b7b940912c2fb3fe24b2f189d3fd5668669 (patch)
tree6112b96d342a87bea824caa405e5786715a0ab2f /packages/cli/src/config/config.test.ts
parent53f8617b249c9f0443f5082a293a30504a118030 (diff)
avoid loading and initializing CLI config twice in non-interactive mode (#5793)
Diffstat (limited to 'packages/cli/src/config/config.test.ts')
-rw-r--r--packages/cli/src/config/config.test.ts142
1 files changed, 141 insertions, 1 deletions
diff --git a/packages/cli/src/config/config.test.ts b/packages/cli/src/config/config.test.ts
index 1d83ccbc..701ae267 100644
--- a/packages/cli/src/config/config.test.ts
+++ b/packages/cli/src/config/config.test.ts
@@ -8,6 +8,7 @@ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
import * as os from 'os';
import * as fs from 'fs';
import * as path from 'path';
+import { ShellTool, EditTool, WriteFileTool } from '@google/gemini-cli-core';
import { loadCliConfig, parseArguments } from './config.js';
import { Settings } from './settings.js';
import { Extension } from './extension.js';
@@ -561,6 +562,17 @@ describe('mergeMcpServers', () => {
});
describe('mergeExcludeTools', () => {
+ const defaultExcludes = [ShellTool.Name, EditTool.Name, WriteFileTool.Name];
+ const originalIsTTY = process.stdin.isTTY;
+
+ beforeEach(() => {
+ process.stdin.isTTY = true;
+ });
+
+ afterEach(() => {
+ process.stdin.isTTY = originalIsTTY;
+ });
+
it('should merge excludeTools from settings and extensions', async () => {
const settings: Settings = { excludeTools: ['tool1', 'tool2'] };
const extensions: Extension[] = [
@@ -655,7 +667,8 @@ describe('mergeExcludeTools', () => {
expect(config.getExcludeTools()).toHaveLength(4);
});
- it('should return an empty array when no excludeTools are specified', async () => {
+ it('should return an empty array when no excludeTools are specified and it is interactive', async () => {
+ process.stdin.isTTY = true;
const settings: Settings = {};
const extensions: Extension[] = [];
process.argv = ['node', 'script.js'];
@@ -669,6 +682,21 @@ describe('mergeExcludeTools', () => {
expect(config.getExcludeTools()).toEqual([]);
});
+ it('should return default excludes when no excludeTools are specified and it is not interactive', async () => {
+ process.stdin.isTTY = false;
+ const settings: Settings = {};
+ const extensions: Extension[] = [];
+ process.argv = ['node', 'script.js', '-p', 'test'];
+ const argv = await parseArguments();
+ const config = await loadCliConfig(
+ settings,
+ extensions,
+ 'test-session',
+ argv,
+ );
+ expect(config.getExcludeTools()).toEqual(defaultExcludes);
+ });
+
it('should handle settings with excludeTools but no extensions', async () => {
process.argv = ['node', 'script.js'];
const argv = await parseArguments();
@@ -1214,3 +1242,115 @@ describe('loadCliConfig chatCompression', () => {
expect(config.getChatCompression()).toBeUndefined();
});
});
+
+describe('loadCliConfig tool exclusions', () => {
+ const originalArgv = process.argv;
+ const originalEnv = { ...process.env };
+ const originalIsTTY = process.stdin.isTTY;
+
+ beforeEach(() => {
+ vi.resetAllMocks();
+ vi.mocked(os.homedir).mockReturnValue('/mock/home/user');
+ process.env.GEMINI_API_KEY = 'test-api-key';
+ process.stdin.isTTY = true;
+ });
+
+ afterEach(() => {
+ process.argv = originalArgv;
+ process.env = originalEnv;
+ process.stdin.isTTY = originalIsTTY;
+ vi.restoreAllMocks();
+ });
+
+ it('should not exclude interactive tools in interactive mode without YOLO', async () => {
+ process.stdin.isTTY = true;
+ process.argv = ['node', 'script.js'];
+ const argv = await parseArguments();
+ const config = await loadCliConfig({}, [], 'test-session', argv);
+ expect(config.getExcludeTools()).not.toContain('run_shell_command');
+ expect(config.getExcludeTools()).not.toContain('replace');
+ expect(config.getExcludeTools()).not.toContain('write_file');
+ });
+
+ it('should not exclude interactive tools in interactive mode with YOLO', async () => {
+ process.stdin.isTTY = true;
+ process.argv = ['node', 'script.js', '--yolo'];
+ const argv = await parseArguments();
+ const config = await loadCliConfig({}, [], 'test-session', argv);
+ expect(config.getExcludeTools()).not.toContain('run_shell_command');
+ expect(config.getExcludeTools()).not.toContain('replace');
+ expect(config.getExcludeTools()).not.toContain('write_file');
+ });
+
+ it('should exclude interactive tools in non-interactive mode without YOLO', async () => {
+ process.stdin.isTTY = false;
+ process.argv = ['node', 'script.js', '-p', 'test'];
+ const argv = await parseArguments();
+ const config = await loadCliConfig({}, [], 'test-session', argv);
+ expect(config.getExcludeTools()).toContain('run_shell_command');
+ expect(config.getExcludeTools()).toContain('replace');
+ expect(config.getExcludeTools()).toContain('write_file');
+ });
+
+ it('should not exclude interactive tools in non-interactive mode with YOLO', async () => {
+ process.stdin.isTTY = false;
+ process.argv = ['node', 'script.js', '-p', 'test', '--yolo'];
+ const argv = await parseArguments();
+ const config = await loadCliConfig({}, [], 'test-session', argv);
+ expect(config.getExcludeTools()).not.toContain('run_shell_command');
+ expect(config.getExcludeTools()).not.toContain('replace');
+ expect(config.getExcludeTools()).not.toContain('write_file');
+ });
+});
+
+describe('loadCliConfig interactive', () => {
+ const originalArgv = process.argv;
+ const originalEnv = { ...process.env };
+ const originalIsTTY = process.stdin.isTTY;
+
+ beforeEach(() => {
+ vi.resetAllMocks();
+ vi.mocked(os.homedir).mockReturnValue('/mock/home/user');
+ process.env.GEMINI_API_KEY = 'test-api-key';
+ process.stdin.isTTY = true;
+ });
+
+ afterEach(() => {
+ process.argv = originalArgv;
+ process.env = originalEnv;
+ process.stdin.isTTY = originalIsTTY;
+ vi.restoreAllMocks();
+ });
+
+ it('should be interactive if isTTY and no prompt', async () => {
+ process.stdin.isTTY = true;
+ process.argv = ['node', 'script.js'];
+ const argv = await parseArguments();
+ const config = await loadCliConfig({}, [], 'test-session', argv);
+ expect(config.isInteractive()).toBe(true);
+ });
+
+ it('should be interactive if prompt-interactive is set', async () => {
+ process.stdin.isTTY = false;
+ process.argv = ['node', 'script.js', '--prompt-interactive', 'test'];
+ const argv = await parseArguments();
+ const config = await loadCliConfig({}, [], 'test-session', argv);
+ expect(config.isInteractive()).toBe(true);
+ });
+
+ it('should not be interactive if not isTTY and no prompt', async () => {
+ process.stdin.isTTY = false;
+ process.argv = ['node', 'script.js'];
+ const argv = await parseArguments();
+ const config = await loadCliConfig({}, [], 'test-session', argv);
+ expect(config.isInteractive()).toBe(false);
+ });
+
+ it('should not be interactive if prompt is set', async () => {
+ process.stdin.isTTY = true;
+ process.argv = ['node', 'script.js', '--prompt', 'test'];
+ const argv = await parseArguments();
+ const config = await loadCliConfig({}, [], 'test-session', argv);
+ expect(config.isInteractive()).toBe(false);
+ });
+});