UNPKG

@executeautomation/playwright-mcp-server

Version:
203 lines (202 loc) 8.6 kB
import { ClickTool, FillTool, SelectTool, HoverTool, EvaluateTool, IframeClickTool } from '../../../tools/browser/interaction.js'; import { NavigationTool } from '../../../tools/browser/navigation.js'; import { jest } from '@jest/globals'; // Mock page functions const mockPageClick = jest.fn().mockImplementation(() => Promise.resolve()); const mockPageFill = jest.fn().mockImplementation(() => Promise.resolve()); const mockPageSelectOption = jest.fn().mockImplementation(() => Promise.resolve()); const mockPageHover = jest.fn().mockImplementation(() => Promise.resolve()); const mockPageWaitForSelector = jest.fn().mockImplementation(() => Promise.resolve()); // Mock locator functions const mockLocatorClick = jest.fn().mockImplementation(() => Promise.resolve()); const mockLocatorFill = jest.fn().mockImplementation(() => Promise.resolve()); const mockLocatorSelectOption = jest.fn().mockImplementation(() => Promise.resolve()); const mockLocatorHover = jest.fn().mockImplementation(() => Promise.resolve()); // Mock locator const mockLocator = jest.fn().mockReturnValue({ click: mockLocatorClick, fill: mockLocatorFill, selectOption: mockLocatorSelectOption, hover: mockLocatorHover }); // Mock iframe locator const mockIframeLocator = jest.fn().mockReturnValue({ click: mockLocatorClick }); // Mock frame locator const mockFrameLocator = jest.fn().mockReturnValue({ locator: mockIframeLocator }); // Mock evaluate function const mockEvaluate = jest.fn().mockImplementation(() => Promise.resolve('test-result')); // Mock the Page object with proper typing const mockGoto = jest.fn().mockImplementation(() => Promise.resolve()); const mockPage = { click: mockPageClick, fill: mockPageFill, selectOption: mockPageSelectOption, hover: mockPageHover, waitForSelector: mockPageWaitForSelector, locator: mockLocator, frameLocator: mockFrameLocator, evaluate: mockEvaluate, goto: mockGoto }; // Mock the server const mockServer = { sendMessage: jest.fn() }; // Mock context const mockContext = { page: mockPage, server: mockServer }; describe('Browser Interaction Tools', () => { let clickTool; let fillTool; let selectTool; let hoverTool; let evaluateTool; let iframeClickTool; beforeEach(() => { jest.clearAllMocks(); clickTool = new ClickTool(mockServer); fillTool = new FillTool(mockServer); selectTool = new SelectTool(mockServer); hoverTool = new HoverTool(mockServer); evaluateTool = new EvaluateTool(mockServer); iframeClickTool = new IframeClickTool(mockServer); }); describe('ClickTool', () => { test('should click an element', async () => { const args = { selector: '#test-button' }; const result = await clickTool.execute(args, mockContext); // The actual implementation uses page.click directly, not locator expect(mockPageClick).toHaveBeenCalledWith('#test-button'); expect(result.isError).toBe(false); expect(result.content[0].text).toContain('Clicked element'); }); test('should handle click errors', async () => { const args = { selector: '#test-button' }; // Mock a click error mockPageClick.mockImplementationOnce(() => Promise.reject(new Error('Click failed'))); const result = await clickTool.execute(args, mockContext); expect(mockPageClick).toHaveBeenCalledWith('#test-button'); expect(result.isError).toBe(true); expect(result.content[0].text).toContain('Operation failed'); }); test('should handle missing page', async () => { const args = { selector: '#test-button' }; const result = await clickTool.execute(args, { server: mockServer }); expect(mockPageClick).not.toHaveBeenCalled(); expect(result.isError).toBe(true); expect(result.content[0].text).toContain('Browser page not initialized'); }); }); describe('IframeClickTool', () => { test('should click an element in an iframe', async () => { const args = { iframeSelector: '#test-iframe', selector: '#test-button' }; const result = await iframeClickTool.execute(args, mockContext); expect(mockFrameLocator).toHaveBeenCalledWith('#test-iframe'); expect(mockIframeLocator).toHaveBeenCalledWith('#test-button'); expect(mockLocatorClick).toHaveBeenCalled(); expect(result.isError).toBe(false); expect(result.content[0].text).toContain('Clicked element'); }); }); describe('FillTool', () => { test('should fill an input field', async () => { const args = { selector: '#test-input', value: 'test value' }; const result = await fillTool.execute(args, mockContext); expect(mockPageWaitForSelector).toHaveBeenCalledWith('#test-input'); expect(mockPageFill).toHaveBeenCalledWith('#test-input', 'test value'); expect(result.isError).toBe(false); expect(result.content[0].text).toContain('Filled'); }); }); describe('SelectTool', () => { test('should select an option', async () => { const args = { selector: '#test-select', value: 'option1' }; const result = await selectTool.execute(args, mockContext); expect(mockPageWaitForSelector).toHaveBeenCalledWith('#test-select'); expect(mockPageSelectOption).toHaveBeenCalledWith('#test-select', 'option1'); expect(result.isError).toBe(false); expect(result.content[0].text).toContain('Selected'); }); }); describe('HoverTool', () => { test('should hover over an element', async () => { const args = { selector: '#test-element' }; const result = await hoverTool.execute(args, mockContext); expect(mockPageWaitForSelector).toHaveBeenCalledWith('#test-element'); expect(mockPageHover).toHaveBeenCalledWith('#test-element'); expect(result.isError).toBe(false); expect(result.content[0].text).toContain('Hovered'); }); }); describe('EvaluateTool', () => { test('should evaluate JavaScript', async () => { const args = { script: 'return document.title' }; const result = await evaluateTool.execute(args, mockContext); expect(mockEvaluate).toHaveBeenCalledWith('return document.title'); expect(result.isError).toBe(false); expect(result.content[0].text).toContain('Executed JavaScript'); }); }); }); describe('NavigationTool', () => { let navigationTool; beforeEach(() => { jest.clearAllMocks(); navigationTool = new NavigationTool(mockServer); }); test('should navigate to a URL', async () => { const args = { url: 'https://example.com', waitUntil: 'networkidle' }; const result = await navigationTool.execute(args, mockContext); expect(mockGoto).toHaveBeenCalledWith('https://example.com', { waitUntil: 'networkidle', timeout: 30000 }); expect(result.isError).toBe(false); expect(result.content[0].text).toContain('Navigated to'); }); test('should handle navigation errors', async () => { const args = { url: 'https://example.com' }; // Mock a navigation error mockGoto.mockImplementationOnce(() => Promise.reject(new Error('Navigation failed'))); const result = await navigationTool.execute(args, mockContext); expect(mockGoto).toHaveBeenCalledWith('https://example.com', { waitUntil: 'load', timeout: 30000 }); expect(result.isError).toBe(true); expect(result.content[0].text).toContain('Operation failed'); }); test('should handle missing page', async () => { const args = { url: 'https://example.com' }; const result = await navigationTool.execute(args, { server: mockServer }); expect(mockGoto).not.toHaveBeenCalled(); expect(result.isError).toBe(true); expect(result.content[0].text).toContain('Browser page not initialized'); }); });