Skip to content

Commit

Permalink
Merge pull request #80 from vastsa/new
Browse files Browse the repository at this point in the history
新版本,前端改vite+vue3
  • Loading branch information
vastsa authored Aug 14, 2023
2 parents c366476 + 0495ea2 commit 642b652
Show file tree
Hide file tree
Showing 80 changed files with 4,208 additions and 5,247 deletions.
4 changes: 3 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,9 @@ docs/_build/
# PyBuilder
.pybuilder/
target/

*.db
./filecodebox.db-shm
./filecodebox.db-wal
# Jupyter Notebook
.ipynb_checkpoints

Expand Down
4 changes: 4 additions & 0 deletions apps/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
# @Time : 2023/8/13 20:43
# @Author : Lan
# @File : __init__.py.py
# @Software: PyCharm
4 changes: 4 additions & 0 deletions apps/admin/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
# @Time : 2023/8/14 14:38
# @Author : Lan
# @File : __init__.py.py
# @Software: PyCharm
10 changes: 10 additions & 0 deletions apps/admin/views.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
# @Time : 2023/8/14 14:38
# @Author : Lan
# @File : views.py
# @Software: PyCharm
from fastapi import APIRouter

admin_api = APIRouter(
prefix='/admin',
tags=['管理'],
)
4 changes: 4 additions & 0 deletions apps/base/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
# @Time : 2023/8/13 20:43
# @Author : Lan
# @File : __init__.py.py
# @Software: PyCharm
17 changes: 7 additions & 10 deletions core/depends.py → apps/base/depends.py
Original file line number Diff line number Diff line change
@@ -1,20 +1,17 @@
# @Time : 2023/8/14 12:20
# @Author : Lan
# @File : depends.py
# @Software: PyCharm
from typing import Union
from datetime import datetime, timedelta

from fastapi import Header, HTTPException, Request

from settings import settings
from core.response import APIResponse


async def admin_required(pwd: Union[str, None] = Header(default=None), request: Request = None):
if 'share' in request.url.path:
if pwd != settings.ADMIN_PASSWORD and not settings.ENABLE_UPLOAD:
raise HTTPException(status_code=403, detail='本站上传功能已关闭,仅管理员可用')
else:
if settings.ADMIN_PASSWORD is None:
raise HTTPException(status_code=404, detail='您未设置管理员密码,无法使用此功能,请更新配置文件后,重启系统')
if not pwd or pwd != settings.ADMIN_PASSWORD:
raise HTTPException(status_code=401, detail="密码错误,请重新登录")
return False


class IPRateLimit:
Expand Down Expand Up @@ -48,5 +45,5 @@ async def remove_expired_ip(self):
def __call__(self, request: Request):
ip = request.headers.get('X-Real-IP', request.headers.get('X-Forwarded-For', request.client.host))
if not self.check_ip(ip):
raise HTTPException(status_code=400, detail=f"请求次数过多,请稍后再试")
raise HTTPException(status_code=423, detail=f"请求次数过多,请稍后再试")
return ip
40 changes: 40 additions & 0 deletions apps/base/models.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
# @Time : 2023/8/13 20:43
# @Author : Lan
# @File : models.py
# @Software: PyCharm
from datetime import datetime
from typing import Optional

from tortoise import fields
from tortoise.models import Model
from tortoise.contrib.pydantic import pydantic_model_creator

from core.utils import get_now


class FileCodes(Model):
id: Optional[int] = fields.IntField(pk=True)
code: Optional[int] = fields.CharField(description='分享码', max_length=255, index=True, unique=True)
prefix: Optional[str] = fields.CharField(max_length=255, description='前缀', default='')
suffix: Optional[str] = fields.CharField(max_length=255, description='后缀', default='')
uuid_file_name: Optional[str] = fields.CharField(max_length=255, description='uuid文件名', null=True)
file_path: Optional[str] = fields.CharField(max_length=255, description='文件路径', null=True)
size: Optional[int] = fields.IntField(description='文件大小', default=0)
text: Optional[str] = fields.TextField(description='文本内容', null=True)
expired_at: Optional[datetime] = fields.DatetimeField(null=True, description='过期时间')
expired_count: Optional[int] = fields.IntField(description='可用次数', default=0)
used_count: Optional[int] = fields.IntField(description='已用次数', default=0)

created_at: Optional[datetime] = fields.DatetimeField(auto_now_add=True, description='创建时间')

async def is_expired(self):
if self.expired_at and (self.expired_count == -1 or self.used_count < self.expired_count):
return self.expired_at < await get_now()
else:
return self.expired_count != -1 and self.used_count >= self.expired_count

async def get_file_path(self):
return f"{self.file_path}/{self.uuid_file_name}"


file_codes_pydantic = pydantic_model_creator(FileCodes, name='FileCodes')
5 changes: 5 additions & 0 deletions apps/base/pydantics.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
from pydantic import BaseModel


class SelectFileModel(BaseModel):
code: str
78 changes: 78 additions & 0 deletions apps/base/utils.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
# @Time : 2023/8/14 01:10
# @Author : Lan
# @File : utils.py
# @Software: PyCharm
import datetime
import uuid
import os
from fastapi import UploadFile

from apps.base.depends import IPRateLimit
from apps.base.models import FileCodes
from core.utils import get_random_num, get_random_string


