2025-06-30 20:03:16 -07:00
|
|
|
/**
|
|
|
|
|
* @license
|
|
|
|
|
* Copyright 2025 Google LLC
|
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
*/
|
|
|
|
|
|
2025-10-17 12:44:31 -07:00
|
|
|
import { vi, describe, it, expect, beforeEach, afterEach } from 'vitest';
|
2025-08-01 20:17:32 -07:00
|
|
|
import { checkForUpdates } from './updateCheck.js';
|
2025-10-17 12:44:31 -07:00
|
|
|
import type { LoadedSettings } from '../../config/settings.js';
|
2025-06-30 20:03:16 -07:00
|
|
|
|
|
|
|
|
const getPackageJson = vi.hoisted(() => vi.fn());
|
2025-10-29 13:23:35 -07:00
|
|
|
const debugLogger = vi.hoisted(() => ({
|
|
|
|
|
warn: vi.fn(),
|
|
|
|
|
}));
|
|
|
|
|
vi.mock('@google/gemini-cli-core', () => ({
|
2025-06-30 20:03:16 -07:00
|
|
|
getPackageJson,
|
2025-10-29 13:23:35 -07:00
|
|
|
debugLogger,
|
2025-06-30 20:03:16 -07:00
|
|
|
}));
|
|
|
|
|
|
2025-10-24 14:23:39 -07:00
|
|
|
const latestVersion = vi.hoisted(() => vi.fn());
|
|
|
|
|
vi.mock('latest-version', () => ({
|
|
|
|
|
default: latestVersion,
|
2025-06-30 20:03:16 -07:00
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
describe('checkForUpdates', () => {
|
2025-10-17 12:44:31 -07:00
|
|
|
let mockSettings: LoadedSettings;
|
|
|
|
|
|
2025-06-30 20:03:16 -07:00
|
|
|
beforeEach(() => {
|
2025-07-29 20:11:15 -04:00
|
|
|
vi.useFakeTimers();
|
2025-06-30 20:03:16 -07:00
|
|
|
vi.resetAllMocks();
|
2025-07-18 09:44:45 +09:00
|
|
|
// Clear DEV environment variable before each test
|
2025-08-17 12:43:21 -04:00
|
|
|
delete process.env['DEV'];
|
2025-10-17 12:44:31 -07:00
|
|
|
|
|
|
|
|
mockSettings = {
|
|
|
|
|
merged: {
|
|
|
|
|
general: {
|
|
|
|
|
disableUpdateNag: false,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
} as LoadedSettings;
|
2025-07-18 09:44:45 +09:00
|
|
|
});
|
|
|
|
|
|
2025-07-29 20:11:15 -04:00
|
|
|
afterEach(() => {
|
|
|
|
|
vi.useRealTimers();
|
|
|
|
|
vi.restoreAllMocks();
|
|
|
|
|
});
|
|
|
|
|
|
2025-10-17 12:44:31 -07:00
|
|
|
it('should return null if disableUpdateNag is true', async () => {
|
|
|
|
|
mockSettings.merged.general!.disableUpdateNag = true;
|
|
|
|
|
const result = await checkForUpdates(mockSettings);
|
|
|
|
|
expect(result).toBeNull();
|
|
|
|
|
expect(getPackageJson).not.toHaveBeenCalled();
|
2025-10-24 14:23:39 -07:00
|
|
|
expect(latestVersion).not.toHaveBeenCalled();
|
2025-10-17 12:44:31 -07:00
|
|
|
});
|
|
|
|
|
|
2025-07-18 09:44:45 +09:00
|
|
|
it('should return null when running from source (DEV=true)', async () => {
|
2025-08-17 12:43:21 -04:00
|
|
|
process.env['DEV'] = 'true';
|
2025-07-18 09:44:45 +09:00
|
|
|
getPackageJson.mockResolvedValue({
|
|
|
|
|
name: 'test-package',
|
|
|
|
|
version: '1.0.0',
|
|
|
|
|
});
|
2025-10-24 14:23:39 -07:00
|
|
|
latestVersion.mockResolvedValue('1.1.0');
|
2025-10-17 12:44:31 -07:00
|
|
|
const result = await checkForUpdates(mockSettings);
|
2025-07-18 09:44:45 +09:00
|
|
|
expect(result).toBeNull();
|
|
|
|
|
expect(getPackageJson).not.toHaveBeenCalled();
|
2025-10-24 14:23:39 -07:00
|
|
|
expect(latestVersion).not.toHaveBeenCalled();
|
2025-06-30 20:03:16 -07:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should return null if package.json is missing', async () => {
|
|
|
|
|
getPackageJson.mockResolvedValue(null);
|
2025-10-17 12:44:31 -07:00
|
|
|
const result = await checkForUpdates(mockSettings);
|
2025-06-30 20:03:16 -07:00
|
|
|
expect(result).toBeNull();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should return null if there is no update', async () => {
|
|
|
|
|
getPackageJson.mockResolvedValue({
|
|
|
|
|
name: 'test-package',
|
|
|
|
|
version: '1.0.0',
|
|
|
|
|
});
|
2025-10-24 14:23:39 -07:00
|
|
|
latestVersion.mockResolvedValue('1.0.0');
|
2025-10-17 12:44:31 -07:00
|
|
|
const result = await checkForUpdates(mockSettings);
|
2025-06-30 20:03:16 -07:00
|
|
|
expect(result).toBeNull();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should return a message if a newer version is available', async () => {
|
|
|
|
|
getPackageJson.mockResolvedValue({
|
|
|
|
|
name: 'test-package',
|
|
|
|
|
version: '1.0.0',
|
|
|
|
|
});
|
2025-10-24 14:23:39 -07:00
|
|
|
latestVersion.mockResolvedValue('1.1.0');
|
2025-07-28 17:56:52 -07:00
|
|
|
|
2025-10-17 12:44:31 -07:00
|
|
|
const result = await checkForUpdates(mockSettings);
|
2025-07-28 17:56:52 -07:00
|
|
|
expect(result?.message).toContain('1.0.0 → 1.1.0');
|
2025-10-24 14:23:39 -07:00
|
|
|
expect(result?.update.current).toEqual('1.0.0');
|
|
|
|
|
expect(result?.update.latest).toEqual('1.1.0');
|
|
|
|
|
expect(result?.update.name).toEqual('test-package');
|
2025-06-30 20:03:16 -07:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should return null if the latest version is the same as the current version', async () => {
|
|
|
|
|
getPackageJson.mockResolvedValue({
|
|
|
|
|
name: 'test-package',
|
|
|
|
|
version: '1.0.0',
|
|
|
|
|
});
|
2025-10-24 14:23:39 -07:00
|
|
|
latestVersion.mockResolvedValue('1.0.0');
|
2025-10-17 12:44:31 -07:00
|
|
|
const result = await checkForUpdates(mockSettings);
|
2025-06-30 20:03:16 -07:00
|
|
|
expect(result).toBeNull();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should return null if the latest version is older than the current version', async () => {
|
|
|
|
|
getPackageJson.mockResolvedValue({
|
|
|
|
|
name: 'test-package',
|
|
|
|
|
version: '1.1.0',
|
|
|
|
|
});
|
2025-10-24 14:23:39 -07:00
|
|
|
latestVersion.mockResolvedValue('1.0.0');
|
2025-10-17 12:44:31 -07:00
|
|
|
const result = await checkForUpdates(mockSettings);
|
2025-06-30 20:03:16 -07:00
|
|
|
expect(result).toBeNull();
|
|
|
|
|
});
|
|
|
|
|
|
2025-10-24 14:23:39 -07:00
|
|
|
it('should return null if latestVersion rejects', async () => {
|
2025-07-29 20:11:15 -04:00
|
|
|
getPackageJson.mockResolvedValue({
|
|
|
|
|
name: 'test-package',
|
|
|
|
|
version: '1.0.0',
|
|
|
|
|
});
|
2025-10-24 14:23:39 -07:00
|
|
|
latestVersion.mockRejectedValue(new Error('Timeout'));
|
2025-08-01 20:17:32 -07:00
|
|
|
|
2025-10-17 12:44:31 -07:00
|
|
|
const result = await checkForUpdates(mockSettings);
|
2025-07-29 20:11:15 -04:00
|
|
|
expect(result).toBeNull();
|
|
|
|
|
});
|
|
|
|
|
|
2025-06-30 20:03:16 -07:00
|
|
|
it('should handle errors gracefully', async () => {
|
|
|
|
|
getPackageJson.mockRejectedValue(new Error('test error'));
|
2025-10-17 12:44:31 -07:00
|
|
|
const result = await checkForUpdates(mockSettings);
|
2025-06-30 20:03:16 -07:00
|
|
|
expect(result).toBeNull();
|
|
|
|
|
});
|
2025-08-01 20:17:32 -07:00
|
|
|
|
|
|
|
|
describe('nightly updates', () => {
|
|
|
|
|
it('should notify for a newer nightly version when current is nightly', async () => {
|
|
|
|
|
getPackageJson.mockResolvedValue({
|
|
|
|
|
name: 'test-package',
|
|
|
|
|
version: '1.2.3-nightly.1',
|
|
|
|
|
});
|
|
|
|
|
|
2025-10-24 14:23:39 -07:00
|
|
|
latestVersion.mockImplementation(async (name, options) => {
|
|
|
|
|
if (options?.version === 'nightly') {
|
|
|
|
|
return '1.2.3-nightly.2';
|
2025-08-01 20:17:32 -07:00
|
|
|
}
|
2025-10-24 14:23:39 -07:00
|
|
|
return '1.2.3';
|
2025-08-01 20:17:32 -07:00
|
|
|
});
|
|
|
|
|
|
2025-10-17 12:44:31 -07:00
|
|
|
const result = await checkForUpdates(mockSettings);
|
2025-08-01 20:17:32 -07:00
|
|
|
expect(result?.message).toContain('1.2.3-nightly.1 → 1.2.3-nightly.2');
|
|
|
|
|
expect(result?.update.latest).toBe('1.2.3-nightly.2');
|
|
|
|
|
});
|
|
|
|
|
});
|
2025-06-30 20:03:16 -07:00
|
|
|
});
|