mirror of
https://github.com/Syngnat/GoNavi.git
synced 2026-06-21 05:53:46 +08:00
- 修复 DuckDB qualified table 在查询结果页丢失 schema 导致无法识别主键的问题 - 打开对象修改前强制刷新最新定义,并避免切换对象失败时沿用旧定义 - 为 DuckDB 元数据链路补充前后端回归测试,并给 app 层真实 runtime 测试增加环境门槛
273 lines
8.9 KiB
TypeScript
273 lines
8.9 KiB
TypeScript
import React from 'react';
|
|
import { act, create } from 'react-test-renderer';
|
|
import { beforeEach, describe, expect, it, vi } from 'vitest';
|
|
|
|
import type { TabData } from '../types';
|
|
import TriggerViewer from './TriggerViewer';
|
|
|
|
const storeState = vi.hoisted(() => ({
|
|
connections: [
|
|
{
|
|
id: 'conn-1',
|
|
name: 'local',
|
|
config: {
|
|
type: 'postgres',
|
|
host: '127.0.0.1',
|
|
port: 5432,
|
|
user: 'postgres',
|
|
password: '',
|
|
database: 'main',
|
|
},
|
|
},
|
|
],
|
|
theme: 'light',
|
|
addTab: vi.fn(),
|
|
setActiveContext: vi.fn(),
|
|
}));
|
|
|
|
const backendApp = vi.hoisted(() => ({
|
|
DBQuery: vi.fn(),
|
|
}));
|
|
|
|
vi.mock('../store', () => ({
|
|
useStore: (selector: (state: typeof storeState) => any) => selector(storeState),
|
|
}));
|
|
|
|
vi.mock('../../wailsjs/go/app/App', () => backendApp);
|
|
|
|
vi.mock('@ant-design/icons', () => ({
|
|
EditOutlined: () => <span data-icon="edit" />,
|
|
}));
|
|
|
|
vi.mock('./MonacoEditor', () => ({
|
|
default: ({ value, options }: any) => (
|
|
<pre data-editor="true" data-readonly={String(options?.readOnly)}>
|
|
{value}
|
|
</pre>
|
|
),
|
|
}));
|
|
|
|
vi.mock('antd', () => ({
|
|
Spin: ({ tip }: any) => <div>{tip}</div>,
|
|
Alert: ({ message, description }: any) => <div>{message}{description}</div>,
|
|
Button: ({ children, onClick, icon }: any) => (
|
|
<button type="button" onClick={onClick}>
|
|
{icon}
|
|
{children}
|
|
</button>
|
|
),
|
|
}));
|
|
|
|
const flushPromises = async (count = 6) => {
|
|
for (let i = 0; i < count; i += 1) {
|
|
await Promise.resolve();
|
|
}
|
|
};
|
|
|
|
const findButtonText = (node: any): string => (
|
|
(node.children || [])
|
|
.map((item: any) => (typeof item === 'string' ? item : findButtonText(item)))
|
|
.join('')
|
|
);
|
|
|
|
const tab: TabData = {
|
|
id: 'trigger-conn-1-main-audit.users_bi',
|
|
title: '触发器: audit.users_bi',
|
|
type: 'trigger',
|
|
connectionId: 'conn-1',
|
|
dbName: 'main',
|
|
triggerName: 'audit.users_bi',
|
|
triggerTableName: 'audit.users',
|
|
schemaName: 'audit',
|
|
};
|
|
|
|
describe('TriggerViewer object edit entry', () => {
|
|
beforeEach(() => {
|
|
storeState.addTab.mockReset();
|
|
storeState.setActiveContext.mockReset();
|
|
storeState.connections[0].config.type = 'postgres';
|
|
backendApp.DBQuery.mockReset();
|
|
backendApp.DBQuery.mockResolvedValue({
|
|
success: true,
|
|
data: [{ trigger_definition: 'CREATE TRIGGER users_bi BEFORE INSERT ON audit.users EXECUTE FUNCTION audit.audit_users();' }],
|
|
});
|
|
});
|
|
|
|
it('opens an editable query tab for trigger definitions', async () => {
|
|
let renderer: any;
|
|
await act(async () => {
|
|
renderer = create(<TriggerViewer tab={tab} />);
|
|
await flushPromises();
|
|
});
|
|
|
|
const button = renderer.root.findAll((node: any) => node.type === 'button' && findButtonText(node).includes('对象修改'))[0];
|
|
|
|
await act(async () => {
|
|
button.props.onClick();
|
|
});
|
|
|
|
expect(storeState.setActiveContext).toHaveBeenCalledWith({ connectionId: 'conn-1', dbName: 'main' });
|
|
expect(storeState.addTab).toHaveBeenCalledWith(expect.objectContaining({
|
|
title: '修改触发器: audit.users_bi',
|
|
type: 'query',
|
|
connectionId: 'conn-1',
|
|
dbName: 'main',
|
|
query: expect.stringContaining('CREATE TRIGGER users_bi BEFORE INSERT'),
|
|
}));
|
|
});
|
|
|
|
it('adds CREATE OR REPLACE for trigger source snippets returned without ddl prefix', async () => {
|
|
storeState.connections[0].config.type = 'oracle';
|
|
backendApp.DBQuery.mockResolvedValue({
|
|
success: true,
|
|
data: [{
|
|
TRIGGER_BODY: 'TRIGGER users_bi\nBEFORE INSERT ON audit.users\nFOR EACH ROW\nBEGIN\n :NEW.created_at := SYSDATE;\nEND;',
|
|
}],
|
|
});
|
|
|
|
let renderer: any;
|
|
await act(async () => {
|
|
renderer = create(<TriggerViewer tab={tab} />);
|
|
await flushPromises();
|
|
});
|
|
|
|
const button = renderer.root.findAll((node: any) => node.type === 'button' && findButtonText(node).includes('对象修改'))[0];
|
|
|
|
await act(async () => {
|
|
button.props.onClick();
|
|
});
|
|
|
|
const query = storeState.addTab.mock.calls[0][0].query;
|
|
expect(query).toContain('CREATE OR REPLACE TRIGGER users_bi');
|
|
expect(query).toContain('BEFORE INSERT ON audit.users');
|
|
expect(query).toContain(':NEW.created_at := SYSDATE;');
|
|
expect(query).not.toContain('请补全 CREATE TRIGGER 语句');
|
|
});
|
|
|
|
it('adds trigger name for trigger body snippets returned without ddl header', async () => {
|
|
storeState.connections[0].config.type = 'oracle';
|
|
backendApp.DBQuery.mockResolvedValue({
|
|
success: true,
|
|
data: [{
|
|
TRIGGER_BODY: 'BEFORE UPDATE ON audit.users\nFOR EACH ROW\nBEGIN\n :NEW.updated_at := SYSDATE;\nEND;',
|
|
}],
|
|
});
|
|
|
|
let renderer: any;
|
|
await act(async () => {
|
|
renderer = create(<TriggerViewer tab={tab} />);
|
|
await flushPromises();
|
|
});
|
|
|
|
const button = renderer.root.findAll((node: any) => node.type === 'button' && findButtonText(node).includes('对象修改'))[0];
|
|
|
|
await act(async () => {
|
|
button.props.onClick();
|
|
});
|
|
|
|
const query = storeState.addTab.mock.calls[0][0].query;
|
|
expect(query).toContain('CREATE OR REPLACE TRIGGER audit.users_bi');
|
|
expect(query).toContain('BEFORE UPDATE ON audit.users');
|
|
expect(query).toContain(':NEW.updated_at := SYSDATE;');
|
|
expect(query).not.toContain('请补全 CREATE TRIGGER 语句');
|
|
});
|
|
|
|
it('reloads the latest trigger definition before opening object edit', async () => {
|
|
backendApp.DBQuery
|
|
.mockResolvedValueOnce({
|
|
success: true,
|
|
data: [{ trigger_definition: 'CREATE TRIGGER users_bi BEFORE INSERT ON audit.users EXECUTE FUNCTION audit.audit_users();' }],
|
|
})
|
|
.mockResolvedValueOnce({
|
|
success: true,
|
|
data: [{ trigger_definition: 'CREATE TRIGGER users_bi BEFORE INSERT OR UPDATE ON audit.users EXECUTE FUNCTION audit.audit_users_v2();' }],
|
|
});
|
|
|
|
let renderer: any;
|
|
await act(async () => {
|
|
renderer = create(<TriggerViewer tab={tab} />);
|
|
await flushPromises();
|
|
});
|
|
|
|
const button = renderer.root.findAll((node: any) => node.type === 'button' && findButtonText(node).includes('对象修改'))[0];
|
|
|
|
await act(async () => {
|
|
await button.props.onClick();
|
|
await flushPromises();
|
|
});
|
|
|
|
expect(backendApp.DBQuery).toHaveBeenCalledTimes(2);
|
|
const query = storeState.addTab.mock.calls[0][0].query;
|
|
expect(query).toContain('CREATE TRIGGER users_bi BEFORE INSERT OR UPDATE ON audit.users');
|
|
expect(query).toContain('audit.audit_users_v2()');
|
|
|
|
const editor = renderer.root.findAll((node: any) => node.props['data-editor'] === 'true')[0];
|
|
expect(String(editor.children.join(''))).toContain('CREATE TRIGGER users_bi BEFORE INSERT OR UPDATE ON audit.users');
|
|
});
|
|
|
|
it('keeps the current trigger definition visible when refresh for object edit fails', async () => {
|
|
backendApp.DBQuery
|
|
.mockResolvedValueOnce({
|
|
success: true,
|
|
data: [{ trigger_definition: 'CREATE TRIGGER users_bi BEFORE INSERT ON audit.users EXECUTE FUNCTION audit.audit_users();' }],
|
|
})
|
|
.mockResolvedValueOnce({
|
|
success: false,
|
|
message: 'refresh failed',
|
|
data: [],
|
|
});
|
|
|
|
let renderer: any;
|
|
await act(async () => {
|
|
renderer = create(<TriggerViewer tab={tab} />);
|
|
await flushPromises();
|
|
});
|
|
|
|
const button = renderer.root.findAll((node: any) => node.type === 'button' && findButtonText(node).includes('对象修改'))[0];
|
|
|
|
await act(async () => {
|
|
await button.props.onClick();
|
|
await flushPromises();
|
|
});
|
|
|
|
expect(storeState.addTab).not.toHaveBeenCalled();
|
|
expect(String(renderer.root.findAll((node: any) => node.props['data-editor'] === 'true')[0].children.join(''))).toContain('CREATE TRIGGER users_bi BEFORE INSERT ON audit.users');
|
|
expect(findButtonText(renderer.root)).toContain('刷新最新定义失败');
|
|
expect(findButtonText(renderer.root)).toContain('refresh failed');
|
|
});
|
|
|
|
it('does not keep the previous trigger definition when switching objects and the new load fails', async () => {
|
|
backendApp.DBQuery
|
|
.mockResolvedValueOnce({
|
|
success: true,
|
|
data: [{ trigger_definition: 'CREATE TRIGGER users_bi BEFORE INSERT ON audit.users EXECUTE FUNCTION audit.audit_users();' }],
|
|
})
|
|
.mockResolvedValueOnce({
|
|
success: false,
|
|
message: 'load failed',
|
|
data: [],
|
|
});
|
|
|
|
let renderer: any;
|
|
await act(async () => {
|
|
renderer = create(<TriggerViewer tab={tab} />);
|
|
await flushPromises();
|
|
});
|
|
|
|
await act(async () => {
|
|
renderer.update(<TriggerViewer tab={{
|
|
...tab,
|
|
id: 'trigger-conn-1-main-audit.users_bu',
|
|
title: '触发器: audit.users_bu',
|
|
triggerName: 'audit.users_bu',
|
|
}} />);
|
|
await flushPromises();
|
|
});
|
|
|
|
expect(findButtonText(renderer.root)).toContain('加载失败');
|
|
expect(findButtonText(renderer.root)).toContain('load failed');
|
|
expect(renderer.root.findAll((node: any) => node.props['data-editor'] === 'true')).toHaveLength(0);
|
|
expect(findButtonText(renderer.root)).not.toContain('CREATE TRIGGER users_bi BEFORE INSERT');
|
|
});
|
|
});
|