mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-05-12 12:54:07 -07:00
feat(extension) - Notify users when there is a new version and update it (#7408)
Co-authored-by: Shi Shu <shii@google.com> Co-authored-by: Shreya <shreyakeshive@google.com>
This commit is contained in:
@@ -50,12 +50,18 @@ vi.mock('vscode', () => ({
|
|||||||
fire: vi.fn(),
|
fire: vi.fn(),
|
||||||
dispose: vi.fn(),
|
dispose: vi.fn(),
|
||||||
})),
|
})),
|
||||||
|
extensions: {
|
||||||
|
getExtension: vi.fn(),
|
||||||
|
},
|
||||||
}));
|
}));
|
||||||
|
|
||||||
describe('activate', () => {
|
describe('activate', () => {
|
||||||
let context: vscode.ExtensionContext;
|
let context: vscode.ExtensionContext;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
vi.mocked(vscode.window.showInformationMessage).mockResolvedValue(
|
||||||
|
undefined,
|
||||||
|
);
|
||||||
context = {
|
context = {
|
||||||
subscriptions: [],
|
subscriptions: [],
|
||||||
environmentVariableCollection: {
|
environmentVariableCollection: {
|
||||||
@@ -68,6 +74,11 @@ describe('activate', () => {
|
|||||||
extensionUri: {
|
extensionUri: {
|
||||||
fsPath: '/path/to/extension',
|
fsPath: '/path/to/extension',
|
||||||
},
|
},
|
||||||
|
extension: {
|
||||||
|
packageJSON: {
|
||||||
|
version: '1.1.0',
|
||||||
|
},
|
||||||
|
},
|
||||||
} as unknown as vscode.ExtensionContext;
|
} as unknown as vscode.ExtensionContext;
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -80,6 +91,9 @@ describe('activate', () => {
|
|||||||
.mocked(vscode.window.showInformationMessage)
|
.mocked(vscode.window.showInformationMessage)
|
||||||
.mockResolvedValue(undefined as never);
|
.mockResolvedValue(undefined as never);
|
||||||
vi.mocked(context.globalState.get).mockReturnValue(undefined);
|
vi.mocked(context.globalState.get).mockReturnValue(undefined);
|
||||||
|
vi.mocked(vscode.extensions.getExtension).mockReturnValue({
|
||||||
|
packageJSON: { version: '1.1.0' },
|
||||||
|
} as vscode.Extension<unknown>);
|
||||||
await activate(context);
|
await activate(context);
|
||||||
expect(showInformationMessageMock).toHaveBeenCalledWith(
|
expect(showInformationMessageMock).toHaveBeenCalledWith(
|
||||||
'Gemini CLI Companion extension successfully installed.',
|
'Gemini CLI Companion extension successfully installed.',
|
||||||
@@ -88,6 +102,9 @@ describe('activate', () => {
|
|||||||
|
|
||||||
it('should not show the info message on subsequent activations', async () => {
|
it('should not show the info message on subsequent activations', async () => {
|
||||||
vi.mocked(context.globalState.get).mockReturnValue(true);
|
vi.mocked(context.globalState.get).mockReturnValue(true);
|
||||||
|
vi.mocked(vscode.extensions.getExtension).mockReturnValue({
|
||||||
|
packageJSON: { version: '1.1.0' },
|
||||||
|
} as vscode.Extension<unknown>);
|
||||||
await activate(context);
|
await activate(context);
|
||||||
expect(vscode.window.showInformationMessage).not.toHaveBeenCalled();
|
expect(vscode.window.showInformationMessage).not.toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
@@ -102,13 +119,143 @@ describe('activate', () => {
|
|||||||
.mocked(vscode.window.showInformationMessage)
|
.mocked(vscode.window.showInformationMessage)
|
||||||
.mockResolvedValue('Re-launch Gemini CLI' as never);
|
.mockResolvedValue('Re-launch Gemini CLI' as never);
|
||||||
vi.mocked(context.globalState.get).mockReturnValue(undefined);
|
vi.mocked(context.globalState.get).mockReturnValue(undefined);
|
||||||
|
vi.mocked(vscode.extensions.getExtension).mockReturnValue({
|
||||||
|
packageJSON: { version: '1.1.0' },
|
||||||
|
} as vscode.Extension<unknown>);
|
||||||
await activate(context);
|
await activate(context);
|
||||||
expect(showInformationMessageMock).toHaveBeenCalled();
|
expect(showInformationMessageMock).toHaveBeenCalledWith(
|
||||||
await new Promise(process.nextTick); // Wait for the promise to resolve
|
'Gemini CLI Companion extension successfully installed.',
|
||||||
const commandCallback = vi
|
);
|
||||||
.mocked(vscode.commands.registerCommand)
|
});
|
||||||
.mock.calls.find((call) => call[0] === 'gemini-cli.runGeminiCLI')?.[1];
|
|
||||||
|
|
||||||
expect(commandCallback).toBeDefined();
|
describe('update notification', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
// Prevent the "installed" message from showing
|
||||||
|
vi.mocked(context.globalState.get).mockReturnValue(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should show an update notification if a newer version is available', async () => {
|
||||||
|
vi.spyOn(global, 'fetch').mockResolvedValue({
|
||||||
|
ok: true,
|
||||||
|
json: async () => ({
|
||||||
|
results: [
|
||||||
|
{
|
||||||
|
extensions: [
|
||||||
|
{
|
||||||
|
versions: [{ version: '1.2.0' }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
const showInformationMessageMock = vi.mocked(
|
||||||
|
vscode.window.showInformationMessage,
|
||||||
|
);
|
||||||
|
|
||||||
|
await activate(context);
|
||||||
|
|
||||||
|
expect(showInformationMessageMock).toHaveBeenCalledWith(
|
||||||
|
'A new version (1.2.0) of the Gemini CLI Companion extension is available.',
|
||||||
|
'Update to latest version',
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not show an update notification if the version is the same', async () => {
|
||||||
|
vi.spyOn(global, 'fetch').mockResolvedValue({
|
||||||
|
ok: true,
|
||||||
|
json: async () => ({
|
||||||
|
results: [
|
||||||
|
{
|
||||||
|
extensions: [
|
||||||
|
{
|
||||||
|
versions: [{ version: '1.1.0' }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
const showInformationMessageMock = vi.mocked(
|
||||||
|
vscode.window.showInformationMessage,
|
||||||
|
);
|
||||||
|
|
||||||
|
await activate(context);
|
||||||
|
|
||||||
|
expect(showInformationMessageMock).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not show an update notification if the version is older', async () => {
|
||||||
|
vi.spyOn(global, 'fetch').mockResolvedValue({
|
||||||
|
ok: true,
|
||||||
|
json: async () => ({
|
||||||
|
results: [
|
||||||
|
{
|
||||||
|
extensions: [
|
||||||
|
{
|
||||||
|
versions: [{ version: '1.0.0' }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
const showInformationMessageMock = vi.mocked(
|
||||||
|
vscode.window.showInformationMessage,
|
||||||
|
);
|
||||||
|
|
||||||
|
await activate(context);
|
||||||
|
|
||||||
|
expect(showInformationMessageMock).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should execute the install command when the user clicks "Update"', async () => {
|
||||||
|
vi.spyOn(global, 'fetch').mockResolvedValue({
|
||||||
|
ok: true,
|
||||||
|
json: async () => ({
|
||||||
|
results: [
|
||||||
|
{
|
||||||
|
extensions: [
|
||||||
|
{
|
||||||
|
versions: [{ version: '1.2.0' }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
} as Response);
|
||||||
|
vi.mocked(vscode.window.showInformationMessage).mockResolvedValue(
|
||||||
|
'Update to latest version' as never,
|
||||||
|
);
|
||||||
|
const executeCommandMock = vi.mocked(vscode.commands.executeCommand);
|
||||||
|
|
||||||
|
await activate(context);
|
||||||
|
|
||||||
|
// Wait for the promise from showInformationMessage.then() to resolve
|
||||||
|
await new Promise(process.nextTick);
|
||||||
|
|
||||||
|
expect(executeCommandMock).toHaveBeenCalledWith(
|
||||||
|
'workbench.extensions.installExtension',
|
||||||
|
'Google.gemini-cli-vscode-ide-companion',
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle fetch errors gracefully', async () => {
|
||||||
|
vi.spyOn(global, 'fetch').mockResolvedValue({
|
||||||
|
ok: false,
|
||||||
|
statusText: 'Internal Server Error',
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
const showInformationMessageMock = vi.mocked(
|
||||||
|
vscode.window.showInformationMessage,
|
||||||
|
);
|
||||||
|
|
||||||
|
await activate(context);
|
||||||
|
|
||||||
|
expect(showInformationMessageMock).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -6,9 +6,11 @@
|
|||||||
|
|
||||||
import * as vscode from 'vscode';
|
import * as vscode from 'vscode';
|
||||||
import { IDEServer } from './ide-server.js';
|
import { IDEServer } from './ide-server.js';
|
||||||
|
import semver from 'semver';
|
||||||
import { DiffContentProvider, DiffManager } from './diff-manager.js';
|
import { DiffContentProvider, DiffManager } from './diff-manager.js';
|
||||||
import { createLogger } from './utils/logger.js';
|
import { createLogger } from './utils/logger.js';
|
||||||
|
|
||||||
|
const CLI_IDE_COMPANION_IDENTIFIER = 'Google.gemini-cli-vscode-ide-companion';
|
||||||
const INFO_MESSAGE_SHOWN_KEY = 'geminiCliInfoMessageShown';
|
const INFO_MESSAGE_SHOWN_KEY = 'geminiCliInfoMessageShown';
|
||||||
export const DIFF_SCHEME = 'gemini-diff';
|
export const DIFF_SCHEME = 'gemini-diff';
|
||||||
|
|
||||||
@@ -17,11 +19,79 @@ let logger: vscode.OutputChannel;
|
|||||||
|
|
||||||
let log: (message: string) => void = () => {};
|
let log: (message: string) => void = () => {};
|
||||||
|
|
||||||
|
async function checkForUpdates(
|
||||||
|
context: vscode.ExtensionContext,
|
||||||
|
log: (message: string) => void,
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
const currentVersion = context.extension.packageJSON.version;
|
||||||
|
|
||||||
|
// Fetch extension details from the VSCode Marketplace.
|
||||||
|
const response = await fetch(
|
||||||
|
'https://marketplace.visualstudio.com/_apis/public/gallery/extensionquery',
|
||||||
|
{
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
Accept: 'application/json;api-version=7.1-preview.1',
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
filters: [
|
||||||
|
{
|
||||||
|
criteria: [
|
||||||
|
{
|
||||||
|
filterType: 7, // Corresponds to ExtensionName
|
||||||
|
value: CLI_IDE_COMPANION_IDENTIFIER,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
// See: https://learn.microsoft.com/en-us/azure/devops/extend/gallery/apis/hyper-linking?view=azure-devops
|
||||||
|
// 946 = IncludeVersions | IncludeFiles | IncludeCategoryAndTags |
|
||||||
|
// IncludeShortDescription | IncludePublisher | IncludeStatistics
|
||||||
|
flags: 946,
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
log(
|
||||||
|
`Failed to fetch latest version info from marketplace: ${response.statusText}`,
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
const extension = data?.results?.[0]?.extensions?.[0];
|
||||||
|
// The versions are sorted by date, so the first one is the latest.
|
||||||
|
const latestVersion = extension?.versions?.[0]?.version;
|
||||||
|
|
||||||
|
if (latestVersion && semver.gt(latestVersion, currentVersion)) {
|
||||||
|
const selection = await vscode.window.showInformationMessage(
|
||||||
|
`A new version (${latestVersion}) of the Gemini CLI Companion extension is available.`,
|
||||||
|
'Update to latest version',
|
||||||
|
);
|
||||||
|
if (selection === 'Update to latest version') {
|
||||||
|
// The install command will update the extension if a newer version is found.
|
||||||
|
await vscode.commands.executeCommand(
|
||||||
|
'workbench.extensions.installExtension',
|
||||||
|
CLI_IDE_COMPANION_IDENTIFIER,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
const message = error instanceof Error ? error.message : String(error);
|
||||||
|
log(`Error checking for extension updates: ${message}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export async function activate(context: vscode.ExtensionContext) {
|
export async function activate(context: vscode.ExtensionContext) {
|
||||||
logger = vscode.window.createOutputChannel('Gemini CLI IDE Companion');
|
logger = vscode.window.createOutputChannel('Gemini CLI IDE Companion');
|
||||||
log = createLogger(context, logger);
|
log = createLogger(context, logger);
|
||||||
log('Extension activated');
|
log('Extension activated');
|
||||||
|
|
||||||
|
checkForUpdates(context, log);
|
||||||
|
|
||||||
const diffContentProvider = new DiffContentProvider();
|
const diffContentProvider = new DiffContentProvider();
|
||||||
const diffManager = new DiffManager(log, diffContentProvider);
|
const diffManager = new DiffManager(log, diffContentProvider);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user