{ "enabled": true, "id": "sales-quotation", "name": "견적관리 테스트", "screenshotPolicy": { "onErrorOnly": true, "captureOn": [ "error", "fail", "timeout", "404", "500", "blocked" ] }, "description": "판매관리 > 견적관리 메뉴의 견적서 조회/등록/수정/삭제 전체 CRUD 테스트", "baseUrl": "https://dev.codebridge-x.com", "menuNavigation": { "level1": "판매관리", "level2": "견적관리", "expectedUrl": "/sales/quote-management", "searchWithinParent": true, "closeOtherMenus": true }, "auth": { "username": "TestUser5", "password": "password123!" }, "testData": { "create": { "clientName": "E2E_TEST_거래처", "itemName": "테스트품목", "quantity": "100", "unitPrice": "10000", "memo": "E2E 자동화 테스트 견적" }, "update": { "quantity": "150", "memo": "E2E 수정된 견적 메모" } }, "steps": [ { "id": 1, "name": "메뉴 진입: 판매관리 > 견적관리", "action": "menu_navigate", "level1": "판매관리", "level2": "견적관리", "expected": { "url_contains": "/sales/quote", "visible": [ "견적관리", "견적" ] } }, { "id": 2, "name": "URL 검증", "action": "verify_url", "expected": { "url_contains": "/sales/quote-management" } }, { "id": 3, "name": "필수 검증 #5: 목업 페이지 감지", "action": "verify_not_mockup", "checks": [ "견적 목록 표시", "견적 등록 버튼 존재", "검색/필터 기능 존재" ], "expected": "정상 페이지 (목업 아님)" }, { "id": 4, "name": "통계 카드 확인", "action": "evaluate", "script": "(() => {\n const cards = document.querySelectorAll('[class*=\"card\"], [class*=\"Card\"], [class*=\"stat\"], [class*=\"Stat\"], [class*=\"summary\"]');\n const texts = Array.from(cards).map(c => c.innerText?.substring(0, 30)).filter(Boolean);\n return texts.length > 0 ? 'Stats: ' + texts.length + ' cards found' : 'No stat cards (ok)';\n })()" }, { "id": 5, "name": "견적 테이블 구조 확인", "action": "verify_table", "checks": [ "견적번호 컬럼", "거래처 컬럼", "품목 컬럼", "금액 컬럼", "상태 컬럼", "작성일 컬럼" ], "expected": "견적 테이블 컬럼 정상 표시" }, { "id": 6, "name": "목록 필터 테스트", "action": "evaluate", "script": "(() => {\n const selects = document.querySelectorAll('select, [role=\"combobox\"], button[class*=\"select\"], button[class*=\"Select\"]');\n if (selects.length > 0) {\n return 'Filters found: ' + selects.length;\n }\n return 'No filter dropdowns (ok)';\n })()" }, { "id": 7, "name": "검색 기능 테스트", "action": "click_if_exists", "target": "input[placeholder*='검색']", "value": "테스트", "expected": { "data_filtered": true } }, { "id": 8, "phase": "CREATE", "name": "[CREATE] 견적 등록 버튼 클릭", "action": "click_if_exists", "target": "button:has-text('등록'), button:has-text('견적 등록'), button:has-text('추가')", "expected": { "modal_or_page": true, "title": "견적 등록" } }, { "id": 9, "phase": "CREATE", "name": "[CREATE] 견적 정보 입력", "action": "fill_form", "fields": [ { "name": "현장명", "value": "E2E_TEST_현장_{timestamp}" }, { "name": "담당자", "value": "E2E 담당자" }, { "name": "연락처", "value": "010-1234-5678" }, { "name": "비고", "value": "E2E 자동화 테스트 견적" } ], "note": "combobox(수주처,부가세,제품코드 등)는 fill_form 미지원" }, { "id": 10, "phase": "CREATE", "name": "[CREATE] 필수 검증 #2: 등록 저장", "action": "click_if_exists", "target": "button:has-text('저장'), button:has-text('등록')", "verify": { "url_maintained": true, "no_error_page": true, "api_call": "POST /api/v1/quotations", "toast": "등록|완료|성공" }, "expected": "견적 등록 완료" }, { "id": 11, "phase": "CREATE", "name": "[CREATE] 저장 완료 토스트 확인", "action": "verify_toast", "verify": { "contains": "등록|완료|성공|저장" } }, { "id": 12, "phase": "CREATE", "name": "[CREATE] 모달 닫기 확인", "action": "close_modal_if_open", "expected": "모달 닫힘" }, { "id": 13, "phase": "CREATE", "name": "[CREATE] 등록 결과 확인", "action": "verify_detail", "search": "E2E 자동화 테스트 견적", "expected": { "row_exists": true, "contains": [ "E2E", "1,000,000" ] } }, { "id": 14, "phase": "READ", "name": "[READ] 견적 상세 페이지 진입", "action": "click_if_exists", "target": "table tbody tr:first-child, table tbody tr:nth-child(1), table tr:nth-child(2)", "expected": { "url_contains": "/sales/quote", "visible": [ "견적 상세", "수정", "삭제" ] } }, { "id": 15, "phase": "READ", "name": "[READ] 상세 정보 확인", "action": "verify_detail", "checks": [ "거래처: E2E_TEST_거래처", "수량: 100", "금액: 1,000,000" ], "expected": "입력한 데이터와 일치" }, { "id": 16, "phase": "UPDATE", "name": "[UPDATE] 수정 모드 진입", "action": "click_if_exists", "target": "button:has-text('수정')", "expected": { "url_contains": "mode=edit", "fields_editable": true } }, { "id": 17, "phase": "UPDATE", "name": "[UPDATE] 수량 수정", "action": "click_if_exists", "target": "input[name*='quantity'], input[placeholder*='수량']" }, { "id": 18, "phase": "UPDATE", "name": "[UPDATE] 메모 수정", "action": "click_if_exists", "target": "textarea[name*='memo'], input[placeholder*='메모']" }, { "id": 19, "phase": "UPDATE", "name": "[UPDATE] 필수 검증 #2: 수정 저장", "action": "click_if_exists", "target": "button:has-text('저장')", "verify": { "url_maintained": true, "no_error_page": true, "api_call": "PUT /api/v1/quotations/", "toast": "수정|완료|성공" }, "expected": "수정 완료" }, { "id": 20, "phase": "UPDATE", "name": "[UPDATE] 수정 완료 토스트 확인", "action": "verify_toast", "verify": { "contains": "수정|완료|성공|저장" } }, { "id": 21, "phase": "UPDATE", "name": "[UPDATE] 수정 결과 확인", "action": "verify_detail", "checks": [ "수량: 150", "금액: 1,500,000" ], "expected": "수정된 데이터 반영" }, { "id": 22, "phase": "DELETE", "name": "[DELETE] 삭제 버튼 클릭", "action": "click_if_exists", "target": "button:has-text('삭제')", "expected": { "confirm_dialog": true, "dialog_message": "삭제|정말" } }, { "id": 23, "phase": "DELETE", "name": "[DELETE] 필수 검증 #6: 삭제 확인", "action": "click_if_exists", "verify": { "api_call": "DELETE /api/v1/quotations/", "toast": "삭제|완료|성공", "redirect": "/sales/quote" }, "expected": "삭제 완료 및 목록 복귀", "target": "[role='alertdialog'] button:has-text('확인'), [role='dialog'] button:has-text('확인'), button:has-text('확인')" }, { "id": 24, "phase": "DELETE", "name": "[DELETE] 삭제 결과 확인", "action": "verify_detail", "search": "E2E 수정된 견적", "expected": { "row_exists": false, "message": "테스트 견적이 목록에서 제거됨" } }, { "id": 25, "name": "콘솔 에러 확인", "action": "verify_element", "target": "body" } ], "expectedAPIs": [ { "method": "GET", "endpoint": "/api/v1/quotations", "description": "견적 목록 조회" }, { "method": "POST", "endpoint": "/api/v1/quotations", "description": "견적 등록" }, { "method": "GET", "endpoint": "/api/v1/quotations/{id}", "description": "견적 상세 조회" }, { "method": "PUT", "endpoint": "/api/v1/quotations/{id}", "description": "견적 수정" }, { "method": "DELETE", "endpoint": "/api/v1/quotations/{id}", "description": "견적 삭제" } ], "requiredVerifications": [ { "id": 2, "name": "등록/저장 버튼", "steps": [ 7, 14 ], "criteria": "API 호출 + 성공 토스트 + 데이터 반영" }, { "id": 3, "name": "검색/필터", "steps": [ 4 ], "criteria": "검색 기능 동작" }, { "id": 5, "name": "목업 페이지 감지", "steps": [ 2 ], "criteria": "견적 목록, 등록 버튼, 필터 존재" }, { "id": 6, "name": "삭제 기능", "steps": [ 16, 17, 18 ], "criteria": "DELETE API + 목록에서 제거" } ], "rollbackPlan": { "onCreateFail": "모달 닫기", "onUpdateFail": "테스트 견적 수동 삭제 필요", "onDeleteFail": "테스트 견적 수동 삭제 필요", "cleanupRequired": "E2E_TEST_ 접두사 견적은 테스트 데이터" } }