Compare commits
4 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c730a5470a | |||
| 81fd75ef89 | |||
| 1ae227885e | |||
| a4a0f2d7cd |
@@ -2,9 +2,9 @@ name: CI
|
|||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches: [main, dev]
|
branches: ['**']
|
||||||
pull_request:
|
pull_request:
|
||||||
branches: [main, dev]
|
branches: [main, dev, uat]
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
workflow_call:
|
workflow_call:
|
||||||
inputs:
|
inputs:
|
||||||
@@ -14,6 +14,9 @@ on:
|
|||||||
type: string
|
type: string
|
||||||
default: '22'
|
default: '22'
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
ci:
|
ci:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|||||||
@@ -1,23 +0,0 @@
|
|||||||
name: E2E Tests
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches: [main]
|
|
||||||
pull_request:
|
|
||||||
branches: [main]
|
|
||||||
workflow_dispatch:
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
concurrency:
|
|
||||||
group: e2e-${{ github.repository }}
|
|
||||||
cancel-in-progress: false
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
e2e:
|
|
||||||
uses: privilegedescalation/.github/.github/workflows/plugin-e2e.yaml@hugh/add-pnpm-support-plugin-e2e
|
|
||||||
with:
|
|
||||||
node-version: '22'
|
|
||||||
headlamp-version: v0.40.1
|
|
||||||
e2e-namespace: headlamp-dev
|
|
||||||
@@ -7,8 +7,38 @@ on:
|
|||||||
description: 'Release version (e.g. 1.0.0)'
|
description: 'Release version (e.g. 1.0.0)'
|
||||||
required: true
|
required: true
|
||||||
type: string
|
type: string
|
||||||
|
node-version:
|
||||||
|
description: 'Node.js version to use'
|
||||||
|
required: false
|
||||||
|
type: string
|
||||||
|
default: '22'
|
||||||
|
upstream-repo:
|
||||||
|
description: 'Upstream repo to fetch appVersion from (e.g. fenio/tns-csi). Leave empty to skip.'
|
||||||
|
required: false
|
||||||
|
type: string
|
||||||
|
default: ''
|
||||||
repository_dispatch:
|
repository_dispatch:
|
||||||
types: [release]
|
types: [release]
|
||||||
|
workflow_call:
|
||||||
|
inputs:
|
||||||
|
version:
|
||||||
|
description: 'Release version (e.g. 1.0.0)'
|
||||||
|
required: true
|
||||||
|
type: string
|
||||||
|
node-version:
|
||||||
|
description: 'Node.js version to use'
|
||||||
|
required: false
|
||||||
|
type: string
|
||||||
|
default: '22'
|
||||||
|
upstream-repo:
|
||||||
|
description: 'Upstream repo to fetch appVersion from (e.g. fenio/tns-csi). Leave empty to skip.'
|
||||||
|
required: false
|
||||||
|
type: string
|
||||||
|
default: ''
|
||||||
|
secrets:
|
||||||
|
GITEA_RELEASE_TOKEN:
|
||||||
|
description: 'Gitea token with write access to repos'
|
||||||
|
required: true
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
contents: write
|
contents: write
|
||||||
@@ -41,7 +71,8 @@ jobs:
|
|||||||
if: needs.check-secrets.outputs.ready == 'true'
|
if: needs.check-secrets.outputs.ready == 'true'
|
||||||
uses: ./.github/workflows/ci.yaml
|
uses: ./.github/workflows/ci.yaml
|
||||||
with:
|
with:
|
||||||
node-version: ${{ inputs.node-version || '22' }}
|
node-version: ${{ inputs.node-version }}
|
||||||
|
secrets: inherit
|
||||||
|
|
||||||
check-token-permissions:
|
check-token-permissions:
|
||||||
needs: check-secrets
|
needs: check-secrets
|
||||||
@@ -83,6 +114,8 @@ jobs:
|
|||||||
needs: check-secrets
|
needs: check-secrets
|
||||||
if: needs.check-secrets.outputs.ready == 'true'
|
if: needs.check-secrets.outputs.ready == 'true'
|
||||||
runs-on: runners-privilegedescalation
|
runs-on: runners-privilegedescalation
|
||||||
|
env:
|
||||||
|
RESOLVED_VERSION: ${{ inputs.version || github.event.client_payload.version }}
|
||||||
outputs:
|
outputs:
|
||||||
skip: ${{ steps.check.outputs.skip }}
|
skip: ${{ steps.check.outputs.skip }}
|
||||||
steps:
|
steps:
|
||||||
@@ -94,9 +127,9 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
HTTP_CODE=$(curl -sf -o /dev/null -w "%{http_code}" \
|
HTTP_CODE=$(curl -sf -o /dev/null -w "%{http_code}" \
|
||||||
-H "Authorization: token ${GITEA_TOKEN}" \
|
-H "Authorization: token ${GITEA_TOKEN}" \
|
||||||
"https://git.farh.net/api/v1/repos/${REPO}/git/refs/tags/v${{ inputs.version }}")
|
"https://git.farh.net/api/v1/repos/${REPO}/git/refs/tags/v${{ env.RESOLVED_VERSION }}")
|
||||||
if [ "$HTTP_CODE" = "200" ]; then
|
if [ "$HTTP_CODE" = "200" ]; then
|
||||||
echo "::notice::Tag v${{ inputs.version }} already exists. Release skipped (not an error)."
|
echo "::notice::Tag v${{ env.RESOLVED_VERSION }} already exists. Release skipped (not an error)."
|
||||||
echo "skip=true" >> $GITHUB_OUTPUT
|
echo "skip=true" >> $GITHUB_OUTPUT
|
||||||
else
|
else
|
||||||
echo "skip=false" >> $GITHUB_OUTPUT
|
echo "skip=false" >> $GITHUB_OUTPUT
|
||||||
@@ -107,11 +140,13 @@ jobs:
|
|||||||
if: needs.check-secrets.outputs.ready == 'true' && needs.check-tag.outputs.skip != 'true' && needs.check-token-permissions.outputs.has_write == 'true'
|
if: needs.check-secrets.outputs.ready == 'true' && needs.check-tag.outputs.skip != 'true' && needs.check-token-permissions.outputs.has_write == 'true'
|
||||||
runs-on: runners-privilegedescalation
|
runs-on: runners-privilegedescalation
|
||||||
timeout-minutes: 10
|
timeout-minutes: 10
|
||||||
|
env:
|
||||||
|
RESOLVED_VERSION: ${{ inputs.version || github.event.client_payload.version }}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Validate version format
|
- name: Validate version format
|
||||||
run: |
|
run: |
|
||||||
if [[ ! "${{ inputs.version }}" =~ ^[0-9]+\.[0-9]+\.[0-9]+$ ]]; then
|
if [[ ! "${{ env.RESOLVED_VERSION }}" =~ ^[0-9]+\.[0-9]+\.[0-9]+$ ]]; then
|
||||||
echo "Error: Version must be in X.Y.Z format"
|
echo "Error: Version must be in X.Y.Z format"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
@@ -138,7 +173,7 @@ jobs:
|
|||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v6
|
uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version: ${{ inputs.node-version || '22' }}
|
node-version: ${{ inputs.node-version }}
|
||||||
cache: ${{ steps.pkg-manager.outputs.manager == 'npm' && 'npm' || '' }}
|
cache: ${{ steps.pkg-manager.outputs.manager == 'npm' && 'npm' || '' }}
|
||||||
|
|
||||||
- name: Setup pnpm (via Corepack)
|
- name: Setup pnpm (via Corepack)
|
||||||
@@ -181,16 +216,16 @@ jobs:
|
|||||||
- name: Update version in package.json
|
- name: Update version in package.json
|
||||||
run: |
|
run: |
|
||||||
if [ "${{ steps.pkg-manager.outputs.manager }}" = "pnpm" ]; then
|
if [ "${{ steps.pkg-manager.outputs.manager }}" = "pnpm" ]; then
|
||||||
pnpm version ${{ inputs.version }} --no-git-tag-version --allow-same-version
|
pnpm version ${{ env.RESOLVED_VERSION }} --no-git-tag-version --allow-same-version
|
||||||
else
|
else
|
||||||
npm version ${{ inputs.version }} --no-git-tag-version --allow-same-version
|
npm version ${{ env.RESOLVED_VERSION }} --no-git-tag-version --allow-same-version
|
||||||
fi
|
fi
|
||||||
|
|
||||||
- name: Update artifacthub-pkg.yml
|
- name: Update artifacthub-pkg.yml
|
||||||
env:
|
env:
|
||||||
REPO: ${{ github.repository }}
|
REPO: ${{ github.repository }}
|
||||||
run: |
|
run: |
|
||||||
VERSION="${{ inputs.version }}"
|
VERSION="${{ env.RESOLVED_VERSION }}"
|
||||||
if [ -f artifacthub-pkg.yml ]; then
|
if [ -f artifacthub-pkg.yml ]; then
|
||||||
PKG_NAME=$(grep '^name:' artifacthub-pkg.yml | cut -d: -f2 | tr -d ' "')
|
PKG_NAME=$(grep '^name:' artifacthub-pkg.yml | cut -d: -f2 | tr -d ' "')
|
||||||
else
|
else
|
||||||
@@ -245,7 +280,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Prepare release tarball
|
- name: Prepare release tarball
|
||||||
run: |
|
run: |
|
||||||
VERSION="${{ inputs.version }}"
|
VERSION="${{ env.RESOLVED_VERSION }}"
|
||||||
if [ -f artifacthub-pkg.yml ]; then
|
if [ -f artifacthub-pkg.yml ]; then
|
||||||
PKG_NAME=$(grep '^name:' artifacthub-pkg.yml | cut -d: -f2 | tr -d ' "')
|
PKG_NAME=$(grep '^name:' artifacthub-pkg.yml | cut -d: -f2 | tr -d ' "')
|
||||||
else
|
else
|
||||||
@@ -280,7 +315,7 @@ jobs:
|
|||||||
env:
|
env:
|
||||||
GITEA_TOKEN: ${{ secrets.GITEA_RELEASE_TOKEN }}
|
GITEA_TOKEN: ${{ secrets.GITEA_RELEASE_TOKEN }}
|
||||||
run: |
|
run: |
|
||||||
VERSION="${{ inputs.version }}"
|
VERSION="${{ env.RESOLVED_VERSION }}"
|
||||||
BRANCH="release/v${VERSION}"
|
BRANCH="release/v${VERSION}"
|
||||||
if git ls-remote --exit-code origin "refs/heads/$BRANCH" 2>/dev/null; then
|
if git ls-remote --exit-code origin "refs/heads/$BRANCH" 2>/dev/null; then
|
||||||
echo "::notice::Branch $BRANCH already exists — deleting for clean re-trigger."
|
echo "::notice::Branch $BRANCH already exists — deleting for clean re-trigger."
|
||||||
@@ -298,7 +333,7 @@ jobs:
|
|||||||
GITEA_TOKEN: ${{ secrets.GITEA_RELEASE_TOKEN }}
|
GITEA_TOKEN: ${{ secrets.GITEA_RELEASE_TOKEN }}
|
||||||
REPO: ${{ github.repository }}
|
REPO: ${{ github.repository }}
|
||||||
run: |
|
run: |
|
||||||
VERSION="${{ inputs.version }}"
|
VERSION="${{ env.RESOLVED_VERSION }}"
|
||||||
TARBALL="${{ env.TARBALL }}"
|
TARBALL="${{ env.TARBALL }}"
|
||||||
RESPONSE=$(curl -sf -X POST \
|
RESPONSE=$(curl -sf -X POST \
|
||||||
-H "Authorization: token ${GITEA_TOKEN}" \
|
-H "Authorization: token ${GITEA_TOKEN}" \
|
||||||
@@ -332,7 +367,7 @@ jobs:
|
|||||||
REPO: ${{ github.repository }}
|
REPO: ${{ github.repository }}
|
||||||
run: |
|
run: |
|
||||||
set -o pipefail
|
set -o pipefail
|
||||||
VERSION="${{ inputs.version }}"
|
VERSION="${{ env.RESOLVED_VERSION }}"
|
||||||
BODY=$(printf "Automated version bump and checksum update for v%s.\n\ncc @cpfarhood" "${VERSION}")
|
BODY=$(printf "Automated version bump and checksum update for v%s.\n\ncc @cpfarhood" "${VERSION}")
|
||||||
RESPONSE=$(curl -sf -X POST \
|
RESPONSE=$(curl -sf -X POST \
|
||||||
-H "Authorization: token ${GITEA_TOKEN}" \
|
-H "Authorization: token ${GITEA_TOKEN}" \
|
||||||
@@ -389,7 +424,7 @@ jobs:
|
|||||||
GITEA_TOKEN: ${{ secrets.GITEA_RELEASE_TOKEN }}
|
GITEA_TOKEN: ${{ secrets.GITEA_RELEASE_TOKEN }}
|
||||||
REPO: ${{ github.repository }}
|
REPO: ${{ github.repository }}
|
||||||
run: |
|
run: |
|
||||||
VERSION="${{ inputs.version }}"
|
VERSION="${{ env.RESOLVED_VERSION }}"
|
||||||
TARBALL_CS=$(sha256sum "${{ env.TARBALL }}" | awk '{print $1}')
|
TARBALL_CS=$(sha256sum "${{ env.TARBALL }}" | awk '{print $1}')
|
||||||
TAG_CS=$(git show "v${VERSION}:artifacthub-pkg.yml" 2>/dev/null \
|
TAG_CS=$(git show "v${VERSION}:artifacthub-pkg.yml" 2>/dev/null \
|
||||||
| grep "archive-checksum" | awk '{print $2}' | sed 's/sha256://')
|
| grep "archive-checksum" | awk '{print $2}' | sed 's/sha256://')
|
||||||
|
|||||||
@@ -1,69 +0,0 @@
|
|||||||
import { test as setup, expect, Page } from '@playwright/test';
|
|
||||||
|
|
||||||
const AUTH_STATE_PATH = 'e2e/.auth/state.json';
|
|
||||||
|
|
||||||
async function authenticateWithOIDC(page: Page, username: string, password: string): Promise<void> {
|
|
||||||
await page.goto('/');
|
|
||||||
await page.waitForURL('**/login');
|
|
||||||
|
|
||||||
const popupPromise = page.waitForEvent('popup');
|
|
||||||
await page.getByRole('button', { name: /sign in/i }).click();
|
|
||||||
const popup = await popupPromise;
|
|
||||||
|
|
||||||
await popup.waitForLoadState('domcontentloaded');
|
|
||||||
await popup.waitForLoadState('networkidle');
|
|
||||||
|
|
||||||
const usernameField = popup.getByRole('textbox', { name: /email or username/i });
|
|
||||||
await usernameField.waitFor({ state: 'visible', timeout: 15_000 });
|
|
||||||
await usernameField.fill(username);
|
|
||||||
await popup.getByRole('button', { name: /log in/i }).click();
|
|
||||||
|
|
||||||
await popup.waitForLoadState('networkidle');
|
|
||||||
const passwordField = popup.getByRole('textbox', { name: /password/i });
|
|
||||||
await passwordField.waitFor({ state: 'visible', timeout: 15_000 });
|
|
||||||
await passwordField.fill(password);
|
|
||||||
await popup.getByRole('button', { name: /continue|log in/i }).click();
|
|
||||||
|
|
||||||
await popup.waitForEvent('close', { timeout: 15_000 });
|
|
||||||
|
|
||||||
await expect(page.getByRole('navigation', { name: 'Navigation' })).toBeVisible({
|
|
||||||
timeout: 15_000,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async function authenticateWithToken(page: Page, token: string): Promise<void> {
|
|
||||||
await page.goto('/');
|
|
||||||
await page.waitForURL(/\/(login|token)$/);
|
|
||||||
|
|
||||||
if (page.url().includes('/login')) {
|
|
||||||
const useTokenBtn = page.getByRole('button', { name: /use a token/i });
|
|
||||||
await useTokenBtn.waitFor({ state: 'visible', timeout: 15_000 });
|
|
||||||
await useTokenBtn.click();
|
|
||||||
await page.waitForURL('**/token');
|
|
||||||
}
|
|
||||||
|
|
||||||
await page.getByRole('textbox', { name: /id token/i }).fill(token);
|
|
||||||
await page.getByRole('button', { name: /authenticate/i }).click();
|
|
||||||
|
|
||||||
await expect(page.getByRole('navigation', { name: 'Navigation' })).toBeVisible({
|
|
||||||
timeout: 15_000,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
setup('authenticate with Headlamp', async ({ page }) => {
|
|
||||||
const username = process.env.AUTHENTIK_USERNAME;
|
|
||||||
const password = process.env.AUTHENTIK_PASSWORD;
|
|
||||||
const token = process.env.HEADLAMP_TOKEN;
|
|
||||||
|
|
||||||
if (username && password) {
|
|
||||||
await authenticateWithOIDC(page, username, password);
|
|
||||||
} else if (token) {
|
|
||||||
await authenticateWithToken(page, token);
|
|
||||||
} else {
|
|
||||||
throw new Error(
|
|
||||||
'Set AUTHENTIK_USERNAME + AUTHENTIK_PASSWORD for OIDC auth, or HEADLAMP_TOKEN for token auth'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
await page.context().storageState({ path: AUTH_STATE_PATH });
|
|
||||||
});
|
|
||||||
@@ -1,63 +0,0 @@
|
|||||||
import { test, expect } from '@playwright/test';
|
|
||||||
|
|
||||||
async function waitForSidebar(page: import('@playwright/test').Page) {
|
|
||||||
const sidebar = page.getByRole('navigation', { name: 'Navigation' });
|
|
||||||
await expect(sidebar).toBeVisible({ timeout: 15_000 });
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
return sidebar;
|
|
||||||
}
|
|
||||||
|
|
||||||
test.describe('Rook plugin smoke tests', () => {
|
|
||||||
test('sidebar contains Rook entry', async ({ page }) => {
|
|
||||||
await page.goto('/');
|
|
||||||
const sidebar = await waitForSidebar(page);
|
|
||||||
await expect(sidebar.getByRole('button', { name: /rook/i })).toBeVisible();
|
|
||||||
});
|
|
||||||
|
|
||||||
test('Rook sidebar entry navigates to overview', async ({ page }) => {
|
|
||||||
await page.goto('/');
|
|
||||||
const sidebar = await waitForSidebar(page);
|
|
||||||
|
|
||||||
const rookEntry = sidebar.getByRole('button', { name: /rook/i });
|
|
||||||
await expect(rookEntry).toBeVisible();
|
|
||||||
await rookEntry.click();
|
|
||||||
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
await expect(page).toHaveURL(/rook-ceph/);
|
|
||||||
await expect(page.getByRole('heading', { name: /overview/i })).toBeVisible();
|
|
||||||
});
|
|
||||||
|
|
||||||
test('overview page renders content', async ({ page }) => {
|
|
||||||
await page.goto('/c/main/rook-ceph');
|
|
||||||
await waitForSidebar(page);
|
|
||||||
|
|
||||||
await expect(page.getByRole('heading', { name: /overview/i })).toBeVisible({
|
|
||||||
timeout: 15_000,
|
|
||||||
});
|
|
||||||
|
|
||||||
const hasContent = await page.locator('text=/cluster|ceph|status/i').first().isVisible().catch(() => false);
|
|
||||||
const hasDashboard = await page.locator('[class*="Mui"]').first().isVisible().catch(() => false);
|
|
||||||
expect(hasContent || hasDashboard).toBe(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('navigation to storage classes view works', async ({ page }) => {
|
|
||||||
await page.goto('/c/main/rook-ceph');
|
|
||||||
|
|
||||||
const sidebar = page.getByRole('navigation', { name: 'Navigation' });
|
|
||||||
const storageClassesLink = sidebar.getByRole('link', { name: /storage classes/i });
|
|
||||||
await expect(storageClassesLink).toBeVisible({ timeout: 10_000 });
|
|
||||||
await storageClassesLink.click();
|
|
||||||
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
await expect(page).toHaveURL(/rook-ceph\/storage-classes/);
|
|
||||||
await expect(page.getByRole('heading', { name: /storage class/i })).toBeVisible({ timeout: 15_000 });
|
|
||||||
});
|
|
||||||
|
|
||||||
test('plugin settings page shows rook plugin entry', async ({ page }) => {
|
|
||||||
await page.goto('/settings/plugins');
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
|
|
||||||
const pluginEntry = page.locator('text=rook').first();
|
|
||||||
await expect(pluginEntry).toBeVisible({ timeout: 30_000 });
|
|
||||||
});
|
|
||||||
});
|
|
||||||
+9
-4
@@ -22,15 +22,12 @@
|
|||||||
"format": "prettier --write src/",
|
"format": "prettier --write src/",
|
||||||
"format:check": "prettier --check src/",
|
"format:check": "prettier --check src/",
|
||||||
"test": "vitest run",
|
"test": "vitest run",
|
||||||
"test:watch": "vitest",
|
"test:watch": "vitest"
|
||||||
"e2e": "playwright test",
|
|
||||||
"e2e:headed": "playwright test --headed"
|
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@headlamp-k8s/eslint-config": "^0.6.0",
|
"@headlamp-k8s/eslint-config": "^0.6.0",
|
||||||
"@kinvolk/headlamp-plugin": "^0.13.0",
|
"@kinvolk/headlamp-plugin": "^0.13.0",
|
||||||
"@mui/material": "^5.15.14",
|
"@mui/material": "^5.15.14",
|
||||||
"@playwright/test": "^1.58.2",
|
|
||||||
"@testing-library/jest-dom": "^6.4.8",
|
"@testing-library/jest-dom": "^6.4.8",
|
||||||
"@testing-library/react": "^16.0.0",
|
"@testing-library/react": "^16.0.0",
|
||||||
"@testing-library/user-event": "^14.5.2",
|
"@testing-library/user-event": "^14.5.2",
|
||||||
@@ -52,5 +49,13 @@
|
|||||||
"vite": ">=6.4.2",
|
"vite": ">=6.4.2",
|
||||||
"lodash": ">=4.18.0",
|
"lodash": ">=4.18.0",
|
||||||
"elliptic": ">=6.6.1"
|
"elliptic": ">=6.6.1"
|
||||||
|
},
|
||||||
|
"packageManager": "pnpm@9.15.4",
|
||||||
|
"pnpm": {
|
||||||
|
"onlyBuiltDependencies": [
|
||||||
|
"@swc/core",
|
||||||
|
"esbuild",
|
||||||
|
"msw"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
import { defineConfig, devices } from '@playwright/test';
|
|
||||||
|
|
||||||
export default defineConfig({
|
|
||||||
testDir: './e2e',
|
|
||||||
timeout: 30_000,
|
|
||||||
expect: { timeout: 10_000 },
|
|
||||||
fullyParallel: false,
|
|
||||||
forbidOnly: !!process.env.CI,
|
|
||||||
retries: process.env.CI ? 1 : 0,
|
|
||||||
reporter: 'list',
|
|
||||||
use: {
|
|
||||||
baseURL: process.env.HEADLAMP_URL || (() => { throw new Error('HEADLAMP_URL is required — run scripts/deploy-e2e-headlamp.sh first'); })(),
|
|
||||||
trace: 'on-first-retry',
|
|
||||||
screenshot: 'only-on-failure',
|
|
||||||
},
|
|
||||||
projects: [
|
|
||||||
{ name: 'setup', testMatch: /auth\.setup\.ts/, timeout: 60_000 },
|
|
||||||
{
|
|
||||||
name: 'chromium',
|
|
||||||
use: {
|
|
||||||
...devices['Desktop Chrome'],
|
|
||||||
storageState: 'e2e/.auth/state.json',
|
|
||||||
},
|
|
||||||
dependencies: ['setup'],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
});
|
|
||||||
Generated
+691
-654
File diff suppressed because it is too large
Load Diff
@@ -1,189 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
# deploy-e2e-headlamp.sh
|
|
||||||
#
|
|
||||||
# Deploys a stock Headlamp instance with the rook plugin loaded via
|
|
||||||
# a ConfigMap volume mount.
|
|
||||||
#
|
|
||||||
# E2E resources are deployed to the `headlamp-dev` namespace. Nothing
|
|
||||||
# persists beyond the test run — teardown cleans up all created resources.
|
|
||||||
#
|
|
||||||
# Prerequisites:
|
|
||||||
# - Plugin built (dist/ exists with plugin-main.js + package.json)
|
|
||||||
# - kubectl configured with cluster access
|
|
||||||
#
|
|
||||||
# Environment:
|
|
||||||
# E2E_NAMESPACE — namespace for E2E Headlamp (default: headlamp-dev)
|
|
||||||
# E2E_RELEASE — release/resource name prefix (default: headlamp-e2e)
|
|
||||||
# HEADLAMP_VERSION — Headlamp image tag (default: latest)
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
REPO_ROOT="$(cd "$(dirname "$0")/.." && pwd)"
|
|
||||||
DIST_DIR="$REPO_ROOT/dist"
|
|
||||||
|
|
||||||
E2E_NAMESPACE="${E2E_NAMESPACE:-headlamp-dev}"
|
|
||||||
E2E_RELEASE="${E2E_RELEASE:-headlamp-e2e}"
|
|
||||||
HEADLAMP_VERSION="${HEADLAMP_VERSION:-latest}"
|
|
||||||
|
|
||||||
if [ ! -d "$DIST_DIR" ]; then
|
|
||||||
echo "ERROR: dist/ not found. Run 'pnpm build' first." >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "Checking RBAC permissions in namespace '${E2E_NAMESPACE}'..."
|
|
||||||
if ! kubectl auth can-i delete configmaps -n "$E2E_NAMESPACE" --quiet 2>/dev/null; then
|
|
||||||
echo "ERROR: Missing RBAC — cannot delete configmaps in namespace '${E2E_NAMESPACE}'." >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "=== E2E Headlamp Deployment ==="
|
|
||||||
echo " Image: ghcr.io/headlamp-k8s/headlamp:${HEADLAMP_VERSION}"
|
|
||||||
echo " Namespace: $E2E_NAMESPACE"
|
|
||||||
echo " Release: $E2E_RELEASE"
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
echo "Creating ConfigMap with plugin files..."
|
|
||||||
|
|
||||||
kubectl delete configmap headlamp-rook-plugin \
|
|
||||||
-n "$E2E_NAMESPACE" --ignore-not-found
|
|
||||||
|
|
||||||
kubectl create configmap headlamp-rook-plugin \
|
|
||||||
-n "$E2E_NAMESPACE" \
|
|
||||||
--from-file="$DIST_DIR" \
|
|
||||||
--from-file=package.json="$REPO_ROOT/package.json"
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
echo "Removing any existing E2E deployment (clean-start)..."
|
|
||||||
kubectl delete deployment "${E2E_RELEASE}" -n "$E2E_NAMESPACE" --ignore-not-found --wait
|
|
||||||
kubectl delete service "${E2E_RELEASE}" -n "$E2E_NAMESPACE" --ignore-not-found --wait
|
|
||||||
kubectl delete serviceaccount "${E2E_RELEASE}" -n "$E2E_NAMESPACE" --ignore-not-found --wait
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
echo "Deploying Headlamp E2E instance..."
|
|
||||||
|
|
||||||
kubectl apply -f - <<EOF
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ServiceAccount
|
|
||||||
metadata:
|
|
||||||
name: ${E2E_RELEASE}
|
|
||||||
namespace: ${E2E_NAMESPACE}
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: ${E2E_RELEASE}
|
|
||||||
namespace: ${E2E_NAMESPACE}
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: headlamp
|
|
||||||
app.kubernetes.io/instance: ${E2E_RELEASE}
|
|
||||||
spec:
|
|
||||||
replicas: 1
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app.kubernetes.io/name: headlamp
|
|
||||||
app.kubernetes.io/instance: ${E2E_RELEASE}
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: headlamp
|
|
||||||
app.kubernetes.io/instance: ${E2E_RELEASE}
|
|
||||||
spec:
|
|
||||||
serviceAccountName: ${E2E_RELEASE}
|
|
||||||
automountServiceAccountToken: true
|
|
||||||
securityContext: {}
|
|
||||||
containers:
|
|
||||||
- name: headlamp
|
|
||||||
image: ghcr.io/headlamp-k8s/headlamp:${HEADLAMP_VERSION}
|
|
||||||
imagePullPolicy: IfNotPresent
|
|
||||||
securityContext:
|
|
||||||
runAsNonRoot: true
|
|
||||||
privileged: false
|
|
||||||
runAsUser: 100
|
|
||||||
runAsGroup: 101
|
|
||||||
args:
|
|
||||||
- "-in-cluster"
|
|
||||||
- "-in-cluster-context-name=main"
|
|
||||||
- "-plugins-dir=/headlamp/plugins"
|
|
||||||
ports:
|
|
||||||
- name: http
|
|
||||||
containerPort: 4466
|
|
||||||
protocol: TCP
|
|
||||||
readinessProbe:
|
|
||||||
httpGet:
|
|
||||||
path: /
|
|
||||||
port: http
|
|
||||||
initialDelaySeconds: 5
|
|
||||||
periodSeconds: 5
|
|
||||||
failureThreshold: 6
|
|
||||||
livenessProbe:
|
|
||||||
httpGet:
|
|
||||||
path: /
|
|
||||||
port: http
|
|
||||||
initialDelaySeconds: 10
|
|
||||||
periodSeconds: 10
|
|
||||||
volumeMounts:
|
|
||||||
- name: rook-plugin
|
|
||||||
mountPath: /headlamp/plugins/headlamp-rook
|
|
||||||
readOnly: true
|
|
||||||
volumes:
|
|
||||||
- name: rook-plugin
|
|
||||||
configMap:
|
|
||||||
name: headlamp-rook-plugin
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
name: ${E2E_RELEASE}
|
|
||||||
namespace: ${E2E_NAMESPACE}
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: headlamp
|
|
||||||
app.kubernetes.io/instance: ${E2E_RELEASE}
|
|
||||||
spec:
|
|
||||||
type: ClusterIP
|
|
||||||
selector:
|
|
||||||
app.kubernetes.io/name: headlamp
|
|
||||||
app.kubernetes.io/instance: ${E2E_RELEASE}
|
|
||||||
ports:
|
|
||||||
- name: http
|
|
||||||
port: 80
|
|
||||||
targetPort: http
|
|
||||||
protocol: TCP
|
|
||||||
EOF
|
|
||||||
|
|
||||||
echo "Waiting for rollout..."
|
|
||||||
kubectl rollout status "deployment/${E2E_RELEASE}" \
|
|
||||||
-n "$E2E_NAMESPACE" --timeout=120s
|
|
||||||
|
|
||||||
SVC_URL="http://${E2E_RELEASE}.${E2E_NAMESPACE}.svc.cluster.local"
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
echo "Waiting for ${SVC_URL} to be reachable..."
|
|
||||||
ATTEMPTS=0
|
|
||||||
MAX_ATTEMPTS=24
|
|
||||||
until curl -sf --max-time 5 "${SVC_URL}" -o /dev/null 2>/dev/null; do
|
|
||||||
ATTEMPTS=$((ATTEMPTS + 1))
|
|
||||||
if [ "$ATTEMPTS" -ge "$MAX_ATTEMPTS" ]; then
|
|
||||||
echo "ERROR: ${SVC_URL} not reachable after $((MAX_ATTEMPTS * 5))s" >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
echo " [${ATTEMPTS}/${MAX_ATTEMPTS}] not yet reachable, retrying in 5s..."
|
|
||||||
sleep 5
|
|
||||||
done
|
|
||||||
echo ""
|
|
||||||
echo "E2E Headlamp is ready at: ${SVC_URL}"
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
echo "Creating service account token for E2E auth..."
|
|
||||||
kubectl create serviceaccount headlamp-e2e-test \
|
|
||||||
-n "$E2E_NAMESPACE" --dry-run=client -o yaml | kubectl apply -f -
|
|
||||||
|
|
||||||
TOKEN=$(kubectl create token headlamp-e2e-test -n "$E2E_NAMESPACE" --duration=1h 2>/dev/null || echo "")
|
|
||||||
if [ -n "$TOKEN" ]; then
|
|
||||||
echo "HEADLAMP_URL=${SVC_URL}" > "$REPO_ROOT/.env.e2e"
|
|
||||||
echo "HEADLAMP_TOKEN=${TOKEN}" >> "$REPO_ROOT/.env.e2e"
|
|
||||||
echo "Wrote .env.e2e with HEADLAMP_URL and HEADLAMP_TOKEN"
|
|
||||||
else
|
|
||||||
echo " WARNING: Could not generate token."
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
echo "E2E deployment complete."
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
# teardown-e2e-headlamp.sh
|
|
||||||
#
|
|
||||||
# Tears down the dedicated E2E Headlamp instance deployed by deploy-e2e-headlamp.sh.
|
|
||||||
#
|
|
||||||
# Environment:
|
|
||||||
# E2E_NAMESPACE — namespace to clean up (default: headlamp-dev)
|
|
||||||
# E2E_RELEASE — release/resource name prefix (default: headlamp-e2e)
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
REPO_ROOT="$(cd "$(dirname "$0")/.." && pwd)"
|
|
||||||
|
|
||||||
E2E_NAMESPACE="${E2E_NAMESPACE:-headlamp-dev}"
|
|
||||||
E2E_RELEASE="${E2E_RELEASE:-headlamp-e2e}"
|
|
||||||
|
|
||||||
echo "=== E2E Headlamp Teardown ==="
|
|
||||||
echo " Namespace: $E2E_NAMESPACE"
|
|
||||||
echo " Release: $E2E_RELEASE"
|
|
||||||
|
|
||||||
echo "Removing Headlamp Deployment, Service, and ServiceAccount..."
|
|
||||||
kubectl delete deployment "${E2E_RELEASE}" -n "$E2E_NAMESPACE" --ignore-not-found
|
|
||||||
kubectl delete service "${E2E_RELEASE}" -n "$E2E_NAMESPACE" --ignore-not-found
|
|
||||||
kubectl delete serviceaccount "${E2E_RELEASE}" -n "$E2E_NAMESPACE" --ignore-not-found
|
|
||||||
|
|
||||||
echo "Cleaning up ConfigMap..."
|
|
||||||
kubectl delete configmap headlamp-rook-plugin -n "$E2E_NAMESPACE" --ignore-not-found
|
|
||||||
|
|
||||||
echo "Cleaning up test service account..."
|
|
||||||
kubectl delete serviceaccount headlamp-e2e-test -n "$E2E_NAMESPACE" --ignore-not-found
|
|
||||||
|
|
||||||
if [ -f "$REPO_ROOT/.env.e2e" ]; then
|
|
||||||
rm "$REPO_ROOT/.env.e2e"
|
|
||||||
echo "Removed .env.e2e"
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
echo "E2E teardown complete."
|
|
||||||
Reference in New Issue
Block a user