Files
sam-scenarios/full-crud-board.json

256 lines
21 KiB
JSON
Raw Normal View History

{
"id": "full-crud-board",
"name": "Full CRUD 테스트: 자유게시판 (Enhanced)",
"version": "2.0.0",
"auth": {
"role": "admin"
},
"menuNavigation": {
"level1": "게시판",
"level2": "자유게시판"
},
"screenshotPolicy": {
"captureOnFail": true,
"captureOnPass": false
},
"steps": [
{
"id": 1,
"name": "[게시판 > 자유게시판] 페이지 로드 대기",
"action": "wait",
"timeout": 3000
},
{
"id": 2,
"name": "[게시판 > 자유게시판] ts 초기화 + sessionStorage 연동",
"action": "evaluate",
"script": "(()=>{try{sessionStorage.removeItem('__E2E_TS__');}catch(e){}delete window.__E2E_TS__;return JSON.stringify({ok:true,cleared:true});})()",
"timeout": 3000
},
{
"id": 3,
"name": "[게시판 > 자유게시판] 테이블 로드 대기",
"action": "wait_for_table",
"timeout": 5000
},
{
"id": 4,
"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__||sessionStorage.getItem('__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;try{sessionStorage.setItem('__E2E_TS__',ts);}catch(e){}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 testTitle='E2E_TEST_게시글_'+ts;const testContent='E2E Full CRUD 테스트 게시글 본문입니다. 타임스탬프: '+ts+'. 자동 삭제 예정.';window.__E2E_TITLE__=testTitle;window.__E2E_CONTENT__=testContent;R.testTitle=testTitle;const btn=Array.from(document.querySelectorAll('button')).find(b=>b.innerText?.trim()==='글쓰기'||/등록|작성/.test(b.innerText?.trim()));if(!btn){R.error='글쓰기 버튼 없음';return JSON.stringify(R);}btn.click();await w(2500);const titleInput=document.querySelector('input[placeholder*=\"제목\"]')||document.querySelector('input[type=\"text\"]');if(!titleInput){R.error='제목 입력란 없음';return JSON.stringify(R);}sv(titleInput,testTitle);await w(200);const ta=document.querySelector('textarea');if(ta){sv(ta,testContent);R.contentFilled=true;await w(200);}const sub=Array.from(document.querySelectorAll('button')).find(b=>b.innerText?.trim()==='등록');if(!sub){R.error='등록 버튼 없음';return JSON.stringify(R);}sub.click();await w(3000);R.toast=toastInfo();R.toastOk=R.toast.text&&(/등록|완료|저장|성공/.test(R.toast.text));R.ok=true;return JSON.stringify(R);})()",
"timeout": 30000,
"phase": "CREATE"
},
{
"id": 5,
"name": "[게시판 > 자유게시판] [CREATE] API POST 검증",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));await w(500);const R={phase:'API_POST_CHECK'};const logs=(window.__E2E__?window.__E2E__.getApiLogs().logs:[]);const postLogs=logs.filter(l=>l.method==='POST'&&l.status>=200&&l.status<300);R.postCount=postLogs.length;R.found=postLogs.length>0;if(postLogs.length>0){const last=postLogs[postLogs.length-1];R.lastPost={url:last.url?.substring(0,80),status:last.status,duration:last.duration};}R.ok=true;R.info=R.found?'POST API '+R.lastPost.status+' ('+R.lastPost.duration+'ms)':'POST API 미감지 (모니터 타이밍 이슈 가능)';return JSON.stringify(R);})()",
"timeout": 10000,
"phase": "VERIFY"
},
{
"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-CREATE] 목록 첫 행에 새 게시글 표시 확인",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||sessionStorage.getItem('__E2E_TS__')||'';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;if(found){R.foundText=found.innerText?.substring(0,100);const idx=Array.from(rows).indexOf(found);R.isFirstRow=idx===0;R.rowIndex=idx;}R.ok=R.found;R.info=R.found?(R.isFirstRow?'새 게시글이 첫 행에 표시됨':'새 게시글 발견 (행 인덱스: '+R.rowIndex+')'):'E2E_TEST_ 데이터 미발견';return JSON.stringify(R);})()",
"timeout": 15000,
"phase": "VERIFY"
},
{
"id": 10,
"name": "[게시판 > 자유게시판] [READ] 상세 페이지 진입 (E2E_TEST_ 행 클릭)",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||sessionStorage.getItem('__E2E_TS__')||'';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__||sessionStorage.getItem('__E2E_TS__')||'';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);const allVals=[pageText,...inputs.map(i=>i.value||'')].join(' ');R.hasTitle=allVals.includes('E2E_TEST_게시글');R.hasContent=allVals.includes('Full CRUD 테스트')||allVals.includes('본문');R.hasTimestamp=allVals.includes(ts);R.hasE2E=allVals.includes('E2E_TEST_');R.ok=R.hasTitle;R.info=[R.hasTitle?'제목 확인':'제목 미감지',R.hasContent?'본문 확인':'본문 미감지',R.hasTimestamp?'타임스탬프 확인':'타임스탬프 미감지'].join(' | ');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__||sessionStorage.getItem('__E2E_TS__')||'';const R={phase:'UPDATE'};const editBtn=Array.from(document.querySelectorAll('button')).find(b=>b.innerText?.trim()==='수정');if(!editBtn){R.error='수정 버튼 없음';R.ok=false;return JSON.stringify(R);}editBtn.click();await w(2000);R.editUrl=location.pathname+location.search;const titleInput=document.querySelector('input[placeholder*=\"제목\"]')||document.querySelector('input[type=\"text\"]');if(!titleInput){R.error='제목 입력란 없음';R.ok=false;return JSON.stringify(R);}const cur=titleInput.value;sv(titleInput,cur.replace('E2E_TEST_게시글','E2E_TEST_수정됨'));R.titleChanged=true;R.newTitle=titleInput.value?.substring(0,50);await w(200);R.ok=true;return JSON.stringify(R);})()",
"timeout": 20000,
"phase": "UPDATE"
},
{
"id": 14,
"name": "[게시판 > 자유게시판] [UPDATE] 저장 클릭",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));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_SAVE'};const editBtn=Array.from(document.querySelectorAll('button')).find(b=>b.innerText?.trim()==='수정');const saveBtn=Array.from(document.querySelectorAll('button')).find(b=>/저장|수정완료|확인/.test(b.innerText?.trim())&&b!==editBtn&&b.offsetParent!==null);if(saveBtn){saveBtn.click();await w(3000);}else{const sub=Array.from(document.querySelectorAll('button')).find(b=>b.innerText?.trim()==='등록');if(sub){sub.click();await w(3000);}}R.toast=toastInfo();R.toastOk=R.toast.text&&(/수정|완료|저장|성공/.test(R.toast.text));R.ok=true;return JSON.stringify(R);})()",
"timeout": 20000,
"phase": "UPDATE"
},
{
"id": 15,
"name": "[게시판 > 자유게시판] [UPDATE] API PUT 검증",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));await w(500);const R={phase:'API_PUT_CHECK'};const logs=(window.__E2E__?window.__E2E__.getApiLogs().logs:[]);const putLogs=logs.filter(l=>(l.method==='PUT'||l.method==='PATCH')&&l.status>=200&&l.status<300);R.putCount=putLogs.length;R.found=putLogs.length>0;if(putLogs.length>0){const last=putLogs[putLogs.length-1];R.lastPut={url:last.url?.substring(0,80),status:last.status,duration:last.duration};}R.ok=true;R.info=R.found?'PUT/PATCH API '+R.lastPut.status+' ('+R.lastPut.duration+'ms)':'PUT/PATCH API 미감지';return JSON.stringify(R);})()",
"timeout": 10000,
"phase": "VERIFY"
},
{
"id": 16,
"name": "[게시판 > 자유게시판] [UPDATE] 저장 후 대기",
"action": "wait",
"timeout": 2000
},
{
"id": 17,
"name": "[게시판 > 자유게시판] [UPDATE] 수정 내용 검증 (수정됨 텍스트)",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const R={phase:'VERIFY_UPDATE'};R.url=location.pathname+location.search;const pageText=document.body.innerText;const inputs=Array.from(document.querySelectorAll('input,textarea')).filter(i=>i.offsetParent!==null);const allVals=[pageText,...inputs.map(i=>i.value||'')].join(' ');R.hasModified=allVals.includes('수정됨');R.ok=R.hasModified;R.info=R.hasModified?'수정됨 텍스트 확인':'수정됨 텍스트 미감지 (토스트로 판정)';return JSON.stringify(R);})()",
"timeout": 15000,
"phase": "VERIFY"
},
{
"id": 18,
"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": 19,
"name": "[게시판 > 자유게시판] [UPDATE] 목록 안정화 대기",
"action": "wait",
"timeout": 2000
},
{
"id": 20,
"name": "[게시판 > 자유게시판] [VERIFY-UPDATE] 목록에서 수정된 제목 반영 확인",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||sessionStorage.getItem('__E2E_TS__')||'';const R={phase:'VERIFY_LIST_UPDATE'};await w(500);const rows=document.querySelectorAll('table tbody tr');R.rowCount=rows.length;const found=Array.from(rows).find(r=>r.innerText?.includes('수정됨'));R.modifiedInList=!!found;if(found){R.foundText=found.innerText?.substring(0,100);}R.ok=R.modifiedInList;R.info=R.modifiedInList?'목록에서 수정된 제목 확인':'목록에서 수정됨 미발견 (상세에서만 반영 가능)';return JSON.stringify(R);})()",
"timeout": 15000,
"phase": "VERIFY"
},
{
"id": 21,
"name": "[게시판 > 자유게시판] [DELETE] 삭제 대상 행 진입",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||sessionStorage.getItem('__E2E_TS__')||'';const R={phase:'DELETE_ENTER'};const rows=Array.from(document.querySelectorAll('table tbody tr'));const row=rows.find(r=>r.innerText?.includes(ts))||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);R.detailUrl=location.pathname+location.search;R.ok=true;return JSON.stringify(R);})()",
"timeout": 15000,
"phase": "DELETE"
},
{
"id": 22,
"name": "[게시판 > 자유게시판] [DELETE] 상세 페이지 대기",
"action": "wait",
"timeout": 2000
},
{
"id": 23,
"name": "[게시판 > 자유게시판] [DELETE] 삭제 버튼 클릭 + 취소 테스트 (삭제 안됨 확인)",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const R={phase:'DELETE_CANCEL_TEST'};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(1500);const dialog=document.querySelector('[role=\"alertdialog\"]');R.dialogFound=!!dialog;if(dialog){R.dialogText=dialog.innerText?.substring(0,100);R.hasDeleteWarning=/삭제|제거|되돌/.test(dialog.innerText||'');}let cancelBtn=Array.from(document.querySelectorAll('[role=\"alertdialog\"] button')).find(b=>/취소|아니오|Cancel/.test(b.innerText?.trim()));if(!cancelBtn){cancelBtn=document.querySelector('[role=\"alertdialog\"] [data-slot=\"alert-dialog-footer\"] button:first-child');}if(cancelBtn){cancelBtn.click();R.cancelClicked=true;await w(1000);}else{document.dispatchEvent(new KeyboardEvent('keydown',{key:'Escape',bubbles:true}));R.escapeSent=true;await w(1000);}const pageText=document.body.innerText;R.stillOnDetail=pageText.includes('E2E_TEST_')||location.search.includes('mode=view')||new RegExp('/[0-9]+$|/[0-9a-f]{8,}$').test(location.pathname);R.ok=R.stillOnDetail;R.info=R.stillOnDetail?'취소 후 데이터 유지 확인':'취소 테스트 불확실';return JSON.stringify(R);})()",
"timeout": 20000,
"phase": "DELETE"
},
{
"id": 24,
"name": "[게시판 > 자유게시판] [DELETE] 삭제 버튼 재클릭 + 확인 (실제 삭제)",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||sessionStorage.getItem('__E2E_TS__')||'';const R={phase:'DELETE_CONFIRM'};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(1500);const dialog=document.querySelector('[role=\"alertdialog\"]');R.dialogFound=!!dialog;let cfm=document.querySelector('[role=\"alertdialog\"] [data-slot=\"alert-dialog-footer\"] button:last-child');if(!cfm){cfm=Array.from(document.querySelectorAll('[role=\"alertdialog\"] button')).find(b=>/삭제/.test(b.innerText?.trim())&&b!==delBtn);}if(!cfm){cfm=Array.from(document.querySelectorAll('button')).find(b=>/확인|삭제|예/.test(b.innerText?.trim())&&b!==delBtn&&b.offsetParent!==null);}if(cfm){cfm.click();await w(4000);}R.ts=ts;R.ok=true;return JSON.stringify(R);})()",
"timeout": 30000,
"phase": "DELETE",
"critical": true
},
{
"id": 25,
"name": "[게시판 > 자유게시판] [DELETE] API DELETE 검증",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));await w(500);const R={phase:'API_DELETE_CHECK'};const logs=(window.__E2E__?window.__E2E__.getApiLogs().logs:[]);const delLogs=logs.filter(l=>l.method==='DELETE'&&l.status>=200&&l.status<300);R.delCount=delLogs.length;R.found=delLogs.length>0;if(delLogs.length>0){const last=delLogs[delLogs.length-1];R.lastDelete={url:last.url?.substring(0,80),status:last.status,duration:last.duration};}R.ok=true;R.info=R.found?'DELETE API '+R.lastDelete.status+' ('+R.lastDelete.duration+'ms)':'DELETE API 미감지';return JSON.stringify(R);})()",
"timeout": 10000,
"phase": "VERIFY"
},
{
"id": 26,
"name": "[게시판 > 자유게시판] [DELETE] 삭제 후 대기",
"action": "wait",
"timeout": 3000
},
{
"id": 27,
"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);}}return JSON.stringify({url:location.pathname+location.search});})()",
"timeout": 10000,
"phase": "DELETE"
},
{
"id": 28,
"name": "[게시판 > 자유게시판] [DELETE] 목록 안정화 대기",
"action": "wait",
"timeout": 2000
},
{
"id": 29,
"name": "[게시판 > 자유게시판] [VERIFY-DELETE] 삭제 후 새로고침",
"action": "reload",
"timeout": 10000
},
{
"id": 30,
"name": "[게시판 > 자유게시판] [VERIFY-DELETE] 새로고침 대기",
"action": "wait",
"timeout": 3000
},
{
"id": 31,
"name": "[게시판 > 자유게시판] [VERIFY-DELETE] 테이블 로드 대기",
"action": "wait_for_table",
"timeout": 10000
},
{
"id": 32,
"name": "[게시판 > 자유게시판] [VERIFY-DELETE] 삭제 확인 (목록에서 E2E_TEST_ 부재 확인)",
"action": "evaluate",
"script": "(async()=>{const w=ms=>new Promise(r=>setTimeout(r,ms));const ts=window.__E2E_TS__||(()=>{try{return sessionStorage.getItem('__E2E_TS__')}catch(e){return null}})()||'E2E_TEST_';const R={phase:'VERIFY_DELETE'};await w(1000);R.url=location.pathname;const rows=document.querySelectorAll('table tbody tr');R.rowCount=rows.length;const found=Array.from(rows).find(r=>r.innerText?.includes('E2E_TEST_')&&r.innerText?.includes(ts));R.stillExists=!!found;R.ok=!found;if(found)R.warn='E2E_TEST_ 데이터가 여전히 존재 - 삭제 실패 가능';R.ts=ts;return JSON.stringify(R);})()",
"timeout": 15000,
"phase": "VERIFY"
},
{
"id": 33,
"name": "[게시판 > 자유게시판] [SUMMARY] API 호출 통계",
"action": "evaluate",
"script": "(async()=>{const R={phase:'API_SUMMARY'};const logs=(window.__E2E__?window.__E2E__.getApiLogs().logs:[]);R.totalCalls=logs.length;R.successCalls=logs.filter(l=>l.status>=200&&l.status<300).length;R.failedCalls=logs.filter(l=>l.status>=400).length;R.avgResponseTime=logs.length>0?Math.round(logs.reduce((s,l)=>s+(l.duration||0),0)/logs.length):0;R.slowCalls=logs.filter(l=>(l.duration||0)>2000).length;R.methods={GET:logs.filter(l=>l.method==='GET').length,POST:logs.filter(l=>l.method==='POST').length,PUT:logs.filter(l=>l.method==='PUT').length,PATCH:logs.filter(l=>l.method==='PATCH').length,DELETE:logs.filter(l=>l.method==='DELETE').length};R.ok=true;R.info='API: '+R.totalCalls+'건 (성공 '+R.successCalls+', 실패 '+R.failedCalls+', 평균 '+R.avgResponseTime+'ms, 느린호출 '+R.slowCalls+'건)';return JSON.stringify(R);})()",
"timeout": 10000,
"phase": "VERIFY"
}
]
}