async def get_file_path_name(file: UploadFile):
"""
获取文件路径和文件名
:param file:
:return: {
'path': 'share/data/2021/08/13',
'suffix': '.jpg',
'prefix': 'test',
'file_uuid': '44a83bbd70e04c8aa7fd93bfd8c88249',
'uuid_file_name': '44a83bbd70e04c8aa7fd93bfd8c88249.jpg',
'save_path': 'share/data/2021/08/13/44a83bbd70e04c8aa7fd93bfd8c88249.jpg'
}
"""
today = datetime.datetime.now()
path = f"share/data/{today.strftime('%Y/%m/%d')}"
prefix, suffix = os.path.splitext(file.filename)
file_uuid = f"{uuid.uuid4().hex}"
uuid_file_name = f"{file_uuid}{suffix}"
save_path = f"{path}/{uuid_file_name}"
return path, suffix, prefix, uuid_file_name, save_path


async def get_expire_info(expire_value: int, expire_style: str):
"""
获取过期信息
:param expire_value:
:param expire_style:
:return: expired_at 过期时间, expired_count 可用次数, used_count 已用次数, code 随机码
"""
expired_count, used_count, now, code = -1, 0, datetime.datetime.now(), None
if expire_style == 'day':
expired_at = now + datetime.timedelta(days=expire_value)
elif expire_style == 'hour':
expired_at = now + datetime.timedelta(hours=expire_value)
elif expire_style == 'minute':
expired_at = now + datetime.timedelta(minutes=expire_value)
elif expire_style == 'count':
expired_at = now + datetime.timedelta(days=1)
expired_count = expire_value
elif expire_style == 'forever':
expired_at = None
code = await get_random_code(style='string')
else:
expired_at = now + datetime.timedelta(days=1)
if not code:
code = await get_random_code()
return expired_at, expired_count, used_count, code


async def get_random_code(style='num'):
"""
获取随机字符串
:return:
"""
while True:
code = await get_random_num() if style == 'num' else await get_random_string()
if not await FileCodes.filter(code=code).exists():
return code


# 错误IP限制器
error_ip_limit = IPRateLimit(1, 1)
# 上传文件限制器
upload_ip_limit = IPRateLimit(10, 1)
91 changes: 91 additions & 0 deletions apps/base/views.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
# @Time : 2023/8/14 03:59
# @Author : Lan
# @File : views.py
# @Software: PyCharm
from fastapi import APIRouter, Form, UploadFile, File, Depends
from starlette.responses import FileResponse

from apps.base.models import FileCodes
from apps.base.pydantics import SelectFileModel
from apps.base.utils import get_expire_info, get_file_path_name, error_ip_limit
from core.response import APIResponse
from core.storage import file_storage

share_api = APIRouter(
prefix='/share',
tags=['分享'],
)


@share_api.post('/text/')
async def share_text(text: str = Form(...), expire_value: int = Form(default=1, gt=0), expire_style: str = Form(default='day')):
expired_at, expired_count, used_count, code = await get_expire_info(expire_value, expire_style)
await FileCodes.create(
code=code,
text=text,
expired_at=expired_at,
expired_count=expired_count,
used_count=used_count,
size=len(text),
prefix='文本分享'
)
return APIResponse(detail={
'code': code,
})


@share_api.post('/file/')
async def share_file(expire_value: int = Form(default=1, gt=0), expire_style: str = Form(default='day'), file: UploadFile = File(...)):
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)
await file_storage.save_file(file, save_path)
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,
)
return APIResponse(detail={
'code': code,
'name': file.filename,
})


@share_api.post('/select/')
async def select_file(data: SelectFileModel, ip: str = Depends(error_ip_limit)):
file_code = await FileCodes.filter(code=data.code).first()
if not file_code:
error_ip_limit.add_ip(ip)
return APIResponse(code=404, detail='文件不存在')
if await file_code.is_expired():
return APIResponse(code=403, detail='文件已过期')
file_code.used_count += 1
await file_code.save()
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),
})


@share_api.get('/download')
async def download_file(key: str, code: str, ip: str = Depends(error_ip_limit)):
is_valid = await file_storage.get_select_token(code) == key
if not is_valid:
error_ip_limit.add_ip(ip)
file_code = await FileCodes.filter(code=code).first()
if not file_code:
return APIResponse(code=404, detail='文件不存在')
if file_code.text:
return APIResponse(detail=file_code.text)
else:
file_path = file_storage.root_path / await file_code.get_file_path()
if not file_path.exists():
return APIResponse(code=404, detail='文件已过期删除')
return FileResponse(file_path, filename=file_code.prefix + file_code.suffix)
4 changes: 4 additions & 0 deletions core/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
# @Time : 2023/8/11 20:06
# @Author : Lan
# @File : __init__.py.py
# @Software: PyCharm
Empty file removed core/base_model.py
Empty file.
84 changes: 0 additions & 84 deletions core/database.py

This file was deleted.

15 changes: 15 additions & 0 deletions core/response.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
# @Time : 2023/8/14 11:48
# @Author : Lan
# @File : response.py
# @Software: PyCharm
from typing import Generic, TypeVar

from pydantic.v1.generics import GenericModel

T = TypeVar('T')


class APIResponse(GenericModel, Generic[T]):
code: int = 200
message: str = 'ok'
detail: T
Loading

0 comments on commit 642b652

Please sign in to comment.