folders.py 8.6 KB

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