/**
* Copyright (c) Microsoft Corporation.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import fs from 'node:fs/promises';
import { expect, test } from './fixtures.js';
test('browser_file_upload', async ({ client, server }, testInfo) => {
server.setContent(
'/',
`
<input type="file" />
<button>Button</button>
`,
'text/html'
);
const navigateResponse = await client.callTool({
name: 'browser_navigate',
arguments: { url: server.PREFIX },
});
expect(navigateResponse).toHaveResponse({
pageState: expect.stringContaining(`- generic [active] [ref=e1]:
- button "Choose File" [ref=e2]
- button "Button" [ref=e3]`),
});
expect(
await client.callTool({
name: 'browser_file_upload',
arguments: { paths: [] },
})
).toHaveResponse({
isError: true,
result: expect.stringContaining(
`The tool "browser_file_upload" can only be used when there is related modal state present.`
),
modalState: expect.stringContaining('- There is no modal state present'),
});
expect(
await client.callTool({
name: 'browser_click',
arguments: {
selectors: [{ ref: 'e2' }],
},
})
).toHaveResponse({
modalState: expect.stringContaining(
`- [File chooser]: can be handled by the "browser_file_upload" tool`
),
});
const filePath = testInfo.outputPath('test.txt');
await fs.writeFile(filePath, 'Hello, world!');
const response = await client.callTool({
name: 'browser_file_upload',
arguments: {
paths: [filePath],
},
});
expect(response).toHaveResponse({
code: expect.stringContaining('await fileChooser.setFiles('),
modalState: undefined,
});
const response2 = await client.callTool({
name: 'browser_click',
arguments: {
selectors: [{ ref: 'e2' }],
},
});
expect(response2).toHaveResponse({
modalState: `- [File chooser]: can be handled by the "browser_file_upload" tool`,
});
const response3 = await client.callTool({
name: 'browser_click',
arguments: {
selectors: [{ ref: 'e3' }],
},
});
expect(response3).toHaveResponse({
result: `Error: Tool "browser_click" does not handle the modal state.`,
modalState: expect.stringContaining(
`- [File chooser]: can be handled by the "browser_file_upload" tool`
),
});
});
test('clicking on download link emits download', async ({
startClient,
server,
}, testInfo) => {
const { client } = await startClient({
config: { outputDir: testInfo.outputPath('output') },
});
server.setContent(
'/',
`<a href="/download" download="test.txt">Download</a>`,
'text/html'
);
server.setContent('/download', 'Data', 'text/plain');
expect(
await client.callTool({
name: 'browser_navigate',
arguments: { url: server.PREFIX },
})
).toHaveResponse({
pageState: expect.stringContaining(`- link "Download" [ref=e2]`),
});
await client.callTool({
name: 'browser_click',
arguments: {
selectors: [{ ref: 'e2' }],
},
});
await expect
.poll(() =>
client.callTool({
name: 'browser_snapshot',
arguments: {
expectation: { includeDownloads: true },
},
})
)
.toHaveResponse({
downloads: `- Downloaded file test.txt to ${testInfo.outputPath('output', 'test.txt')}`,
});
});
test('navigating to download link emits download', async ({
startClient,
server,
mcpBrowser,
}, testInfo) => {
const { client } = await startClient({
config: { outputDir: testInfo.outputPath('output') },
});
test.skip(mcpBrowser !== 'chromium', 'This test is racy');
server.route('/download', (_req, res) => {
res.writeHead(200, {
'Content-Type': 'text/plain',
'Content-Disposition': 'attachment; filename=test.txt',
});
res.end('Hello world!');
});
expect(
await client.callTool({
name: 'browser_navigate',
arguments: {
url: `${server.PREFIX}download`,
},
})
).toHaveResponse({
downloads: expect.stringContaining('- Downloaded file test.txt to'),
});
});