portfolio.py 4.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. from fastapi import APIRouter, Depends, HTTPException, Form, UploadFile, File
  2. import db
  3. import schemas
  4. import auth_utils
  5. import config
  6. import os
  7. import uuid
  8. import shutil
  9. router = APIRouter(tags=["portfolio"])
  10. @router.get("/portfolio")
  11. async def get_public_portfolio():
  12. query = """
  13. SELECT p.id, p.file_path, COALESCE(o.material_name, 'Showcase') as material_name, p.order_id
  14. FROM order_photos p
  15. LEFT JOIN orders o ON p.order_id = o.id
  16. WHERE p.is_public = TRUE AND (o.id IS NULL OR o.allow_portfolio = TRUE)
  17. ORDER BY p.created_at DESC
  18. """
  19. return db.execute_query(query)
  20. @router.get("/admin/all-photos")
  21. async def admin_get_all_photos(token: str = Depends(auth_utils.oauth2_scheme)):
  22. payload = auth_utils.decode_token(token)
  23. if not payload or payload.get("role") != 'admin':
  24. raise HTTPException(status_code=403, detail="Admin role required")
  25. query = """
  26. SELECT p.id, p.file_path, p.is_public, p.order_id, o.allow_portfolio,
  27. o.first_name, o.last_name, COALESCE(o.material_name, 'Manual') as material_name
  28. FROM order_photos p
  29. JOIN orders o ON p.order_id = o.id
  30. ORDER BY p.created_at DESC
  31. """
  32. return db.execute_query(query)
  33. @router.post("/admin/orders/{order_id}/photos")
  34. async def admin_upload_order_photo(
  35. order_id: int,
  36. is_public: bool = Form(False),
  37. file: UploadFile = File(...),
  38. token: str = Depends(auth_utils.oauth2_scheme)
  39. ):
  40. payload = auth_utils.decode_token(token)
  41. if not payload or payload.get("role") != 'admin':
  42. raise HTTPException(status_code=403, detail="Admin role required")
  43. order = db.execute_query("SELECT allow_portfolio FROM orders WHERE id = %s", (order_id,))
  44. if not order: raise HTTPException(status_code=404, detail="Order not found")
  45. if is_public and not order[0]['allow_portfolio']:
  46. raise HTTPException(status_code=400, detail="Cannot make public: User did not consent to portfolio usage")
  47. if not file.filename: raise HTTPException(status_code=400, detail="Invalid file")
  48. unique_filename = f"{uuid.uuid4()}{os.path.splitext(file.filename)[1]}"
  49. disk_path = os.path.join(config.UPLOAD_DIR, unique_filename)
  50. db_file_path = f"uploads/{unique_filename}"
  51. with open(disk_path, "wb") as buffer:
  52. shutil.copyfileobj(file.file, buffer)
  53. query = "INSERT INTO order_photos (order_id, file_path, is_public) VALUES (%s, %s, %s)"
  54. photo_id = db.execute_commit(query, (order_id, db_file_path, is_public))
  55. return {"id": photo_id, "file_path": db_file_path, "is_public": is_public}
  56. @router.patch("/admin/photos/{photo_id}")
  57. async def admin_update_photo_status(photo_id: int, data: schemas.PhotoUpdate, token: str = Depends(auth_utils.oauth2_scheme)):
  58. payload = auth_utils.decode_token(token)
  59. if not payload or payload.get("role") != 'admin':
  60. raise HTTPException(status_code=403, detail="Admin role required")
  61. query = "SELECT p.*, o.allow_portfolio FROM order_photos p JOIN orders o ON p.order_id = o.id WHERE p.id = %s"
  62. photo_data = db.execute_query(query, (photo_id,))
  63. if not photo_data: raise HTTPException(status_code=404, detail="Photo not found")
  64. if data.is_public and not photo_data[0]['allow_portfolio']:
  65. raise HTTPException(status_code=400, detail="Cannot make public: User did not consent to portfolio usage")
  66. db.execute_commit("UPDATE order_photos SET is_public = %s WHERE id = %s", (data.is_public, photo_id))
  67. return {"id": photo_id, "is_public": data.is_public}
  68. @router.delete("/admin/photos/{photo_id}")
  69. async def admin_delete_photo(photo_id: int, token: str = Depends(auth_utils.oauth2_scheme)):
  70. payload = auth_utils.decode_token(token)
  71. if not payload or payload.get("role") != 'admin':
  72. raise HTTPException(status_code=403, detail="Admin role required")
  73. photo = db.execute_query("SELECT file_path FROM order_photos WHERE id = %s", (photo_id,))
  74. if not photo:
  75. raise HTTPException(status_code=404, detail="Photo not found")
  76. try:
  77. path = os.path.join(config.BASE_DIR, photo[0]['file_path'])
  78. if os.path.exists(path):
  79. os.remove(path)
  80. except Exception as e:
  81. print(f"Error deleting photo file: {e}")
  82. db.execute_commit("DELETE FROM order_photos WHERE id = %s", (photo_id,))
  83. return {"id": photo_id, "status": "deleted"}