Optimize gateway comms reload behavior and strengthen regression coverage (#496)
This commit is contained in:
committed by
GitHub
Unverified
parent
08960d700f
commit
1dbe4a8466
145
tests/unit/provider-runtime-sync.test.ts
Normal file
145
tests/unit/provider-runtime-sync.test.ts
Normal file
@@ -0,0 +1,145 @@
|
||||
import { beforeEach, describe, expect, it, vi } from 'vitest';
|
||||
import type { GatewayManager } from '@electron/gateway/manager';
|
||||
import type { ProviderConfig } from '@electron/utils/secure-storage';
|
||||
|
||||
const mocks = vi.hoisted(() => ({
|
||||
getProviderAccount: vi.fn(),
|
||||
listProviderAccounts: vi.fn(),
|
||||
getProviderSecret: vi.fn(),
|
||||
getAllProviders: vi.fn(),
|
||||
getApiKey: vi.fn(),
|
||||
getDefaultProvider: vi.fn(),
|
||||
getProvider: vi.fn(),
|
||||
getProviderConfig: vi.fn(),
|
||||
getProviderDefaultModel: vi.fn(),
|
||||
removeProviderFromOpenClaw: vi.fn(),
|
||||
saveOAuthTokenToOpenClaw: vi.fn(),
|
||||
saveProviderKeyToOpenClaw: vi.fn(),
|
||||
setOpenClawDefaultModel: vi.fn(),
|
||||
setOpenClawDefaultModelWithOverride: vi.fn(),
|
||||
syncProviderConfigToOpenClaw: vi.fn(),
|
||||
updateAgentModelProvider: vi.fn(),
|
||||
}));
|
||||
|
||||
vi.mock('@electron/services/providers/provider-store', () => ({
|
||||
getProviderAccount: mocks.getProviderAccount,
|
||||
listProviderAccounts: mocks.listProviderAccounts,
|
||||
}));
|
||||
|
||||
vi.mock('@electron/services/secrets/secret-store', () => ({
|
||||
getProviderSecret: mocks.getProviderSecret,
|
||||
}));
|
||||
|
||||
vi.mock('@electron/utils/secure-storage', () => ({
|
||||
getAllProviders: mocks.getAllProviders,
|
||||
getApiKey: mocks.getApiKey,
|
||||
getDefaultProvider: mocks.getDefaultProvider,
|
||||
getProvider: mocks.getProvider,
|
||||
}));
|
||||
|
||||
vi.mock('@electron/utils/provider-registry', () => ({
|
||||
getProviderConfig: mocks.getProviderConfig,
|
||||
getProviderDefaultModel: mocks.getProviderDefaultModel,
|
||||
}));
|
||||
|
||||
vi.mock('@electron/utils/openclaw-auth', () => ({
|
||||
removeProviderFromOpenClaw: mocks.removeProviderFromOpenClaw,
|
||||
saveOAuthTokenToOpenClaw: mocks.saveOAuthTokenToOpenClaw,
|
||||
saveProviderKeyToOpenClaw: mocks.saveProviderKeyToOpenClaw,
|
||||
setOpenClawDefaultModel: mocks.setOpenClawDefaultModel,
|
||||
setOpenClawDefaultModelWithOverride: mocks.setOpenClawDefaultModelWithOverride,
|
||||
syncProviderConfigToOpenClaw: mocks.syncProviderConfigToOpenClaw,
|
||||
updateAgentModelProvider: mocks.updateAgentModelProvider,
|
||||
}));
|
||||
|
||||
vi.mock('@electron/utils/logger', () => ({
|
||||
logger: {
|
||||
debug: vi.fn(),
|
||||
info: vi.fn(),
|
||||
warn: vi.fn(),
|
||||
error: vi.fn(),
|
||||
},
|
||||
}));
|
||||
|
||||
import {
|
||||
syncDefaultProviderToRuntime,
|
||||
syncDeletedProviderToRuntime,
|
||||
syncSavedProviderToRuntime,
|
||||
} from '@electron/services/providers/provider-runtime-sync';
|
||||
|
||||
function createProvider(overrides: Partial<ProviderConfig> = {}): ProviderConfig {
|
||||
return {
|
||||
id: 'moonshot',
|
||||
name: 'Moonshot',
|
||||
type: 'moonshot',
|
||||
model: 'kimi-k2.5',
|
||||
enabled: true,
|
||||
createdAt: '2026-03-14T00:00:00.000Z',
|
||||
updatedAt: '2026-03-14T00:00:00.000Z',
|
||||
...overrides,
|
||||
};
|
||||
}
|
||||
|
||||
function createGateway(state: 'running' | 'stopped' = 'running'): Pick<GatewayManager, 'debouncedReload' | 'debouncedRestart' | 'getStatus'> {
|
||||
return {
|
||||
debouncedReload: vi.fn(),
|
||||
debouncedRestart: vi.fn(),
|
||||
getStatus: vi.fn(() => ({ state } as ReturnType<GatewayManager['getStatus']>)),
|
||||
};
|
||||
}
|
||||
|
||||
describe('provider-runtime-sync refresh strategy', () => {
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
mocks.getProviderAccount.mockResolvedValue(null);
|
||||
mocks.getProviderSecret.mockResolvedValue(undefined);
|
||||
mocks.getAllProviders.mockResolvedValue([]);
|
||||
mocks.getApiKey.mockResolvedValue('sk-test');
|
||||
mocks.getDefaultProvider.mockResolvedValue('moonshot');
|
||||
mocks.getProvider.mockResolvedValue(createProvider());
|
||||
mocks.getProviderDefaultModel.mockReturnValue('kimi-k2.5');
|
||||
mocks.getProviderConfig.mockReturnValue({
|
||||
api: 'openai-completions',
|
||||
baseUrl: 'https://api.moonshot.cn/v1',
|
||||
apiKeyEnv: 'MOONSHOT_API_KEY',
|
||||
});
|
||||
mocks.syncProviderConfigToOpenClaw.mockResolvedValue(undefined);
|
||||
mocks.setOpenClawDefaultModel.mockResolvedValue(undefined);
|
||||
mocks.setOpenClawDefaultModelWithOverride.mockResolvedValue(undefined);
|
||||
mocks.saveProviderKeyToOpenClaw.mockResolvedValue(undefined);
|
||||
mocks.removeProviderFromOpenClaw.mockResolvedValue(undefined);
|
||||
mocks.updateAgentModelProvider.mockResolvedValue(undefined);
|
||||
});
|
||||
|
||||
it('uses debouncedReload after saving provider config', async () => {
|
||||
const gateway = createGateway('running');
|
||||
await syncSavedProviderToRuntime(createProvider(), undefined, gateway as GatewayManager);
|
||||
|
||||
expect(gateway.debouncedReload).toHaveBeenCalledTimes(1);
|
||||
expect(gateway.debouncedRestart).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('uses debouncedRestart after deleting provider config', async () => {
|
||||
const gateway = createGateway('running');
|
||||
await syncDeletedProviderToRuntime(createProvider(), 'moonshot', gateway as GatewayManager);
|
||||
|
||||
expect(gateway.debouncedRestart).toHaveBeenCalledTimes(1);
|
||||
expect(gateway.debouncedReload).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('uses debouncedReload after switching default provider when gateway is running', async () => {
|
||||
const gateway = createGateway('running');
|
||||
await syncDefaultProviderToRuntime('moonshot', gateway as GatewayManager);
|
||||
|
||||
expect(gateway.debouncedReload).toHaveBeenCalledTimes(1);
|
||||
expect(gateway.debouncedRestart).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('skips refresh after switching default provider when gateway is stopped', async () => {
|
||||
const gateway = createGateway('stopped');
|
||||
await syncDefaultProviderToRuntime('moonshot', gateway as GatewayManager);
|
||||
|
||||
expect(gateway.debouncedReload).not.toHaveBeenCalled();
|
||||
expect(gateway.debouncedRestart).not.toHaveBeenCalled();
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user