191 lines
5.0 KiB
TypeScript
Raw Normal View History

import { join } from 'node:path';
import { assert } from '@midscene/shared/utils';
import { randomUUID } from 'node:crypto';
import { existsSync } from 'node:fs';
feat(web-integration): add support for new agent method(aiBoolean, aiString, aiNumber, aiLocate) (#658) * feat(web-integration): add support for new agent method(aiBoolean, aiString, aiNumber, aiLocate) * fix(core, web-integration): update data extraction logic and refine return types for agent methods * feat(site): add detailed descriptions and examples * fix(web-integration): remove unused properties from aiLocate return type and update UI titles * fix(core): enhance agent methods for better extraction and querying * fix(core): update test scripts and refine data extraction prompts * fix(core): refine return type in ExecutionTaskApply and remove unused import in inspect * fix(core): update data extraction logic and enhance type handling in Insight class * fix(web-integration): update import path for puppeteerAgentForTarget in player test file * fix(core): tests fix * fix(core): fix tests * fix(shared): enhance imageInfo tests by validating individual properties and updating snapshots * fix(site): remove waitForNavigationTimeout from Playwright integration examples * fix(site): update timeout configuration examples in FAQ for clarity * test(web-integration): remove unused client-extractor test and related snapshots * chore(core): fix query implementation * fix(report): improve data handling in DetailSide component and enhance output display * fix(core): refine data extraction prompts and improve error handling in PageTaskExecutor * fix(core): update data extraction prompt for clarity and type validation --------- Co-authored-by: yutao <yutao.tao@bytedance.com>
2025-04-29 13:36:49 +08:00
import { puppeteerAgentForTarget } from '@/puppeteer/agent-launcher';
import { ScriptPlayer, buildYaml, parseYamlScript } from '@/yaml';
import type { MidsceneYamlScriptWebEnv } from '@midscene/core';
import { describe, expect, test, vi } from 'vitest';
const serverRoot = join(__dirname, 'server_root');
const runYaml = async (yamlString: string, ignoreStatusAssertion = false) => {
const script = parseYamlScript(yamlString);
const statusUpdate = vi.fn();
const player = new ScriptPlayer<MidsceneYamlScriptWebEnv>(
script,
puppeteerAgentForTarget,
statusUpdate,
);
await player.run();
if (!ignoreStatusAssertion) {
assert(
player.status === 'done',
player.errorInSetup?.message || 'unknown error',
);
expect(statusUpdate).toHaveBeenCalled();
}
return {
player,
statusUpdate,
};
};
const shouldRunAITest =
process.platform !== 'linux' || process.env.AITEST === 'true';
describe('yaml utils', () => {
test('basic build && load', () => {
const script = buildYaml(
{
url: 'https://www.baidu.com',
waitForNetworkIdle: {
timeout: 1000,
continueOnNetworkIdleError: true,
},
},
[
{
name: 'search',
flow: [
{
aiAction: 'type "hello" in search box, hit enter',
},
],
},
],
);
expect(script).toMatchSnapshot();
const loadedScript = parseYamlScript(script);
expect(loadedScript).toMatchSnapshot();
});
test('load error with filePath', () => {
expect(() => {
parseYamlScript(
`
target:
a: 1
`,
'some_error_path',
);
}).toThrow(/some_error_path/);
});
test('player - bad params', async () => {
expect(async () => {
await runYaml(`
target:
serve: ${serverRoot}
`);
}).rejects.toThrow();
expect(async () => {
await runYaml(`
target:
serve: ${serverRoot}
viewportWidth: 0
`);
}).rejects.toThrow();
});
});
describe.skipIf(!shouldRunAITest)(
'player - e2e',
() => {
test('flush output even if assertion failed', async () => {
const outputPath = `./midscene_run/output/${randomUUID()}.json`;
const yamlString = `
target:
url: https://www.bing.com
output: ${outputPath}
tasks:
- name: local page
flow:
- aiQuery: >
the background color of the page, { color: 'white' | 'black' | 'red' | 'green' | 'blue' | 'yellow' | 'purple' | 'orange' | 'pink' | 'brown' | 'gray' | 'black'
- name: check content
flow:
- aiAssert: this is a food delivery service app
`;
await expect(async () => {
await runYaml(yamlString);
}).rejects.toThrow();
expect(existsSync(outputPath)).toBe(true);
});
test('cookie', async () => {
const yamlString = `
target:
url: http://httpbin.dev/cookies
cookie: ./tests/unit-test/fixtures/cookie/httpbin.dev_cookies.json
tasks:
- name: check cookie
flow:
- aiAssert: the value of midscene_foo is "bar"
`;
await runYaml(yamlString);
});
test('online server - lazy response', async () => {
const yamlString = `
target:
url: https://httpbin.org/delay/60000
waitForNetworkIdle:
timeout: 10
continueOnNetworkIdleError: false
tasks:
- name: check content
flow:
- aiAssert: the response is "Hello, world!"
`;
expect(async () => {
await runYaml(yamlString);
}).rejects.toThrow(/TimeoutError/i);
});
test('stop on task error', async () => {
const yamlString = `
target:
url: https://www.baidu.com
tasks:
- name: assert1
flow:
- aiAssert: this is a food delivery service app
- name: assert2
flow:
- aiAssert: this is a search engine
`;
const { player } = await runYaml(yamlString, true);
expect(player.status).toBe('error');
expect(player.taskStatusList[0].status).toBe('error');
expect(player.taskStatusList[1].status).toBe('init');
});
test('allow continue on task error', async () => {
const yamlString = `
target:
url: https://www.baidu.com
tasks:
- name: assert1
continueOnError: true
flow:
- aiAssert: this is a food delivery service app
- name: assert2
flow:
- aiAssert: this is a search engine
`;
const { player } = await runYaml(yamlString, true);
expect(player.status).toBe('done');
expect(player.taskStatusList[0].status).toBe('error');
expect(player.taskStatusList[1].status).toBe('done');
});
},
60 * 1000,
);