# @Time : 2023/8/14 03:59 # @Author : Lan # @File : views.py # @Software: PyCharm # 导入所需的库和模块 from fastapi import APIRouter, Form, UploadFile, File, Depends, HTTPException from apps.admin.depends import admin_required from apps.base.models import FileCodes from apps.base.pydantics import SelectFileModel from apps.base.utils import get_expire_info, get_file_path_name, ip_limit from core.response import APIResponse from core.settings import settings from core.storage import storages, FileStorageInterface from core.utils import get_select_token # 创建一个API路由 share_api = APIRouter( prefix='/share', # 路由前缀 tags=['分享'], # 标签 ) # 分享文本的API @share_api.post('/text/', dependencies=[Depends(admin_required)]) async def share_text(text: str = Form(...), expire_value: int = Form(default=1, gt=0), expire_style: str = Form(default='day'), ip: str = Depends(ip_limit['upload'])): # 获取大小 text_size = len(text.encode('utf-8')) # 限制 222KB max_txt_size = 222 * 1024 # 转换为字节 if text_size > max_txt_size: raise HTTPException(status_code=403, detail=f'内容过多,建议采用文件形式') # 获取过期信息 expired_at, expired_count, used_count, code = await get_expire_info(expire_value, expire_style) # 创建一个新的FileCodes实例 await FileCodes.create( code=code, text=text, expired_at=expired_at, expired_count=expired_count, used_count=used_count, size=len(text), prefix='文本分享' ) # 添加IP到限制列表 ip_limit['upload'].add_ip(ip) # 返回API响应 return APIResponse(detail={ 'code': code, }) # 分享文件的API @share_api.post('/file/', dependencies=[Depends(admin_required)]) async def share_file(expire_value: int = Form(default=1, gt=0), expire_style: str = Form(default='day'), file: UploadFile = File(...), ip: str = Depends(ip_limit['upload'])): if file.size > settings.uploadSize: # 转换为 MB 并格式化输出 max_size_mb = settings.uploadSize / (1024 * 1024) raise HTTPException(status_code=403, detail=f'大小超过限制,最大为{max_size_mb:.2f} MB') # 获取过期信息 if expire_style not in settings.expireStyle: raise HTTPException(status_code=400, detail='过期时间类型错误') expired_at, expired_count, used_count, code = await get_expire_info(expire_value, expire_style) # 获取文件路径和名称 path, suffix, prefix, uuid_file_name, save_path = await get_file_path_name(file) # 保存文件 file_storage: FileStorageInterface = storages[settings.file_storage]() await file_storage.save_file(file, save_path) # 创建一个新的FileCodes实例 await FileCodes.create( code=code, prefix=prefix, suffix=suffix, uuid_file_name=uuid_file_name, file_path=path, size=file.size, expired_at=expired_at, expired_count=expired_count, used_count=used_count, ) # 添加IP到限制列表 ip_limit['upload'].add_ip(ip) # 返回API响应 return APIResponse(detail={ 'code': code, 'name': file.filename, }) # 根据code获取文件 async def get_code_file_by_code(code, check=True): # 查询文件 file_code = await FileCodes.filter(code=code).first() # 检查文件是否存在 if not file_code: return False, '文件不存在' # 检查文件是否过期 if await file_code.is_expired() and check: return False, '文件已过期', return True, file_code # 获取文件的API @share_api.get('/select/') async def get_code_file(code: str, ip: str = Depends(ip_limit['error'])): file_storage: FileStorageInterface = storages[settings.file_storage]() # 获取文件 has, file_code = await get_code_file_by_code(code) # 检查文件是否存在 if not has: # 添加IP到限制列表 ip_limit['error'].add_ip(ip) # 返回API响应 return APIResponse(code=404, detail=file_code) # 更新文件的使用次数和过期次数 file_code.used_count += 1 if file_code.expired_count > 0: file_code.expired_count -= 1 # 保存文件 await file_code.save() # 返回文件响应 return await file_storage.get_file_response(file_code) # 选择文件的API @share_api.post('/select/') async def select_file(data: SelectFileModel, ip: str = Depends(ip_limit['error'])): file_storage: FileStorageInterface = storages[settings.file_storage]() # 获取文件 has, file_code = await get_code_file_by_code(data.code) # 检查文件是否存在 if not has: # 添加IP到限制列表 ip_limit['error'].add_ip(ip) # 返回API响应 return APIResponse(code=404, detail=file_code) # 更新文件的使用次数和过期次数 file_code.used_count += 1 if file_code.expired_count > 0: file_code.expired_count -= 1 # 保存文件 await file_code.save() # 返回API响应 return APIResponse(detail={ 'code': file_code.code, 'name': file_code.prefix + file_code.suffix, 'size': file_code.size, 'text': file_code.text if file_code.text is not None else await file_storage.get_file_url(file_code), }) # 下载文件的API @share_api.get('/download') async def download_file(key: str, code: str, ip: str = Depends(ip_limit['error'])): file_storage: FileStorageInterface = storages[settings.file_storage]() # 检查token是否有效 is_valid = await get_select_token(code) == key if not is_valid: # 添加IP到限制列表 ip_limit['error'].add_ip(ip) # 获取文件 has, file_code = await get_code_file_by_code(code, False) # 检查文件是否存在 if not has: # 返回API响应 return APIResponse(code=404, detail='文件不存在') # 如果文件是文本,返回文本内容,否则返回文件响应 if file_code.text: return APIResponse(detail=file_code.text) else: return await file_storage.get_file_response(file_code)