Skip to content

Commit a8d3613

Browse files
committed
chore(test,ci): enable authentication via browser in e2e tests
Signed-off-by: Ondrej Dockal <[email protected]>
1 parent b590c6e commit a8d3613

File tree

4 files changed

+409
-21
lines changed

4 files changed

+409
-21
lines changed

.github/workflows/e2e-main.yaml

+24-2
Original file line numberDiff line numberDiff line change
@@ -92,7 +92,7 @@ jobs:
9292

9393
- uses: actions/setup-node@v4
9494
with:
95-
node-version: 20
95+
node-version: 22
9696

9797
- name: Update podman v. 5.x
9898
run: |
@@ -111,16 +111,19 @@ jobs:
111111
sudo apt-get update && \
112112
sudo apt-get -y install podman; }
113113
podman version
114+
114115
- name: Revert unprivileged user namespace restrictions in Ubuntu 24.04
115116
run: |
116117
# allow unprivileged user namespace
117118
sudo sysctl -w kernel.apparmor_restrict_unprivileged_userns=0
119+
118120
- name: Build Podman Desktop for E2E tests Development Mode
119121
working-directory: ./podman-desktop
120122
if: ${{ env.MODE == 'development' }}
121123
run: |
122124
pnpm install
123125
pnpm test:e2e:build
126+
124127
- name: Build Podman Desktop for E2E tests Production Mode
125128
working-directory: ./podman-desktop
126129
if: ${{ env.MODE == 'production' }}
@@ -132,6 +135,7 @@ jobs:
132135
path=$(realpath ./dist/linux-unpacked/podman-desktop)
133136
echo "Podman Desktop built binary: $path"
134137
echo "PODMAN_DESKTOP_BINARY_PATH=$path" >> $GITHUB_ENV
138+
135139
- name: Execute pnpm in Red Hat Account Extension
136140
working-directory: ${{ env.REPOSITORY }}
137141
run: |
@@ -140,18 +144,36 @@ jobs:
140144
echo "Version of @podman-desktop/tests-playwright to be used: $version"
141145
jq --arg version "$version" '.devDependencies."@podman-desktop/tests-playwright" = $version' package.json > package.json_tmp && mv package.json_tmp package.json
142146
pnpm install --no-frozen-lockfile
147+
148+
- name: Set default browser
149+
run: |
150+
which google-chrome
151+
xdg-settings set default-web-browser google-chrome.desktop
152+
sudo update-alternatives --set x-www-browser /usr/bin/google-chrome-stable
153+
143154
- name: Run All E2E tests in Red Hat Account Extension in Development Mode
144-
working-directory: ${{ env.REPOSITORY }}
155+
# uses: GabrielBB/[email protected]
145156
if: ${{ env.MODE == 'development' }}
146157
env:
147158
PODMAN_DESKTOP_ARGS: ${{ github.workspace }}/podman-desktop
159+
DVLPR_USERNAME: ${{ secrets.DVLPR_USERNAME }}
160+
DVLPR_PASSWORD: ${{ secrets.DVLPR_PASSWORD }}
161+
# with:
162+
working-directory: ${{ env.REPOSITORY }}
148163
run: pnpm test:e2e
149164

165+
- name: Setup tmate session
166+
if: ${{ failure() }}
167+
uses: mxschmitt/action-tmate@v3
168+
timeout-minutes: 25
169+
150170
- name: Run All E2E tests in Red Hat Account Extension in Production mode
151171
working-directory: ${{ env.REPOSITORY }}
152172
if: ${{ env.MODE == 'production' }}
153173
env:
154174
PODMAN_DESKTOP_BINARY: ${{ env.PODMAN_DESKTOP_BINARY_PATH }}
175+
DVLPR_USERNAME: ${{ secrets.DVLPR_USERNAME }}
176+
DVLPR_PASSWORD: ${{ secrets.DVLPR_PASSWORD }}
155177
run: pnpm test:e2e
156178

