Jotzy/backend/app/routes/folders.py

127 lines
3.8 KiB
Python
Raw Normal View History

2025-11-23 09:08:01 +00:00
from typing import List
from fastapi import APIRouter, Depends, HTTPException # type: ignore
from sqlmodel import Session, select # type: ignore
from app.database import get_session
from app.models import (
Folder,
FolderCreate,
FolderTreeNode,
FolderTreeResponse,
FolderUpdate,
2025-11-23 09:08:01 +00:00
Note,
NoteRead,
)
router = APIRouter(prefix="/folders", tags=["folders"])
def build_folder_tree_node(folder: Folder) -> FolderTreeNode:
"""Recursively build a folder tree node with notes and children"""
return FolderTreeNode(
id=folder.id,
name=folder.name,
notes=[NoteRead.model_validate(note) for note in folder.notes],
children=[build_folder_tree_node(child) for child in folder.children],
)
@router.get("/tree", response_model=FolderTreeResponse)
def get_folder_tree(session: Session = Depends(get_session)):
"""Get complete folder tree with notes"""
# Get all top-level folders (parent_id is None)
top_level_folders = session.exec(
select(Folder).where(Folder.parent_id == None)
).all()
# Get all orphaned notes (folder_id is None)
orphaned_notes = session.exec(select(Note).where(Note.folder_id == None)).all()
# Build tree recursively
tree = [build_folder_tree_node(folder) for folder in top_level_folders]
return FolderTreeResponse(
folders=tree,
orphaned_notes=[NoteRead.model_validate(note) for note in orphaned_notes],
)
@router.get("/", response_model=List[Folder])
def list_folders(session: Session = Depends(get_session)):
"""Get flat list of all folders"""
folders = session.exec(select(Folder)).all()
return folders
@router.post("/", response_model=Folder)
def create_folder(folder: FolderCreate, session: Session = Depends(get_session)):
"""Create a new folder"""
db_folder = Folder.model_validate(folder)
session.add(db_folder)
session.commit()
session.refresh(db_folder)
return db_folder
@router.delete("/{folder_id}")
def delete_folder(folder_id: int, session: Session = Depends(get_session)):
"""Delete a folder"""
folder = session.get(Folder, folder_id)
if not folder:
raise HTTPException(status_code=404, detail="Folder not found")
session.delete(folder)
session.commit()
return {"message": "Folder deleted"}
@router.patch("/{folder_id}")
def update_folder(
folder_id: int, folder_update: FolderUpdate, session: Session = Depends(get_session)
):
"""Update a folder"""
print(f"=== UPDATE FOLDER CALLED ===")
print(f"Folder ID: {folder_id}")
print(f"Update data received: {folder_update}")
folder = session.get(Folder, folder_id)
if not folder:
raise HTTPException(status_code=404, detail="Folder not found")
print(
f"Found folder: id={folder.id}, name={folder.name}, parent_id={folder.parent_id}"
)
# Update folder attributes from the request body
update_data = folder_update.model_dump(exclude_unset=True)
print(f"Update data dict (exclude_unset): {update_data}")
print(f"Update data keys: {list(update_data.keys())}")
for key, value in update_data.items():
print(f"Setting {key} = {value}")
setattr(folder, key, value)
print(
f"After setattr: id={folder.id}, name={folder.name}, parent_id={folder.parent_id}"
)
session.add(folder)
session.commit()
print(f"Committed changes to database")
session.refresh(folder)
print(
f"After refresh: id={folder.id}, name={folder.name}, parent_id={folder.parent_id}"
)
# Verify the change persisted
verification = session.get(Folder, folder_id)
print(
f"Verification query: id={verification.id}, name={verification.name}, parent_id={verification.parent_id}"
)
print(f"=== UPDATE COMPLETE ===")
return folder