open-webui/backend/open_webui/apps/webui/routers/chats.py

483 lines
14 KiB
Python
Raw Normal View History

2023-12-26 09:27:43 +00:00
import json
import logging
2024-08-27 22:10:27 +00:00
from typing import Optional
2023-12-26 05:44:28 +00:00
from open_webui.apps.webui.models.chats import (
ChatForm,
ChatResponse,
Chats,
ChatTitleIdResponse,
)
from open_webui.apps.webui.models.tags import (
ChatIdTagForm,
ChatIdTagModel,
TagModel,
Tags,
)
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
2024-08-27 22:10:27 +00:00
from fastapi import APIRouter, Depends, HTTPException, Request, status
from pydantic import BaseModel
from open_webui.utils.utils import get_admin_user, get_verified_user
2024-03-31 08:13:39 +00:00
log = logging.getLogger(__name__)
log.setLevel(SRC_LOG_LEVELS["MODELS"])
2023-12-26 05:44:28 +00:00
router = APIRouter()
############################
2024-04-27 22:12:57 +00:00
# GetChatList
2023-12-26 05:44:28 +00:00
############################
2024-08-14 12:46:31 +00:00
@router.get("/", response_model=list[ChatTitleIdResponse])
@router.get("/list", response_model=list[ChatTitleIdResponse])
2024-04-27 22:12:57 +00:00
async def get_session_user_chat_list(
2024-08-04 14:58:08 +00:00
user=Depends(get_verified_user), page: Optional[int] = None
2024-01-17 22:47:56 +00:00
):
2024-08-04 15:04:15 +00:00
if page is not None:
2024-08-04 15:35:26 +00:00
limit = 60
2024-08-04 14:58:08 +00:00
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)
2024-04-27 22:12:57 +00:00
2024-04-27 22:14:15 +00:00
############################
# DeleteAllChats
############################
@router.delete("/", response_model=bool)
2024-06-27 18:29:59 +00:00
async def delete_all_user_chats(request: Request, user=Depends(get_verified_user)):
2024-10-01 16:51:58 +00:00
if user.role == "user" and not request.app.state.config.USER_PERMISSIONS.get(
"chat", {}
).get("deletion", {}):
2024-04-27 22:14:15 +00:00
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED,
detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
)
result = Chats.delete_chats_by_user_id(user.id)
2024-04-27 22:14:15 +00:00
return result
2024-04-27 22:12:57 +00:00
############################
# GetUserChatList
############################
2024-08-14 12:46:31 +00:00
@router.get("/list/user/{user_id}", response_model=list[ChatTitleIdResponse])
2024-04-27 22:12:57 +00:00
async def get_user_chat_list_by_user_id(
user_id: str,
user=Depends(get_admin_user),
skip: int = 0,
limit: int = 50,
2024-04-27 22:12:57 +00:00
):
if not ENABLE_ADMIN_CHAT_ACCESS:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED,
detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
)
2024-05-26 09:00:31 +00:00
return Chats.get_chat_list_by_user_id(
user_id, include_archived=True, skip=skip, limit=limit
2024-05-26 09:00:31 +00:00
)
2023-12-26 05:44:28 +00:00
2024-04-20 23:24:18 +00:00
############################
2024-05-26 09:00:31 +00:00
# CreateNewChat
2024-05-03 22:34:08 +00:00
############################
2024-05-26 09:00:31 +00:00
@router.post("/new", response_model=Optional[ChatResponse])
2024-06-27 18:29:59 +00:00
async def create_new_chat(form_data: ChatForm, user=Depends(get_verified_user)):
2024-05-26 09:00:31 +00:00
try:
chat = Chats.insert_new_chat(user.id, form_data)
2024-05-03 22:34:08 +00:00
return ChatResponse(**{**chat.model_dump(), "chat": json.loads(chat.chat)})
2024-05-26 09:00:31 +00:00
except Exception as e:
log.exception(e)
2024-05-03 22:34:08 +00:00
raise HTTPException(
2024-05-26 09:00:31 +00:00
status_code=status.HTTP_400_BAD_REQUEST, detail=ERROR_MESSAGES.DEFAULT()
2024-05-03 22:34:08 +00:00
)
2023-12-27 06:10:22 +00:00
############################
2024-04-27 22:12:57 +00:00
# GetChats
2023-12-27 06:10:22 +00:00
############################
2024-08-14 12:46:31 +00:00
@router.get("/all", response_model=list[ChatResponse])
2024-06-27 18:29:59 +00:00
async def get_user_chats(user=Depends(get_verified_user)):
return [
2024-01-17 22:47:56 +00:00
ChatResponse(**{**chat.model_dump(), "chat": json.loads(chat.chat)})
for chat in Chats.get_chats_by_user_id(user.id)
2024-06-03 04:39:09 +00:00
]
############################
# GetArchivedChats
############################
2024-08-14 12:46:31 +00:00
@router.get("/all/archived", response_model=list[ChatResponse])
async def get_user_archived_chats(user=Depends(get_verified_user)):
2024-06-03 04:39:09 +00:00
return [
ChatResponse(**{**chat.model_dump(), "chat": json.loads(chat.chat)})
for chat in Chats.get_archived_chats_by_user_id(user.id)
]
2023-12-27 06:10:22 +00:00
############################
# GetAllChatsInDB
############################
2024-08-14 12:46:31 +00:00
@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(), "chat": json.loads(chat.chat)})
for chat in Chats.get_chats()
]
2023-12-26 05:44:28 +00:00
############################
2024-05-26 09:00:31 +00:00
# GetArchivedChats
2023-12-26 05:44:28 +00:00
############################
2024-08-14 12:46:31 +00:00
@router.get("/archived", response_model=list[ChatTitleIdResponse])
2024-05-26 09:00:31 +00:00
async def get_archived_session_user_chat_list(
2024-06-27 18:29:59 +00:00
user=Depends(get_verified_user), skip: int = 0, limit: int = 50
2024-05-26 09:00:31 +00:00
):
return Chats.get_archived_chat_list_by_user_id(user.id, skip, limit)
2024-05-26 09:00:31 +00:00
############################
# ArchiveAllChats
############################
2024-06-11 08:18:05 +00:00
@router.post("/archive/all", response_model=bool)
2024-06-27 18:29:59 +00:00
async def archive_all_chats(user=Depends(get_verified_user)):
return Chats.archive_all_chats_by_user_id(user.id)
2024-05-26 09:00:31 +00:00
############################
# GetSharedChatById
############################
@router.get("/share/{share_id}", response_model=Optional[ChatResponse])
2024-06-27 18:29:59 +00:00
async def get_shared_chat_by_id(share_id: str, user=Depends(get_verified_user)):
2024-05-26 09:00:31 +00:00
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)
2024-05-26 09:00:31 +00:00
if chat:
2024-01-17 22:47:56 +00:00
return ChatResponse(**{**chat.model_dump(), "chat": json.loads(chat.chat)})
2024-05-26 09:00:31 +00:00
else:
2024-01-17 22:47:56 +00:00
raise HTTPException(
2024-05-26 09:00:31 +00:00
status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
2024-01-17 22:47:56 +00:00
)
2023-12-26 05:44:28 +00:00
2024-05-03 22:34:08 +00:00
############################
# GetChatsByTags
############################
class TagNameForm(BaseModel):
name: str
skip: Optional[int] = 0
limit: Optional[int] = 50
2024-08-14 12:46:31 +00:00
@router.post("/tags", response_model=list[ChatTitleIdResponse])
2024-05-03 22:34:08 +00:00
async def get_user_chat_list_by_tag_name(
2024-06-27 18:29:59 +00:00
form_data: TagNameForm, user=Depends(get_verified_user)
2024-05-03 22:34:08 +00:00
):
chat_ids = [
chat_id_tag.chat_id
for chat_id_tag in Tags.get_chat_ids_by_tag_name_and_user_id(
form_data.name, user.id
2024-05-03 22:34:08 +00:00
)
]
2024-06-24 07:57:08 +00:00
chats = Chats.get_chat_list_by_chat_ids(chat_ids, form_data.skip, form_data.limit)
2024-05-03 22:34:08 +00:00
if len(chats) == 0:
Tags.delete_tag_by_tag_name_and_user_id(form_data.name, user.id)
2024-05-03 22:34:08 +00:00
return chats
############################
# GetAllTags
############################
2024-08-14 12:46:31 +00:00
@router.get("/tags/all", response_model=list[TagModel])
2024-06-27 18:29:59 +00:00
async def get_all_tags(user=Depends(get_verified_user)):
2024-05-03 22:34:08 +00:00
try:
tags = Tags.get_tags_by_user_id(user.id)
2024-05-03 22:34:08 +00:00
return tags
except Exception as e:
log.exception(e)
raise HTTPException(
status_code=status.HTTP_400_BAD_REQUEST, detail=ERROR_MESSAGES.DEFAULT()
)
2023-12-26 05:44:28 +00:00
############################
# GetChatById
############################
2023-12-26 09:27:43 +00:00
@router.get("/{id}", response_model=Optional[ChatResponse])
2024-06-27 18:29:59 +00:00
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)
2023-12-26 05:44:28 +00:00
if chat:
2024-01-17 22:47:56 +00:00
return ChatResponse(**{**chat.model_dump(), "chat": json.loads(chat.chat)})
2023-12-26 05:44:28 +00:00
else:
2024-01-17 22:47:56 +00:00
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
)
2023-12-26 05:44:28 +00:00
############################
# UpdateChatById
############################
2023-12-26 09:27:43 +00:00
@router.post("/{id}", response_model=Optional[ChatResponse])
2024-01-17 22:47:56 +00:00
async def update_chat_by_id(
2024-06-27 18:29:59 +00:00
id: str, form_data: ChatForm, user=Depends(get_verified_user)
2024-01-17 22:47:56 +00:00
):
chat = Chats.get_chat_by_id_and_user_id(id, user.id)
if chat:
updated_chat = {**json.loads(chat.chat), **form_data.chat}
2023-12-26 18:41:55 +00:00
chat = Chats.update_chat_by_id(id, updated_chat)
2024-01-17 22:47:56 +00:00
return ChatResponse(**{**chat.model_dump(), "chat": json.loads(chat.chat)})
else:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED,
detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
)
2023-12-26 09:27:43 +00:00
############################
# DeleteChatById
############################
@router.delete("/{id}", response_model=bool)
2024-06-27 18:29:59 +00:00
async def delete_chat_by_id(request: Request, id: str, user=Depends(get_verified_user)):
2024-04-27 22:24:59 +00:00
if user.role == "admin":
result = Chats.delete_chat_by_id(id)
2024-04-27 22:24:59 +00:00
return result
else:
2024-10-01 16:51:58 +00:00
if not request.app.state.config.USER_PERMISSIONS.get("chat", {}).get(
"deletion", {}
):
2024-04-27 22:24:59 +00:00
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED,
detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
)
result = Chats.delete_chat_by_id_and_user_id(id, user.id)
2024-04-27 22:24:59 +00:00
return result
2023-12-30 08:15:37 +00:00
2024-05-31 17:30:42 +00:00
############################
# CloneChat
############################
@router.get("/{id}/clone", response_model=Optional[ChatResponse])
2024-06-27 18:29:59 +00:00
async def clone_chat_by_id(id: str, user=Depends(get_verified_user)):
chat = Chats.get_chat_by_id_and_user_id(id, user.id)
2024-05-31 17:30:42 +00:00
if chat:
chat_body = json.loads(chat.chat)
updated_chat = {
**chat_body,
"originalChatId": chat.id,
"branchPointMessageId": chat_body["history"]["currentId"],
"title": f"Clone of {chat.title}",
}
chat = Chats.insert_new_chat(user.id, ChatForm(**{"chat": updated_chat}))
2024-05-31 17:30:42 +00:00
return ChatResponse(**{**chat.model_dump(), "chat": json.loads(chat.chat)})
else:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
)
2024-04-20 22:03:39 +00:00
############################
# ArchiveChat
############################
@router.get("/{id}/archive", response_model=Optional[ChatResponse])
2024-06-27 18:29:59 +00:00
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)
2024-04-20 22:03:39 +00:00
if chat:
chat = Chats.toggle_chat_archive_by_id(id)
2024-04-20 22:03:39 +00:00
return ChatResponse(**{**chat.model_dump(), "chat": json.loads(chat.chat)})
else:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
)
############################
# ShareChatById
############################
@router.post("/{id}/share", response_model=Optional[ChatResponse])
2024-06-27 18:29:59 +00:00
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(), "chat": json.loads(shared_chat.chat)}
)
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(), "chat": json.loads(shared_chat.chat)}
)
else:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED,
detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
)
############################
# DeletedSharedChatById
############################
2024-04-02 14:16:25 +00:00
@router.delete("/{id}/share", response_model=Optional[bool])
2024-06-27 18:29:59 +00:00
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)
2024-04-02 14:16:25 +00:00
return result and update_result != None
else:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED,
detail=ERROR_MESSAGES.ACCESS_PROHIBITED,
)
2024-01-18 08:58:45 +00:00
############################
# GetChatTagsById
############################
2024-08-14 12:46:31 +00:00
@router.get("/{id}/tags", response_model=list[TagModel])
2024-06-27 18:29:59 +00:00
async def get_chat_tags_by_id(id: str, user=Depends(get_verified_user)):
tags = Tags.get_tags_by_chat_id_and_user_id(id, user.id)
2024-01-18 08:58:45 +00:00
2024-01-18 10:10:07 +00:00
if tags != None:
2024-01-18 08:58:45 +00:00
return tags
else:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
)
############################
# AddChatTagById
############################
2024-01-18 10:10:07 +00:00
@router.post("/{id}/tags", response_model=Optional[ChatIdTagModel])
2024-01-18 08:58:45 +00:00
async def add_chat_tag_by_id(
2024-06-27 18:29:59 +00:00
id: str, form_data: ChatIdTagForm, user=Depends(get_verified_user)
2024-01-18 08:58:45 +00:00
):
tags = Tags.get_tags_by_chat_id_and_user_id(id, user.id)
2024-01-18 10:10:07 +00:00
if form_data.tag_name not in tags:
tag = Tags.add_tag_to_chat(user.id, form_data)
2024-01-18 08:58:45 +00:00
2024-01-18 10:10:07 +00:00
if tag:
return tag
else:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED,
detail=ERROR_MESSAGES.NOT_FOUND,
)
2024-01-18 08:58:45 +00:00
else:
raise HTTPException(
2024-01-18 10:10:07 +00:00
status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.DEFAULT()
2024-01-18 08:58:45 +00:00
)
############################
# DeleteChatTagById
############################
@router.delete("/{id}/tags", response_model=Optional[bool])
2024-01-18 09:04:24 +00:00
async def delete_chat_tag_by_id(
2024-06-27 18:29:59 +00:00
id: str, form_data: ChatIdTagForm, user=Depends(get_verified_user)
2024-01-18 08:58:45 +00:00
):
2024-01-18 09:04:24 +00:00
result = Tags.delete_tag_by_tag_name_and_chat_id_and_user_id(
form_data.tag_name, id, user.id
2024-01-18 08:58:45 +00:00
)
2024-01-18 09:04:24 +00:00
if result:
return result
else:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
)
############################
# DeleteAllChatTagsById
############################
@router.delete("/{id}/tags/all", response_model=Optional[bool])
2024-06-27 18:29:59 +00:00
async def delete_all_chat_tags_by_id(id: str, user=Depends(get_verified_user)):
result = Tags.delete_tags_by_chat_id_and_user_id(id, user.id)
2024-01-18 09:04:24 +00:00
if result:
return result
2024-01-18 08:58:45 +00:00
else:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED, detail=ERROR_MESSAGES.NOT_FOUND
)