123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698 |
- import json
- import logging
- from typing import Optional
- from open_webui.models.chats import (
- ChatForm,
- ChatImportForm,
- ChatResponse,
- Chats,
- ChatTitleIdResponse,
- )
- from open_webui.models.tags import TagModel, Tags
- from open_webui.models.folders import Folders
- from open_webui.config import ENABLE_ADMIN_CHAT_ACCESS, ENABLE_ADMIN_EXPORT
- from open_webui.constants import ERROR_MESSAGES
- from open_webui.env import SRC_LOG_LEVELS
- from fastapi import APIRouter, Depends, HTTPException, Request, status
- from pydantic import BaseModel
- from open_webui.utils.auth import get_admin_user, get_verified_user
- from open_webui.utils.access_control import has_permission
- log = logging.getLogger(__name__)
- log.setLevel(SRC_LOG_LEVELS["MODELS"])
- router = APIRouter()
- ############################
- # GetChatList
- ############################
- @router.get("/", response_model=list[ChatTitleIdResponse])
- @router.get("/list", response_model=list[ChatTitleIdResponse])
- async def get_session_user_chat_list(
- user=Depends(get_verified_user), page: Optional[int] = None
- ):
- if page is not None:
- limit = 60
- skip = (page - 1) * limit
- return Chats.get_chat_title_id_list_by_user_id(user.id, skip=skip, limit=limit)
- else:
- return Chats.get_chat_title_id_list_by_user_id(user.id)
- ############################
- # DeleteAllChats
- ############################
- @router.delete("/", response_model=bool)
- async def delete_all_user_chats(request: Request, user=Depends(get_verified_user)):
- if user.role == "user" and not has_permission(
- user.id, "chat.delete", request.app.state.config.USER_PERMISSIONS
- ):
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED,
- detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
- )
- result = Chats.delete_chats_by_user_id(user.id)
- return result
- ############################
- # GetUserChatList
- ############################
- @router.get("/list/user/{user_id}", response_model=list[ChatTitleIdResponse])
- async def get_user_chat_list_by_user_id(
- user_id: str,
- user=Depends(get_admin_user),
- skip: int = 0,
- limit: int = 50,
- ):
- if not ENABLE_ADMIN_CHAT_ACCESS:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED,
- detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
- )
- return Chats.get_chat_list_by_user_id(
- user_id, include_archived=True, skip=skip, limit=limit
- )
- ############################
- # CreateNewChat
- ############################
- @router.post("/new", response_model=Optional[ChatResponse])
- async def create_new_chat(form_data: ChatForm, user=Depends(get_verified_user)):
- try:
- chat = Chats.insert_new_chat(user.id, form_data)
- return ChatResponse(**chat.model_dump())
- except Exception as e:
- log.exception(e)
- raise HTTPException(
- status_code=status.HTTP_400_BAD_REQUEST, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # ImportChat
- ############################
- @router.post("/import", response_model=Optional[ChatResponse])
- async def import_chat(form_data: ChatImportForm, user=Depends(get_verified_user)):
- try:
- chat = Chats.import_chat(user.id, form_data)
- if chat:
- tags = chat.meta.get("tags", [])
- for tag_id in tags:
- tag_id = tag_id.replace(" ", "_").lower()
- tag_name = " ".join([word.capitalize() for word in tag_id.split("_")])
- if (
- tag_id != "none"
- and Tags.get_tag_by_name_and_user_id(tag_name, user.id) is None
- ):
- Tags.insert_new_tag(tag_name, user.id)
- return ChatResponse(**chat.model_dump())
- except Exception as e:
- log.exception(e)
- raise HTTPException(
- status_code=status.HTTP_400_BAD_REQUEST, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # GetChats
- ############################
- @router.get("/search", response_model=list[ChatTitleIdResponse])
- async def search_user_chats(
- text: str, page: Optional[int] = None, user=Depends(get_verified_user)
- ):
- if page is None:
- page = 1
- limit = 60
- skip = (page - 1) * limit
- chat_list = [
- ChatTitleIdResponse(**chat.model_dump())
- for chat in Chats.get_chats_by_user_id_and_search_text(
- user.id, text, skip=skip, limit=limit
- )
- ]
- # Delete tag if no chat is found
- words = text.strip().split(" ")
- if page == 1 and len(words) == 1 and words[0].startswith("tag:"):
- tag_id = words[0].replace("tag:", "")
- if len(chat_list) == 0:
- if Tags.get_tag_by_name_and_user_id(tag_id, user.id):
- log.debug(f"deleting tag: {tag_id}")
- Tags.delete_tag_by_name_and_user_id(tag_id, user.id)
- return chat_list
- ############################
- # GetChatsByFolderId
- ############################
- @router.get("/folder/{folder_id}", response_model=list[ChatResponse])
- async def get_chats_by_folder_id(folder_id: str, user=Depends(get_verified_user)):
- folder_ids = [folder_id]
- children_folders = Folders.get_children_folders_by_id_and_user_id(
- folder_id, user.id
- )
- if children_folders:
- folder_ids.extend([folder.id for folder in children_folders])
- return [
- ChatResponse(**chat.model_dump())
- for chat in Chats.get_chats_by_folder_ids_and_user_id(folder_ids, user.id)
- ]
- ############################
- # GetPinnedChats
- ############################
- @router.get("/pinned", response_model=list[ChatResponse])
- async def get_user_pinned_chats(user=Depends(get_verified_user)):
- return [
- ChatResponse(**chat.model_dump())
- for chat in Chats.get_pinned_chats_by_user_id(user.id)
- ]
- ############################
- # GetChats
- ############################
- @router.get("/all", response_model=list[ChatResponse])
- async def get_user_chats(user=Depends(get_verified_user)):
- return [
- ChatResponse(**chat.model_dump())
- for chat in Chats.get_chats_by_user_id(user.id)
- ]
- ############################
- # GetArchivedChats
- ############################
- @router.get("/all/archived", response_model=list[ChatResponse])
- async def get_user_archived_chats(user=Depends(get_verified_user)):
- return [
- ChatResponse(**chat.model_dump())
- for chat in Chats.get_archived_chats_by_user_id(user.id)
- ]
- ############################
- # GetAllTags
- ############################
- @router.get("/all/tags", response_model=list[TagModel])
- async def get_all_user_tags(user=Depends(get_verified_user)):
- try:
- tags = Tags.get_tags_by_user_id(user.id)
- return tags
- except Exception as e:
- log.exception(e)
- raise HTTPException(
- status_code=status.HTTP_400_BAD_REQUEST, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # GetAllChatsInDB
- ############################
- @router.get("/all/db", response_model=list[ChatResponse])
- async def get_all_user_chats_in_db(user=Depends(get_admin_user)):
- if not ENABLE_ADMIN_EXPORT:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED,
- detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
- )
- return [ChatResponse(**chat.model_dump()) for chat in Chats.get_chats()]
- ############################
- # GetArchivedChats
- ############################
- @router.get("/archived", response_model=list[ChatTitleIdResponse])
- async def get_archived_session_user_chat_list(
- user=Depends(get_verified_user), skip: int = 0, limit: int = 50
- ):
- return Chats.get_archived_chat_list_by_user_id(user.id, skip, limit)
- ############################
- # ArchiveAllChats
- ############################
- @router.post("/archive/all", response_model=bool)
- async def archive_all_chats(user=Depends(get_verified_user)):
- return Chats.archive_all_chats_by_user_id(user.id)
- ############################
- # GetSharedChatById
- ############################
- @router.get("/share/{share_id}", response_model=Optional[ChatResponse])
- async def get_shared_chat_by_id(share_id: str, user=Depends(get_verified_user)):
- if user.role == "pending":
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
- )
- if user.role == "user" or (user.role == "admin" and not ENABLE_ADMIN_CHAT_ACCESS):
- chat = Chats.get_chat_by_share_id(share_id)
- elif user.role == "admin" and ENABLE_ADMIN_CHAT_ACCESS:
- chat = Chats.get_chat_by_id(share_id)
- if chat:
- return ChatResponse(**chat.model_dump())
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
- )
- ############################
- # GetChatsByTags
- ############################
- class TagForm(BaseModel):
- name: str
- class TagFilterForm(TagForm):
- skip: Optional[int] = 0
- limit: Optional[int] = 50
- @router.post("/tags", response_model=list[ChatTitleIdResponse])
- async def get_user_chat_list_by_tag_name(
- form_data: TagFilterForm, user=Depends(get_verified_user)
- ):
- chats = Chats.get_chat_list_by_user_id_and_tag_name(
- user.id, form_data.name, form_data.skip, form_data.limit
- )
- if len(chats) == 0:
- Tags.delete_tag_by_name_and_user_id(form_data.name, user.id)
- return chats
- ############################
- # GetChatById
- ############################
- @router.get("/{id}", response_model=Optional[ChatResponse])
- async def get_chat_by_id(id: str, user=Depends(get_verified_user)):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- return ChatResponse(**chat.model_dump())
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
- )
- ############################
- # UpdateChatById
- ############################
- @router.post("/{id}", response_model=Optional[ChatResponse])
- async def update_chat_by_id(
- id: str, form_data: ChatForm, user=Depends(get_verified_user)
- ):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- updated_chat = {**chat.chat, **form_data.chat}
- chat = Chats.update_chat_by_id(id, updated_chat)
- return ChatResponse(**chat.model_dump())
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED,
- detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
- )
- ############################
- # DeleteChatById
- ############################
- @router.delete("/{id}", response_model=bool)
- async def delete_chat_by_id(request: Request, id: str, user=Depends(get_verified_user)):
- if user.role == "admin":
- chat = Chats.get_chat_by_id(id)
- for tag in chat.meta.get("tags", []):
- if Chats.count_chats_by_tag_name_and_user_id(tag, user.id) == 1:
- Tags.delete_tag_by_name_and_user_id(tag, user.id)
- result = Chats.delete_chat_by_id(id)
- return result
- else:
- if not has_permission(
- user.id, "chat.delete", request.app.state.config.USER_PERMISSIONS
- ):
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED,
- detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
- )
- chat = Chats.get_chat_by_id(id)
- for tag in chat.meta.get("tags", []):
- if Chats.count_chats_by_tag_name_and_user_id(tag, user.id) == 1:
- Tags.delete_tag_by_name_and_user_id(tag, user.id)
- result = Chats.delete_chat_by_id_and_user_id(id, user.id)
- return result
- ############################
- # GetPinnedStatusById
- ############################
- @router.get("/{id}/pinned", response_model=Optional[bool])
- async def get_pinned_status_by_id(id: str, user=Depends(get_verified_user)):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- return chat.pinned
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # PinChatById
- ############################
- @router.post("/{id}/pin", response_model=Optional[ChatResponse])
- async def pin_chat_by_id(id: str, user=Depends(get_verified_user)):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- chat = Chats.toggle_chat_pinned_by_id(id)
- return chat
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # CloneChat
- ############################
- class CloneForm(BaseModel):
- title: Optional[str] = None
- @router.post("/{id}/clone", response_model=Optional[ChatResponse])
- async def clone_chat_by_id(
- form_data: CloneForm, id: str, user=Depends(get_verified_user)
- ):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- updated_chat = {
- **chat.chat,
- "originalChatId": chat.id,
- "branchPointMessageId": chat.chat["history"]["currentId"],
- "title": form_data.title if form_data.title else f"Clone of {chat.title}",
- }
- chat = Chats.insert_new_chat(user.id, ChatForm(**{"chat": updated_chat}))
- return ChatResponse(**chat.model_dump())
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # CloneSharedChatById
- ############################
- @router.post("/{id}/clone/shared", response_model=Optional[ChatResponse])
- async def clone_shared_chat_by_id(id: str, user=Depends(get_verified_user)):
- chat = Chats.get_chat_by_share_id(id)
- if chat:
- updated_chat = {
- **chat.chat,
- "originalChatId": chat.id,
- "branchPointMessageId": chat.chat["history"]["currentId"],
- "title": f"Clone of {chat.title}",
- }
- chat = Chats.insert_new_chat(user.id, ChatForm(**{"chat": updated_chat}))
- return ChatResponse(**chat.model_dump())
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # ArchiveChat
- ############################
- @router.post("/{id}/archive", response_model=Optional[ChatResponse])
- async def archive_chat_by_id(id: str, user=Depends(get_verified_user)):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- chat = Chats.toggle_chat_archive_by_id(id)
- # Delete tags if chat is archived
- if chat.archived:
- for tag_id in chat.meta.get("tags", []):
- if Chats.count_chats_by_tag_name_and_user_id(tag_id, user.id) == 0:
- log.debug(f"deleting tag: {tag_id}")
- Tags.delete_tag_by_name_and_user_id(tag_id, user.id)
- else:
- for tag_id in chat.meta.get("tags", []):
- tag = Tags.get_tag_by_name_and_user_id(tag_id, user.id)
- if tag is None:
- log.debug(f"inserting tag: {tag_id}")
- tag = Tags.insert_new_tag(tag_id, user.id)
- return ChatResponse(**chat.model_dump())
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # ShareChatById
- ############################
- @router.post("/{id}/share", response_model=Optional[ChatResponse])
- async def share_chat_by_id(id: str, user=Depends(get_verified_user)):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- if chat.share_id:
- shared_chat = Chats.update_shared_chat_by_chat_id(chat.id)
- return ChatResponse(**shared_chat.model_dump())
- shared_chat = Chats.insert_shared_chat_by_chat_id(chat.id)
- if not shared_chat:
- raise HTTPException(
- status_code=status.HTTP_500_INTERNAL_SERVER_ERROR,
- detail=ERROR_MESSAGES.DEFAULT(),
- )
- return ChatResponse(**shared_chat.model_dump())
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED,
- detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
- )
- ############################
- # DeletedSharedChatById
- ############################
- @router.delete("/{id}/share", response_model=Optional[bool])
- async def delete_shared_chat_by_id(id: str, user=Depends(get_verified_user)):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- if not chat.share_id:
- return False
- result = Chats.delete_shared_chat_by_chat_id(id)
- update_result = Chats.update_chat_share_id_by_id(id, None)
- return result and update_result != None
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED,
- detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
- )
- ############################
- # UpdateChatFolderIdById
- ############################
- class ChatFolderIdForm(BaseModel):
- folder_id: Optional[str] = None
- @router.post("/{id}/folder", response_model=Optional[ChatResponse])
- async def update_chat_folder_id_by_id(
- id: str, form_data: ChatFolderIdForm, user=Depends(get_verified_user)
- ):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- chat = Chats.update_chat_folder_id_by_id_and_user_id(
- id, user.id, form_data.folder_id
- )
- return ChatResponse(**chat.model_dump())
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # GetChatTagsById
- ############################
- @router.get("/{id}/tags", response_model=list[TagModel])
- async def get_chat_tags_by_id(id: str, user=Depends(get_verified_user)):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- tags = chat.meta.get("tags", [])
- return Tags.get_tags_by_ids_and_user_id(tags, user.id)
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
- )
- ############################
- # AddChatTagById
- ############################
- @router.post("/{id}/tags", response_model=list[TagModel])
- async def add_tag_by_id_and_tag_name(
- id: str, form_data: TagForm, user=Depends(get_verified_user)
- ):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- tags = chat.meta.get("tags", [])
- tag_id = form_data.name.replace(" ", "_").lower()
- if tag_id == "none":
- raise HTTPException(
- status_code=status.HTTP_400_BAD_REQUEST,
- detail=ERROR_MESSAGES.DEFAULT("Tag name cannot be 'None'"),
- )
- if tag_id not in tags:
- Chats.add_chat_tag_by_id_and_user_id_and_tag_name(
- id, user.id, form_data.name
- )
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- tags = chat.meta.get("tags", [])
- return Tags.get_tags_by_ids_and_user_id(tags, user.id)
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
- )
- ############################
- # DeleteChatTagById
- ############################
- @router.delete("/{id}/tags", response_model=list[TagModel])
- async def delete_tag_by_id_and_tag_name(
- id: str, form_data: TagForm, user=Depends(get_verified_user)
- ):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- Chats.delete_tag_by_id_and_user_id_and_tag_name(id, user.id, form_data.name)
- if Chats.count_chats_by_tag_name_and_user_id(form_data.name, user.id) == 0:
- Tags.delete_tag_by_name_and_user_id(form_data.name, user.id)
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- tags = chat.meta.get("tags", [])
- return Tags.get_tags_by_ids_and_user_id(tags, user.id)
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
- )
- ############################
- # DeleteAllTagsById
- ############################
- @router.delete("/{id}/tags/all", response_model=Optional[bool])
- async def delete_all_tags_by_id(id: str, user=Depends(get_verified_user)):
- chat = Chats.get_chat_by_id_and_user_id(id, user.id)
- if chat:
- Chats.delete_all_tags_by_id_and_user_id(id, user.id)
- for tag in chat.meta.get("tags", []):
- if Chats.count_chats_by_tag_name_and_user_id(tag, user.id) == 0:
- Tags.delete_tag_by_name_and_user_id(tag, user.id)
- return True
- else:
- raise HTTPException(
- status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
- )
|