- Added a 'source' field to PluginCatalogItem to distinguish between official and third-party plugins. - Refactored ChangeExtractorService to improve caching mechanisms and normalize file paths for better consistency. - Updated TaskBoundaryParser to support task IDs with underscores, enhancing task identification. - Enhanced TeamMcpConfigBuilder to merge user-defined MCP configurations with generated ones, improving configuration management. - Improved UI components to display source information for plugins and MCP servers, enhancing user experience and clarity.
170 lines
4.9 KiB
TypeScript
170 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',
|
|
source: 'official',
|
|
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');
|
|
});
|
|
});
|