157179
- uses: actions/upload-artifact@v4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
/**********************************************************************
2+
* Copyright (C) 2025 Red Hat, Inc.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*
16+
* SPDX-License-Identifier: Apache-2.0
17+
***********************************************************************/
18+
19+
import type { Locator, Page } from '@playwright/test';
20+
import { BasePage } from '@podman-desktop/tests-playwright';
21+
import test, { expect as playExpect } from '@playwright/test';
22+
23+
export class SSOAuthenticationProviderCardPage extends BasePage {
24+
readonly parent: Locator;
25+
readonly providerInformation: Locator;
26+
readonly providerActions: Locator;
27+
readonly signinButton: Locator;
28+
readonly providerName: Locator;
29+
readonly providerStatus: Locator;
30+
readonly logoutButton: Locator;
31+
readonly userName: Locator;
32+
readonly signoutButton: Locator;
33+
34+
constructor(page: Page) {
35+
super(page);
36+
this.parent = this.page.getByRole('listitem', { name: 'Red Hat SSO' });
37+
this.providerInformation = this.parent.getByLabel('Provider Information');
38+
this.providerActions = this.parent.getByLabel('Provider Actions');
39+
this.signinButton = this.providerActions.getByRole('button', { name: 'Sign in' });
40+
this.providerName = this.providerInformation.getByLabel('Provider Name');
41+
this.providerStatus = this.providerInformation.getByLabel('Provider Status');
42+
this.userName = this.providerInformation.getByLabel('Logged In Username');
43+
this.signoutButton = this.providerInformation.getByRole('button', { name: 'Sign out of ', exact: false });
44+
}
45+
46+
public async signIn(): Promise<void> {
47+
test.step('Perform Sign In', async () => {
48+
console.log(`Signin Button is enabled`);
49+
await playExpect(this.signinButton).toBeEnabled();
50+
console.log(`Clicking on the button...`);
51+
await this.signinButton.click();
52+
console.log(`Button clicked`);
53+
});
54+
}
55+
56+
public async logout(): Promise<void> {
57+
test.step('Perform Sign Out', async () => {
58+
await playExpect(this.signoutButton).toBeEnabled();
59+
await this.signoutButton.click();
60+
});
61+
}
62+
63+
public async checkUserIsLoggedIn(loggedIn = true): Promise<void> {
64+
test.step(loggedIn ? 'User si logged In' : 'User is logged out', async () => {
65+
await playExpect(this.providerStatus).toBeVisible();
66+
console.log(`Status text: ${await this.providerStatus.innerText()}`);
67+
await playExpect(this.providerStatus).toContainText(loggedIn ? 'logged in' : 'logged out', { ignoreCase: true });
68+
});
69+
}
70+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,106 @@
1+
/**********************************************************************
2+
* Copyright (C) 2025 Red Hat, Inc.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*
16+
* SPDX-License-Identifier: Apache-2.0
17+
***********************************************************************/
18+
19+
import type { Locator, Page } from '@playwright/test';
20+
import { expect as playExpect } from '@playwright/test';
21+
22+
import { BasePage } from '@podman-desktop/tests-playwright';
23+
24+
export class TroubleshootingPage extends BasePage {
25+
readonly heading: Locator;
26+
readonly header: Locator;
27+
readonly tabs: Locator;
28+
readonly tabContent: Locator;
29+
30+
constructor(page: Page) {
31+
super(page);
32+
this.header = this.page.getByRole('region', { name: 'Header' });
33+
this.heading = this.header.getByRole('heading', { name: 'Troubleshooting' });
34+
this.tabs = this.page.getByRole('region', { name: 'Tabs' });
35+
this.tabContent = this.page.getByRole('region', { name: 'Tab Content' });
36+
}
37+
38+
public async openRepairConnections(): Promise<void> {
39+
await this.openTab('Repair & Connections');
40+
await playExpect(this.tabContent.getByRole('status', { name: 'container connections' })).toBeVisible();
41+
}
42+
43+
public async openStores(): Promise<void> {
44+
await this.openTab('Stores');
45+
await playExpect(this.tabContent.getByRole('status', { name: 'stores' })).toBeVisible();
46+
}
47+
48+
public async openLogs(): Promise<void> {
49+
await this.openTab('Logs');
50+
await playExpect(this.tabContent.getByText('Logs', { exact: true })).toBeVisible();
51+
}
52+
53+
public async openGatherLogs(): Promise<void> {
54+
await this.openTab('Gather Logs');
55+
await playExpect(this.tabContent.getByText('Gather Log Files')).toBeVisible();
56+
}
57+
58+
private async openTab(tabName: string): Promise<void> {
59+
const link = this.tabs.getByRole('link', { name: tabName, exact: true });
60+
await playExpect(link, `Tab Link ${tabName} is not visible`).toBeVisible();
61+
await link.click();
62+
}
63+
64+
// return locator for better processing in playwright assertions
65+
public async getLogs(): Promise<Locator> {
66+
await this.openLogs();
67+
const list = this.tabContent.getByRole('list', { name: 'logs', exact: true });
68+
return list;
69+
}
70+
71+
public async refreshStore(storeName: string): Promise<void> {
72+
await this.openStores();
73+
const stores = this.tabContent.getByRole('list', { name: 'stores' });
74+
let store = stores.getByRole('listitem', { name: storeName });
75+
if (await store.count() <= 0) {
76+
const details = this.tabContent.getByRole('button', { name: 'open details' }).and(this.tabContent.getByText('auth providers'));
77+
await playExpect(details).toBeVisible();
78+
store = details.locator('..').locator('..');
79+
console.log(`Store: ${await store.innerHTML()}`);
80+
}
81+
await playExpect(store).toBeVisible();
82+
await store.scrollIntoViewIfNeeded();
83+
const refreshButton = store.getByRole('button', { name: 'Refresh' });
84+
await playExpect(refreshButton).toBeEnabled();
85+
await refreshButton.click();
86+
}
87+
88+
public async getContainerConnectionsStatus(): Promise<string> {
89+
await this.openRepairConnections();
90+
const connectProviderRegion = this.tabContent.getByRole('region', { name: 'Container connections' });
91+
const status = connectProviderRegion.getByRole('status', { name: 'container connections' });
92+
await playExpect(status).toBeVisible();
93+
return await status.innerText();
94+
}
95+
96+
public async reconnetProviders(): Promise<string> {
97+
await this.openRepairConnections();
98+
const containerConnectionsRegion = this.tabContent.getByRole('region', { name: 'Container connections' });
99+
const reconnectButton = containerConnectionsRegion.getByRole('button', { name: 'Reconnect providers' });
100+
await playExpect(reconnectButton).toBeEnabled();
101+
await reconnectButton.click();
102+
const status = containerConnectionsRegion.getByRole('status', { name: 'Reconnect Providers' });
103+
await playExpect(status).toBeVisible();
104+
return await status.innerText();
105+
}
106+
}

0 commit comments

Comments
 (0)