""" Knowledge Base API 知識タイル一覧表示と検証マーク機構 """ from fastapi import APIRouter, Depends, HTTPException, Query from fastapi.responses import FileResponse, StreamingResponse from typing import List, Optional from datetime import datetime import os import json import io from sqlalchemy.orm import Session from backend.app.middleware.auth import get_current_user_optional, get_current_user, User from backend.app.config import settings from backend.app.database.session import get_db from backend.app.services.knowledge_service import KnowledgeService, get_knowledge_service from backend.app.schemas.knowledge import KnowledgeTile, KnowledgeListResponse, KnowledgeDetailResponse, EditRequest, VerificationMark router = APIRouter() @router.get("/", response_model=KnowledgeListResponse) async def list_knowledge_tiles( domain_id: Optional[str] = Query(None, description="ドメインでフィルタ"), verification_type: Optional[str] = Query(None, description="検証タイプでフィルタ"), search: Optional[str] = Query(None, description="検索クエリ"), page: int = Query(1, ge=1, description="ページ番号"), page_size: int = Query(20, ge=1, le=100, description="ページサイズ"), db: Session = Depends(get_db), service: KnowledgeService = Depends(get_knowledge_service) ): tiles_orm, total_count = service.list_tiles( db=db, page=page, page_size=page_size, domain_id=domain_id, verification_type=verification_type, search=search ) tiles_pydantic = [KnowledgeTile.from_orm(t) for t in tiles_orm] return KnowledgeListResponse( tiles=tiles_pydantic, total_count=total_count, page=page, page_size=page_size, has_more=(page * page_size) < total_count ) @router.get("/{tile_id}", response_model=KnowledgeDetailResponse) async def get_knowledge_tile( tile_id: str, db: Session = Depends(get_db), service: KnowledgeService = Depends(get_knowledge_service) ): tile_orm = service.get_tile(db, tile_id=tile_id) if not tile_orm: raise HTTPException(status_code=404, detail="Knowledge tile not found") tile_pydantic = KnowledgeTile.from_orm(tile_orm) return KnowledgeDetailResponse( tile=tile_pydantic, full_content=tile_orm.content, # TODO: Implement sources, related_tiles, and edit_history from DB sources=[], related_tiles=[], edit_history=[] ) @router.put("/{tile_id}", response_model=KnowledgeTile) async def update_knowledge_tile( tile_id: str, request: EditRequest, current_user: User = Depends(get_current_user), db: Session = Depends(get_db), service: KnowledgeService = Depends(get_knowledge_service) ): updated_tile_orm = service.update_tile( db=db, tile_id=tile_id, content=request.content, user=current_user ) if not updated_tile_orm: raise HTTPException(status_code=404, detail="Knowledge tile not found") return KnowledgeTile.from_orm(updated_tile_orm) @router.get("/coordinates") async def get_coordinates_for_3d_visualization( domain_id: Optional[str] = Query(None, description="特定ドメインのみ取得"), db: Session = Depends(get_db), service: KnowledgeService = Depends(get_knowledge_service) ): """ 3D可視化用の座標データを取得 座標を持つタイルのみを返し、必要最小限の情報のみを含める """ # Fetch all tiles (large page size to get all) tiles_orm, _ = service.list_tiles(db=db, page_size=10000, domain_id=domain_id) # Filter tiles that have coordinates and extract minimal data coordinates_data = [] for tile in tiles_orm: if tile.coordinates: # Only include tiles with coordinates coordinates_data.append({ "tile_id": tile.id, "topic": tile.topic, "domain_id": tile.domain_id, "coordinates": tile.coordinates, # [x, y, z, c, g, v] "confidence_score": tile.confidence_score, "verification_type": tile.verification_type }) return { "tiles": coordinates_data, "count": len(coordinates_data), "domain_id": domain_id or "all" } @router.get("/export/json") async def export_db_json( domain_id: Optional[str] = Query(None, description="特定ドメインのみエクスポート"), db: Session = Depends(get_db), service: KnowledgeService = Depends(get_knowledge_service) ): # Fetch all tiles for export tiles_orm, _ = service.list_tiles(db=db, page_size=10000, domain_id=domain_id) # A large page size to get all tiles_pydantic = [KnowledgeTile.from_orm(t).dict() for t in tiles_orm] export_data = { "metadata": { "export_date": datetime.now().isoformat(), "source": "NullAI Knowledge Base", "domain_filter": domain_id or "all", "tile_count": len(tiles_pydantic) }, "tiles": tiles_pydantic } json_str = json.dumps(export_data, indent=2, ensure_ascii=False, default=str) return StreamingResponse( io.BytesIO(json_str.encode('utf-8')), media_type="application/json", headers={ "Content-Disposition": f"attachment; filename=null_ai_knowledge_{datetime.now().strftime('%Y%m%d')}.json" } )