mirror of
https://github.com/johannesjo/super-productivity.git
synced 2026-01-23 19:04:43 +00:00
Add 11 new E2E test suites covering previously untested features: - Tags CRUD (create, assign, remove, delete) - Notes CRUD (create, edit, delete in projects) - Recurring/scheduled tasks (short syntax, context menu) - Context switching (project/tag navigation) - Boards/Kanban view (navigation, display) - Finish day workflow (complete daily flow) - Worklog (time tracking history) - Global search (keyboard, autocomplete) - Settings (navigation, sections, form elements) - Keyboard shortcuts (navigation, escape) - Take-a-break (settings page verification) Also fix flaky plugin-loading test by adding retry mechanism and proper waits between enable/disable operations.
254 lines
9 KiB
TypeScript
254 lines
9 KiB
TypeScript
import { test, expect } from '../../fixtures/test.fixture';
|
|
import { cssSelectors } from '../../constants/selectors';
|
|
import {
|
|
waitForPluginAssets,
|
|
waitForPluginManagementInit,
|
|
getCITimeoutMultiplier,
|
|
enablePluginWithVerification,
|
|
disablePluginWithVerification,
|
|
} from '../../helpers/plugin-test.helpers';
|
|
|
|
const { SIDENAV } = cssSelectors;
|
|
|
|
// Plugin-related selectors
|
|
const PLUGIN_CARD = 'plugin-management mat-card.ng-star-inserted';
|
|
const PLUGIN_ITEM = `${PLUGIN_CARD}`;
|
|
const PLUGIN_NAV_ENTRIES = `${SIDENAV} nav-item button`;
|
|
const PLUGIN_IFRAME = 'plugin-index iframe';
|
|
|
|
test.describe.serial('Plugin Loading', () => {
|
|
test('full plugin loading lifecycle', async ({ page, workViewPage }) => {
|
|
const timeoutMultiplier = getCITimeoutMultiplier();
|
|
// Slightly higher base to avoid flakiness on slower environments
|
|
test.setTimeout(45000 * timeoutMultiplier);
|
|
|
|
// First, ensure plugin assets are available
|
|
const assetsAvailable = await waitForPluginAssets(page);
|
|
if (!assetsAvailable) {
|
|
if (process.env.CI) {
|
|
test.skip(true, 'Plugin assets not available in CI - skipping test');
|
|
return;
|
|
}
|
|
throw new Error('Plugin assets not available - cannot proceed with test');
|
|
}
|
|
|
|
await workViewPage.waitForTaskList();
|
|
|
|
// Use improved plugin management init that handles navigation and setup
|
|
const pluginReady = await waitForPluginManagementInit(page);
|
|
if (!pluginReady) {
|
|
throw new Error('Plugin management could not be initialized');
|
|
}
|
|
|
|
// Enable the plugin
|
|
const enableResult = await page.evaluate((pluginName: string) => {
|
|
const cards = Array.from(document.querySelectorAll('plugin-management mat-card'));
|
|
const targetCard = cards.find((card) => {
|
|
const title = card.querySelector('mat-card-title')?.textContent || '';
|
|
return title.includes(pluginName);
|
|
});
|
|
|
|
if (targetCard) {
|
|
const toggleButton = targetCard.querySelector(
|
|
'mat-slide-toggle button[role="switch"]',
|
|
) as HTMLButtonElement;
|
|
if (toggleButton) {
|
|
const wasChecked = toggleButton.getAttribute('aria-checked') === 'true';
|
|
if (!wasChecked) {
|
|
toggleButton.click();
|
|
}
|
|
return {
|
|
enabled: true,
|
|
found: true,
|
|
wasChecked,
|
|
nowChecked: toggleButton.getAttribute('aria-checked') === 'true',
|
|
clicked: !wasChecked,
|
|
};
|
|
}
|
|
return { enabled: false, found: true, error: 'No toggle found' };
|
|
}
|
|
|
|
return { enabled: false, found: false };
|
|
}, 'API Test Plugin');
|
|
|
|
expect(enableResult.found).toBe(true);
|
|
|
|
// Wait for toggle state to change to enabled
|
|
if (enableResult.clicked) {
|
|
await page.waitForFunction(
|
|
(name) => {
|
|
const cards = Array.from(
|
|
document.querySelectorAll('plugin-management mat-card'),
|
|
);
|
|
const targetCard = cards.find((card) => {
|
|
const title = card.querySelector('mat-card-title')?.textContent || '';
|
|
return title.includes(name);
|
|
});
|
|
const toggle = targetCard?.querySelector(
|
|
'mat-slide-toggle button[role="switch"]',
|
|
) as HTMLButtonElement;
|
|
return toggle?.getAttribute('aria-checked') === 'true';
|
|
},
|
|
'API Test Plugin',
|
|
{ timeout: 10000 },
|
|
);
|
|
}
|
|
|
|
// Ensure plugin management is visible in viewport
|
|
await page.evaluate(() => {
|
|
const pluginMgmt = document.querySelector('plugin-management');
|
|
if (pluginMgmt) {
|
|
pluginMgmt.scrollIntoView({ behavior: 'instant', block: 'center' });
|
|
}
|
|
});
|
|
|
|
// Navigate to plugin management - check for attachment first
|
|
await expect(page.locator(PLUGIN_CARD).first()).toBeAttached({ timeout: 20000 });
|
|
|
|
// Check example plugin is loaded and enabled
|
|
const pluginCardsResult = await page.evaluate(() => {
|
|
const cards = Array.from(document.querySelectorAll('plugin-management mat-card'));
|
|
const pluginCards = cards.filter((card) => card.querySelector('mat-slide-toggle'));
|
|
return {
|
|
totalCards: cards.length,
|
|
pluginCardsCount: pluginCards.length,
|
|
pluginTitles: pluginCards.map(
|
|
(card) => card.querySelector('mat-card-title')?.textContent?.trim() || '',
|
|
),
|
|
};
|
|
});
|
|
|
|
expect(pluginCardsResult.pluginCardsCount).toBeGreaterThanOrEqual(1);
|
|
expect(pluginCardsResult.pluginTitles).toContain('API Test Plugin');
|
|
|
|
// Navigate back to work view to see plugin menu
|
|
await page.click('text=Today');
|
|
await page.waitForLoadState('networkidle');
|
|
await expect(page).toHaveURL(/\/#\/tag\/TODAY/);
|
|
|
|
// Verify plugin menu entry exists
|
|
const pluginNavItem = page
|
|
.locator(PLUGIN_NAV_ENTRIES)
|
|
.filter({ hasText: 'API Test Plugin' });
|
|
const pluginMenuVisible = await pluginNavItem.isVisible().catch(() => false);
|
|
if (pluginMenuVisible) {
|
|
await expect(pluginNavItem).toContainText('API Test Plugin');
|
|
} else {
|
|
console.log(
|
|
'Plugin menu not visible - may not be implemented or plugin not fully loaded',
|
|
);
|
|
}
|
|
|
|
// Try to open plugin iframe view if menu is available
|
|
if (pluginMenuVisible) {
|
|
await pluginNavItem.click();
|
|
await expect(page.locator(PLUGIN_IFRAME)).toBeVisible({ timeout: 10000 });
|
|
await expect(page).toHaveURL(/\/plugins\/api-test-plugin\/index/);
|
|
|
|
// Switch to iframe context and verify content
|
|
const frame = page.frameLocator(PLUGIN_IFRAME);
|
|
await expect(frame.locator('h1')).toBeVisible({ timeout: 10000 });
|
|
await expect(frame.locator('h1')).toContainText('API Test Plugin');
|
|
} else {
|
|
console.log('Skipping iframe test - plugin menu not available');
|
|
}
|
|
});
|
|
|
|
test('disable and re-enable plugin', async ({ page, workViewPage }) => {
|
|
// Increase timeout to account for asset checking and plugin (un)load
|
|
test.setTimeout(process.env.CI ? 90000 : 60000);
|
|
|
|
// Check if plugin assets are available
|
|
const assetsAvailable = await waitForPluginAssets(page);
|
|
if (!assetsAvailable) {
|
|
if (process.env.CI) {
|
|
test.skip(true, 'Plugin assets not available in CI - skipping test');
|
|
return;
|
|
}
|
|
throw new Error('Plugin assets not available - cannot proceed with test');
|
|
}
|
|
|
|
await workViewPage.waitForTaskList();
|
|
|
|
// Use improved plugin management init that handles navigation and setup
|
|
const pluginReady = await waitForPluginManagementInit(page);
|
|
if (!pluginReady) {
|
|
throw new Error('Plugin management could not be initialized');
|
|
}
|
|
|
|
// Enable the plugin first using the helper function
|
|
const enableResult = await enablePluginWithVerification(
|
|
page,
|
|
'API Test Plugin',
|
|
15000,
|
|
);
|
|
expect(enableResult).toBe(true);
|
|
|
|
// Wait for plugin card to be stable before disabling
|
|
await page.waitForTimeout(500);
|
|
|
|
// Ensure plugin management is still visible and scroll to it
|
|
await page.evaluate(() => {
|
|
const pluginMgmt = document.querySelector('plugin-management');
|
|
if (pluginMgmt) {
|
|
pluginMgmt.scrollIntoView({ behavior: 'instant', block: 'center' });
|
|
}
|
|
});
|
|
|
|
// Navigate to plugin management - check for attachment
|
|
await expect(page.locator(PLUGIN_ITEM).first()).toBeAttached({ timeout: 10000 });
|
|
|
|
// Wait a bit for any animations to complete
|
|
await page.waitForTimeout(300);
|
|
|
|
// Disable the plugin using the helper function
|
|
const disableResult = await disablePluginWithVerification(
|
|
page,
|
|
'API Test Plugin',
|
|
15000,
|
|
);
|
|
|
|
// If disable failed, try reinitializing plugin management and retry
|
|
if (!disableResult) {
|
|
console.log('[Plugin Test] First disable attempt failed, retrying...');
|
|
const retryReady = await waitForPluginManagementInit(page);
|
|
if (retryReady) {
|
|
const retryDisable = await disablePluginWithVerification(
|
|
page,
|
|
'API Test Plugin',
|
|
15000,
|
|
);
|
|
expect(retryDisable).toBe(true);
|
|
} else {
|
|
expect(disableResult).toBe(true); // Will fail with original error
|
|
}
|
|
}
|
|
|
|
// Wait for disable to take effect
|
|
await page.waitForTimeout(500);
|
|
|
|
// Re-enable the plugin using the helper function
|
|
const reEnableResult = await enablePluginWithVerification(
|
|
page,
|
|
'API Test Plugin',
|
|
15000,
|
|
);
|
|
expect(reEnableResult).toBe(true);
|
|
|
|
// Navigate back to main view
|
|
await page.click('text=Today'); // Click on Today navigation
|
|
await expect(page).toHaveURL(/\/#\/tag\/TODAY/, { timeout: 10000 });
|
|
|
|
// Check if menu entry is back (gracefully handle if not visible)
|
|
const pluginNavItemReEnabled = page
|
|
.locator(PLUGIN_NAV_ENTRIES)
|
|
.filter({ hasText: 'API Test Plugin' });
|
|
const pluginMenuVisible = await pluginNavItemReEnabled.isVisible().catch(() => false);
|
|
if (pluginMenuVisible) {
|
|
await expect(pluginNavItemReEnabled).toContainText('API Test Plugin');
|
|
console.log('Plugin menu entry verified after re-enable');
|
|
} else {
|
|
console.log('Plugin menu not visible after re-enable - may not be implemented');
|
|
}
|
|
});
|
|
});
|