Files
FastGPT/packages/global/test/core/app/utils.test.ts
T
Ryo 289da0f7b0 chore: bump pro submodule for hydration stability (#6808)
* sandbox-sync-agent

* refactor: host pro as submodule

* chore: checkpoint host pro restructure

* refactor workspace test layout and startup init

* chore: update next turbopack setup

* chore: snapshot current work before actions fix

* chore: update pro submodule

* chore: point pro submodule url to upstream https

* fix: Dockerfile

* chore: update pro submodule

* ci: support private pro submodule token and skip fork jobs

* fix(ci): build sdk workspace deps before code-sandbox bundle

* fix(app): exclude vitest configs from production typecheck

* fix(app-image): build sdk packages before next build

* fix(ci): align dockerfiles with workspace sdk build flow

* chore(docker): upgrade node20 docker images to node24

* fix(ci): read admin coverage output path in pro test workflow

* fix(app-image): include next-i18next config and locale assets

* chore: update pro submodule

* chore: do not specify branch for submodule

* chore: remove most ts-nocheck sign

* chore: update pro submodule

* chore: remove sandbox-agent-sync package

* chore: do not modify "pushData" file logic

* fix: health check

* chore: restore dev axios proxy state

* fix: test-fastgpt report workflow

* fix: use valid vitest coverage action inputs
2026-04-27 17:44:12 +08:00

177 lines
5.2 KiB
TypeScript

import { describe, expect, it, vi } from 'vitest';
import { getDefaultAppForm, getAppType, formatToolError } from '@fastgpt/global/core/app/utils';
import { AppTypeEnum } from '@fastgpt/global/core/app/constants';
import { DatasetSearchModeEnum } from '@fastgpt/global/core/dataset/constants';
describe('getDefaultAppForm', () => {
it('should return default app form with correct structure', () => {
const result = getDefaultAppForm();
expect(result).toHaveProperty('aiSettings');
expect(result).toHaveProperty('dataset');
expect(result).toHaveProperty('selectedTools');
expect(result).toHaveProperty('chatConfig');
});
it('should return correct aiSettings defaults', () => {
const result = getDefaultAppForm();
expect(result.aiSettings).toEqual({
model: '',
isResponseAnswerText: true,
maxHistories: 6
});
});
it('should return correct dataset defaults', () => {
const result = getDefaultAppForm();
expect(result.dataset).toEqual({
datasets: [],
similarity: 0.4,
limit: 3000,
searchMode: DatasetSearchModeEnum.embedding,
usingReRank: true,
rerankModel: '',
rerankWeight: 0.5,
datasetSearchUsingExtensionQuery: true,
datasetSearchExtensionBg: ''
});
});
it('should return empty selectedTools array', () => {
const result = getDefaultAppForm();
expect(result.selectedTools).toEqual([]);
});
it('should return empty chatConfig object', () => {
const result = getDefaultAppForm();
expect(result.chatConfig).toEqual({});
});
});
describe('getAppType', () => {
it('should return empty string when config is undefined', () => {
const result = getAppType(undefined);
expect(result).toBe('');
});
it('should return simple type when config has aiSettings', () => {
const config = {
aiSettings: {
model: 'gpt-4',
isResponseAnswerText: true,
maxHistories: 6
},
dataset: {
datasets: [],
similarity: 0.4,
limit: 3000,
searchMode: DatasetSearchModeEnum.embedding,
usingReRank: true,
rerankModel: '',
rerankWeight: 0.5,
datasetSearchUsingExtensionQuery: true,
datasetSearchExtensionBg: ''
},
selectedTools: [],
chatConfig: {}
};
const result = getAppType(config);
expect(result).toBe(AppTypeEnum.simple);
});
it('should return empty string when config has no nodes and no aiSettings', () => {
const config = {} as any;
const result = getAppType(config);
expect(result).toBe('');
});
it('should return workflow type when nodes contain workflowStart', () => {
const config = {
nodes: [
{ flowNodeType: 'workflowStart', nodeId: '1' },
{ flowNodeType: 'aiChat', nodeId: '2' }
],
edges: []
};
const result = getAppType(config as any);
expect(result).toBe(AppTypeEnum.workflow);
});
it('should return workflowTool type when nodes contain pluginInput', () => {
const config = {
nodes: [
{ flowNodeType: 'pluginInput', nodeId: '1' },
{ flowNodeType: 'pluginOutput', nodeId: '2' }
],
edges: []
};
const result = getAppType(config as any);
expect(result).toBe(AppTypeEnum.workflowTool);
});
it('should return empty string when nodes exist but no workflowStart or pluginInput', () => {
const config = {
nodes: [
{ flowNodeType: 'aiChat', nodeId: '1' },
{ flowNodeType: 'textOutput', nodeId: '2' }
],
edges: []
};
const result = getAppType(config as any);
expect(result).toBe('');
});
it('should prioritize workflow type over workflowTool when both exist', () => {
const config = {
nodes: [
{ flowNodeType: 'workflowStart', nodeId: '1' },
{ flowNodeType: 'pluginInput', nodeId: '2' }
],
edges: []
};
const result = getAppType(config as any);
expect(result).toBe(AppTypeEnum.workflow);
});
});
describe('formatToolError', () => {
it('should return undefined when error is undefined', () => {
const result = formatToolError(undefined);
expect(result).toBeUndefined();
});
it('should return undefined when error is null', () => {
const result = formatToolError(null);
expect(result).toBeUndefined();
});
it('should return undefined when error is not a string', () => {
const result = formatToolError({ message: 'error' });
expect(result).toBeUndefined();
});
it('should return undefined when error is a number', () => {
const result = formatToolError(123);
expect(result).toBeUndefined();
});
it('should return the original error string when not found in error lists', () => {
const result = formatToolError('unknownError');
expect(result).toBe('unknownError');
});
it('should return formatted message for known app error', () => {
const result = formatToolError('appUnExist');
expect(result).toBeDefined();
expect(typeof result).toBe('string');
});
it('should return formatted message for known plugin error', () => {
const result = formatToolError('pluginUnExist');
expect(result).toBeDefined();
expect(typeof result).toBe('string');
});
});