Files
jongryangje/e2e/qa_sweep.spec.js

96 lines
4.1 KiB
JavaScript
Raw Normal View History

const { test, expect } = require('@playwright/test');
const { login } = require('./helpers/auth');
/**
* 전체 점검 스윕 주요 화면 콘솔에러·차단 오버레이 점검 + 매뉴얼/도움말/워크스페이스 통합 검증.
*/
async function selectDaegu(page) {
await page.goto('/admin/select-local-government');
await page.evaluate(() => {
const r = document.querySelector('input[name="lg_idx"][value="1"]');
if (r) { r.checked = true; r.form.submit(); }
});
await page.waitForTimeout(700);
}
// kakao 외부 SDK 관련(도메인 미등록 환경) 잡음은 제외
function appError(msg) {
return !/kakao|dapi\.kakao|sdk\.js|OPEN_MAP_AND_LOCAL|appkey/i.test(msg);
}
const PAGES = [
'/', '/bag/inventory', '/bag/order/create', '/bag/bag-orders',
'/bag/receiving/scanner', '/bag/receiving/batch', '/bag/sale/designated',
'/bag/issue/create', '/bag/issue', '/bag/flow', '/bag/sales',
'/bag/reports/daily-summary', '/bag/reports/lot-flow', '/bag/reports/returns',
'/bag/bag-prices', '/bag/packaging-units', '/bag/code-kinds',
'/bag/designated-shops', '/bag/designated-shops/browse', '/bag/number-lookup',
'/bag/manual', '/admin', '/admin/menus', '/admin/users', '/admin/access/login-history',
];
test.describe('QA 스윕', () => {
test('주요 화면 콘솔 에러·차단 오버레이 점검', async ({ page }) => {
await login(page, 'admin');
await selectDaegu(page);
const problems = [];
for (const url of PAGES) {
const errs = [];
page.removeAllListeners('pageerror');
page.on('pageerror', (e) => { if (appError(String(e))) errs.push(String(e)); });
const res = await page.goto(url, { waitUntil: 'domcontentloaded' });
await page.waitForTimeout(700);
const status = res ? res.status() : 0;
const cover = await page.evaluate(() => {
const cx = Math.floor(innerWidth / 2), cy = Math.floor(innerHeight / 2);
let n = 0;
document.querySelectorAll('*').forEach((el) => {
const s = getComputedStyle(el);
if ((s.position === 'fixed' || s.position === 'absolute') && s.display !== 'none' &&
s.visibility !== 'hidden' && parseFloat(s.opacity || '1') > 0.1 && s.pointerEvents !== 'none') {
const r = el.getBoundingClientRect();
if (r.width >= innerWidth * 0.85 && r.height >= innerHeight * 0.7 &&
!/portal-header|sidebar|ws-/.test(el.className || '')) n++;
}
});
return n;
});
if (status >= 400) problems.push(`${url} → HTTP ${status}`);
if (errs.length) problems.push(`${url} → JS오류: ${errs[0]}`);
if (cover > 0) problems.push(`${url} → 화면 덮는 오버레이 ${cover}`);
}
console.log('>>> SWEEP problems=' + JSON.stringify(problems, null, 0));
expect(problems, problems.join('\n')).toEqual([]);
});
test('매뉴얼 전체 페이지 렌더', async ({ page }) => {
await login(page, 'user');
const slugs = ['overview', 'flow', 'order', 'inventory', 'sales', 'reports', 'basic', 'codes', 'faq'];
for (const s of slugs) {
const res = await page.goto('/bag/manual/' + s, { waitUntil: 'domcontentloaded' });
expect(res.status(), s).toBe(200);
await expect(page.locator('.manual-prose')).not.toBeEmpty();
}
// 미등록 slug → 404
const bad = await page.goto('/bag/manual/zzz-none', { waitUntil: 'domcontentloaded' });
expect(bad.status()).toBe(404);
});
test('이 화면 설명 매핑 정확', async ({ page }) => {
await login(page, 'admin');
await selectDaegu(page);
const cases = [
['/bag/inventory', 'inventory'],
['/bag/order/create', 'order'],
['/bag/sale/designated', 'sales'],
['/bag/flow', 'reports'],
['/bag/bag-prices', 'basic'],
['/bag/number-lookup', 'codes'],
];
for (const [url, slug] of cases) {
await page.goto(url, { waitUntil: 'domcontentloaded' });
const href = await page.locator('a.no-print', { hasText: '이 화면 설명' }).first().getAttribute('href');
expect(href, url).toContain('/bag/manual/' + slug);
}
});
});