/** * UI smoke test - launches the real app and checks DOM elements via webContents. * Run with: node tests/ui-smoke.js * (This spawns Electron as a child process) */ const { execSync } = require('child_process'); const path = require('path'); const fs = require('fs'); // Create a temp script that the real Electron app will execute via --eval const testScript = ` const { app, BrowserWindow } = require('electron'); // Monkey-patch: after the real window loads, run tests const origReady = app.whenReady; async function runAfterDelay(win, delayMs) { await new Promise(r => setTimeout(r, delayMs)); return win; } // Wait for app to be ready, then wait for the real window to load setTimeout(async () => { const windows = BrowserWindow.getAllWindows(); if (windows.length === 0) { console.log('ERROR: No windows found'); process.exit(1); } const win = windows[0]; const wc = win.webContents; // Wait for renderer init await new Promise(r => setTimeout(r, 2000)); let passed = 0; let failed = 0; const results = []; function check(name, condition) { if (condition) { passed++; results.push(' PASS: ' + name); } else { failed++; results.push(' FAIL: ' + name); } } try { console.log('\\n=== Upload View ==='); const tabCount = await wc.executeJavaScript('document.querySelectorAll(".tab").length'); check('3 tabs exist', tabCount === 3); const activeTab = await wc.executeJavaScript('document.querySelector(".tab.active")?.textContent?.trim()'); check('Upload tab active by default', activeTab === 'Upload'); const dropVisible = await wc.executeJavaScript('document.getElementById("dropZone")?.style.display !== "none"'); check('Drop zone visible (no files)', dropVisible); const queueHidden = await wc.executeJavaScript('document.getElementById("queueContainer")?.style.display'); check('Queue hidden (no files)', queueHidden === 'none'); const chips = await wc.executeJavaScript('document.querySelectorAll(".hoster-chip").length'); check('4 hoster chips', chips === 4); const startDisabled = await wc.executeJavaScript('document.getElementById("startUploadBtn")?.disabled'); check('Start button disabled initially', startDisabled === true); const sbState = await wc.executeJavaScript('document.getElementById("sbState")?.textContent'); check('Statusbar: Bereit', sbState === 'Bereit'); const version = await wc.executeJavaScript('document.getElementById("versionLabel")?.textContent'); check('Version label present', version && version.startsWith('v')); const ctxHidden = await wc.executeJavaScript('document.getElementById("contextMenu")?.style.display'); check('Context menu hidden', ctxHidden === 'none'); console.log('\\n=== Settings View ==='); await wc.executeJavaScript('document.querySelector(".tab[data-view=\\'settings\\']").click()'); await new Promise(r => setTimeout(r, 300)); const settingsActive = await wc.executeJavaScript('document.getElementById("settings-view")?.classList.contains("active")'); check('Settings tab active', settingsActive); const panels = await wc.executeJavaScript('document.querySelectorAll(".hoster-settings-panel").length'); check('4 hoster panels', panels === 4); const hsInputCount = await wc.executeJavaScript('document.querySelectorAll(".hs-input").length'); check('24 per-hoster inputs (6x4)', hsInputCount === 24); await wc.executeJavaScript('document.querySelector(".hoster-panel-header").click()'); await new Promise(r => setTimeout(r, 200)); const panelBody = await wc.executeJavaScript('document.querySelector(".hoster-panel-body").style.display'); check('Panel expands on click', panelBody !== 'none'); const retries = await wc.executeJavaScript('document.querySelector(".hs-input[data-hs=\\'retries\\']")?.value'); check('Retries default 3', retries === '3'); const parallel = await wc.executeJavaScript('document.querySelector(".hs-input[data-hs=\\'parallelCount\\']")?.value'); check('ParallelCount default 2', parallel === '2'); // Test save await wc.executeJavaScript('document.getElementById("saveSettingsBtn").click()'); await new Promise(r => setTimeout(r, 500)); const feedback = await wc.executeJavaScript('document.getElementById("saveFeedback")?.textContent'); check('Save shows Gespeichert!', feedback === 'Gespeichert!'); console.log('\\n=== History View ==='); await wc.executeJavaScript('document.querySelector(".tab[data-view=\\'history\\']").click()'); await new Promise(r => setTimeout(r, 1000)); // Wait for async loadHistory const historyActive = await wc.executeJavaScript('document.getElementById("history-view")?.classList.contains("active")'); check('History tab active', historyActive); const emptyState = await wc.executeJavaScript('document.querySelector("#historyContainer .empty-state")?.textContent'); check('Empty state or history table shown', emptyState === 'Noch keine Uploads.' || emptyState === undefined); console.log('\\n=== Global UI ==='); const shutdownHidden = await wc.executeJavaScript('document.getElementById("shutdownOverlay")?.style.display'); check('Shutdown overlay hidden', shutdownHidden === 'none'); const toastHidden = await wc.executeJavaScript('!document.getElementById("copyToast")?.classList.contains("show")'); check('Copy toast hidden', toastHidden); const updateHidden = await wc.executeJavaScript('document.getElementById("updateBanner")?.style.display'); check('Update banner hidden', updateHidden === 'none'); } catch (err) { console.error('Test error:', err.message); failed++; } console.log('\\n=== Results ==='); results.forEach(r => console.log(r)); console.log('\\nTotal: ' + (passed + failed) + ' | Passed: ' + passed + ' | Failed: ' + failed); if (failed > 0) process.exitCode = 1; app.quit(); }, 5000); `; // Write the injection script const injectPath = path.join(__dirname, '_ui-inject.tmp.js'); fs.writeFileSync(injectPath, testScript, 'utf-8'); // Run the real app with the injection try { const electronPath = path.join(__dirname, '..', 'node_modules', '.bin', 'electron'); const mainPath = path.join(__dirname, '..', 'main.js'); // We'll use --require to inject the test after the main process loads const result = execSync( `"${electronPath}" --require "${injectPath}" "${mainPath}"`, { cwd: path.join(__dirname, '..'), timeout: 20000, encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] } ); console.log(result); } catch (err) { // timeout or exit code - still print output if (err.stdout) console.log(err.stdout); if (err.stderr) { const filtered = err.stderr.split('\n') .filter(l => !l.includes('cache_util') && !l.includes('disk_cache') && !l.includes('gpu_disk_cache')) .join('\n'); if (filtered.trim()) console.error(filtered); } if (err.status && err.status !== 0 && !err.killed) { process.exit(err.status); } } finally { try { fs.unlinkSync(injectPath); } catch {} }