kofdai's picture
Upload folder using huggingface_hub
594ed40 verified
"""
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"
}
)