Files
moreminimore-vibe/e2e-tests/app_search.spec.ts
Kunthawat Greethong d22227bb13 feat: implement fuzzy search and replace functionality with Levenshtein distance
- Added `applySearchReplace` function to handle search and replace operations with fuzzy matching capabilities.
- Introduced tests for various scenarios including fuzzy matching with typos, exact matches, and handling whitespace differences.
- Created a parser for search/replace blocks to facilitate the new functionality.
- Updated prompts for search-replace operations to clarify usage and examples.
- Added utility functions for text normalization and language detection based on file extensions.
- Implemented a minimal stdio MCP server for local testing with tools for adding numbers and printing environment variables.
2025-12-05 11:28:57 +07:00

292 lines
9.4 KiB
TypeScript

import { test } from "./helpers/test_helper";
test("app search - basic search dialog functionality", async ({ po }) => {
await po.setUp({ autoApprove: true });
await po.goToAppsTab();
await po.page.getByTestId("search-apps-button").waitFor();
// Create some apps for testing
await po.sendPrompt("create a todo application");
// Go back to apps list
await po.goToAppsTab();
await po.page.getByTestId("search-apps-button").waitFor();
// Create second app
await po.sendPrompt("build a weather dashboard");
// Go back to apps list
await po.goToAppsTab();
await po.page.getByTestId("search-apps-button").waitFor();
// Create third app
await po.sendPrompt("create a blog system");
// Go back to apps list
await po.goToAppsTab();
await po.page.getByTestId("search-apps-button").waitFor();
// Test 1: Open search dialog using the search button
const searchButton = po.page.getByTestId("search-apps-button");
await searchButton.click();
// Wait for search dialog to appear
const dialog = po.page.getByTestId("app-search-dialog");
await dialog.waitFor({ state: "visible", timeout: 10000 });
// Test 2: Close dialog with Ctrl+K (shortcut toggles)
await po.page.keyboard.press("Control+k");
await dialog.waitFor({ state: "hidden", timeout: 5000 });
// Test 3: Open dialog again with Ctrl+K (shortcut toggles)
await po.page.keyboard.press("Control+k");
await dialog.waitFor({ state: "visible", timeout: 10000 });
// Test 4: Search for specific term
await po.page.getByPlaceholder("Search apps").fill("app");
await po.page.waitForTimeout(500);
// Test 5: Clear search and close with Escape
await po.page.getByPlaceholder("Search apps").clear();
await po.page.keyboard.press("Escape");
await dialog.waitFor({ state: "hidden", timeout: 5000 });
});
test("app search - search functionality with different terms", async ({
po,
}) => {
await po.setUp({ autoApprove: true });
// Create apps with specific content for testing
await po.sendPrompt("create a calculator application with advanced features");
await po.goToAppsTab();
await po.sendPrompt("build a task management system with priority levels");
await po.goToAppsTab();
await po.sendPrompt("create a weather monitoring dashboard");
await po.goToAppsTab();
// Open search dialog
await po.page.getByTestId("search-apps-button").click();
await po.page.getByTestId("app-search-dialog").waitFor();
// Search for "calculator" - should find the calculator app through chat content
await po.page.getByPlaceholder("Search apps").fill("calculator");
await po.page.waitForTimeout(500);
// Search for "task" - should find the task management app
await po.page.getByPlaceholder("Search apps").fill("task");
await po.page.waitForTimeout(500);
// Search for "weather" - should find the weather dashboard
await po.page.getByPlaceholder("Search apps").fill("weather");
await po.page.waitForTimeout(500);
// Search for non-existent term
await po.page.getByPlaceholder("Search apps").fill("nonexistent");
await po.page.waitForTimeout(500);
// Should show empty state
await po.page.getByTestId("app-search-empty").waitFor();
await po.page.keyboard.press("Escape");
});
test("app search - keyboard shortcut functionality", async ({ po }) => {
await po.setUp({ autoApprove: true });
// Create an app first
await po.sendPrompt("create sample application");
await po.goToAppsTab();
// Test keyboard shortcut (Ctrl+K) to open dialog
await po.page.keyboard.press("Control+k");
await po.page.getByTestId("app-search-dialog").waitFor();
// Close with escape
await po.page.keyboard.press("Escape");
await po.page.getByTestId("app-search-dialog").waitFor({ state: "hidden" });
// Test keyboard shortcut again
await po.page.keyboard.press("Control+k");
await po.page.getByTestId("app-search-dialog").waitFor();
// Close with Ctrl+K (toggle)
await po.page.keyboard.press("Control+k");
await po.page.getByTestId("app-search-dialog").waitFor({ state: "hidden" });
});
test("app search - navigation and selection", async ({ po }) => {
await po.setUp({ autoApprove: true });
// Create multiple apps
await po.sendPrompt("create first application");
await po.goToAppsTab();
await po.sendPrompt("create second application");
await po.goToAppsTab();
await po.sendPrompt("create third application");
await po.goToAppsTab();
// Open search dialog
await po.page.getByTestId("search-apps-button").click();
await po.page.getByTestId("app-search-dialog").waitFor();
// Get all app items in the search results
const searchItems = await po.page.getByTestId(/^app-search-item-/).all();
if (searchItems.length > 0) {
// Click on the first search result
await searchItems[0].click();
// Dialog should close after selection
await po.page.getByTestId("app-search-dialog").waitFor({ state: "hidden" });
// Should navigate to the selected app
await po.page.waitForTimeout(1000);
} else {
// If no items found, just close the dialog
await po.page.keyboard.press("Escape");
}
});
test("app search - empty search shows all apps", async ({ po }) => {
await po.setUp({ autoApprove: true });
// Create a few apps
await po.sendPrompt("create alpha application");
await po.goToAppsTab();
await po.sendPrompt("create beta application");
await po.goToAppsTab();
await po.sendPrompt("create gamma application");
await po.goToAppsTab();
// Open search dialog
await po.page.getByTestId("search-apps-button").click();
await po.page.getByTestId("app-search-dialog").waitFor();
// Clear any existing search (should show all apps)
await po.page.getByPlaceholder("Search apps").clear();
await po.page.waitForTimeout(500);
// Should show all apps in the list
const searchItems = await po.page.getByTestId(/^app-search-item-/).all();
console.log(`Found ${searchItems.length} apps in search results`);
await po.page.keyboard.press("Escape");
});
test("app search - case insensitive search", async ({ po }) => {
await po.setUp({ autoApprove: true });
// Create an app with mixed case content
await po.sendPrompt("create a Test Application with Mixed Case Content");
await po.goToAppsTab();
// Open search dialog
await po.page.getByTestId("search-apps-button").click();
await po.page.getByTestId("app-search-dialog").waitFor();
// Search with different cases
await po.page.getByPlaceholder("Search apps").fill("test");
await po.page.waitForTimeout(500);
await po.page.getByPlaceholder("Search apps").fill("TEST");
await po.page.waitForTimeout(500);
await po.page.getByPlaceholder("Search apps").fill("Test");
await po.page.waitForTimeout(500);
await po.page.getByPlaceholder("Search apps").fill("MIXED");
await po.page.waitForTimeout(500);
await po.page.keyboard.press("Escape");
});
test("app search - partial word matching", async ({ po }) => {
await po.setUp({ autoApprove: true });
// Create an app with a long descriptive name
await po.sendPrompt("create a comprehensive project management solution");
await po.goToAppsTab();
// Open search dialog
await po.page.getByTestId("search-apps-button").click();
await po.page.getByTestId("app-search-dialog").waitFor();
// Search with partial words
await po.page.getByPlaceholder("Search apps").fill("proj");
await po.page.waitForTimeout(500);
await po.page.getByPlaceholder("Search apps").fill("manage");
await po.page.waitForTimeout(500);
await po.page.getByPlaceholder("Search apps").fill("comp");
await po.page.waitForTimeout(500);
await po.page.getByPlaceholder("Search apps").fill("sol");
await po.page.waitForTimeout(500);
await po.page.keyboard.press("Escape");
});
test("app search - search by app name", async ({ po }) => {
await po.setUp({ autoApprove: true });
// Create apps - note that app names are randomly generated
await po.sendPrompt("create a todo application");
await po.goToAppsTab();
await po.sendPrompt("build a weather dashboard");
await po.goToAppsTab();
await po.sendPrompt("create a blog system");
await po.goToAppsTab();
// Get the actual app names from the UI (these are randomly generated)
const appItems = await po.page.getByTestId(/^app-list-item-/).all();
const appNames: string[] = [];
for (const item of appItems) {
const testId = await item.getAttribute("data-testid");
if (testId) {
const appName = testId.replace("app-list-item-", "");
appNames.push(appName);
}
}
// Open search dialog
await po.page.getByTestId("search-apps-button").click();
await po.page.getByTestId("app-search-dialog").waitFor();
// Test searching by actual app names (randomly generated)
if (appNames.length > 0) {
// Search for the first few characters of the first app name
const firstAppName = appNames[0];
const searchTerm = firstAppName.substring(
0,
Math.min(4, firstAppName.length),
);
await po.page.getByPlaceholder("Search apps").fill(searchTerm);
await po.page.waitForTimeout(500);
// Clear and search for second app if available
if (appNames.length > 1) {
await po.page.getByPlaceholder("Search apps").clear();
const secondAppName = appNames[1];
const secondSearchTerm = secondAppName.substring(
0,
Math.min(4, secondAppName.length),
);
await po.page.getByPlaceholder("Search apps").fill(secondSearchTerm);
await po.page.waitForTimeout(500);
}
}
await po.page.keyboard.press("Escape");
});