Files
sam-scenarios/login.json
kimbokon 2084439482 fix: login 시나리오 - 쿠키/스토리지 클리어 후 로그인 페이지 이동
로그아웃 클릭 대신 세션 직접 클리어 방식으로 변경
(로그아웃 비동기 리다이렉트 문제 해결)

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-08 01:35:39 +09:00

171 lines
5.7 KiB
JSON

{
"id": "login-test",
"name": "로그인 테스트",
"screenshotPolicy": {
"onErrorOnly": true,
"captureOn": ["error", "fail", "timeout", "404", "500", "blocked"]
},
"description": "로그아웃 → 로그인 페이지 UI 검증 → 로그인 실패/성공 → 대시보드 진입 테스트",
"baseUrl": "https://dev.codebridge-x.com",
"timeout": 30000,
"tags": ["auth", "login", "critical"],
"auth": {
"username": "TestUser5",
"password": "password123!"
},
"steps": [
{
"id": 1,
"name": "세션 클리어 (로그아웃 효과)",
"action": "evaluate",
"script": "(async () => { document.cookie.split(';').forEach(c => { document.cookie = c.trim().split('=')[0] + '=;expires=Thu, 01 Jan 1970 00:00:00 GMT;path=/'; }); try { localStorage.clear(); } catch(e) {} try { sessionStorage.clear(); } catch(e) {} return 'Session cleared'; })()"
},
{
"id": 2,
"name": "로그인 페이지 이동",
"action": "navigate",
"target": "/ko/login"
},
{
"id": 3,
"name": "로그인 페이지 도착 대기",
"action": "wait",
"timeout": 2000
},
{
"id": 4,
"name": "로그인 폼 존재 확인",
"action": "evaluate",
"script": "(() => { const uid = document.querySelector('#userId'); const pwd = document.querySelector('#password'); const submit = document.querySelector('button[type=\"submit\"]'); if (uid && pwd && submit) return 'Login form ready'; return 'Form not found - inputs: ' + document.querySelectorAll('input').length; })()"
},
{
"id": 5,
"name": "필수 검증 #5: 목업 페이지 감지",
"action": "verify_not_mockup",
"checks": [
"아이디 입력 필드 존재 및 입력 가능",
"비밀번호 입력 필드 존재 및 입력 가능",
"로그인 버튼 존재 및 클릭 가능"
],
"expected": "정상 페이지 (목업 아님)"
},
{
"id": 6,
"name": "UI 요소 검증 - 입력 필드",
"action": "evaluate",
"script": "(() => { const uid = document.querySelector('#userId'); const pwd = document.querySelector('#password'); const submit = document.querySelector('button[type=\"submit\"]'); return 'userId: ' + !!uid + ', password: ' + !!pwd + ', submit: ' + !!submit + ', inputs: ' + document.querySelectorAll('input').length; })()"
},
{
"id": 7,
"name": "비밀번호 토글 버튼 테스트",
"action": "evaluate",
"script": "(() => { const pwd = document.querySelector('#password'); if (!pwd) return 'No password field'; const toggleBtn = pwd.parentElement?.querySelector('button') || pwd.nextElementSibling; if (toggleBtn) { toggleBtn.click(); const type1 = pwd.type; toggleBtn.click(); const type2 = pwd.type; return 'Toggle works: ' + type1 + ' → ' + type2; } return 'No toggle button found'; })()"
},
{
"id": 8,
"name": "로그인 실패 테스트 - 빈 필드",
"action": "click",
"target": "button[type='submit']"
},
{
"id": 9,
"name": "빈 필드 제출 후 에러 확인",
"action": "evaluate",
"script": "(() => { const body = document.body.innerText; const hasError = ['필수', '입력', '오류', '실패', 'required'].some(t => body.includes(t)); return 'Error displayed: ' + hasError; })()"
},
{
"id": 10,
"name": "아이디 입력",
"action": "fill",
"target": "#userId",
"value": "TestUser5"
},
{
"id": 11,
"name": "잘못된 비밀번호 입력",
"action": "fill",
"target": "#password",
"value": "wrongpassword"
},
{
"id": 12,
"name": "잘못된 비밀번호로 로그인 시도",
"action": "click",
"target": "button[type='submit']"
},
{
"id": 13,
"name": "로그인 실패 메시지 확인",
"action": "wait",
"timeout": 2000
},
{
"id": 14,
"name": "실패 후 로그인 페이지 유지 확인",
"action": "verify_url",
"expected": {
"url_contains": "/login"
}
},
{
"id": 15,
"name": "비밀번호 필드 초기화",
"action": "clear",
"target": "#password"
},
{
"id": 16,
"name": "올바른 비밀번호 입력",
"action": "fill",
"target": "#password",
"value": "password123!"
},
{
"id": 17,
"name": "로그인 버튼 클릭",
"action": "click",
"target": "button[type='submit']"
},
{
"id": 18,
"name": "대시보드 페이지 도착 대기",
"action": "wait_for_navigation",
"expected": {
"url_contains": "/dashboard"
}
},
{
"id": 19,
"name": "대시보드 UI 확인",
"action": "verify_elements",
"checks": [
"사용자명 '홍킬동' 표시",
"메뉴 영역 표시"
],
"expected": "사용자 정보 및 메인 레이아웃 정상"
},
{
"id": 20,
"name": "세션 유지 확인 - 페이지 새로고침",
"action": "reload"
},
{
"id": 21,
"name": "새로고침 후 대시보드 유지 확인",
"action": "verify_url",
"expected": {
"url_contains": "/dashboard"
}
}
],
"expectedAPIs": [
{ "method": "POST", "endpoint": "/api/v1/auth/login", "description": "로그인 인증" },
{ "method": "GET", "endpoint": "/api/v1/auth/me", "description": "현재 사용자 정보 조회" },
{ "method": "POST", "endpoint": "/api/v1/auth/logout", "description": "로그아웃" }
],
"testData": {
"validUser": { "username": "TestUser5", "password": "password123!", "displayName": "홍킬동" },
"invalidPassword": "wrongpassword"
}
}