UnisKB/apps/chat/views/chat.py

258 lines
10 KiB
Python
Raw Normal View History

2025-06-06 14:28:21 +00:00
# coding=utf-8
"""
@project: MaxKB
@Author虎虎
@file chat.py
@date2025/6/6 11:18
@desc:
"""
2025-08-27 06:23:58 +00:00
import requests
from django.http import HttpResponse, StreamingHttpResponse
2025-06-06 14:28:21 +00:00
from django.utils.translation import gettext_lazy as _
from drf_spectacular.utils import extend_schema
2025-07-04 11:52:45 +00:00
from rest_framework.parsers import MultiPartParser
2025-06-06 14:28:21 +00:00
from rest_framework.request import Request
from rest_framework.views import APIView
from application.api.application_api import SpeechToTextAPI, TextToSpeechAPI
2025-06-09 08:18:43 +00:00
from chat.api.chat_api import ChatAPI
2025-07-10 11:16:47 +00:00
from chat.api.chat_authentication_api import ChatAuthenticationAPI, ChatAuthenticationProfileAPI, ChatOpenAPI, OpenAIAPI
from chat.serializers.chat import OpenChatSerializers, ChatSerializers, SpeechToTextSerializers, \
TextToSpeechSerializers, OpenAIChatSerializer
2025-06-09 08:18:43 +00:00
from chat.serializers.chat_authentication import AnonymousAuthenticationSerializer, ApplicationProfileSerializer, \
AuthProfileSerializer
2025-06-06 14:28:21 +00:00
from common.auth import TokenAuth
2025-06-09 08:18:43 +00:00
from common.constants.permission_constants import ChatAuth
2025-06-06 14:28:21 +00:00
from common.exception.app_exception import AppAuthenticationFailed
from common.result import result
2025-07-04 11:52:45 +00:00
from knowledge.models import FileSourceType
from oss.serializers.file import FileSerializer
2025-06-17 07:01:59 +00:00
from users.api import CaptchaAPI
from users.serializers.login import CaptchaSerializer
2025-06-06 14:28:21 +00:00
2025-08-27 06:23:58 +00:00
def stream_image(response):
"""生成器函数,用于流式传输图片数据"""
for chunk in response.iter_content(chunk_size=4096):
if chunk: # 过滤掉保持连接的空块
yield chunk
class ResourceProxy(APIView):
def get(self, request: Request):
image_url = request.query_params.get("url")
if not image_url:
return result.error("Missing 'url' parameter")
try:
# 发送GET请求流式获取图片内容
response = requests.get(
image_url,
stream=True, # 启用流式响应
allow_redirects=True,
timeout=10
)
content_type = response.headers.get('Content-Type', '').split(';')[0]
# 创建Django流式响应
django_response = StreamingHttpResponse(
stream_image(response), # 使用生成器
content_type=content_type
)
return django_response
except Exception as e:
return result.error(f"Image request failed: {str(e)}")
2025-07-10 11:16:47 +00:00
class OpenAIView(APIView):
authentication_classes = [TokenAuth]
@extend_schema(
methods=['POST'],
description=_('OpenAI Interface Dialogue'),
summary=_('OpenAI Interface Dialogue'),
operation_id=_('OpenAI Interface Dialogue'), # type: ignore
request=OpenAIAPI.get_request(),
responses=None,
tags=[_('Chat')] # type: ignore
)
def post(self, request: Request, application_id: str):
return OpenAIChatSerializer(data={'application_id': application_id, 'chat_user_id': request.auth.chat_user_id,
'chat_user_type': request.auth.chat_user_type}).chat(request.data)
2025-06-09 08:18:43 +00:00
class AnonymousAuthentication(APIView):
2025-06-06 14:28:21 +00:00
def options(self, request, *args, **kwargs):
return HttpResponse(
headers={"Access-Control-Allow-Origin": "*", "Access-Control-Allow-Credentials": "true",
"Access-Control-Allow-Methods": "POST",
"Access-Control-Allow-Headers": "Origin,Content-Type,Cookie,Accept,Token"}, )
@extend_schema(
methods=['POST'],
2025-06-09 08:18:43 +00:00
description=_('Application Anonymous Certification'),
summary=_('Application Anonymous Certification'),
operation_id=_('Application Anonymous Certification'), # type: ignore
2025-06-06 14:28:21 +00:00
request=ChatAuthenticationAPI.get_request(),
responses=None,
tags=[_('Chat')] # type: ignore
)
def post(self, request: Request):
return result.success(
2025-06-09 08:18:43 +00:00
AnonymousAuthenticationSerializer(data={'access_token': request.data.get("access_token")}).auth(
2025-06-06 14:28:21 +00:00
request),
headers={"Access-Control-Allow-Origin": "*", "Access-Control-Allow-Credentials": "true",
"Access-Control-Allow-Methods": "POST",
"Access-Control-Allow-Headers": "Origin,Content-Type,Cookie,Accept,Token"}
)
class ApplicationProfile(APIView):
authentication_classes = [TokenAuth]
@extend_schema(
methods=['GET'],
description=_("Get application related information"),
summary=_("Get application related information"),
operation_id=_("Get application related information"), # type: ignore
request=None,
responses=None,
tags=[_('Chat')] # type: ignore
)
def get(self, request: Request):
2025-06-09 08:18:43 +00:00
if isinstance(request.auth, ChatAuth):
2025-06-06 14:28:21 +00:00
return result.success(ApplicationProfileSerializer(
2025-06-09 08:18:43 +00:00
data={'application_id': request.auth.application_id}).profile())
2025-06-06 14:28:21 +00:00
raise AppAuthenticationFailed(401, "身份异常")
2025-06-09 08:18:43 +00:00
class AuthProfile(APIView):
@extend_schema(
methods=['GET'],
description=_("Get application authentication information"),
summary=_("Get application authentication information"),
operation_id=_("Get application authentication information"), # type: ignore
parameters=ChatAuthenticationProfileAPI.get_parameters(),
responses=None,
tags=[_('Chat')] # type: ignore
)
def get(self, request: Request):
return result.success(
AuthProfileSerializer(data={'access_token': request.query_params.get("access_token")}).profile())
2025-06-06 14:28:21 +00:00
class ChatView(APIView):
2025-06-09 08:18:43 +00:00
authentication_classes = [TokenAuth]
@extend_schema(
methods=['POST'],
description=_("dialogue"),
summary=_("dialogue"),
operation_id=_("dialogue"), # type: ignore
request=ChatAPI.get_request(),
parameters=ChatAPI.get_parameters(),
responses=None,
tags=[_('Chat')] # type: ignore
)
def post(self, request: Request, chat_id: str):
return ChatSerializers(data={'chat_id': chat_id,
'chat_user_id': request.auth.chat_user_id,
'chat_user_type': request.auth.chat_user_type,
2025-06-17 03:58:35 +00:00
'application_id': request.auth.application_id,
'debug': False}
2025-06-09 08:18:43 +00:00
).chat(request.data)
class OpenView(APIView):
authentication_classes = [TokenAuth]
@extend_schema(
methods=['GET'],
description=_("Get the session id according to the application id"),
summary=_("Get the session id according to the application id"),
operation_id=_("Get the session id according to the application id"), # type: ignore
parameters=ChatOpenAPI.get_parameters(),
responses=None,
tags=[_('Chat')] # type: ignore
)
2025-06-17 03:58:35 +00:00
def get(self, request: Request):
2025-06-09 08:18:43 +00:00
return result.success(OpenChatSerializers(
2025-06-17 03:58:35 +00:00
data={'application_id': request.auth.application_id,
'chat_user_id': request.auth.chat_user_id, 'chat_user_type': request.auth.chat_user_type,
'debug': False}).open())
2025-06-17 07:01:59 +00:00
class CaptchaView(APIView):
@extend_schema(methods=['GET'],
2025-06-23 12:19:32 +00:00
summary=_("Get Chat captcha"),
description=_("Get Chat captcha"),
operation_id=_("Get Chat captcha"), # type: ignore
2025-07-04 11:52:45 +00:00
tags=[_("Chat")], # type: ignore
2025-06-17 07:01:59 +00:00
responses=CaptchaAPI.get_response())
def get(self, request: Request):
username = request.query_params.get('username', None)
accessToken = request.query_params.get('accessToken', None)
return result.success(CaptchaSerializer().chat_generate(username, 'chat', accessToken))
class SpeechToText(APIView):
authentication_classes = [TokenAuth]
@extend_schema(
methods=['POST'],
description=_("speech to text"),
summary=_("speech to text"),
operation_id=_("speech to text"), # type: ignore
request=SpeechToTextAPI.get_request(),
responses=SpeechToTextAPI.get_response(),
2025-07-04 11:52:45 +00:00
tags=[_('Chat')] # type: ignore
)
def post(self, request: Request):
return result.success(
SpeechToTextSerializers(
data={'application_id': request.auth.application_id})
.speech_to_text({'file': request.FILES.get('file')}))
class TextToSpeech(APIView):
authentication_classes = [TokenAuth]
@extend_schema(
methods=['POST'],
description=_("text to speech"),
summary=_("text to speech"),
operation_id=_("text to speech"), # type: ignore
request=TextToSpeechAPI.get_request(),
responses=TextToSpeechAPI.get_response(),
2025-07-04 11:52:45 +00:00
tags=[_('Chat')] # type: ignore
)
def post(self, request: Request):
byte_data = TextToSpeechSerializers(
data={'application_id': request.auth.application_id}).text_to_speech(request.data)
return HttpResponse(byte_data, status=200, headers={'Content-Type': 'audio/mp3',
'Content-Disposition': 'attachment; filename="abc.mp3"'})
2025-07-04 11:52:45 +00:00
class UploadFile(APIView):
authentication_classes = [TokenAuth]
parser_classes = [MultiPartParser]
@extend_schema(
methods=['POST'],
description=_("Upload files"),
summary=_("Upload files"),
operation_id=_("Upload files"), # type: ignore
request=TextToSpeechAPI.get_request(),
responses=TextToSpeechAPI.get_response(),
tags=[_('Application')] # type: ignore
)
def post(self, request: Request, chat_id: str):
files = request.FILES.getlist('file')
file_ids = []
meta = {}
for file in files:
file_url = FileSerializer(
data={'file': file, 'meta': meta, 'source_id': chat_id, 'source_type': FileSourceType.CHAT, }).upload()
file_ids.append({'name': file.name, 'url': file_url, 'file_id': file_url.split('/')[-1]})
return result.success(file_ids)