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,
|
2025-11-30 19:40:10 +00:00
|
|
|
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"}
|
2025-11-30 19:40:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@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
|