mirror of
https://github.com/Azure/cosmos-explorer.git
synced 2025-12-27 21:01:57 +00:00
case1 working
This commit is contained in:
@@ -1,4 +1,4 @@
|
||||
import { expect, test } from "@playwright/test";
|
||||
import { expect, test, type Page } from "@playwright/test";
|
||||
|
||||
import { DataExplorer, TestAccount } from "../fx";
|
||||
|
||||
@@ -7,54 +7,208 @@ import { DataExplorer, TestAccount } from "../fx";
|
||||
const DATABASE_ID = process.env.INDEX_ADVISOR_TEST_DATABASE || "t_db05_1765364190570";
|
||||
const CONTAINER_ID = process.env.INDEX_ADVISOR_TEST_CONTAINER || "testcontainer";
|
||||
|
||||
test("Index Advisor tab loads without errors", async ({ page }) => {
|
||||
// Mock SDK response structure based on IndexMetricsResponse interface
|
||||
const mockIndexMetricsWithSingleUtilized = {
|
||||
UtilizedIndexes: {
|
||||
SingleIndexes: [
|
||||
{ IndexSpec: "/_partitionKey/?", IndexImpactScore: "High" },
|
||||
{ IndexSpec: "/name/?", IndexImpactScore: "Medium" },
|
||||
],
|
||||
},
|
||||
PotentialIndexes: {},
|
||||
};
|
||||
|
||||
const mockIndexMetricsWithCompositeUtilized = {
|
||||
UtilizedIndexes: {
|
||||
CompositeIndexes: [
|
||||
{
|
||||
IndexSpecs: ["/category asc", "/price desc"],
|
||||
IndexImpactScore: "High",
|
||||
},
|
||||
],
|
||||
},
|
||||
PotentialIndexes: {},
|
||||
};
|
||||
|
||||
const mockIndexMetricsWithSinglePotential = {
|
||||
UtilizedIndexes: {},
|
||||
PotentialIndexes: {
|
||||
SingleIndexes: [
|
||||
{ IndexSpec: "/email/?", IndexImpactScore: "High" },
|
||||
{ IndexSpec: "/status/?", IndexImpactScore: "Medium" },
|
||||
{ IndexSpec: "/createdDate/?", IndexImpactScore: "Low" },
|
||||
],
|
||||
},
|
||||
};
|
||||
|
||||
const mockIndexMetricsWithCompositePotential = {
|
||||
UtilizedIndexes: {},
|
||||
PotentialIndexes: {
|
||||
CompositeIndexes: [
|
||||
{
|
||||
IndexSpecs: ["/userId asc", "/timestamp desc"],
|
||||
IndexImpactScore: "High",
|
||||
},
|
||||
{
|
||||
IndexSpecs: ["/country asc", "/city asc", "/zipCode asc"],
|
||||
IndexImpactScore: "Medium",
|
||||
},
|
||||
],
|
||||
},
|
||||
};
|
||||
|
||||
const mockIndexMetricsComplete = {
|
||||
UtilizedIndexes: {
|
||||
SingleIndexes: [
|
||||
{ IndexSpec: "/_partitionKey/?", IndexImpactScore: "High" },
|
||||
{ IndexSpec: "/name/?", IndexImpactScore: "Medium" },
|
||||
],
|
||||
CompositeIndexes: [
|
||||
{
|
||||
IndexSpecs: ["/category asc", "/price desc"],
|
||||
IndexImpactScore: "High",
|
||||
},
|
||||
],
|
||||
},
|
||||
PotentialIndexes: {
|
||||
SingleIndexes: [
|
||||
{ IndexSpec: "/email/?", IndexImpactScore: "High" },
|
||||
{ IndexSpec: "/status/?", IndexImpactScore: "Medium" },
|
||||
{ IndexSpec: "/createdDate/?", IndexImpactScore: "Low" },
|
||||
],
|
||||
CompositeIndexes: [
|
||||
{
|
||||
IndexSpecs: ["/userId asc", "/timestamp desc"],
|
||||
IndexImpactScore: "High",
|
||||
},
|
||||
{
|
||||
IndexSpecs: ["/country asc", "/city asc", "/zipCode asc"],
|
||||
IndexImpactScore: "Medium",
|
||||
},
|
||||
],
|
||||
},
|
||||
};
|
||||
|
||||
// Helper function to intercept SDK calls and inject mock response
|
||||
async function interceptIndexMetrics(
|
||||
page: Page,
|
||||
mockResponse: any,
|
||||
): Promise<void> {
|
||||
await page.route("**/dbs/*/colls/*/docs", async (route) => {
|
||||
const request = route.request();
|
||||
|
||||
// Check if this is a query request with populateIndexMetrics
|
||||
if (request.method() === "POST") {
|
||||
const postData = request.postData();
|
||||
|
||||
if (postData) {
|
||||
try {
|
||||
const body = JSON.parse(postData);
|
||||
|
||||
// If this is a query request, we'll intercept it
|
||||
if (body.query) {
|
||||
// Fetch the actual response
|
||||
const response = await route.fetch();
|
||||
const responseBody = await response.json();
|
||||
|
||||
// Add our mock indexMetrics to the response
|
||||
const modifiedResponse = {
|
||||
...responseBody,
|
||||
indexMetrics: mockResponse ? JSON.stringify(mockResponse) : undefined,
|
||||
};
|
||||
|
||||
await route.fulfill({
|
||||
status: response.status(),
|
||||
headers: response.headers(),
|
||||
body: JSON.stringify(modifiedResponse),
|
||||
});
|
||||
return;
|
||||
}
|
||||
} catch (e) {
|
||||
// If parsing fails, continue with normal route
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
await route.continue();
|
||||
});
|
||||
}
|
||||
|
||||
// Helper function to set up query tab and navigate to Index Advisor
|
||||
async function setupIndexAdvisorTab(page: Page, mockResponse?: any) {
|
||||
if (mockResponse !== undefined) {
|
||||
await interceptIndexMetrics(page, mockResponse);
|
||||
}
|
||||
|
||||
const explorer = await DataExplorer.open(page, TestAccount.SQL);
|
||||
|
||||
// Wait for the database node
|
||||
const databaseNode = await explorer.waitForNode(DATABASE_ID);
|
||||
console.log("Database node found");
|
||||
|
||||
// Expand the database
|
||||
await databaseNode.expand();
|
||||
console.log("Database expanded");
|
||||
|
||||
// Wait a moment for container to appear
|
||||
await page.waitForTimeout(2000);
|
||||
|
||||
// Wait for the container node
|
||||
const containerNode = await explorer.waitForNode(`${DATABASE_ID}/${CONTAINER_ID}`);
|
||||
console.log("Container node found");
|
||||
|
||||
// Click on "New SQL Query" from the container's context menu
|
||||
const containerNode = await explorer.waitForNode(`${DATABASE_ID}/${CONTAINER_ID}`);
|
||||
await containerNode.openContextMenu();
|
||||
await containerNode.contextMenuItem("New SQL Query").click();
|
||||
await page.waitForTimeout(2000);
|
||||
|
||||
// Wait for the query tab to fully load
|
||||
await page.waitForTimeout(3000);
|
||||
|
||||
// Wait for the query tab to load
|
||||
const queryTab = explorer.queryTab("tab0");
|
||||
await queryTab.editor().locator.waitFor({ timeout: 30 * 1000 });
|
||||
await queryTab.executeCTA.waitFor();
|
||||
|
||||
// Click on the specific query tab (tab0) to make sure it's active
|
||||
const queryTabHeader = explorer.frame.getByRole("tab", { name: "Query 1" });
|
||||
await queryTabHeader.click();
|
||||
await page.waitForTimeout(1000);
|
||||
|
||||
// Click in the editor and execute the query
|
||||
await queryTab.editor().locator.click();
|
||||
const executeQueryButton = explorer.commandBarButton("Execute Query");
|
||||
await executeQueryButton.waitFor({ state: "visible", timeout: 10000 });
|
||||
await executeQueryButton.click();
|
||||
|
||||
// Wait for results to load
|
||||
const executeQueryButton = explorer.commandBarButton("Execute Query");
|
||||
await executeQueryButton.click();
|
||||
await expect(queryTab.resultsEditor.locator).toBeAttached({ timeout: 60 * 1000 });
|
||||
|
||||
// Click on the Index Advisor tab
|
||||
const indexAdvisorTab = queryTab.resultsView.getByTestId("QueryTab/ResultsPane/ResultsView/IndexAdvisorTab");
|
||||
await indexAdvisorTab.click();
|
||||
await page.waitForTimeout(2000);
|
||||
|
||||
// Verify the Index Advisor tab is visible and loaded
|
||||
return { explorer, queryTab, indexAdvisorTab };
|
||||
}
|
||||
|
||||
test("Index Advisor tab loads without errors", async ({ page }) => {
|
||||
const { indexAdvisorTab } = await setupIndexAdvisorTab(page, mockIndexMetricsComplete);
|
||||
await expect(indexAdvisorTab).toHaveAttribute("aria-selected", "true");
|
||||
});
|
||||
});
|
||||
|
||||
test("Verify UI sections are collapsible", async ({ page }) => {
|
||||
const { explorer } = await setupIndexAdvisorTab(page, mockIndexMetricsComplete);
|
||||
|
||||
// Verify both section headers exist
|
||||
const includedHeader = explorer.frame.getByText("Included in Current Policy", { exact: true });
|
||||
const notIncludedHeader = explorer.frame.getByText("Not Included in Current Policy", { exact: true });
|
||||
|
||||
await expect(includedHeader).toBeVisible();
|
||||
await expect(notIncludedHeader).toBeVisible();
|
||||
|
||||
// Test collapsibility by checking if chevron/arrow icon changes state
|
||||
// Both sections should be expandable/collapsible regardless of content
|
||||
await includedHeader.click();
|
||||
await page.waitForTimeout(300);
|
||||
await includedHeader.click();
|
||||
await page.waitForTimeout(300);
|
||||
|
||||
await notIncludedHeader.click();
|
||||
await page.waitForTimeout(300);
|
||||
await notIncludedHeader.click();
|
||||
await page.waitForTimeout(300);
|
||||
});
|
||||
|
||||
test("Verify SDK response structure - Case 1: Empty response", async ({ page }) => {
|
||||
const { explorer } = await setupIndexAdvisorTab(page, {
|
||||
UtilizedIndexes: {},
|
||||
PotentialIndexes: {},
|
||||
});
|
||||
|
||||
// Verify both section headers still exist even with no data
|
||||
await expect(explorer.frame.getByText("Included in Current Policy", { exact: true })).toBeVisible();
|
||||
await expect(explorer.frame.getByText("Not Included in Current Policy", { exact: true })).toBeVisible();
|
||||
|
||||
// Verify table headers
|
||||
const table = explorer.frame.locator("table");
|
||||
await expect(table.getByText("Index", { exact: true })).toBeVisible();
|
||||
await expect(table.getByText("Estimated Impact", { exact: true })).toBeVisible();
|
||||
|
||||
// Verify "Update Indexing Policy" button is NOT visible when there are no potential indexes
|
||||
const updateButton = explorer.frame.getByRole("button", { name: /Update Indexing Policy/i });
|
||||
await expect(updateButton).not.toBeVisible();
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user