agent-ecosystem/test/shared/utils/extensionNormalizers.test.ts
iliya d3f19834ff feat: enhance extension installation and uninstallation flows
- Refactored plugin and MCP server installation/uninstallation handlers to use async/await for improved readability and error handling.
- Added cache invalidation logic upon successful installation of plugins and MCP servers to ensure up-to-date state.
- Introduced error message handling for installation failures, providing users with clearer feedback on issues encountered during the process.
- Updated UI components to display error messages related to installation and uninstallation, enhancing user experience.
- Implemented sanitization for MCP server names to ensure compliance with CLI requirements.
2026-03-08 01:44:58 +02:00

169 lines
4.9 KiB
TypeScript

import { describe, expect, it } from 'vitest';
import type { PluginCatalogItem } from '@shared/types/extensions';
import {
buildPluginId,
formatInstallCount,
getCapabilityLabel,
getPrimaryCapabilityLabel,
inferCapabilities,
normalizeCategory,
normalizeRepoUrl,
sanitizeMcpServerName,
} from '@shared/utils/extensionNormalizers';
describe('normalizeRepoUrl', () => {
it('lowercases and strips .git', () => {
expect(normalizeRepoUrl('https://GitHub.com/Org/Repo.git')).toBe(
'https://github.com/org/repo',
);
});
it('strips trailing slashes', () => {
expect(normalizeRepoUrl('https://github.com/org/repo/')).toBe(
'https://github.com/org/repo',
);
});
it('handles already clean URLs', () => {
expect(normalizeRepoUrl('https://github.com/org/repo')).toBe(
'https://github.com/org/repo',
);
});
});
describe('inferCapabilities', () => {
const makePlugin = (overrides: Partial<PluginCatalogItem>): PluginCatalogItem => ({
pluginId: 'test@marketplace',
marketplaceId: 'test@marketplace',
qualifiedName: 'test@marketplace',
name: 'test',
description: 'test',
category: 'development',
hasLspServers: false,
hasMcpServers: false,
hasAgents: false,
hasCommands: false,
hasHooks: false,
isExternal: false,
...overrides,
});
it('returns "skill" fallback when no capabilities', () => {
expect(inferCapabilities(makePlugin({}))).toEqual(['skill']);
});
it('detects LSP capability', () => {
expect(inferCapabilities(makePlugin({ hasLspServers: true }))).toEqual(['lsp']);
});
it('detects multiple capabilities', () => {
expect(
inferCapabilities(makePlugin({ hasLspServers: true, hasMcpServers: true })),
).toEqual(['lsp', 'mcp']);
});
it('preserves capability order', () => {
expect(
inferCapabilities(
makePlugin({
hasHooks: true,
hasAgents: true,
hasLspServers: true,
}),
),
).toEqual(['lsp', 'agent', 'hook']);
});
});
describe('getPrimaryCapabilityLabel', () => {
it('returns "Skill" for empty array', () => {
expect(getPrimaryCapabilityLabel([])).toBe('Skill');
});
it('returns label for first capability', () => {
expect(getPrimaryCapabilityLabel(['lsp', 'mcp'])).toBe('LSP');
});
});
describe('getCapabilityLabel', () => {
it('maps all capabilities', () => {
expect(getCapabilityLabel('lsp')).toBe('LSP');
expect(getCapabilityLabel('mcp')).toBe('MCP');
expect(getCapabilityLabel('agent')).toBe('Agent');
expect(getCapabilityLabel('command')).toBe('Command');
expect(getCapabilityLabel('hook')).toBe('Hook');
expect(getCapabilityLabel('skill')).toBe('Skill');
});
});
describe('formatInstallCount', () => {
it('formats small numbers as-is', () => {
expect(formatInstallCount(0)).toBe('0');
expect(formatInstallCount(42)).toBe('42');
expect(formatInstallCount(999)).toBe('999');
});
it('formats thousands with K suffix', () => {
expect(formatInstallCount(1_000)).toBe('1K');
expect(formatInstallCount(1_500)).toBe('1.5K');
expect(formatInstallCount(10_000)).toBe('10K');
expect(formatInstallCount(277_472)).toBe('277K');
});
it('formats millions with M suffix', () => {
expect(formatInstallCount(1_000_000)).toBe('1M');
expect(formatInstallCount(1_200_000)).toBe('1.2M');
expect(formatInstallCount(15_000_000)).toBe('15M');
});
it('removes trailing .0 in formatted numbers', () => {
expect(formatInstallCount(5_000)).toBe('5K');
expect(formatInstallCount(2_000_000)).toBe('2M');
});
});
describe('normalizeCategory', () => {
it('lowercases and trims', () => {
expect(normalizeCategory(' Development ')).toBe('development');
});
it('returns "other" for undefined', () => {
expect(normalizeCategory(undefined)).toBe('other');
});
it('returns "other" for empty string', () => {
expect(normalizeCategory('')).toBe('other');
expect(normalizeCategory(' ')).toBe('other');
});
});
describe('buildPluginId', () => {
it('creates qualifiedName format', () => {
expect(buildPluginId('context7', 'claude-plugins-official')).toBe(
'context7@claude-plugins-official',
);
});
});
describe('sanitizeMcpServerName', () => {
it('lowercases and replaces spaces with dashes', () => {
expect(sanitizeMcpServerName('My Server')).toBe('my-server');
});
it('strips special characters', () => {
expect(sanitizeMcpServerName('Stripe (Beta)')).toBe('stripe-beta');
expect(sanitizeMcpServerName('MCP/Server')).toBe('mcpserver');
expect(sanitizeMcpServerName("O'Reilly")).toBe('oreilly');
});
it('keeps dots, underscores, and dashes', () => {
expect(sanitizeMcpServerName('v2.0-server_name')).toBe('v2.0-server_name');
});
it('handles simple names', () => {
expect(sanitizeMcpServerName('Alpic')).toBe('alpic');
expect(sanitizeMcpServerName('Context7')).toBe('context7');
});
});