folders.py 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. import logging
  2. import os
  3. import shutil
  4. import uuid
  5. from pathlib import Path
  6. from typing import Optional
  7. from pydantic import BaseModel
  8. import mimetypes
  9. from open_webui.models.folders import (
  10. FolderForm,
  11. FolderUpdateForm,
  12. FolderModel,
  13. FolderNameIdResponse,
  14. Folders,
  15. )
  16. from open_webui.models.chats import Chats
  17. from open_webui.models.files import Files
  18. from open_webui.models.knowledge import Knowledges
  19. from open_webui.config import UPLOAD_DIR
  20. from open_webui.env import SRC_LOG_LEVELS
  21. from open_webui.constants import ERROR_MESSAGES
  22. from fastapi import APIRouter, Depends, File, HTTPException, UploadFile, status, Request
  23. from fastapi.responses import FileResponse, StreamingResponse
  24. from open_webui.utils.auth import get_admin_user, get_verified_user
  25. from open_webui.utils.access_control import has_permission
  26. log = logging.getLogger(__name__)
  27. log.setLevel(SRC_LOG_LEVELS["MODELS"])
  28. router = APIRouter()
  29. ############################
  30. # Get Folders
  31. ############################
  32. @router.get("/", response_model=list[FolderNameIdResponse])
  33. async def get_folders(user=Depends(get_verified_user)):
  34. folders = Folders.get_folders_by_user_id(user.id)
  35. # Verify folder data integrity
  36. for folder in folders:
  37. if folder.data:
  38. if "files" in folder.data:
  39. valid_files = []
  40. for file in folder.data["files"]:
  41. if file.get("type") == "file":
  42. if Files.check_access_by_user_id(
  43. file.get("id"), user.id, "read"
  44. ):
  45. valid_files.append(file)
  46. elif file.get("type") == "collection":
  47. if Knowledges.check_access_by_user_id(
  48. file.get("id"), user.id, "read"
  49. ):
  50. valid_files.append(file)
  51. else:
  52. valid_files.append(file)
  53. folder.data["files"] = valid_files
  54. Folders.update_folder_by_id_and_user_id(
  55. folder.id, user.id, FolderUpdateForm(data=folder.data)
  56. )
  57. return [
  58. {
  59. **folder.model_dump(),
  60. }
  61. for folder in folders
  62. ]
  63. ############################
  64. # Create Folder
  65. ############################
  66. @router.post("/")
  67. def create_folder(form_data: FolderForm, user=Depends(get_verified_user)):
  68. folder = Folders.get_folder_by_parent_id_and_user_id_and_name(
  69. None, user.id, form_data.name
  70. )
  71. if folder:
  72. raise HTTPException(
  73. status_code=status.HTTP_400_BAD_REQUEST,
  74. detail=ERROR_MESSAGES.DEFAULT("Folder already exists"),
  75. )
  76. try:
  77. folder = Folders.insert_new_folder(user.id, form_data)
  78. return folder
  79. except Exception as e:
  80. log.exception(e)
  81. log.error("Error creating folder")
  82. raise HTTPException(
  83. status_code=status.HTTP_400_BAD_REQUEST,
  84. detail=ERROR_MESSAGES.DEFAULT("Error creating folder"),
  85. )
  86. ############################
  87. # Get Folders By Id
  88. ############################
  89. @router.get("/{id}", response_model=Optional[FolderModel])
  90. async def get_folder_by_id(id: str, user=Depends(get_verified_user)):
  91. folder = Folders.get_folder_by_id_and_user_id(id, user.id)
  92. if folder:
  93. return folder
  94. else:
  95. raise HTTPException(
  96. status_code=status.HTTP_404_NOT_FOUND,
  97. detail=ERROR_MESSAGES.NOT_FOUND,
  98. )
  99. ############################
  100. # Update Folder Name By Id
  101. ############################
  102. @router.post("/{id}/update")
  103. async def update_folder_name_by_id(
  104. id: str, form_data: FolderUpdateForm, user=Depends(get_verified_user)
  105. ):
  106. folder = Folders.get_folder_by_id_and_user_id(id, user.id)
  107. if folder:
  108. if form_data.name is not None:
  109. # Check if folder with same name exists
  110. existing_folder = Folders.get_folder_by_parent_id_and_user_id_and_name(
  111. folder.parent_id, user.id, form_data.name
  112. )
  113. if existing_folder and existing_folder.id != id:
  114. raise HTTPException(
  115. status_code=status.HTTP_400_BAD_REQUEST,
  116. detail=ERROR_MESSAGES.DEFAULT("Folder already exists"),
  117. )
  118. try:
  119. folder = Folders.update_folder_by_id_and_user_id(id, user.id, form_data)
  120. return folder
  121. except Exception as e:
  122. log.exception(e)
  123. log.error(f"Error updating folder: {id}")
  124. raise HTTPException(
  125. status_code=status.HTTP_400_BAD_REQUEST,
  126. detail=ERROR_MESSAGES.DEFAULT("Error updating folder"),
  127. )
  128. else:
  129. raise HTTPException(
  130. status_code=status.HTTP_404_NOT_FOUND,
  131. detail=ERROR_MESSAGES.NOT_FOUND,
  132. )
  133. ############################
  134. # Update Folder Parent Id By Id
  135. ############################
  136. class FolderParentIdForm(BaseModel):
  137. parent_id: Optional[str] = None
  138. @router.post("/{id}/update/parent")
  139. async def update_folder_parent_id_by_id(
  140. id: str, form_data: FolderParentIdForm, user=Depends(get_verified_user)
  141. ):
  142. folder = Folders.get_folder_by_id_and_user_id(id, user.id)
  143. if folder:
  144. existing_folder = Folders.get_folder_by_parent_id_and_user_id_and_name(
  145. form_data.parent_id, user.id, folder.name
  146. )
  147. if existing_folder:
  148. raise HTTPException(
  149. status_code=status.HTTP_400_BAD_REQUEST,
  150. detail=ERROR_MESSAGES.DEFAULT("Folder already exists"),
  151. )
  152. try:
  153. folder = Folders.update_folder_parent_id_by_id_and_user_id(
  154. id, user.id, form_data.parent_id
  155. )
  156. return folder
  157. except Exception as e:
  158. log.exception(e)
  159. log.error(f"Error updating folder: {id}")
  160. raise HTTPException(
  161. status_code=status.HTTP_400_BAD_REQUEST,
  162. detail=ERROR_MESSAGES.DEFAULT("Error updating folder"),
  163. )
  164. else:
  165. raise HTTPException(
  166. status_code=status.HTTP_404_NOT_FOUND,
  167. detail=ERROR_MESSAGES.NOT_FOUND,
  168. )
  169. ############################
  170. # Update Folder Is Expanded By Id
  171. ############################
  172. class FolderIsExpandedForm(BaseModel):
  173. is_expanded: bool
  174. @router.post("/{id}/update/expanded")
  175. async def update_folder_is_expanded_by_id(
  176. id: str, form_data: FolderIsExpandedForm, user=Depends(get_verified_user)
  177. ):
  178. folder = Folders.get_folder_by_id_and_user_id(id, user.id)
  179. if folder:
  180. try:
  181. folder = Folders.update_folder_is_expanded_by_id_and_user_id(
  182. id, user.id, form_data.is_expanded
  183. )
  184. return folder
  185. except Exception as e:
  186. log.exception(e)
  187. log.error(f"Error updating folder: {id}")
  188. raise HTTPException(
  189. status_code=status.HTTP_400_BAD_REQUEST,
  190. detail=ERROR_MESSAGES.DEFAULT("Error updating folder"),
  191. )
  192. else:
  193. raise HTTPException(
  194. status_code=status.HTTP_404_NOT_FOUND,
  195. detail=ERROR_MESSAGES.NOT_FOUND,
  196. )
  197. ############################
  198. # Delete Folder By Id
  199. ############################
  200. @router.delete("/{id}")
  201. async def delete_folder_by_id(
  202. request: Request, id: str, user=Depends(get_verified_user)
  203. ):
  204. if Chats.count_chats_by_folder_id_and_user_id(id, user.id):
  205. chat_delete_permission = has_permission(
  206. user.id, "chat.delete", request.app.state.config.USER_PERMISSIONS
  207. )
  208. if user.role != "admin" and not chat_delete_permission:
  209. raise HTTPException(
  210. status_code=status.HTTP_403_FORBIDDEN,
  211. detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
  212. )
  213. folder = Folders.get_folder_by_id_and_user_id(id, user.id)
  214. if folder:
  215. try:
  216. folder_ids = Folders.delete_folder_by_id_and_user_id(id, user.id)
  217. for folder_id in folder_ids:
  218. Chats.delete_chats_by_user_id_and_folder_id(user.id, folder_id)
  219. return True
  220. except Exception as e:
  221. log.exception(e)
  222. log.error(f"Error deleting folder: {id}")
  223. raise HTTPException(
  224. status_code=status.HTTP_400_BAD_REQUEST,
  225. detail=ERROR_MESSAGES.DEFAULT("Error deleting folder"),
  226. )
  227. else:
  228. raise HTTPException(
  229. status_code=status.HTTP_404_NOT_FOUND,
  230. detail=ERROR_MESSAGES.NOT_FOUND,
  231. )