diff --git a/.gitignore b/.gitignore index 7c78742..11ffffc 100644 --- a/.gitignore +++ b/.gitignore @@ -1,2 +1,4 @@ .vscode node_modules +test-results +output \ No newline at end of file diff --git a/podman-desktop-extension/tests/src/image-checker.spec.ts b/podman-desktop-extension/tests/src/image-checker.spec.ts new file mode 100644 index 0000000..95e7abf --- /dev/null +++ b/podman-desktop-extension/tests/src/image-checker.spec.ts @@ -0,0 +1,128 @@ +/********************************************************************** + * Copyright (C) 2025 Red Hat, Inc. + * + * 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. + * + * SPDX-License-Identifier: Apache-2.0 + ***********************************************************************/ + +import type { NavigationBar } from '@podman-desktop/tests-playwright'; +import { + expect as playExpect, + ExtensionCardPage, + RunnerOptions, + test, +} from '@podman-desktop/tests-playwright'; +import { ImageCheckerExtensionPage } from './pages/image-checker-extension-page'; +import { ImageCheckerImageDetailsPage } from './pages/image-checker-image-details-page'; + +let extensionInstalled = false; +let extensionCard: ExtensionCardPage; +const imageName = 'ghcr.io/redhat-developer/podman-desktop-image-checker-openshift-ext:latest'; +const extensionLabel = 'redhat.openshift-checker'; +const extensionLabelName = 'openshift-checker'; +const activeExtensionStatus = 'ACTIVE'; +const disabledExtensionStatus = 'DISABLED'; +const imageToCheck = 'ghcr.io/linuxcontainers/alpine'; +test.use({ + runnerOptions: new RunnerOptions({ customFolder: 'image-checker-tests-pd', autoUpdate: false, autoCheckUpdates: false }), +}); +test.beforeAll(async ({ runner, page, welcomePage }) => { + runner.setVideoAndTraceName('image-checker-e2e'); + await welcomePage.handleWelcomePage(true); + extensionCard = new ExtensionCardPage(page, extensionLabelName, extensionLabel); +}); + +test.afterAll(async ({ runner }) => { + await runner.close(); +}); + +test.describe.serial('Red Hat Image Checker extension installation', () => { + test('Check if extension is already installed', async ({ navigationBar }) => { + const extensions = await navigationBar.openExtensions(); + if (await extensions.extensionIsInstalled(extensionLabel)) { + extensionInstalled = true; + } + }); + + test('Remove old version of the extension', async ({ navigationBar }) => { + test.skip(!extensionInstalled); + await removeExtension(navigationBar); + }); + + test('Extension can be installed from an OCI image', async ({ navigationBar }) => { + test.setTimeout(180000); + const extensions = await navigationBar.openExtensions(); + await extensions.installExtensionFromOCIImage(imageName); + await playExpect(extensionCard.card).toBeVisible(); + }); + + test('Extension is installed and active, extension card is present', async ({ navigationBar }) => { + const extensions = await navigationBar.openExtensions(); + await playExpect + .poll(async () => await extensions.extensionIsInstalled(extensionLabel), { timeout: 60000 }) + .toBeTruthy(); + const extensionCard = await extensions.getInstalledExtension(extensionLabelName, extensionLabel); + await playExpect(extensionCard.status, `Extension status is: ${extensionCard.status}`).toHaveText(activeExtensionStatus); + }); + + test("Extension details show correct status, no error", async ({ page, navigationBar }) => { + const extensions = await navigationBar.openExtensions(); + const extensionCard = await extensions.getInstalledExtension(extensionLabelName, extensionLabel); + await extensionCard.openExtensionDetails('Red Hat OpenShift Checker extension'); + const detailsPage = new ImageCheckerExtensionPage(page); + await playExpect(detailsPage.heading).toBeVisible(); + await playExpect(detailsPage.status).toHaveText(activeExtensionStatus); + const errorTab = detailsPage.tabs.getByRole('button', { name: 'Error' }); + // we would like to propagate the error's stack trace into test failure message + let stackTrace = ''; + if ((await errorTab.count()) > 0) { + await detailsPage.activateTab('Error'); + stackTrace = await detailsPage.errorStackTrace.innerText(); + } + await playExpect(errorTab, `Error Tab was present with stackTrace: ${stackTrace}`).not.toBeVisible(); + }); + + test('Image Checker tab is present in Image Details Page', async ({ navigationBar }) => { + await imageCheckerTabPresent(navigationBar); + }); + + test('Extension can be removed', async ({ navigationBar }) => { + await removeExtension(navigationBar); + }); +}); + +async function removeExtension(navigationBar: NavigationBar): Promise { + const extensions = await navigationBar.openExtensions(); + const extensionCard = await extensions.getInstalledExtension(extensionLabelName, extensionLabel); + await extensionCard.disableExtension(); + await extensionCard.removeExtension(); + await playExpect + .poll(async () => await extensions.extensionIsInstalled(extensionLabel), { timeout: 15000 }) + .toBeFalsy(); +} + +async function imageCheckerTabPresent(navigationBar: NavigationBar) { + const imagesPage = await navigationBar.openImages(); + await playExpect(imagesPage.heading).toBeVisible(); + + const pullImagePage = await imagesPage.openPullImage(); + const updatedImages = await pullImagePage.pullImage(imageToCheck); + + const exists = await updatedImages.waitForImageExists(imageToCheck); + playExpect(exists, `${imageToCheck} image not found in the image list`).toBeTruthy(); + + const imageDetailPage = await imagesPage.openImageDetails(imageToCheck); + const imageDetailsPage = new ImageCheckerImageDetailsPage(imageDetailPage.page, imageToCheck); + await playExpect(imageDetailsPage.imageCheckerTab).toBeVisible(); +} diff --git a/podman-desktop-extension/tests/src/pages/image-checker-extension-page.ts b/podman-desktop-extension/tests/src/pages/image-checker-extension-page.ts new file mode 100644 index 0000000..2dc50c7 --- /dev/null +++ b/podman-desktop-extension/tests/src/pages/image-checker-extension-page.ts @@ -0,0 +1,26 @@ +/********************************************************************** + * Copyright (C) 2025 Red Hat, Inc. + * + * 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. + * + * SPDX-License-Identifier: Apache-2.0 + ***********************************************************************/ + +import type { Page } from '@playwright/test'; +import { ExtensionDetailsPage } from '@podman-desktop/tests-playwright'; + +export class ImageCheckerExtensionPage extends ExtensionDetailsPage { + constructor(page: Page) { + super(page, 'Red Hat OpenShift Checker extension'); + } +} \ No newline at end of file diff --git a/podman-desktop-extension/tests/src/pages/image-checker-image-details-page.ts b/podman-desktop-extension/tests/src/pages/image-checker-image-details-page.ts new file mode 100644 index 0000000..732f5fd --- /dev/null +++ b/podman-desktop-extension/tests/src/pages/image-checker-image-details-page.ts @@ -0,0 +1,29 @@ +/********************************************************************** + * Copyright (C) 2025 Red Hat, Inc. + * + * 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. + * + * SPDX-License-Identifier: Apache-2.0 + ***********************************************************************/ + +import type { Locator, Page } from '@playwright/test'; +import { DetailsPage } from '@podman-desktop/tests-playwright'; + +export class ImageCheckerImageDetailsPage extends DetailsPage { + readonly imageCheckerTab: Locator; + + constructor(page: Page, name: string) { + super(page, name); + this.imageCheckerTab = this.tabs.getByText('Check'); + } +} \ No newline at end of file diff --git a/podman-desktop-extension/tests/tsconfig.json b/podman-desktop-extension/tests/tsconfig.json new file mode 100644 index 0000000..e80b4fe --- /dev/null +++ b/podman-desktop-extension/tests/tsconfig.json @@ -0,0 +1,16 @@ +{ + "compilerOptions": { + "strictNullChecks": true, + "lib": [ "ES2020", "webworker" ], + "module": "esnext", + "target": "esnext", + "sourceMap": true, + "rootDir": "src", + "outDir": "dist", + "skipLibCheck": true, + "types": [ "node" ], + "allowSyntheticDefaultImports": true, + "moduleResolution": "Node", + "esModuleInterop": true + } +}