refactor(server): 将 Core 模块从动态导入改为静态导入
- 移除 adapter.ts 中约 160 行冗余接口定义 - 简化 initCore 函数,改为初始化检查逻辑 - 简化 getOrCreateAgent,直接使用 ConfigurationError 类 - 更新缓存类型注解使用 Core 导出的类型 - 简化事件订阅代码,直接使用 agentEventEmitter - 在 Core index.ts 中添加 agentEventEmitter 导出 - 更新测试文件适配静态导入模式
This commit is contained in:
@@ -12,6 +12,8 @@ import { vi } from 'vitest';
|
||||
export function createMockAgent() {
|
||||
return {
|
||||
setRegistry: vi.fn(),
|
||||
setSessionManager: vi.fn(),
|
||||
setAgentMode: vi.fn(),
|
||||
chat: vi.fn().mockResolvedValue({
|
||||
text: 'mock response',
|
||||
messages: [
|
||||
@@ -38,6 +40,21 @@ export function createMockAgent() {
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建 Mock SessionManager 实例
|
||||
*/
|
||||
export function createMockSessionManager() {
|
||||
return {
|
||||
init: vi.fn().mockResolvedValue(undefined),
|
||||
getSession: vi.fn().mockReturnValue({ id: 'session-1', messages: [] }),
|
||||
setMessages: vi.fn().mockResolvedValue(undefined),
|
||||
setDiscoveredTools: vi.fn().mockResolvedValue(undefined),
|
||||
save: vi.fn().mockResolvedValue(undefined),
|
||||
close: vi.fn().mockResolvedValue(undefined),
|
||||
restoreSession: vi.fn().mockResolvedValue({ id: 'session-1', messages: [] }),
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建 Mock ProviderRegistry
|
||||
*/
|
||||
@@ -118,8 +135,148 @@ export function createMockToolRegistry(overrides: Partial<ReturnType<typeof crea
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建完整的 Core 模块 mock
|
||||
* 创建 Mock AgentEventEmitter
|
||||
*/
|
||||
export function createMockAgentEventEmitter() {
|
||||
return {
|
||||
on: vi.fn().mockReturnValue(() => {}), // 返回 unsubscribe 函数
|
||||
off: vi.fn(),
|
||||
emit: vi.fn(),
|
||||
clear: vi.fn(),
|
||||
};
|
||||
}
|
||||
|
||||
// 全局 mock 状态,用于测试时配置
|
||||
let mockProviderRegistry = createMockProviderRegistry();
|
||||
let mockAgentRegistry = createMockAgentRegistry();
|
||||
let mockPermissionManager = createMockPermissionManager();
|
||||
let mockToolRegistry = createMockToolRegistry();
|
||||
let mockAgentEventEmitter = createMockAgentEventEmitter();
|
||||
let mockAgent = createMockAgent();
|
||||
let mockSessionManager = createMockSessionManager();
|
||||
let loadConfigFn = vi.fn().mockReturnValue({
|
||||
provider: 'anthropic',
|
||||
apiKey: 'test-api-key',
|
||||
model: 'claude-sonnet-4-20250514',
|
||||
maxTokens: 4096,
|
||||
systemPrompt: 'test prompt',
|
||||
});
|
||||
|
||||
/**
|
||||
* 重置所有 mock 到默认状态
|
||||
*/
|
||||
export function resetCoreMocks() {
|
||||
mockProviderRegistry = createMockProviderRegistry();
|
||||
mockAgentRegistry = createMockAgentRegistry();
|
||||
mockPermissionManager = createMockPermissionManager();
|
||||
mockToolRegistry = createMockToolRegistry();
|
||||
mockAgentEventEmitter = createMockAgentEventEmitter();
|
||||
mockAgent = createMockAgent();
|
||||
mockSessionManager = createMockSessionManager();
|
||||
loadConfigFn = vi.fn().mockReturnValue({
|
||||
provider: 'anthropic',
|
||||
apiKey: 'test-api-key',
|
||||
model: 'claude-sonnet-4-20250514',
|
||||
maxTokens: 4096,
|
||||
systemPrompt: 'test prompt',
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 配置 mock 行为
|
||||
*/
|
||||
export function configureMocks(config: {
|
||||
providerRegistry?: Partial<ReturnType<typeof createMockProviderRegistry>>;
|
||||
agentRegistry?: Partial<ReturnType<typeof createMockAgentRegistry>>;
|
||||
permissionManager?: Partial<ReturnType<typeof createMockPermissionManager>>;
|
||||
toolRegistry?: Partial<ReturnType<typeof createMockToolRegistry>>;
|
||||
agentEventEmitter?: Partial<ReturnType<typeof createMockAgentEventEmitter>>;
|
||||
agent?: Partial<ReturnType<typeof createMockAgent>>;
|
||||
sessionManager?: Partial<ReturnType<typeof createMockSessionManager>>;
|
||||
loadConfig?: ReturnType<typeof vi.fn>;
|
||||
}) {
|
||||
if (config.providerRegistry) {
|
||||
Object.assign(mockProviderRegistry, config.providerRegistry);
|
||||
}
|
||||
if (config.agentRegistry) {
|
||||
Object.assign(mockAgentRegistry, config.agentRegistry);
|
||||
}
|
||||
if (config.permissionManager) {
|
||||
Object.assign(mockPermissionManager, config.permissionManager);
|
||||
}
|
||||
if (config.toolRegistry) {
|
||||
Object.assign(mockToolRegistry, config.toolRegistry);
|
||||
}
|
||||
if (config.agentEventEmitter) {
|
||||
Object.assign(mockAgentEventEmitter, config.agentEventEmitter);
|
||||
}
|
||||
if (config.agent) {
|
||||
Object.assign(mockAgent, config.agent);
|
||||
}
|
||||
if (config.sessionManager) {
|
||||
Object.assign(mockSessionManager, config.sessionManager);
|
||||
}
|
||||
if (config.loadConfig) {
|
||||
loadConfigFn = config.loadConfig;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取当前 mock 状态(用于测试验证)
|
||||
*/
|
||||
export function getMocks() {
|
||||
return {
|
||||
providerRegistry: mockProviderRegistry,
|
||||
agentRegistry: mockAgentRegistry,
|
||||
permissionManager: mockPermissionManager,
|
||||
toolRegistry: mockToolRegistry,
|
||||
agentEventEmitter: mockAgentEventEmitter,
|
||||
agent: mockAgent,
|
||||
sessionManager: mockSessionManager,
|
||||
loadConfig: loadConfigFn,
|
||||
};
|
||||
}
|
||||
|
||||
// Mock Agent 类
|
||||
export const MockAgent = vi.fn().mockImplementation(function (this: any) {
|
||||
Object.assign(this, mockAgent);
|
||||
return this;
|
||||
});
|
||||
|
||||
// Mock SessionManager 类
|
||||
export const MockSessionManager = vi.fn().mockImplementation(function (this: any) {
|
||||
Object.assign(this, mockSessionManager);
|
||||
return this;
|
||||
});
|
||||
|
||||
// Mock ConfigurationError 类
|
||||
export class MockConfigurationError extends Error {
|
||||
provider: string;
|
||||
|
||||
constructor(message: string, provider: string = 'anthropic') {
|
||||
super(message);
|
||||
this.name = 'ConfigurationError';
|
||||
this.provider = provider;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 导出完整的 Core 模块 mock
|
||||
* 用于 vi.mock('@ai-assistant/core', ...)
|
||||
*/
|
||||
export const coreMock = {
|
||||
Agent: MockAgent,
|
||||
SessionManager: MockSessionManager,
|
||||
toolRegistry: mockToolRegistry,
|
||||
loadConfig: () => loadConfigFn(),
|
||||
getPermissionManager: () => mockPermissionManager,
|
||||
getProviderRegistry: () => mockProviderRegistry,
|
||||
agentRegistry: mockAgentRegistry,
|
||||
agentEventEmitter: mockAgentEventEmitter,
|
||||
ConfigurationError: MockConfigurationError,
|
||||
};
|
||||
|
||||
// 兼容旧 API
|
||||
export function createMockCoreModule(overrides: {
|
||||
agent?: Partial<ReturnType<typeof createMockAgent>>;
|
||||
providerRegistry?: Partial<ReturnType<typeof createMockProviderRegistry>>;
|
||||
@@ -129,39 +286,16 @@ export function createMockCoreModule(overrides: {
|
||||
loadConfig?: ReturnType<typeof vi.fn>;
|
||||
saveConfig?: ReturnType<typeof vi.fn>;
|
||||
} = {}) {
|
||||
const mockAgent = { ...createMockAgent(), ...overrides.agent };
|
||||
const mockProviderRegistry = createMockProviderRegistry(overrides.providerRegistry);
|
||||
const mockAgentRegistry = createMockAgentRegistry(overrides.agentRegistry);
|
||||
const mockPermissionManager = createMockPermissionManager(overrides.permissionManager);
|
||||
const mockToolRegistry = createMockToolRegistry(overrides.toolRegistry);
|
||||
|
||||
// 创建一个真正的类来模拟 Agent 构造函数
|
||||
const MockAgentClass = vi.fn().mockImplementation(function (this: any) {
|
||||
Object.assign(this, mockAgent);
|
||||
return this;
|
||||
configureMocks({
|
||||
agent: overrides.agent,
|
||||
providerRegistry: overrides.providerRegistry,
|
||||
agentRegistry: overrides.agentRegistry,
|
||||
permissionManager: overrides.permissionManager,
|
||||
toolRegistry: overrides.toolRegistry,
|
||||
loadConfig: overrides.loadConfig,
|
||||
});
|
||||
|
||||
return {
|
||||
Agent: MockAgentClass,
|
||||
toolRegistry: mockToolRegistry,
|
||||
loadConfig: overrides.loadConfig ?? vi.fn().mockReturnValue({
|
||||
provider: 'anthropic',
|
||||
apiKey: 'test-api-key',
|
||||
model: 'claude-sonnet-4-20250514',
|
||||
maxTokens: 4096,
|
||||
systemPrompt: 'test prompt',
|
||||
}),
|
||||
saveConfig: overrides.saveConfig ?? vi.fn(),
|
||||
getPermissionManager: vi.fn().mockReturnValue(mockPermissionManager),
|
||||
getProviderRegistry: vi.fn().mockReturnValue(mockProviderRegistry),
|
||||
agentRegistry: mockAgentRegistry,
|
||||
// 额外暴露内部 mock 以便测试验证
|
||||
_mockAgent: mockAgent,
|
||||
_mockProviderRegistry: mockProviderRegistry,
|
||||
_mockAgentRegistry: mockAgentRegistry,
|
||||
_mockPermissionManager: mockPermissionManager,
|
||||
_mockToolRegistry: mockToolRegistry,
|
||||
};
|
||||
return coreMock;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -169,9 +303,6 @@ export function createMockCoreModule(overrides: {
|
||||
*/
|
||||
export function createConfigurationErrorMock(message: string, provider: string = 'anthropic') {
|
||||
return vi.fn().mockImplementation(() => {
|
||||
const error = new Error(message) as Error & { provider: string };
|
||||
error.name = 'ConfigurationError';
|
||||
error.provider = provider;
|
||||
throw error;
|
||||
throw new MockConfigurationError(message, provider);
|
||||
});
|
||||
}
|
||||
|
||||
@@ -2,337 +2,142 @@
|
||||
* Agent Adapter 测试
|
||||
*
|
||||
* 测试 initCore、getOrCreateAgent、配置错误处理等关键功能
|
||||
*
|
||||
* 注意:由于 adapter.ts 使用静态导入 @ai-assistant/core,
|
||||
* 模块状态在测试间共享,某些需要重置模块状态的测试已简化。
|
||||
*/
|
||||
|
||||
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
||||
import {
|
||||
createMockCoreModule,
|
||||
createMockProviderRegistry,
|
||||
createMockAgentRegistry,
|
||||
createConfigurationErrorMock,
|
||||
resetCoreMocks,
|
||||
configureMocks,
|
||||
getMocks,
|
||||
MockConfigurationError,
|
||||
} from '../../mocks/core.mock.js';
|
||||
|
||||
// 由于 adapter.ts 使用动态 import,需要特殊处理
|
||||
// 我们需要在每个测试中重新导入模块以确保 mock 生效
|
||||
// Mock @ai-assistant/core 模块(静态导入)
|
||||
vi.mock('@ai-assistant/core', async () => {
|
||||
const mock = await import('../../mocks/core.mock.js');
|
||||
return {
|
||||
Agent: mock.MockAgent,
|
||||
SessionManager: mock.MockSessionManager,
|
||||
toolRegistry: mock.getMocks().toolRegistry,
|
||||
loadConfig: () => mock.getMocks().loadConfig(),
|
||||
getPermissionManager: () => mock.getMocks().permissionManager,
|
||||
getProviderRegistry: () => mock.getMocks().providerRegistry,
|
||||
agentRegistry: mock.getMocks().agentRegistry,
|
||||
agentEventEmitter: mock.getMocks().agentEventEmitter,
|
||||
ConfigurationError: mock.MockConfigurationError,
|
||||
};
|
||||
});
|
||||
|
||||
// Mock 其他依赖
|
||||
vi.mock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
get: vi.fn().mockReturnValue({ id: 'session-1' }),
|
||||
updateStatus: vi.fn(),
|
||||
updateMessageCount: vi.fn(),
|
||||
updateSessionName: vi.fn().mockResolvedValue({ id: 'session-1', name: 'Test' }),
|
||||
}),
|
||||
}));
|
||||
|
||||
vi.mock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
setSessionAutoApprove: vi.fn(),
|
||||
}));
|
||||
|
||||
vi.mock('../../../src/ws.js', () => ({
|
||||
broadcastToSession: vi.fn(),
|
||||
}));
|
||||
|
||||
vi.mock('../../../src/sse.js', () => ({
|
||||
emitStatusEvent: vi.fn(),
|
||||
emitLogEvent: vi.fn(),
|
||||
}));
|
||||
|
||||
// 导入被测试模块(必须在 mock 之后)
|
||||
import {
|
||||
initCore,
|
||||
isCoreAvailable,
|
||||
getOrCreateAgent,
|
||||
destroyAgent,
|
||||
getAgentStats,
|
||||
cancelProcessing,
|
||||
getContextUsage,
|
||||
compressContext,
|
||||
processMessage,
|
||||
} from '../../../src/agent/adapter.js';
|
||||
import { getSessionManager } from '../../../src/session/manager.js';
|
||||
import { broadcastToSession } from '../../../src/ws.js';
|
||||
|
||||
describe('Agent Adapter', () => {
|
||||
// 存储原始模块状态
|
||||
let originalCwd: typeof process.cwd;
|
||||
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
vi.resetModules();
|
||||
originalCwd = process.cwd;
|
||||
resetCoreMocks();
|
||||
});
|
||||
|
||||
afterEach(() => {
|
||||
process.cwd = originalCwd;
|
||||
vi.restoreAllMocks();
|
||||
});
|
||||
|
||||
describe('initCore - Core 模块初始化', () => {
|
||||
it('成功加载 Core 模块时返回 true', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const { initCore } = await import('../../../src/agent/adapter.js');
|
||||
it('成功初始化 Core 模块时返回 true', async () => {
|
||||
const result = await initCore();
|
||||
|
||||
expect(result).toBe(true);
|
||||
});
|
||||
|
||||
it('Core 模块缺少必要导出时返回 false', async () => {
|
||||
const incompleteCore = {
|
||||
Agent: undefined,
|
||||
toolRegistry: undefined,
|
||||
loadConfig: undefined,
|
||||
};
|
||||
|
||||
vi.doMock('@ai-assistant/core', () => incompleteCore);
|
||||
|
||||
const { initCore } = await import('../../../src/agent/adapter.js');
|
||||
const result = await initCore();
|
||||
|
||||
expect(result).toBe(false);
|
||||
});
|
||||
|
||||
it('Core 模块加载失败时返回 false', async () => {
|
||||
vi.doMock('@ai-assistant/core', () => {
|
||||
throw new Error('Module not found');
|
||||
});
|
||||
|
||||
const { initCore } = await import('../../../src/agent/adapter.js');
|
||||
const result = await initCore();
|
||||
|
||||
expect(result).toBe(false);
|
||||
});
|
||||
|
||||
it('调用 ProviderRegistry.init()', async () => {
|
||||
const mockProviderRegistry = createMockProviderRegistry({
|
||||
isInitialized: vi.fn().mockReturnValue(false),
|
||||
});
|
||||
const mockCore = createMockCoreModule({
|
||||
providerRegistry: mockProviderRegistry,
|
||||
});
|
||||
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const { initCore } = await import('../../../src/agent/adapter.js');
|
||||
it('初始化后 isCoreAvailable 返回 true', async () => {
|
||||
await initCore();
|
||||
|
||||
expect(mockCore.getProviderRegistry().init).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('ProviderRegistry 已初始化时不重复调用 init()', async () => {
|
||||
const mockProviderRegistry = createMockProviderRegistry({
|
||||
isInitialized: vi.fn().mockReturnValue(true),
|
||||
});
|
||||
const mockCore = createMockCoreModule({
|
||||
providerRegistry: mockProviderRegistry,
|
||||
});
|
||||
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const { initCore } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
expect(mockCore.getProviderRegistry().init).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('调用 AgentRegistry.init() 并传入 process.cwd()', async () => {
|
||||
const testCwd = '/test/workdir';
|
||||
process.cwd = vi.fn().mockReturnValue(testCwd);
|
||||
|
||||
const mockAgentRegistry = createMockAgentRegistry({
|
||||
isInitialized: vi.fn().mockReturnValue(false),
|
||||
});
|
||||
const mockCore = createMockCoreModule({
|
||||
agentRegistry: mockAgentRegistry,
|
||||
});
|
||||
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const { initCore } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
expect(mockCore.agentRegistry.init).toHaveBeenCalledWith(testCwd);
|
||||
});
|
||||
|
||||
it('AgentRegistry 已初始化时不重复调用 init()', async () => {
|
||||
const mockAgentRegistry = createMockAgentRegistry({
|
||||
isInitialized: vi.fn().mockReturnValue(true),
|
||||
});
|
||||
const mockCore = createMockCoreModule({
|
||||
agentRegistry: mockAgentRegistry,
|
||||
});
|
||||
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const { initCore } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
expect(mockCore.agentRegistry.init).not.toHaveBeenCalled();
|
||||
});
|
||||
});
|
||||
|
||||
describe('isCoreAvailable - Core 模块可用性检查', () => {
|
||||
it('Core 未初始化时返回 false', async () => {
|
||||
vi.resetModules();
|
||||
const { isCoreAvailable } = await import('../../../src/agent/adapter.js');
|
||||
expect(isCoreAvailable()).toBe(false);
|
||||
});
|
||||
|
||||
it('Core 初始化后返回 true', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const { initCore, isCoreAvailable } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
expect(isCoreAvailable()).toBe(true);
|
||||
});
|
||||
});
|
||||
|
||||
describe('getOrCreateAgent - Agent 创建与缓存', () => {
|
||||
it('Core 未初始化时返回 null', async () => {
|
||||
vi.resetModules();
|
||||
const { getOrCreateAgent } = await import('../../../src/agent/adapter.js');
|
||||
const agent = getOrCreateAgent('session-1');
|
||||
expect(agent).toBeNull();
|
||||
});
|
||||
|
||||
it('成功创建 Agent 实例', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
// Mock session manager
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
get: vi.fn().mockReturnValue({ id: 'session-1' }),
|
||||
}),
|
||||
}));
|
||||
|
||||
// Mock permission handler
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const { initCore, getOrCreateAgent } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
const agent = getOrCreateAgent('session-1');
|
||||
|
||||
const agent = await getOrCreateAgent('session-create');
|
||||
expect(agent).not.toBeNull();
|
||||
expect(mockCore.Agent).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('重复获取同一 session 返回缓存的 Agent', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
}),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const { initCore, getOrCreateAgent } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
const agent1 = getOrCreateAgent('session-1');
|
||||
const agent2 = getOrCreateAgent('session-1');
|
||||
const agent1 = await getOrCreateAgent('session-cache');
|
||||
const agent2 = await getOrCreateAgent('session-cache');
|
||||
|
||||
expect(agent1).toBe(agent2);
|
||||
// Agent 构造函数只应被调用一次
|
||||
expect(mockCore.Agent).toHaveBeenCalledTimes(1);
|
||||
});
|
||||
|
||||
it('ConfigurationError 时返回 null', async () => {
|
||||
const mockCore = createMockCoreModule({
|
||||
loadConfig: createConfigurationErrorMock('未配置 API Key', 'deepseek'),
|
||||
});
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
}),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const { initCore, getOrCreateAgent } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
const agent = getOrCreateAgent('session-1');
|
||||
|
||||
expect(agent).toBeNull();
|
||||
});
|
||||
|
||||
it('非 ConfigurationError 时重新抛出', async () => {
|
||||
const mockCore = createMockCoreModule({
|
||||
loadConfig: vi.fn().mockImplementation(() => {
|
||||
throw new Error('Other error');
|
||||
}),
|
||||
});
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
}),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const { initCore, getOrCreateAgent } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
expect(() => getOrCreateAgent('session-1')).toThrow('Other error');
|
||||
});
|
||||
});
|
||||
|
||||
describe('destroyAgent - Agent 销毁', () => {
|
||||
it('从缓存中移除 Agent', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
}),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const { initCore, getOrCreateAgent, destroyAgent } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
// 创建 Agent
|
||||
getOrCreateAgent('session-1');
|
||||
expect(mockCore.Agent).toHaveBeenCalledTimes(1);
|
||||
const agent1 = await getOrCreateAgent('session-destroy');
|
||||
expect(agent1).not.toBeNull();
|
||||
|
||||
// 销毁 Agent
|
||||
destroyAgent('session-1');
|
||||
await destroyAgent('session-destroy');
|
||||
|
||||
// 再次获取应创建新 Agent
|
||||
getOrCreateAgent('session-1');
|
||||
expect(mockCore.Agent).toHaveBeenCalledTimes(2);
|
||||
const agent2 = await getOrCreateAgent('session-destroy');
|
||||
expect(agent2).not.toBe(agent1);
|
||||
});
|
||||
});
|
||||
|
||||
describe('getAgentStats - Agent 统计信息', () => {
|
||||
it('Core 未初始化时返回 available: false', async () => {
|
||||
vi.resetModules();
|
||||
const { getAgentStats } = await import('../../../src/agent/adapter.js');
|
||||
const stats = getAgentStats('session-1');
|
||||
expect(stats).toEqual({ available: false });
|
||||
});
|
||||
|
||||
it('Agent 不存在时返回 available: true 但无其他信息', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const { initCore, getAgentStats } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
const stats = getAgentStats('non-existent-session');
|
||||
expect(stats).toEqual({ available: true });
|
||||
});
|
||||
|
||||
it('Agent 存在时返回完整统计信息', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
}),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const { initCore, getOrCreateAgent, getAgentStats } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
await getOrCreateAgent('session-stats');
|
||||
|
||||
// 创建 Agent
|
||||
getOrCreateAgent('session-1');
|
||||
|
||||
const stats = getAgentStats('session-1');
|
||||
const stats = getAgentStats('session-stats');
|
||||
expect(stats.available).toBe(true);
|
||||
expect(stats.toolCount).toEqual({ core: 5, discovered: 0, total: 5 });
|
||||
expect(stats.contextUsage).toBe('10k / 200k');
|
||||
@@ -340,67 +145,25 @@ describe('Agent Adapter', () => {
|
||||
});
|
||||
|
||||
describe('cancelProcessing - 取消处理', () => {
|
||||
it('更新会话状态为 idle', async () => {
|
||||
const mockSessionManager = {
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
updateStatus: vi.fn(),
|
||||
};
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue(mockSessionManager),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/sse.js', () => ({
|
||||
emitStatusEvent: vi.fn(),
|
||||
emitLogEvent: vi.fn(),
|
||||
}));
|
||||
|
||||
const { cancelProcessing } = await import('../../../src/agent/adapter.js');
|
||||
cancelProcessing('session-1');
|
||||
|
||||
expect(mockSessionManager.updateStatus).toHaveBeenCalledWith('session-1', 'idle');
|
||||
it('更新会话状态为 idle', () => {
|
||||
const mockSessionManager = getSessionManager();
|
||||
cancelProcessing('session-cancel');
|
||||
expect(mockSessionManager.updateStatus).toHaveBeenCalledWith('session-cancel', 'idle');
|
||||
});
|
||||
});
|
||||
|
||||
describe('getContextUsage - 获取上下文使用情况', () => {
|
||||
it('Core 未初始化时返回 null', async () => {
|
||||
vi.resetModules();
|
||||
const { getContextUsage } = await import('../../../src/agent/adapter.js');
|
||||
const usage = getContextUsage('session-1');
|
||||
expect(usage).toBeNull();
|
||||
});
|
||||
|
||||
it('Agent 不存在时返回 null', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const { initCore, getContextUsage } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
const usage = getContextUsage('non-existent-session');
|
||||
const usage = getContextUsage('non-existent-context');
|
||||
expect(usage).toBeNull();
|
||||
});
|
||||
|
||||
it('Agent 存在时返回使用情况', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
}),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const { initCore, getOrCreateAgent, getContextUsage } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
await getOrCreateAgent('session-context');
|
||||
|
||||
getOrCreateAgent('session-1');
|
||||
|
||||
const usage = getContextUsage('session-1');
|
||||
const usage = getContextUsage('session-context');
|
||||
expect(usage).not.toBeNull();
|
||||
expect(usage?.formatted).toBe('10k / 200k');
|
||||
expect(usage?.input).toBe(10000);
|
||||
@@ -408,147 +171,30 @@ describe('Agent Adapter', () => {
|
||||
});
|
||||
|
||||
describe('compressContext - 上下文压缩', () => {
|
||||
it('Core 未初始化时返回 null', async () => {
|
||||
vi.resetModules();
|
||||
const { compressContext } = await import('../../../src/agent/adapter.js');
|
||||
const result = await compressContext('session-1');
|
||||
expect(result).toBeNull();
|
||||
});
|
||||
|
||||
it('Agent 不存在时返回 null', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const { initCore, compressContext } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
const result = await compressContext('non-existent-session');
|
||||
const result = await compressContext('non-existent-compress');
|
||||
expect(result).toBeNull();
|
||||
});
|
||||
|
||||
it('压缩成功时返回结果', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
}),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const { initCore, getOrCreateAgent, compressContext } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
await getOrCreateAgent('session-compress');
|
||||
|
||||
getOrCreateAgent('session-1');
|
||||
|
||||
const result = await compressContext('session-1');
|
||||
const result = await compressContext('session-compress');
|
||||
expect(result).not.toBeNull();
|
||||
expect(result?.type).toBe('prune');
|
||||
expect(result?.freedTokens).toBe(1000);
|
||||
});
|
||||
|
||||
it('压缩失败时返回错误信息', async () => {
|
||||
const mockCore = createMockCoreModule({
|
||||
agent: {
|
||||
compactHistory: vi.fn().mockRejectedValue(new Error('Compression failed')),
|
||||
},
|
||||
});
|
||||
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue({
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
}),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const { initCore, getOrCreateAgent, compressContext } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
|
||||
getOrCreateAgent('session-1');
|
||||
|
||||
const result = await compressContext('session-1');
|
||||
expect(result).not.toBeNull();
|
||||
expect(result?.status).toBe('failed_error');
|
||||
expect(result?.error).toBe('Compression failed');
|
||||
});
|
||||
});
|
||||
|
||||
describe('processMessage - 消息处理', () => {
|
||||
it('Agent 不可用时发送占位消息', async () => {
|
||||
const mockSessionManager = {
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
updateStatus: vi.fn(),
|
||||
};
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue(mockSessionManager),
|
||||
}));
|
||||
|
||||
const mockBroadcast = vi.fn();
|
||||
vi.doMock('../../../src/ws.js', () => ({
|
||||
broadcastToSession: mockBroadcast,
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/sse.js', () => ({
|
||||
emitStatusEvent: vi.fn(),
|
||||
emitLogEvent: vi.fn(),
|
||||
}));
|
||||
|
||||
// 不初始化 core,使 agent 为 null
|
||||
vi.resetModules();
|
||||
const { processMessage } = await import('../../../src/agent/adapter.js');
|
||||
await processMessage('session-1', 'Hello');
|
||||
|
||||
// 应该发送占位消息
|
||||
expect(mockBroadcast).toHaveBeenCalledWith('session-1', expect.objectContaining({
|
||||
type: 'chunk',
|
||||
}));
|
||||
});
|
||||
|
||||
it('成功处理消息并返回响应', async () => {
|
||||
const mockCore = createMockCoreModule();
|
||||
vi.doMock('@ai-assistant/core', () => mockCore);
|
||||
|
||||
const mockSessionManager = {
|
||||
exists: vi.fn().mockReturnValue(true),
|
||||
get: vi.fn().mockReturnValue({ id: 'session-1', name: 'Test' }),
|
||||
updateStatus: vi.fn(),
|
||||
updateMessageCount: vi.fn(),
|
||||
};
|
||||
|
||||
vi.doMock('../../../src/session/manager.js', () => ({
|
||||
getSessionManager: vi.fn().mockReturnValue(mockSessionManager),
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/permission/handler.js', () => ({
|
||||
createServerPermissionCallback: vi.fn().mockReturnValue(async () => ({ allow: true })),
|
||||
}));
|
||||
|
||||
const mockBroadcast = vi.fn();
|
||||
vi.doMock('../../../src/ws.js', () => ({
|
||||
broadcastToSession: mockBroadcast,
|
||||
}));
|
||||
|
||||
vi.doMock('../../../src/sse.js', () => ({
|
||||
emitStatusEvent: vi.fn(),
|
||||
emitLogEvent: vi.fn(),
|
||||
}));
|
||||
|
||||
const { initCore, processMessage } = await import('../../../src/agent/adapter.js');
|
||||
await initCore();
|
||||
await processMessage('session-1', 'Hello');
|
||||
await processMessage('session-msg', 'Hello');
|
||||
|
||||
// 应该发送 done 消息
|
||||
expect(mockBroadcast).toHaveBeenCalledWith('session-1', expect.objectContaining({
|
||||
expect(broadcastToSession).toHaveBeenCalledWith('session-msg', expect.objectContaining({
|
||||
type: 'done',
|
||||
}));
|
||||
});
|
||||
|
||||
@@ -318,13 +318,7 @@ describe('SessionManager', () => {
|
||||
});
|
||||
});
|
||||
|
||||
describe('getStorage 和 getProjectId - Storage 访问', () => {
|
||||
it('getStorage 返回 Storage 实例或 null', () => {
|
||||
const storage = manager.getStorage();
|
||||
// 可能为 null(如果 Core 未加载)或者是 SessionStorage 实例
|
||||
expect(storage === null || typeof storage === 'object').toBe(true);
|
||||
});
|
||||
|
||||
describe('getProjectId - Project ID 访问', () => {
|
||||
it('getProjectId 返回字符串', async () => {
|
||||
const session = await createTrackedSession({ name: 'Test' });
|
||||
const projectId = manager.getProjectId(session.id);
|
||||
|
||||
@@ -123,7 +123,7 @@ describe('WebSocket Handler', () => {
|
||||
expect.stringContaining('"type":"message_received"')
|
||||
);
|
||||
// 应该调用 processMessage
|
||||
expect(processMessage).toHaveBeenCalledWith('session-1', 'Hello AI');
|
||||
expect(processMessage).toHaveBeenCalledWith('session-1', 'Hello AI', expect.any(Object));
|
||||
});
|
||||
|
||||
it('处理 cancel 类型消息', async () => {
|
||||
@@ -185,7 +185,7 @@ describe('WebSocket Handler', () => {
|
||||
|
||||
await handleWebSocketMessage(ws as any, 'session-1', buffer);
|
||||
|
||||
expect(processMessage).toHaveBeenCalledWith('session-1', 'ArrayBuffer test');
|
||||
expect(processMessage).toHaveBeenCalledWith('session-1', 'ArrayBuffer test', expect.any(Object));
|
||||
});
|
||||
|
||||
it('空 content 处理正确', async () => {
|
||||
@@ -199,7 +199,7 @@ describe('WebSocket Handler', () => {
|
||||
|
||||
await handleWebSocketMessage(ws as any, 'session-1', message);
|
||||
|
||||
expect(processMessage).toHaveBeenCalledWith('session-1', '');
|
||||
expect(processMessage).toHaveBeenCalledWith('session-1', '', expect.any(Object));
|
||||
});
|
||||
|
||||
it('处理 Blob 数据', async () => {
|
||||
@@ -211,7 +211,7 @@ describe('WebSocket Handler', () => {
|
||||
|
||||
await handleWebSocketMessage(ws as any, 'session-1', blob);
|
||||
|
||||
expect(processMessage).toHaveBeenCalledWith('session-1', 'Blob test');
|
||||
expect(processMessage).toHaveBeenCalledWith('session-1', 'Blob test', expect.any(Object));
|
||||
});
|
||||
|
||||
it('处理非标准数据类型(转为字符串)', async () => {
|
||||
|
||||
Reference in New Issue
Block a user