Files
tk-factory-services/ai-service/services/report_service.py
Hyungi Ahn b3012b8320 feat: AI 서비스 및 AI 어시스턴트 전용 페이지 추가
- ai-service: Ollama 기반 AI 서비스 (분류, 시맨틱 검색, RAG Q&A, 패턴 분석)
- AI 어시스턴트 페이지: 채팅형 Q&A, 시맨틱 검색, 패턴 분석, 분류 테스트
- 권한 시스템에 ai_assistant 페이지 등록 (기본 비활성)
- 기존 페이지에 AI 기능 통합 (대시보드, 수신함, 관리함)
- docker-compose, gateway, nginx 설정 업데이트

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 09:38:30 +09:00

123 lines
3.8 KiB
Python

import httpx
from services.ollama_client import ollama_client
from services.db_client import get_daily_qc_stats, get_issues_for_date
from config import settings
REPORT_PROMPT_PATH = "prompts/daily_report.txt"
def _load_prompt(path: str) -> str:
with open(path, "r", encoding="utf-8") as f:
return f.read()
async def _fetch_system1_data(date_str: str, token: str) -> dict:
headers = {"Authorization": f"Bearer {token}"}
data = {"attendance": None, "work_reports": None, "patrol": None}
try:
async with httpx.AsyncClient(timeout=15.0) as client:
# 근태
try:
r = await client.get(
f"{settings.SYSTEM1_API_URL}/api/attendance/daily-status",
params={"date": date_str},
headers=headers,
)
if r.status_code == 200:
data["attendance"] = r.json()
except Exception:
pass
# 작업보고
try:
r = await client.get(
f"{settings.SYSTEM1_API_URL}/api/daily-work-reports/summary",
params={"date": date_str},
headers=headers,
)
if r.status_code == 200:
data["work_reports"] = r.json()
except Exception:
pass
# 순회점검
try:
r = await client.get(
f"{settings.SYSTEM1_API_URL}/api/patrol/today-status",
params={"date": date_str},
headers=headers,
)
if r.status_code == 200:
data["patrol"] = r.json()
except Exception:
pass
except Exception:
pass
return data
def _format_attendance(data) -> str:
if not data:
return "데이터 없음"
if isinstance(data, dict):
parts = []
for k, v in data.items():
parts.append(f" {k}: {v}")
return "\n".join(parts)
return str(data)
def _format_work_reports(data) -> str:
if not data:
return "데이터 없음"
return str(data)
def _format_qc_issues(issues: list[dict], stats: dict) -> str:
lines = []
lines.append(f"전체: {stats.get('total', 0)}")
lines.append(f"금일 신규: {stats.get('new_today', 0)}")
lines.append(f"진행중: {stats.get('in_progress', 0)}")
lines.append(f"완료: {stats.get('completed', 0)}")
lines.append(f"미검토: {stats.get('pending', 0)}")
if issues:
lines.append("\n금일 신규 이슈:")
for iss in issues[:10]:
cat = iss.get("category", "")
desc = (iss.get("description") or "")[:50]
status = iss.get("review_status", "")
lines.append(f" - [{cat}] {desc} (상태: {status})")
return "\n".join(lines)
def _format_patrol(data) -> str:
if not data:
return "데이터 없음"
return str(data)
async def generate_daily_report(
date_str: str, project_id: int = None, token: str = ""
) -> dict:
system1_data = await _fetch_system1_data(date_str, token)
qc_stats = get_daily_qc_stats(date_str)
qc_issues = get_issues_for_date(date_str)
template = _load_prompt(REPORT_PROMPT_PATH)
prompt = template.format(
date=date_str,
attendance_data=_format_attendance(system1_data["attendance"]),
work_report_data=_format_work_reports(system1_data["work_reports"]),
qc_issue_data=_format_qc_issues(qc_issues, qc_stats),
patrol_data=_format_patrol(system1_data["patrol"]),
)
report_text = await ollama_client.generate_text(prompt)
return {
"date": date_str,
"report": report_text,
"stats": {
"qc": qc_stats,
"new_issues_count": len(qc_issues),
},
}