{ "id": "full-crud-acc-sales", "name": "Full CRUD 테스트: 매출관리", "version": "1.0.0", "auth": { "role": "admin" }, "menuNavigation": { "level1": "회계관리", "level2": "매출관리" }, "screenshotPolicy": { "captureOnFail": true, "captureOnPass": false }, "steps": [ { "id": 1, "name": "[회계관리 > 매출관리] 페이지 로드 대기", "action": "wait", "timeout": 5000 }, { "id": 2, "name": "[회계관리 > 매출관리] 테이블 로드 대기", "action": "wait_for_table", "timeout": 20000 }, { "id": 3, "name": "[회계관리 > 매출관리] [CREATE] 매출 등록 버튼 클릭", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const R={phase:'CREATE_OPEN'};const btn=Array.from(document.querySelectorAll('button')).find(b=>/매출.*등록|등록/.test(b.innerText?.trim())&&b.offsetParent!==null&&!b.disabled);if(!btn){R.error='매출 등록 버튼 없음';R.ok=false;return JSON.stringify(R);}R.btnText=btn.innerText?.trim();btn.click();await w(2500);R.url=location.pathname+location.search;R.ok=true;return JSON.stringify(R);})()", "timeout": 15000, "phase": "CREATE" }, { "id": 4, "name": "[회계관리 > 매출관리] [CREATE] 등록 폼 로드 대기", "action": "wait", "timeout": 2000 }, { "id": 5, "name": "[회계관리 > 매출관리] [CREATE] 거래처 선택 + 매출유형 + 품목 입력 + 등록", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const sv=(el,v)=>{const p=el.tagName==='TEXTAREA'?HTMLTextAreaElement.prototype:HTMLInputElement.prototype;const ns=Object.getOwnPropertyDescriptor(p,'value')?.set;if(ns)ns.call(el,v);else el.value=v;el.dispatchEvent(new Event('input',{bubbles:true}));el.dispatchEvent(new Event('change',{bubbles:true}));};const ts=window.__E2E_TS__||(()=>{const n=new Date();const p=v=>v.toString().padStart(2,'0');return n.getFullYear()+p(n.getMonth()+1)+p(n.getDate())+'_'+p(n.getHours())+p(n.getMinutes())+p(n.getSeconds());})();window.__E2E_TS__=ts;const toastInfo=()=>{const t=document.querySelectorAll('[data-sonner-toast],[role=\"status\"],[class*=\"toast\"],[class*=\"Toast\"],[class*=\"Toaster\"] [data-content]');return{count:t.length,text:t.length>0?Array.from(t).pop()?.innerText?.trim().substring(0,100):''};};const R={phase:'CREATE',ts};const combos=Array.from(document.querySelectorAll('button[role=\"combobox\"]')).filter(b=>b.offsetParent!==null);R.comboCount=combos.length;for(let i=0;io.innerText?.includes('제품매출'))||opts[0];if(target){target.click();await w(400);R.typeSelected=true;}}else{document.dispatchEvent(new KeyboardEvent('keydown',{key:'Escape',bubbles:true}));await w(200);}break;}}await w(300);const inputs=Array.from(document.querySelectorAll('input[type=\"text\"],input[type=\"number\"],input:not([type])'));const visInputs=inputs.filter(i=>i.offsetParent!==null&&!i.readOnly&&!i.disabled);for(const inp of visInputs){const ph=inp.placeholder||'';const nm=inp.name||'';const lbl=inp.closest('[class*=field],[class*=Field],[class*=form-item]')?.querySelector('label')?.innerText||'';if(ph.includes('품목')||nm.includes('item')||lbl.includes('품목')){sv(inp,'E2E_TEST_품목_'+ts);R.itemFilled=true;await w(200);break;}}for(const inp of visInputs){const ph=inp.placeholder||'';const nm=inp.name||'';const lbl=inp.closest('[class*=field],[class*=Field],[class*=form-item]')?.querySelector('label')?.innerText||'';if(ph.includes('수량')||nm.includes('quantity')||nm.includes('qty')||lbl.includes('수량')){sv(inp,'5');R.qtyFilled=true;await w(200);break;}}for(const inp of visInputs){const ph=inp.placeholder||'';const nm=inp.name||'';const lbl=inp.closest('[class*=field],[class*=Field],[class*=form-item]')?.querySelector('label')?.innerText||'';if(ph.includes('단가')||nm.includes('price')||nm.includes('unitPrice')||lbl.includes('단가')){sv(inp,'100000');R.priceFilled=true;await w(200);break;}}const noteInput=visInputs.find(i=>{const ph=i.placeholder||'';const nm=i.name||'';const lbl=i.closest('[class*=field],[class*=Field],[class*=form-item]')?.querySelector('label')?.innerText||'';return ph.includes('적요')||nm.includes('note')||nm.includes('remark')||lbl.includes('적요');});if(noteInput){sv(noteInput,'E2E_TEST_매출_'+ts);R.noteFilled=true;await w(200);}await w(500);const sub=Array.from(document.querySelectorAll('button')).find(b=>b.innerText?.trim()==='등록'&&b.offsetParent!==null&&!b.disabled);if(!sub){R.error='등록 버튼 없음';return JSON.stringify(R);}sub.click();await w(3000);R.toast=toastInfo();R.ok=true;return JSON.stringify(R);})()", "timeout": 30000, "phase": "CREATE" }, { "id": 6, "name": "[회계관리 > 매출관리] [CREATE] 생성 후 대기", "action": "wait", "timeout": 3000 }, { "id": 7, "name": "[회계관리 > 매출관리] [CREATE] 목록 복귀", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const onForm=location.search.includes('mode=new')||location.search.includes('mode=edit')||location.search.includes('mode=view')||new RegExp('/(new|[0-9]+|[0-9a-f]{8,})$').test(location.pathname);if(onForm){const btn=Array.from(document.querySelectorAll('button,a')).find(b=>/목록|취소|뒤로/.test(b.innerText?.trim()));if(btn){btn.click();await w(2000);}else{history.back();await w(2000);}}return JSON.stringify({url:location.pathname+location.search});})()", "timeout": 10000, "phase": "CREATE" }, { "id": 8, "name": "[회계관리 > 매출관리] [CREATE] 목록 안정화 대기", "action": "wait", "timeout": 2000 }, { "id": 9, "name": "[회계관리 > 매출관리] [VERIFY] 생성 데이터 확인", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||'E2E_TEST_';const R={phase:'VERIFY_CREATE'};await w(500);const rows=document.querySelectorAll('table tbody tr');R.rowCount=rows.length;const found=Array.from(rows).find(r=>r.innerText?.includes('E2E_TEST_'));R.found=!!found;R.ok=R.found;if(found)R.foundText=found.innerText?.substring(0,100);return JSON.stringify(R);})()", "timeout": 15000, "phase": "VERIFY" }, { "id": 10, "name": "[회계관리 > 매출관리] [READ] 상세 페이지 진입", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||'E2E_TEST_';const R={phase:'READ'};let row;for(let i=0;i<3;i++){const rows=Array.from(document.querySelectorAll('table tbody tr'));row=rows.find(r=>r.innerText?.includes('E2E_TEST_'));if(row)break;await w(1000);}if(!row){R.error='E2E_TEST_ 행 없음';R.ok=false;return JSON.stringify(R);}row.click();await w(2500);R.detailUrl=location.pathname+location.search;R.ok=true;return JSON.stringify(R);})()", "timeout": 15000, "phase": "READ" }, { "id": 11, "name": "[회계관리 > 매출관리] [READ] 상세 페이지 대기", "action": "wait", "timeout": 2000 }, { "id": 12, "name": "[회계관리 > 매출관리] [READ] 상세 데이터 검증 (품목/수량/단가/공급가액)", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||'E2E_TEST_';const R={phase:'READ_VERIFY'};R.url=location.pathname+location.search;const pageText=document.body.innerText;const inputs=Array.from(document.querySelectorAll('input,textarea')).filter(i=>i.offsetParent!==null);R.fieldCount=inputs.length;R.hasE2EData=pageText.includes('E2E_TEST_')||inputs.some(i=>i.value?.includes('E2E_TEST_'));const hasQty=pageText.includes('5')||inputs.some(i=>i.value==='5');const hasPrice=pageText.includes('100,000')||pageText.includes('100000')||inputs.some(i=>i.value?.includes('100000'));const hasSupply=pageText.includes('500,000')||pageText.includes('500000');const hasVat=pageText.includes('50,000')||pageText.includes('50000');R.fieldChecks={qty:hasQty,price:hasPrice,supply:hasSupply,vat:hasVat};R.ok=R.hasE2EData;return JSON.stringify(R);})()", "timeout": 15000, "phase": "READ" }, { "id": 13, "name": "[회계관리 > 매출관리] [UPDATE] 수정 모드 진입 + 수량 변경 + 저장", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const sv=(el,v)=>{const p=el.tagName==='TEXTAREA'?HTMLTextAreaElement.prototype:HTMLInputElement.prototype;const ns=Object.getOwnPropertyDescriptor(p,'value')?.set;if(ns)ns.call(el,v);else el.value=v;el.dispatchEvent(new Event('input',{bubbles:true}));el.dispatchEvent(new Event('change',{bubbles:true}));};const ts=window.__E2E_TS__||'E2E_TEST_';const toastInfo=()=>{const t=document.querySelectorAll('[data-sonner-toast],[role=\"status\"],[class*=\"toast\"],[class*=\"Toast\"],[class*=\"Toaster\"] [data-content]');return{count:t.length,text:t.length>0?Array.from(t).pop()?.innerText?.trim().substring(0,100):''};};const R={phase:'UPDATE'};const editBtn=Array.from(document.querySelectorAll('button')).find(b=>b.innerText?.trim()==='수정'&&b.offsetParent!==null);if(!editBtn){R.error='수정 버튼 없음';R.ok=false;return JSON.stringify(R);}editBtn.click();await w(2000);R.editUrl=location.pathname+location.search;const inputs=Array.from(document.querySelectorAll('input[type=\"text\"],input[type=\"number\"],input:not([type])')).filter(i=>i.offsetParent!==null&&!i.readOnly&&!i.disabled);const qtyInput=inputs.find(i=>{const ph=i.placeholder||'';const nm=i.name||'';const lbl=i.closest('[class*=field],[class*=Field],[class*=form-item]')?.querySelector('label')?.innerText||'';return ph.includes('수량')||nm.includes('quantity')||nm.includes('qty')||lbl.includes('수량');});if(qtyInput){R.oldQty=qtyInput.value;sv(qtyInput,'10');R.newQty='10';await w(300);}const noteInput=inputs.find(i=>{const ph=i.placeholder||'';const nm=i.name||'';const lbl=i.closest('[class*=field],[class*=Field],[class*=form-item]')?.querySelector('label')?.innerText||'';return ph.includes('적요')||nm.includes('note')||nm.includes('remark')||lbl.includes('적요');});if(noteInput){sv(noteInput,'E2E_수정됨_'+ts);R.noteModified=true;await w(200);}await w(500);const saveBtn=Array.from(document.querySelectorAll('button')).find(b=>/저장|수정|확인/.test(b.innerText?.trim())&&b!==editBtn&&b.offsetParent!==null&&!b.disabled);if(saveBtn){saveBtn.click();await w(3000);}R.toast=toastInfo();R.ok=true;return JSON.stringify(R);})()", "timeout": 30000, "phase": "UPDATE" }, { "id": 14, "name": "[회계관리 > 매출관리] [UPDATE] 저장 후 대기", "action": "wait", "timeout": 3000 }, { "id": 15, "name": "[회계관리 > 매출관리] [UPDATE] 수정 내용 검증 (공급가액 1,000,000 재계산)", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const R={phase:'VERIFY_UPDATE'};const pageText=document.body.innerText;const inputs=Array.from(document.querySelectorAll('input,textarea')).filter(i=>i.offsetParent!==null);const hasModified=pageText.includes('수정됨')||inputs.some(i=>i.value?.includes('수정됨'));const hasNewSupply=pageText.includes('1,000,000')||pageText.includes('1000000');const toasts=document.querySelectorAll('[data-sonner-toast],[role=\"status\"],[class*=\"toast\"],[class*=\"Toast\"]');const toastOk=Array.from(toasts).some(t=>/수정|완료|저장|성공/.test(t.innerText));R.hasModified=hasModified;R.hasNewSupply=hasNewSupply;R.toastOk=toastOk;R.ok=hasModified||toastOk;return JSON.stringify(R);})()", "timeout": 15000, "phase": "UPDATE" }, { "id": 16, "name": "[회계관리 > 매출관리] [UPDATE] 목록 복귀", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const onForm=location.search.includes('mode=new')||location.search.includes('mode=edit')||location.search.includes('mode=view')||new RegExp('/(new|[0-9]+|[0-9a-f]{8,})$').test(location.pathname);if(onForm){const btn=Array.from(document.querySelectorAll('button,a')).find(b=>/목록|취소|뒤로/.test(b.innerText?.trim()));if(btn){btn.click();await w(2000);}else{history.back();await w(2000);}}return JSON.stringify({url:location.pathname+location.search});})()", "timeout": 10000, "phase": "UPDATE" }, { "id": 17, "name": "[회계관리 > 매출관리] [UPDATE] 목록 안정화 대기", "action": "wait", "timeout": 2000 }, { "id": 18, "name": "[회계관리 > 매출관리] [DELETE] 데이터 삭제", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||'E2E_TEST_';const toastInfo=()=>{const t=document.querySelectorAll('[data-sonner-toast],[role=\"status\"],[class*=\"toast\"],[class*=\"Toast\"],[class*=\"Toaster\"] [data-content]');return{count:t.length,text:t.length>0?Array.from(t).pop()?.innerText?.trim().substring(0,100):''};};const R={phase:'DELETE'};if(!location.search.includes('mode=view')&&!location.search.includes('mode=edit')){const rows=Array.from(document.querySelectorAll('table tbody tr'));const row=rows.find(r=>r.innerText?.includes('E2E_TEST_'));if(!row){R.error='E2E_TEST_ 행 없음';R.ok=false;return JSON.stringify(R);}row.click();await w(2500);}const delBtn=Array.from(document.querySelectorAll('button')).find(b=>b.innerText?.trim()==='삭제'&&b.offsetParent!==null);if(!delBtn){R.error='삭제 버튼 없음';R.ok=false;return JSON.stringify(R);}delBtn.click();await w(1000);const cfm=Array.from(document.querySelectorAll('[role=\"alertdialog\"] button,[role=\"dialog\"] button,button')).find(b=>/확인|삭제|예|Yes/.test(b.innerText?.trim())&&b!==delBtn&&b.offsetParent!==null);if(cfm){cfm.click();await w(3000);}R.toast=toastInfo();R.ok=true;return JSON.stringify(R);})()", "timeout": 30000, "phase": "DELETE", "critical": true }, { "id": 19, "name": "[회계관리 > 매출관리] [DELETE] 목록 복귀 + 대기", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const onForm=location.search.includes('mode=new')||location.search.includes('mode=edit')||location.search.includes('mode=view')||new RegExp('/(new|[0-9]+|[0-9a-f]{8,})$').test(location.pathname);if(onForm){const btn=Array.from(document.querySelectorAll('button,a')).find(b=>/목록|취소|뒤로/.test(b.innerText?.trim()));if(btn){btn.click();await w(2000);}else{history.back();await w(2000);}}await w(2000);return JSON.stringify({url:location.pathname+location.search});})()", "timeout": 10000, "phase": "DELETE" }, { "id": 20, "name": "[회계관리 > 매출관리] [VERIFY] 삭제 확인", "action": "evaluate", "script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||'';const R={phase:'VERIFY_DELETE'};await w(1000);const rows=document.querySelectorAll('table tbody tr');R.rowCount=rows.length;const found=Array.from(rows).find(r=>r.innerText?.includes(ts));R.stillExists=!!found;R.ok=!found;R.ts=ts;return JSON.stringify(R);})()", "timeout": 15000, "phase": "VERIFY" } ] }