SurfSense/surfsense_backend/app/routes/editor_routes.py

167 lines
5 KiB
Python
Raw Normal View History

2025-11-23 15:23:31 +05:30
"""
Editor routes for BlockNote document editing.
"""
2025-11-23 16:39:23 +05:30
2025-11-23 15:23:31 +05:30
from datetime import UTC, datetime
from typing import Any
from fastapi import APIRouter, Depends, HTTPException
from sqlalchemy import select
from sqlalchemy.ext.asyncio import AsyncSession
from sqlalchemy.orm import selectinload
2025-11-23 15:23:31 +05:30
from app.db import Document, Permission, User, get_async_session
2025-11-23 15:23:31 +05:30
from app.users import current_active_user
from app.utils.rbac import check_permission
2025-11-23 15:23:31 +05:30
router = APIRouter()
@router.get("/search-spaces/{search_space_id}/documents/{document_id}/editor-content")
2025-11-23 15:23:31 +05:30
async def get_editor_content(
search_space_id: int,
2025-11-23 15:23:31 +05:30
document_id: int,
session: AsyncSession = Depends(get_async_session),
user: User = Depends(current_active_user),
):
"""
Get document content for editing.
2025-11-23 16:39:23 +05:30
2025-11-23 15:23:31 +05:30
Returns BlockNote JSON document. If blocknote_document is NULL,
attempts to generate it from chunks (lazy migration).
Requires DOCUMENTS_READ permission.
2025-11-23 15:23:31 +05:30
"""
# Check RBAC permission
await check_permission(
session,
user,
search_space_id,
Permission.DOCUMENTS_READ.value,
"You don't have permission to read documents in this search space",
)
2025-11-23 15:23:31 +05:30
result = await session.execute(
select(Document)
.options(selectinload(Document.chunks))
.filter(
Document.id == document_id,
Document.search_space_id == search_space_id,
)
2025-11-23 15:23:31 +05:30
)
document = result.scalars().first()
2025-11-23 16:39:23 +05:30
2025-11-23 15:23:31 +05:30
if not document:
raise HTTPException(status_code=404, detail="Document not found")
2025-11-23 16:39:23 +05:30
2025-11-23 15:23:31 +05:30
# If blocknote_document exists, return it
if document.blocknote_document:
return {
"document_id": document.id,
"title": document.title,
"blocknote_document": document.blocknote_document,
2025-11-23 16:39:23 +05:30
"last_edited_at": document.last_edited_at.isoformat()
if document.last_edited_at
else None,
2025-11-23 15:23:31 +05:30
}
2025-11-23 16:39:23 +05:30
# Lazy migration: Try to generate blocknote_document from chunks
from app.utils.blocknote_converter import convert_markdown_to_blocknote
chunks = sorted(document.chunks, key=lambda c: c.id)
if not chunks:
raise HTTPException(
status_code=400,
detail="This document has no chunks and cannot be edited. Please re-upload to enable editing.",
)
# Reconstruct markdown from chunks
markdown_content = "\n\n".join(chunk.content for chunk in chunks)
if not markdown_content.strip():
raise HTTPException(
status_code=400,
detail="This document has empty content and cannot be edited.",
)
# Convert to BlockNote
blocknote_json = await convert_markdown_to_blocknote(markdown_content)
if not blocknote_json:
raise HTTPException(
status_code=500,
detail="Failed to convert document to editable format. Please try again later.",
)
# Save the generated blocknote_document (lazy migration)
document.blocknote_document = blocknote_json
document.content_needs_reindexing = False
document.last_edited_at = None
await session.commit()
return {
"document_id": document.id,
"title": document.title,
"blocknote_document": blocknote_json,
"last_edited_at": None,
}
2025-11-23 15:23:31 +05:30
@router.post("/search-spaces/{search_space_id}/documents/{document_id}/save")
async def save_document(
search_space_id: int,
2025-11-23 15:23:31 +05:30
document_id: int,
data: dict[str, Any],
session: AsyncSession = Depends(get_async_session),
user: User = Depends(current_active_user),
):
"""
Save BlockNote document and trigger reindexing.
Called when user clicks 'Save & Exit'.
Requires DOCUMENTS_UPDATE permission.
2025-11-23 15:23:31 +05:30
"""
from app.tasks.celery_tasks.document_reindex_tasks import reindex_document_task
# Check RBAC permission
await check_permission(
session,
user,
search_space_id,
Permission.DOCUMENTS_UPDATE.value,
"You don't have permission to update documents in this search space",
)
2025-11-23 15:23:31 +05:30
result = await session.execute(
select(Document).filter(
Document.id == document_id,
Document.search_space_id == search_space_id,
)
2025-11-23 15:23:31 +05:30
)
document = result.scalars().first()
2025-11-23 15:23:31 +05:30
if not document:
raise HTTPException(status_code=404, detail="Document not found")
2025-11-23 15:23:31 +05:30
blocknote_document = data.get("blocknote_document")
if not blocknote_document:
raise HTTPException(status_code=400, detail="blocknote_document is required")
# Save BlockNote document
2025-11-23 15:23:31 +05:30
document.blocknote_document = blocknote_document
document.last_edited_at = datetime.now(UTC)
document.content_needs_reindexing = True
2025-11-23 15:23:31 +05:30
await session.commit()
# Queue reindex task
reindex_document_task.delay(document_id, str(user.id))
return {
"status": "saved",
"document_id": document_id,
"message": "Document saved and will be reindexed in the background",
"last_edited_at": document.last_edited_at.isoformat(),
}