2026-01-30 10:50:38 +09:00
|
|
|
{
|
|
|
|
|
"id": "employee-register",
|
|
|
|
|
"name": "직원 등록 테스트",
|
|
|
|
|
"screenshotPolicy": {
|
|
|
|
|
"onErrorOnly": true,
|
2026-02-09 15:05:03 +09:00
|
|
|
"captureOn": [
|
|
|
|
|
"error",
|
|
|
|
|
"fail",
|
|
|
|
|
"timeout",
|
|
|
|
|
"404",
|
|
|
|
|
"500",
|
|
|
|
|
"blocked"
|
|
|
|
|
]
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
"description": "신규 직원 정보를 입력하고 등록하는 E2E 테스트",
|
|
|
|
|
"baseUrl": "https://dev.codebridge-x.com",
|
|
|
|
|
"url": "/ko/hr/employee-management",
|
2026-01-30 21:47:29 +09:00
|
|
|
"navigation": {
|
|
|
|
|
"targetUrl": "/hr/employee-management",
|
|
|
|
|
"urlPattern": "/hr/employee-management|/ko/hr/employee-management",
|
2026-02-09 15:05:03 +09:00
|
|
|
"menuHints": [
|
|
|
|
|
"사원관리",
|
|
|
|
|
"사원 관리",
|
|
|
|
|
"인사관리"
|
|
|
|
|
]
|
2026-01-30 21:47:29 +09:00
|
|
|
},
|
2026-01-30 10:50:38 +09:00
|
|
|
"menuNavigation": {
|
|
|
|
|
"level1": "인사관리",
|
|
|
|
|
"level2": "사원관리",
|
2026-02-04 21:59:56 +09:00
|
|
|
"expectedUrl": "/ko/hr/employee-management",
|
|
|
|
|
"searchWithinParent": true,
|
|
|
|
|
"closeOtherMenus": true
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
"menuNavigationEnhanced": {
|
|
|
|
|
"strategy": "scroll-and-search",
|
|
|
|
|
"level1": {
|
|
|
|
|
"text": "인사관리",
|
|
|
|
|
"scrollContainer": ".sidebar-scroll, [class*='sidebar'], nav",
|
|
|
|
|
"maxScrollAttempts": 5,
|
|
|
|
|
"scrollStep": 200
|
|
|
|
|
},
|
|
|
|
|
"level2": {
|
|
|
|
|
"text": "사원관리",
|
|
|
|
|
"waitAfterLevel1": 500
|
|
|
|
|
},
|
|
|
|
|
"fallbackUrl": "/ko/hr/employee-management",
|
|
|
|
|
"timeout": 10000
|
|
|
|
|
},
|
|
|
|
|
"timeout": 60000,
|
2026-02-09 15:05:03 +09:00
|
|
|
"tags": [
|
|
|
|
|
"hr",
|
|
|
|
|
"employee",
|
|
|
|
|
"crud"
|
|
|
|
|
],
|
2026-02-04 21:59:56 +09:00
|
|
|
"auth": {
|
|
|
|
|
"username": "TestUser5",
|
|
|
|
|
"password": "password123!"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
"steps": [
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 1,
|
2026-01-30 16:26:52 +09:00
|
|
|
"name": "사이드바 메뉴 전체 펼치기",
|
|
|
|
|
"description": "모두 펼치기 버튼을 클릭하여 전체 메뉴를 펼친 후 메뉴 탐색 준비",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "evaluate",
|
|
|
|
|
"script": "(async () => { document.querySelector('.sidebar-scroll, [class*=\"sidebar\"], nav')?.scrollTo({top: 0, behavior: 'instant'}); await new Promise(r => setTimeout(r, 300)); Array.from(document.querySelectorAll('button')).find(b => b.innerText?.includes('모두 펼치기'))?.click(); await new Promise(r => setTimeout(r, 2000)); return 'Menu expanded'; })()"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 2,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "인사관리 메뉴 진입",
|
2026-02-28 17:21:01 +09:00
|
|
|
"description": "인사관리 > 직원관리 메뉴로 이동",
|
|
|
|
|
"action": "menu_navigate",
|
|
|
|
|
"level1": "인사관리",
|
|
|
|
|
"level2": "직원관리"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 3,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "사원 등록 페이지 이동",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "click_if_exists",
|
|
|
|
|
"target": "사원 등록"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 4,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "사원 정보 입력",
|
|
|
|
|
"description": "기본 사원 정보 입력",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "fill_form",
|
|
|
|
|
"fields": [
|
|
|
|
|
{
|
|
|
|
|
"name": "이름 *",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "E2E_TEST_사원"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"name": "주민등록번호",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "900101-1234567"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"name": "휴대폰",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "010-1234-5678"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"name": "이메일 *",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "e2e_test_employee@codebridge-x.com"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"name": "연봉",
|
|
|
|
|
"type": "number",
|
|
|
|
|
"value": "50000000"
|
|
|
|
|
}
|
|
|
|
|
]
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 5,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "급여계좌 정보 입력",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "fill_form",
|
|
|
|
|
"fields": [
|
|
|
|
|
{
|
|
|
|
|
"name": "은행명",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "신한은행"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"name": "계좌번호",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "110-123-456789"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"name": "예금주",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "E2E_TEST_사원"
|
|
|
|
|
}
|
|
|
|
|
]
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 6,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "사원 상세 정보 입력",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "fill_form",
|
|
|
|
|
"fields": [
|
2026-02-09 15:05:03 +09:00
|
|
|
{
|
2026-02-28 17:21:01 +09:00
|
|
|
"name": "사원코드",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "E2E_TEST_EMP001"
|
2026-02-09 15:05:03 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-28 17:21:01 +09:00
|
|
|
"name": "남성",
|
|
|
|
|
"type": "radio",
|
|
|
|
|
"value": "true"
|
2026-02-09 15:05:03 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-28 17:21:01 +09:00
|
|
|
"name": "상세주소를 입력해주세요",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "123번지 4층"
|
2026-02-09 15:05:03 +09:00
|
|
|
}
|
2026-01-30 10:50:38 +09:00
|
|
|
]
|
|
|
|
|
},
|
2026-02-28 17:21:01 +09:00
|
|
|
{
|
|
|
|
|
"id": 7,
|
|
|
|
|
"name": "인사 정보 입력",
|
|
|
|
|
"action": "evaluate",
|
|
|
|
|
"script": "(async () => { const fillField = (label, value) => { const labels = Array.from(document.querySelectorAll('label, span, div')); const found = labels.find(l => l.innerText?.includes(label)); if (found) { const input = found.closest('.form-group, .field, [class*=field], [class*=form]')?.querySelector('input, textarea, select') || found.parentElement?.querySelector('input, textarea, select'); if (input) { input.focus(); input.value = value; input.dispatchEvent(new Event('input', {bubbles:true})); input.dispatchEvent(new Event('change', {bubbles:true})); return true; } } return false; }; fillField('입사일', '2026-01-14'); await new Promise(r => setTimeout(r, 300)); const clickText = (text) => { const el = Array.from(document.querySelectorAll('button, [role=button], [role=option], li, div[class*=option], span')).find(e => e.innerText?.trim() === text || e.innerText?.includes(text)); if (el) { el.click(); return true; } return false; }; clickText('고용형태 선택'); await new Promise(r => setTimeout(r, 300)); clickText('정규직'); await new Promise(r => setTimeout(r, 300)); clickText('직급 선택'); await new Promise(r => setTimeout(r, 300)); clickText('사원'); await new Promise(r => setTimeout(r, 300)); return 'HR info filled'; })()"
|
|
|
|
|
},
|
2026-01-30 10:50:38 +09:00
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 8,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "사용자 정보 입력",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "fill_form",
|
|
|
|
|
"fields": [
|
|
|
|
|
{
|
|
|
|
|
"name": "아이디 *",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "e2e_test_user001"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"name": "비밀번호 *",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "password123!"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"name": "비밀번호 확인 *",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "password123!"
|
|
|
|
|
}
|
|
|
|
|
]
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 9,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "등록 완료",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "click_if_exists",
|
|
|
|
|
"target": "등록"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 10,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "검색 기간 설정 - 유효 기간",
|
|
|
|
|
"description": "등록된 사원의 입사일(2026-01-14)이 포함되는 기간으로 검색",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "evaluate",
|
|
|
|
|
"script": "(async () => { const startInput = document.querySelector(\"input[placeholder*='시작'], input[name*='startDate'], input[type='date']:first-of-type\"); if (startInput) { startInput.click(); await new Promise(r => setTimeout(r, 200)); } const endInput = document.querySelector(\"input[placeholder*='종료'], input[name*='endDate'], input[type='date']:last-of-type\"); if (endInput) { endInput.click(); await new Promise(r => setTimeout(r, 200)); } const searchBtn = Array.from(document.querySelectorAll('button')).find(b => b.innerText?.includes('검색')); if (searchBtn) searchBtn.click(); await new Promise(r => setTimeout(r, 1000)); return document.body.innerText.includes('E2E_TEST_사원') ? 'PASS: Found in search results' : 'WARN: Not found in search results'; })()",
|
2026-01-30 10:50:38 +09:00
|
|
|
"onFail": {
|
|
|
|
|
"record": true,
|
|
|
|
|
"message": "검색 기간 2026-01-01 ~ 2026-01-31 내 입사일(2026-01-14) 사원이 검색되지 않음",
|
|
|
|
|
"severity": "HIGH",
|
|
|
|
|
"bugType": "검색 기간 필터링 오류"
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 11,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "검색 기간 설정 - 범위 외 기간",
|
|
|
|
|
"description": "등록된 사원의 입사일이 포함되지 않는 기간으로 검색하여 검색되지 않음을 확인",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "evaluate",
|
|
|
|
|
"script": "(async () => { const startInput = document.querySelector(\"input[placeholder*='시작'], input[name*='startDate'], input[type='date']:first-of-type\"); if (startInput) { startInput.click(); await new Promise(r => setTimeout(r, 200)); } const endInput = document.querySelector(\"input[placeholder*='종료'], input[name*='endDate'], input[type='date']:last-of-type\"); if (endInput) { endInput.click(); await new Promise(r => setTimeout(r, 200)); } const searchBtn = Array.from(document.querySelectorAll('button')).find(b => b.innerText?.includes('검색')); if (searchBtn) searchBtn.click(); await new Promise(r => setTimeout(r, 1000)); return !document.body.innerText.includes('E2E_TEST_사원') ? 'PASS: Not found (expected)' : 'WARN: Still found in out-of-range period'; })()",
|
2026-01-30 10:50:38 +09:00
|
|
|
"onFail": {
|
|
|
|
|
"record": true,
|
|
|
|
|
"message": "검색 기간 2025-01-01 ~ 2025-12-31 범위 외 입사일(2026-01-14) 사원이 검색됨 - 기간 필터 미작동",
|
|
|
|
|
"severity": "HIGH",
|
|
|
|
|
"bugType": "검색 기간 필터링 미작동"
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 12,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "검색 기간 초기화 및 전체 조회",
|
|
|
|
|
"description": "검색 조건 초기화하여 등록된 사원이 다시 표시되는지 확인",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "evaluate",
|
|
|
|
|
"script": "(async () => { const resetBtn = Array.from(document.querySelectorAll('button')).find(b => b.innerText?.includes('초기화') || b.innerText?.includes('Reset')); if (resetBtn) { resetBtn.click(); await new Promise(r => setTimeout(r, 500)); } const searchBtn = Array.from(document.querySelectorAll('button')).find(b => b.innerText?.includes('검색')); if (searchBtn) { searchBtn.click(); await new Promise(r => setTimeout(r, 1000)); } return document.body.innerText.includes('E2E_TEST_사원') ? 'PASS: Found after reset' : 'WARN: Not found after reset'; })()",
|
2026-01-30 10:50:38 +09:00
|
|
|
"onFail": {
|
|
|
|
|
"record": true,
|
|
|
|
|
"message": "검색 초기화 후 전체 조회에서 등록된 사원이 표시되지 않음",
|
|
|
|
|
"severity": "MEDIUM",
|
|
|
|
|
"bugType": "검색 초기화 오류"
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 13,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "등록된 직원 상세 페이지 이동",
|
|
|
|
|
"description": "등록된 직원을 클릭하여 상세 페이지로 이동",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "click_if_exists",
|
|
|
|
|
"target": "table tbody tr:has-text('E2E_TEST_사원')"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 14,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "직원 수정 모드 전환",
|
|
|
|
|
"description": "수정 버튼 클릭하여 편집 모드로 전환",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "click_if_exists",
|
|
|
|
|
"target": "수정"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 15,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "직원 정보 수정",
|
|
|
|
|
"description": "휴대폰 번호 변경",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "fill_form",
|
|
|
|
|
"fields": [
|
|
|
|
|
{
|
|
|
|
|
"name": "휴대폰",
|
|
|
|
|
"type": "text",
|
|
|
|
|
"value": "010-9999-8888",
|
|
|
|
|
"clear": true
|
|
|
|
|
}
|
|
|
|
|
]
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 16,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "수정 저장",
|
|
|
|
|
"description": "수정된 직원 정보 저장",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "click_if_exists",
|
|
|
|
|
"target": "저장"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 17,
|
2026-02-28 17:21:01 +09:00
|
|
|
"name": "필수 검증: 수정 데이터 반영 확인",
|
2026-01-30 10:50:38 +09:00
|
|
|
"note": "토스트 성공 메시지만으로 PASS 판정 불가. 실제 데이터 변경 확인 필수!",
|
|
|
|
|
"description": "상세 페이지에서 수정된 휴대폰 번호 확인",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "verify_detail",
|
|
|
|
|
"checks": [
|
|
|
|
|
"휴대폰: 010-9999-8888"
|
|
|
|
|
]
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 18,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "직원 삭제",
|
|
|
|
|
"description": "삭제 버튼 클릭하여 직원 삭제",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "click_if_exists",
|
|
|
|
|
"target": "삭제"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 19,
|
2026-01-30 10:50:38 +09:00
|
|
|
"name": "삭제 확인",
|
|
|
|
|
"description": "삭제 확인 다이얼로그에서 확인 클릭",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "click_dialog_confirm"
|
2026-01-30 10:50:38 +09:00
|
|
|
},
|
|
|
|
|
{
|
2026-02-09 15:05:03 +09:00
|
|
|
"id": 20,
|
2026-02-28 17:21:01 +09:00
|
|
|
"name": "필수 검증: 삭제 데이터 반영 확인",
|
2026-01-30 10:50:38 +09:00
|
|
|
"note": "토스트 성공 메시지만으로 PASS 판정 불가. 실제 데이터 삭제 확인 필수!",
|
|
|
|
|
"description": "목록에서 삭제된 직원이 없어졌는지 확인",
|
2026-02-28 17:21:01 +09:00
|
|
|
"action": "verify_element",
|
|
|
|
|
"target": "body",
|
2026-01-30 10:50:38 +09:00
|
|
|
"verify": {
|
2026-02-19 21:55:15 +09:00
|
|
|
"tableNotContains": "E2E_TEST_사원"
|
2026-01-30 10:50:38 +09:00
|
|
|
}
|
2026-02-09 15:05:03 +09:00
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"id": 21,
|
|
|
|
|
"name": "콘솔 에러 확인",
|
|
|
|
|
"action": "verify_element",
|
|
|
|
|
"target": "body"
|
2026-01-30 10:50:38 +09:00
|
|
|
}
|
|
|
|
|
],
|
|
|
|
|
"assertions": [
|
|
|
|
|
{
|
|
|
|
|
"type": "url",
|
|
|
|
|
"expected": "/hr/employee-management",
|
|
|
|
|
"message": "등록 후 직원 목록 페이지로 이동해야 함"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"type": "text",
|
|
|
|
|
"target": "body",
|
2026-02-19 21:55:15 +09:00
|
|
|
"expected": "E2E_TEST_사원",
|
2026-01-30 10:50:38 +09:00
|
|
|
"message": "등록된 직원이 목록에 표시되어야 함"
|
|
|
|
|
}
|
|
|
|
|
]
|
2026-02-28 17:21:01 +09:00
|
|
|
}
|