mirror of
https://github.com/1Panel-dev/MaxKB.git
synced 2025-12-26 18:32:48 +00:00
161 lines
8.3 KiB
Python
161 lines
8.3 KiB
Python
# coding=utf-8
|
||
import re
|
||
import urllib
|
||
|
||
import uuid_utils.compat as uuid
|
||
from django.db.models import QuerySet
|
||
from django.http import HttpResponse
|
||
from django.utils.translation import gettext_lazy as _
|
||
from rest_framework import serializers
|
||
|
||
from common.exception.app_exception import NotFound404
|
||
from knowledge.models import File, FileSourceType
|
||
from tools.serializers.tool import UploadedFileField
|
||
|
||
mime_types = {
|
||
"html": "text/html", "htm": "text/html", "shtml": "text/html", "css": "text/css", "xml": "text/xml",
|
||
"gif": "image/gif", "jpeg": "image/jpeg", "jpg": "image/jpeg", "js": "application/javascript",
|
||
"atom": "application/atom+xml", "rss": "application/rss+xml", "mml": "text/mathml", "txt": "text/plain",
|
||
"jad": "text/vnd.sun.j2me.app-descriptor", "wml": "text/vnd.wap.wml", "htc": "text/x-component",
|
||
"avif": "image/avif", "png": "image/png", "svg": "image/svg+xml", "svgz": "image/svg+xml",
|
||
"tif": "image/tiff", "tiff": "image/tiff", "wbmp": "image/vnd.wap.wbmp", "webp": "image/webp",
|
||
"ico": "image/x-icon", "jng": "image/x-jng", "bmp": "image/x-ms-bmp", "woff": "font/woff",
|
||
"woff2": "font/woff2", "jar": "application/java-archive", "war": "application/java-archive",
|
||
"ear": "application/java-archive", "json": "application/json", "hqx": "application/mac-binhex40",
|
||
"doc": "application/msword", "pdf": "application/pdf", "ps": "application/postscript",
|
||
"docx": "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
|
||
"xlsx": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
|
||
"pptx": "application/vnd.openxmlformats-officedocument.presentationml.presentation",
|
||
"eps": "application/postscript", "ai": "application/postscript", "rtf": "application/rtf",
|
||
"m3u8": "application/vnd.apple.mpegurl", "kml": "application/vnd.google-earth.kml+xml",
|
||
"kmz": "application/vnd.google-earth.kmz", "xls": "application/vnd.ms-excel",
|
||
"eot": "application/vnd.ms-fontobject", "ppt": "application/vnd.ms-powerpoint",
|
||
"odg": "application/vnd.oasis.opendocument.graphics",
|
||
"odp": "application/vnd.oasis.opendocument.presentation",
|
||
"ods": "application/vnd.oasis.opendocument.spreadsheet", "odt": "application/vnd.oasis.opendocument.text",
|
||
"wmlc": "application/vnd.wap.wmlc", "wasm": "application/wasm", "7z": "application/x-7z-compressed",
|
||
"cco": "application/x-cocoa", "jardiff": "application/x-java-archive-diff",
|
||
"jnlp": "application/x-java-jnlp-file", "run": "application/x-makeself", "pl": "application/x-perl",
|
||
"pm": "application/x-perl", "prc": "application/x-pilot", "pdb": "application/x-pilot",
|
||
"rar": "application/x-rar-compressed", "rpm": "application/x-redhat-package-manager",
|
||
"sea": "application/x-sea", "swf": "application/x-shockwave-flash", "sit": "application/x-stuffit",
|
||
"tcl": "application/x-tcl", "tk": "application/x-tcl", "der": "application/x-x509-ca-cert",
|
||
"pem": "application/x-x509-ca-cert", "crt": "application/x-x509-ca-cert",
|
||
"xpi": "application/x-xpinstall", "xhtml": "application/xhtml+xml", "xspf": "application/xspf+xml",
|
||
"zip": "application/zip", "bin": "application/octet-stream", "exe": "application/octet-stream",
|
||
"dll": "application/octet-stream", "deb": "application/octet-stream", "dmg": "application/octet-stream",
|
||
"iso": "application/octet-stream", "img": "application/octet-stream", "msi": "application/octet-stream",
|
||
"msp": "application/octet-stream", "msm": "application/octet-stream", "mid": "audio/midi",
|
||
"midi": "audio/midi", "kar": "audio/midi", "mp3": "audio/mp3", "ogg": "audio/ogg", "m4a": "audio/x-m4a",
|
||
"ra": "audio/x-realaudio", "3gpp": "video/3gpp", "3gp": "video/3gpp", "ts": "video/mp2t",
|
||
"mp4": "video/mp4", "mpeg": "video/mpeg", "mpg": "video/mpeg", "mov": "video/quicktime",
|
||
"webm": "video/webm", "flv": "video/x-flv", "m4v": "video/x-m4v", "mng": "video/x-mng",
|
||
"asx": "video/x-ms-asf", "asf": "video/x-ms-asf", "wmv": "video/x-ms-wmv", "avi": "video/x-msvideo",
|
||
"wav": "audio/wav", "flac": "audio/flac", "aac": "audio/aac", "opus": "audio/opus",
|
||
"csv": "text/csv", "tsv": "text/tab-separated-values", "ics": "text/calendar",
|
||
}
|
||
|
||
# 如果是音频文件并且有range请求,处理部分内容
|
||
audio_types = ['mp3', 'wav', 'ogg', 'flac', 'aac', 'opus', 'm4a']
|
||
|
||
|
||
class FileSerializer(serializers.Serializer):
|
||
file = UploadedFileField(required=True, label=_('file'))
|
||
meta = serializers.JSONField(required=False, allow_null=True)
|
||
source_id = serializers.CharField(
|
||
required=False, allow_null=True, label=_('source id'), default=FileSourceType.TEMPORARY_120_MINUTE.value
|
||
)
|
||
source_type = serializers.ChoiceField(
|
||
choices=FileSourceType.choices, required=False, allow_null=True, label=_('source type'),
|
||
default=FileSourceType.TEMPORARY_120_MINUTE
|
||
)
|
||
|
||
def upload(self, with_valid=True):
|
||
if with_valid:
|
||
self.is_valid(raise_exception=True)
|
||
meta = self.data.get('meta', None)
|
||
if not meta:
|
||
meta = {'debug': True}
|
||
file_id = meta.get('file_id', uuid.uuid7())
|
||
file = File(
|
||
id=file_id,
|
||
file_name=self.data.get('file').name,
|
||
meta=meta,
|
||
source_id=self.data.get('source_id') or FileSourceType.TEMPORARY_120_MINUTE.value,
|
||
source_type=self.data.get('source_type') or FileSourceType.TEMPORARY_120_MINUTE
|
||
)
|
||
file.save(self.data.get('file').read())
|
||
return f'./oss/file/{file_id}'
|
||
|
||
class Operate(serializers.Serializer):
|
||
id = serializers.UUIDField(required=True)
|
||
http_range = serializers.CharField(
|
||
required=False, allow_blank=True, allow_null=True, label=_('HTTP Range'),
|
||
help_text=_('HTTP Range header for partial content requests, e.g., "bytes=0-1023"')
|
||
)
|
||
|
||
def get(self, with_valid=True):
|
||
if with_valid:
|
||
self.is_valid(raise_exception=True)
|
||
file_id = self.data.get('id')
|
||
file = QuerySet(File).filter(id=file_id).first()
|
||
if file is None:
|
||
raise NotFound404(404, _('File not found'))
|
||
file_type = file.file_name.split(".")[-1].lower()
|
||
content_type = mime_types.get(file_type, 'application/octet-stream')
|
||
encoded_filename = urllib.parse.quote(file.file_name)
|
||
# 获取文件内容
|
||
file_bytes = file.get_bytes()
|
||
file_size = len(file_bytes)
|
||
|
||
response = None
|
||
if file_type in audio_types and self.data.get('http_range'):
|
||
response = self.handle_audio(file_size, file_bytes, content_type, encoded_filename)
|
||
if response:
|
||
return response
|
||
|
||
# 对于非范围请求或其他类型文件,返回完整内容
|
||
headers = {
|
||
'Content-Type': content_type,
|
||
'Content-Disposition': f'{"inline" if file_type == "pdf" else "attachment"}; filename={encoded_filename}'
|
||
}
|
||
return HttpResponse(
|
||
file_bytes,
|
||
status=200,
|
||
headers=headers
|
||
)
|
||
|
||
def handle_audio(self, file_size, file_bytes, content_type, encoded_filename):
|
||
|
||
# 解析range请求 (格式如 "bytes=0-1023")
|
||
range_match = re.match(r'bytes=(\d+)-(\d*)', self.data.get('http_range', ''))
|
||
if range_match:
|
||
start = int(range_match.group(1))
|
||
end = int(range_match.group(2)) if range_match.group(2) else file_size - 1
|
||
|
||
# 确保范围合法
|
||
end = min(end, file_size - 1)
|
||
length = end - start + 1
|
||
|
||
# 创建部分响应
|
||
response = HttpResponse(
|
||
file_bytes[start:start + length],
|
||
status=206,
|
||
content_type=content_type
|
||
)
|
||
|
||
# 设置部分内容响应头
|
||
response['Content-Range'] = f'bytes {start}-{end}/{file_size}'
|
||
response['Accept-Ranges'] = 'bytes'
|
||
response['Content-Length'] = str(length)
|
||
response['Content-Disposition'] = f'inline; filename={encoded_filename}'
|
||
return response
|
||
|
||
def delete(self):
|
||
self.is_valid(raise_exception=True)
|
||
file_id = self.data.get('id')
|
||
file = QuerySet(File).filter(id=file_id).first()
|
||
if file is not None:
|
||
file.delete()
|
||
return True
|