mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-04-20 18:14:29 -07:00
Add initial /extensions explore implementation
This commit is contained in:
@@ -0,0 +1,147 @@
|
||||
/**
|
||||
* @license
|
||||
* Copyright 2026 Google LLC
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
import { renderWithProviders as render } from '../../../test-utils/render.js';
|
||||
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
||||
import { act } from 'react';
|
||||
import { ExtensionRegistryView } from './ExtensionRegistryView.js';
|
||||
import {
|
||||
ExtensionRegistryClient,
|
||||
type RegistryExtension,
|
||||
} from '../../../config/extensionRegistryClient.js';
|
||||
import { type ExtensionManager } from '../../../config/extension-manager.js';
|
||||
|
||||
vi.mock('../../config/extensionRegistryClient.js');
|
||||
|
||||
const mockExtensions = [
|
||||
{
|
||||
id: 'ext-1',
|
||||
extensionName: 'Extension 1',
|
||||
extensionDescription: 'Description 1',
|
||||
repoDescription: 'Repo Description 1',
|
||||
},
|
||||
{
|
||||
id: 'ext-2',
|
||||
extensionName: 'Extension 2',
|
||||
extensionDescription: 'Description 2',
|
||||
repoDescription: 'Repo Description 2',
|
||||
},
|
||||
];
|
||||
|
||||
describe('ExtensionRegistryView', () => {
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
});
|
||||
|
||||
it('should render loading state initially', async () => {
|
||||
const mockExtensionManager = {
|
||||
getExtensions: vi.fn().mockReturnValue([]),
|
||||
};
|
||||
|
||||
const { lastFrame } = render(
|
||||
<ExtensionRegistryView
|
||||
extensionManager={mockExtensionManager as unknown as ExtensionManager}
|
||||
/>,
|
||||
);
|
||||
expect(lastFrame()).toContain('Loading extensions...');
|
||||
});
|
||||
|
||||
it('should render extensions after fetching', async () => {
|
||||
vi.spyOn(
|
||||
ExtensionRegistryClient.prototype,
|
||||
'getExtensions',
|
||||
).mockResolvedValue({
|
||||
extensions: mockExtensions as unknown as RegistryExtension[],
|
||||
total: 2,
|
||||
});
|
||||
|
||||
const mockExtensionManager = {
|
||||
getExtensions: vi.fn().mockReturnValue([]),
|
||||
};
|
||||
|
||||
const { lastFrame } = render(
|
||||
<ExtensionRegistryView
|
||||
extensionManager={mockExtensionManager as unknown as ExtensionManager}
|
||||
/>,
|
||||
);
|
||||
|
||||
await act(async () => {
|
||||
await Promise.resolve();
|
||||
await Promise.resolve();
|
||||
});
|
||||
|
||||
const frame = lastFrame();
|
||||
expect(frame).toContain('Extension 1');
|
||||
expect(frame).toContain('Description 1');
|
||||
expect(frame).toContain('Extension 2');
|
||||
expect(frame).toContain('Description 2');
|
||||
});
|
||||
|
||||
it('should render error message on fetch failure', async () => {
|
||||
vi.spyOn(
|
||||
ExtensionRegistryClient.prototype,
|
||||
'getExtensions',
|
||||
).mockRejectedValue(new Error('Fetch failed'));
|
||||
|
||||
const mockExtensionManager = {
|
||||
getExtensions: vi.fn().mockReturnValue([]),
|
||||
};
|
||||
|
||||
const { lastFrame } = render(
|
||||
<ExtensionRegistryView
|
||||
extensionManager={mockExtensionManager as unknown as ExtensionManager}
|
||||
/>,
|
||||
);
|
||||
|
||||
await act(async () => {
|
||||
await Promise.resolve();
|
||||
await Promise.resolve();
|
||||
await Promise.resolve();
|
||||
});
|
||||
|
||||
const frame = lastFrame();
|
||||
expect(frame).toContain('Error loading extensions:');
|
||||
expect(frame).toContain('Fetch failed');
|
||||
});
|
||||
|
||||
it('should call onSelect when an item is selected', async () => {
|
||||
vi.spyOn(
|
||||
ExtensionRegistryClient.prototype,
|
||||
'getExtensions',
|
||||
).mockResolvedValue({
|
||||
extensions: mockExtensions as unknown as RegistryExtension[],
|
||||
total: 2,
|
||||
});
|
||||
const onSelect = vi.fn();
|
||||
|
||||
const mockExtensionManager = {
|
||||
getExtensions: vi.fn().mockReturnValue([]),
|
||||
};
|
||||
|
||||
const { stdin } = render(
|
||||
<ExtensionRegistryView
|
||||
onSelect={onSelect}
|
||||
extensionManager={mockExtensionManager as unknown as ExtensionManager}
|
||||
/>,
|
||||
);
|
||||
|
||||
await act(async () => {
|
||||
await Promise.resolve();
|
||||
await Promise.resolve();
|
||||
await Promise.resolve();
|
||||
});
|
||||
|
||||
// Press Enter to select the first item
|
||||
await act(async () => {
|
||||
stdin.write('\r');
|
||||
await Promise.resolve();
|
||||
await Promise.resolve();
|
||||
await Promise.resolve();
|
||||
});
|
||||
|
||||
expect(onSelect).toHaveBeenCalledWith(mockExtensions[0]);
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user