/** * Project Master Analysis JS * P-WAR (Project Performance Above Replacement) 분석 엔진 */ document.addEventListener('DOMContentLoaded', () => { console.log("Analysis engine initialized..."); loadPWarData(); }); async function loadPWarData() { try { const response = await fetch('/api/analysis/p-war'); const data = await response.json(); if (data.error) throw new Error(data.error); updateSummaryMetrics(data); renderPWarLeaderboard(data); renderRiskSignals(data); // 시스템 평균 정보 표시 if (data.length > 0 && data[0].avg_info) { const avg = data[0].avg_info; document.getElementById('avg-system-info').textContent = `* 0.0 = 시스템 평균 (파일 ${avg.avg_files.toLocaleString()}개 / 방치 ${avg.avg_stagnant}일 / 리스크 ${avg.avg_risk}건)`; } } catch (e) { console.error("분석 데이터 로딩 실패:", e); } } function updateSummaryMetrics(data) { // 1. 평균 P-WAR 산출 const avgPWar = data.reduce((acc, cur) => acc + cur.p_war, 0) / data.length; document.querySelector('.metric-card.sra .value').textContent = avgPWar.toFixed(2); // 2. 고위험 좀비 프로젝트 비율 (P-WAR < -1.0 기준) const zombieCount = data.filter(p => p.p_war < -1.0).length; const zombieRate = (zombieCount / data.length) * 100; document.querySelector('.metric-card.stability .value').textContent = `${zombieRate.toFixed(1)}%`; // 3. 총 활성 리소스 규모 const totalActiveFiles = data.filter(p => p.p_war > 0).reduce((acc, cur) => acc + cur.file_count, 0); document.querySelector('.metric-card.piso .value').textContent = (totalActiveFiles / 1000).toFixed(1) + "k"; // 4. 방치 리스크 총합 const totalRisks = data.reduce((acc, cur) => acc + cur.risk_count, 0); document.querySelector('.metric-card.iwar .value').textContent = totalRisks; } function renderPWarLeaderboard(data) { const container = document.querySelector('.timeline-analysis .card-body'); const sortedData = [...data].sort((a, b) => b.p_war - a.p_war); container.innerHTML = `
| 프로젝트명 | 관리 상태 | 파일 수 | 방치일 | 미결리스크 | P-WAR (기여도) |
|---|---|---|---|---|---|
| ${p.project_nm} | ${statusBadge} | ${p.file_count.toLocaleString()}개 | ${p.days_stagnant}일 | ${p.risk_count}건 | ${p.p_war > 0 ? '+' : ''}${p.p_war} |