mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-03-18 18:11:02 -07:00
This commit is contained in:
@@ -9,6 +9,10 @@ import os from 'node:os';
|
||||
import {
|
||||
isWindows10,
|
||||
isJetBrainsTerminal,
|
||||
isTmux,
|
||||
isGnuScreen,
|
||||
isLowColorTmux,
|
||||
isDumbTerminal,
|
||||
supports256Colors,
|
||||
supportsTrueColor,
|
||||
getCompatibilityWarnings,
|
||||
@@ -67,20 +71,104 @@ describe('compatibility', () => {
|
||||
});
|
||||
|
||||
describe('isJetBrainsTerminal', () => {
|
||||
it.each<{ env: string; expected: boolean; desc: string }>([
|
||||
beforeEach(() => {
|
||||
vi.stubEnv('TERMINAL_EMULATOR', '');
|
||||
vi.stubEnv('JETBRAINS_IDE', '');
|
||||
});
|
||||
it.each<{
|
||||
env: Record<string, string>;
|
||||
expected: boolean;
|
||||
desc: string;
|
||||
}>([
|
||||
{
|
||||
env: 'JetBrains-JediTerm',
|
||||
env: { TERMINAL_EMULATOR: 'JetBrains-JediTerm' },
|
||||
expected: true,
|
||||
desc: 'TERMINAL_EMULATOR is JetBrains-JediTerm',
|
||||
desc: 'TERMINAL_EMULATOR starts with JetBrains',
|
||||
},
|
||||
{ env: 'something-else', expected: false, desc: 'other terminals' },
|
||||
{ env: '', expected: false, desc: 'TERMINAL_EMULATOR is not set' },
|
||||
{
|
||||
env: { JETBRAINS_IDE: 'IntelliJ' },
|
||||
expected: true,
|
||||
desc: 'JETBRAINS_IDE is set',
|
||||
},
|
||||
{
|
||||
env: { TERMINAL_EMULATOR: 'xterm' },
|
||||
expected: false,
|
||||
desc: 'other terminals',
|
||||
},
|
||||
{ env: {}, expected: false, desc: 'no env vars set' },
|
||||
])('should return $expected when $desc', ({ env, expected }) => {
|
||||
vi.stubEnv('TERMINAL_EMULATOR', env);
|
||||
vi.stubEnv('TERMINAL_EMULATOR', '');
|
||||
vi.stubEnv('JETBRAINS_IDE', '');
|
||||
for (const [key, value] of Object.entries(env)) {
|
||||
vi.stubEnv(key, value);
|
||||
}
|
||||
expect(isJetBrainsTerminal()).toBe(expected);
|
||||
});
|
||||
});
|
||||
|
||||
describe('isTmux', () => {
|
||||
it('should return true when TMUX is set', () => {
|
||||
vi.stubEnv('TMUX', '/tmp/tmux-1001/default,1425,0');
|
||||
expect(isTmux()).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when TMUX is not set', () => {
|
||||
vi.stubEnv('TMUX', '');
|
||||
expect(isTmux()).toBe(false);
|
||||
});
|
||||
});
|
||||
|
||||
describe('isGnuScreen', () => {
|
||||
it('should return true when STY is set', () => {
|
||||
vi.stubEnv('STY', '1234.pts-0.host');
|
||||
expect(isGnuScreen()).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when STY is not set', () => {
|
||||
vi.stubEnv('STY', '');
|
||||
expect(isGnuScreen()).toBe(false);
|
||||
});
|
||||
});
|
||||
|
||||
describe('isLowColorTmux', () => {
|
||||
it('should return true when TERM=screen and COLORTERM is not set', () => {
|
||||
vi.stubEnv('TERM', 'screen');
|
||||
vi.stubEnv('TMUX', '1');
|
||||
vi.stubEnv('COLORTERM', '');
|
||||
expect(isLowColorTmux()).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when TERM=screen and COLORTERM is set', () => {
|
||||
vi.stubEnv('TERM', 'screen');
|
||||
vi.stubEnv('TMUX', '1');
|
||||
vi.stubEnv('COLORTERM', 'truecolor');
|
||||
expect(isLowColorTmux()).toBe(false);
|
||||
});
|
||||
|
||||
it('should return false when TERM=xterm-256color', () => {
|
||||
vi.stubEnv('TERM', 'xterm-256color');
|
||||
vi.stubEnv('COLORTERM', '');
|
||||
expect(isLowColorTmux()).toBe(false);
|
||||
});
|
||||
});
|
||||
|
||||
describe('isDumbTerminal', () => {
|
||||
it('should return true when TERM=dumb', () => {
|
||||
vi.stubEnv('TERM', 'dumb');
|
||||
expect(isDumbTerminal()).toBe(true);
|
||||
});
|
||||
|
||||
it('should return true when TERM=vt100', () => {
|
||||
vi.stubEnv('TERM', 'vt100');
|
||||
expect(isDumbTerminal()).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when TERM=xterm', () => {
|
||||
vi.stubEnv('TERM', 'xterm');
|
||||
expect(isDumbTerminal()).toBe(false);
|
||||
});
|
||||
});
|
||||
|
||||
describe('supports256Colors', () => {
|
||||
it.each<{
|
||||
depth: number;
|
||||
@@ -110,6 +198,8 @@ describe('compatibility', () => {
|
||||
process.stdout.getColorDepth = vi.fn().mockReturnValue(depth);
|
||||
if (term !== undefined) {
|
||||
vi.stubEnv('TERM', term);
|
||||
} else {
|
||||
vi.stubEnv('TERM', '');
|
||||
}
|
||||
expect(supports256Colors()).toBe(expected);
|
||||
});
|
||||
@@ -158,6 +248,14 @@ describe('compatibility', () => {
|
||||
|
||||
describe('getCompatibilityWarnings', () => {
|
||||
beforeEach(() => {
|
||||
// Clear out potential local environment variables that might trigger warnings
|
||||
vi.stubEnv('TERMINAL_EMULATOR', '');
|
||||
vi.stubEnv('JETBRAINS_IDE', '');
|
||||
vi.stubEnv('TMUX', '');
|
||||
vi.stubEnv('STY', '');
|
||||
vi.stubEnv('TERM', 'xterm-256color'); // Prevent dumb terminal warning
|
||||
vi.stubEnv('TERM_PROGRAM', '');
|
||||
|
||||
// Default to supporting true color to keep existing tests simple
|
||||
vi.stubEnv('COLORTERM', 'truecolor');
|
||||
process.stdout.getColorDepth = vi.fn().mockReturnValue(24);
|
||||
@@ -177,44 +275,71 @@ describe('compatibility', () => {
|
||||
);
|
||||
});
|
||||
|
||||
it.each<{
|
||||
platform: NodeJS.Platform;
|
||||
release: string;
|
||||
externalTerminal: string;
|
||||
desc: string;
|
||||
}>([
|
||||
{
|
||||
platform: 'darwin',
|
||||
release: '20.6.0',
|
||||
externalTerminal: 'iTerm2 or Ghostty',
|
||||
desc: 'macOS',
|
||||
},
|
||||
{
|
||||
platform: 'win32',
|
||||
release: '10.0.22000',
|
||||
externalTerminal: 'Windows Terminal',
|
||||
desc: 'Windows',
|
||||
}, // Valid Windows 11 release to not trigger the Windows 10 warning
|
||||
{
|
||||
platform: 'linux',
|
||||
release: '5.10.0',
|
||||
externalTerminal: 'Ghostty',
|
||||
desc: 'Linux',
|
||||
},
|
||||
])(
|
||||
'should return JetBrains warning when detected and in alternate buffer ($desc)',
|
||||
({ platform, release, externalTerminal }) => {
|
||||
vi.mocked(os.platform).mockReturnValue(platform);
|
||||
vi.mocked(os.release).mockReturnValue(release);
|
||||
vi.stubEnv('TERMINAL_EMULATOR', 'JetBrains-JediTerm');
|
||||
it('should return JetBrains warning when detected and in alternate buffer', () => {
|
||||
vi.mocked(os.platform).mockReturnValue('darwin');
|
||||
vi.stubEnv('TERMINAL_EMULATOR', 'JetBrains-JediTerm');
|
||||
|
||||
const warnings = getCompatibilityWarnings({ isAlternateBuffer: true });
|
||||
const warnings = getCompatibilityWarnings({ isAlternateBuffer: true });
|
||||
expect(warnings).toContainEqual(
|
||||
expect.objectContaining({
|
||||
id: 'jetbrains-terminal',
|
||||
message: expect.stringContaining('JetBrains terminal detected'),
|
||||
priority: WarningPriority.High,
|
||||
}),
|
||||
);
|
||||
});
|
||||
|
||||
it('should return tmux warning when detected and in alternate buffer', () => {
|
||||
vi.stubEnv('TMUX', '/tmp/tmux-1001/default,1,0');
|
||||
|
||||
const warnings = getCompatibilityWarnings({ isAlternateBuffer: true });
|
||||
expect(warnings).toContainEqual(
|
||||
expect.objectContaining({
|
||||
id: 'tmux-alternate-buffer',
|
||||
message: expect.stringContaining('tmux detected'),
|
||||
priority: WarningPriority.High,
|
||||
}),
|
||||
);
|
||||
});
|
||||
|
||||
it('should return low-color tmux warning when detected', () => {
|
||||
vi.stubEnv('TERM', 'screen');
|
||||
vi.stubEnv('TMUX', '1');
|
||||
vi.stubEnv('COLORTERM', '');
|
||||
|
||||
const warnings = getCompatibilityWarnings();
|
||||
expect(warnings).toContainEqual(
|
||||
expect.objectContaining({
|
||||
id: 'low-color-tmux',
|
||||
message: expect.stringContaining('Limited color support detected'),
|
||||
priority: WarningPriority.High,
|
||||
}),
|
||||
);
|
||||
});
|
||||
|
||||
it('should return GNU screen warning when detected', () => {
|
||||
vi.stubEnv('STY', '1234.pts-0.host');
|
||||
|
||||
const warnings = getCompatibilityWarnings();
|
||||
expect(warnings).toContainEqual(
|
||||
expect.objectContaining({
|
||||
id: 'gnu-screen',
|
||||
message: expect.stringContaining('GNU screen detected'),
|
||||
priority: WarningPriority.Low,
|
||||
}),
|
||||
);
|
||||
});
|
||||
|
||||
it.each(['dumb', 'vt100'])(
|
||||
'should return dumb terminal warning when TERM=%s',
|
||||
(term) => {
|
||||
vi.stubEnv('TERM', term);
|
||||
|
||||
const warnings = getCompatibilityWarnings();
|
||||
expect(warnings).toContainEqual(
|
||||
expect.objectContaining({
|
||||
id: 'jetbrains-terminal',
|
||||
message: expect.stringContaining(
|
||||
`Warning: JetBrains mouse scrolling is unreliable. Disabling alternate buffer mode in settings or using an external terminal (e.g., ${externalTerminal}) is recommended.`,
|
||||
),
|
||||
id: 'dumb-terminal',
|
||||
message: `Warning: Basic terminal detected (TERM=${term}). Visual rendering will be limited. For the best experience, use a terminal emulator with truecolor support.`,
|
||||
priority: WarningPriority.High,
|
||||
}),
|
||||
);
|
||||
|
||||
@@ -27,7 +27,40 @@ export function isWindows10(): boolean {
|
||||
* Detects if the current terminal is a JetBrains-based IDE terminal.
|
||||
*/
|
||||
export function isJetBrainsTerminal(): boolean {
|
||||
return process.env['TERMINAL_EMULATOR'] === 'JetBrains-JediTerm';
|
||||
const env = process.env;
|
||||
return !!(
|
||||
env['TERMINAL_EMULATOR']?.startsWith('JetBrains') || env['JETBRAINS_IDE']
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Detects if the current terminal is running inside tmux.
|
||||
*/
|
||||
export function isTmux(): boolean {
|
||||
return !!process.env['TMUX'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Detects if the current terminal is running inside GNU screen.
|
||||
*/
|
||||
export function isGnuScreen(): boolean {
|
||||
return !!process.env['STY'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Detects if the terminal is low-color mode (TERM=screen* with no COLORTERM).
|
||||
*/
|
||||
export function isLowColorTmux(): boolean {
|
||||
const term = process.env['TERM'] || '';
|
||||
return isTmux() && term.startsWith('screen') && !process.env['COLORTERM'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Detects if the terminal is a "dumb" terminal.
|
||||
*/
|
||||
export function isDumbTerminal(): boolean {
|
||||
const term = process.env['TERM'] || '';
|
||||
return term === 'dumb' || term === 'vt100';
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -104,17 +137,46 @@ export function getCompatibilityWarnings(options?: {
|
||||
}
|
||||
|
||||
if (isJetBrainsTerminal() && options?.isAlternateBuffer) {
|
||||
const platformTerminals: Partial<Record<NodeJS.Platform, string>> = {
|
||||
win32: 'Windows Terminal',
|
||||
darwin: 'iTerm2 or Ghostty',
|
||||
linux: 'Ghostty',
|
||||
};
|
||||
const suggestion = platformTerminals[os.platform()];
|
||||
const suggestedTerminals = suggestion ? ` (e.g., ${suggestion})` : '';
|
||||
|
||||
warnings.push({
|
||||
id: 'jetbrains-terminal',
|
||||
message: `Warning: JetBrains mouse scrolling is unreliable. Disabling alternate buffer mode in settings or using an external terminal${suggestedTerminals} is recommended.`,
|
||||
message:
|
||||
'Warning: JetBrains terminal detected — alternate buffer mode may cause scroll wheel issues and rendering artifacts. If you experience problems, disable it in /settings → "Use Alternate Screen Buffer".',
|
||||
priority: WarningPriority.High,
|
||||
});
|
||||
}
|
||||
|
||||
if (isTmux() && options?.isAlternateBuffer) {
|
||||
warnings.push({
|
||||
id: 'tmux-alternate-buffer',
|
||||
message:
|
||||
'Warning: tmux detected — alternate buffer mode may cause unexpected scrollback loss and flickering. If you experience issues, disable it in /settings → "Use Alternate Screen Buffer".\n Tip: Use Ctrl-b [ to access tmux copy mode for scrolling history.',
|
||||
priority: WarningPriority.High,
|
||||
});
|
||||
}
|
||||
|
||||
if (isLowColorTmux()) {
|
||||
warnings.push({
|
||||
id: 'low-color-tmux',
|
||||
message:
|
||||
'Warning: Limited color support detected (TERM=screen). Some visual elements may not render correctly. For better color support in tmux, add to ~/.tmux.conf:\n set -g default-terminal "tmux-256color"\n set -ga terminal-overrides ",*256col*:Tc"',
|
||||
priority: WarningPriority.High,
|
||||
});
|
||||
}
|
||||
|
||||
if (isGnuScreen()) {
|
||||
warnings.push({
|
||||
id: 'gnu-screen',
|
||||
message:
|
||||
'Warning: GNU screen detected. Some keyboard shortcuts and visual features may behave unexpectedly. For the best experience, consider using tmux or running Gemini CLI directly in your terminal.',
|
||||
priority: WarningPriority.Low,
|
||||
});
|
||||
}
|
||||
|
||||
if (isDumbTerminal()) {
|
||||
const term = process.env['TERM'] || 'dumb';
|
||||
warnings.push({
|
||||
id: 'dumb-terminal',
|
||||
message: `Warning: Basic terminal detected (TERM=${term}). Visual rendering will be limited. For the best experience, use a terminal emulator with truecolor support.`,
|
||||
priority: WarningPriority.High,
|
||||
});
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user