Files
2020-09-01 23:50:00 +08:00

174 lines
6.0 KiB
Python

import io
import xlsxwriter
from fastapi import APIRouter, Depends
from fastapi.responses import JSONResponse
from starlette.responses import StreamingResponse
from starlette.status import HTTP_409_CONFLICT
from tortoise import Model
from tortoise.contrib.pydantic import pydantic_model_creator
from tortoise.exceptions import IntegrityError
from tortoise.fields import ManyToManyRelation
from ..common import handle_m2m_fields_create_or_update
from ..depends import (
QueryItem,
admin_log_create,
admin_log_delete,
admin_log_update,
create_checker,
delete_checker,
get_model,
get_query,
parse_body,
read_checker,
update_checker,
)
from ..factory import app
from ..responses import GetManyOut
from ..schemas import BulkIn
from ..shortcuts import get_object_or_404
router = APIRouter()
@router.get("/{resource}/export")
async def export(resource: str, query: QueryItem = Depends(get_query), model=Depends(get_model)):
qs = model.all()
if query.where:
qs = qs.filter(**query.where)
resource = await app.get_resource(resource)
result = await qs
creator = pydantic_model_creator(
model, include=resource.resource_fields.keys(), exclude=model._meta.m2m_fields
)
data = map(lambda x: creator.from_orm(x).dict(), result)
output = io.BytesIO()
workbook = xlsxwriter.Workbook(output)
worksheet = workbook.add_worksheet()
for row, item in enumerate(data):
col = 0
for k, v in item.items():
if row == 0:
worksheet.write(row, col, k)
worksheet.write(row + 1, col, v)
col += 1
workbook.close()
output.seek(0)
return StreamingResponse(output)
@router.get("/{resource}", dependencies=[Depends(read_checker)])
async def get_resource(
resource: str, query: QueryItem = Depends(get_query), model=Depends(get_model)
):
menu = app.model_menu_mapping[resource]
qs = model.all()
if query.where:
qs = qs.filter(**query.where)
sort = query.sort
for k, v in sort.items():
if k in menu.sort_fields:
if v == -1:
qs = qs.order_by(f"-{k}")
elif v == 1:
qs = qs.order_by(k)
resource = await app.get_resource(resource)
result = await qs.limit(query.size).offset((query.page - 1) * query.size)
creator = pydantic_model_creator(
model, include=resource.resource_fields.keys(), exclude=model._meta.m2m_fields
)
data = []
for item in result:
item_dict = creator.from_orm(item).dict()
item_dict["_rowVariant"] = item_dict.pop("rowVariant", None)
item_dict["_cellVariants"] = item_dict.pop("cellVariants", None)
data.append(item_dict)
return GetManyOut(total=await qs.count(), data=data)
@router.get("/{resource}/form", dependencies=[Depends(read_checker)])
async def form(resource: str,):
resource = await app.get_resource(
resource, exclude_pk=True, exclude_m2m_field=False, exclude_actions=True
)
return resource.dict(by_alias=True, exclude_unset=True)
@router.get("/{resource}/grid", dependencies=[Depends(read_checker)])
async def grid(resource: str,):
resource = await app.get_resource(resource)
return resource.dict(by_alias=True, exclude_unset=True)
@router.get("/{resource}/view", dependencies=[Depends(read_checker)])
async def view(resource: str,):
resource = await app.get_resource(resource)
return resource.dict(by_alias=True, exclude_unset=True)
@router.post(
"/{resource}/bulk/delete", dependencies=[Depends(delete_checker), Depends(admin_log_delete)]
)
async def bulk_delete(bulk_in: BulkIn, model=Depends(get_model)):
await model.filter(pk__in=bulk_in.pk_list).delete()
return {"success": True}
@router.delete(
"/{resource}/{id}", dependencies=[Depends(delete_checker), Depends(admin_log_delete)]
)
async def delete_one(id: int, model=Depends(get_model)):
await model.filter(pk=id).delete()
return {"success": True}
@router.put("/{resource}/{id}", dependencies=[Depends(update_checker), Depends(admin_log_update)])
async def update_one(id: int, parsed=Depends(parse_body), model=Depends(get_model)):
body, resource_fields = parsed
m2m_fields = model._meta.m2m_fields
try:
obj = await handle_m2m_fields_create_or_update(
body, m2m_fields, model, app.user_model, False, id
)
except IntegrityError as e:
return JSONResponse(
status_code=HTTP_409_CONFLICT, content=dict(message=f"Update Error,{e}")
)
creator = pydantic_model_creator(model, include=resource_fields, exclude=m2m_fields)
return creator.from_orm(obj).dict()
@router.post("/{resource}", dependencies=[Depends(create_checker), Depends(admin_log_create)])
async def create_one(parsed=Depends(parse_body), model=Depends(get_model)):
body, resource_fields = parsed
m2m_fields = model._meta.m2m_fields
creator = pydantic_model_creator(model, include=resource_fields, exclude=m2m_fields)
try:
obj = await handle_m2m_fields_create_or_update(body, m2m_fields, model, app.user_model)
except IntegrityError as e:
return JSONResponse(
status_code=HTTP_409_CONFLICT, content=dict(message=f"Create Error,{e}")
)
return creator.from_orm(obj).dict()
@router.get("/{resource}/{id}", dependencies=[Depends(read_checker)])
async def get_one(id: int, resource: str, model=Depends(get_model)):
obj = await get_object_or_404(model, pk=id) # type:Model
m2m_fields = model._meta.m2m_fields
resource = await app.get_resource(resource, exclude_m2m_field=False)
include = resource.resource_fields.keys()
creator = pydantic_model_creator(model, include=include, exclude=m2m_fields)
ret = creator.from_orm(obj).dict()
for m2m_field in m2m_fields:
if m2m_field in include:
relate_model = getattr(obj, m2m_field) # type:ManyToManyRelation
ids = await relate_model.all().values_list(relate_model.remote_model._meta.pk_attr)
ret[m2m_field] = list(map(lambda x: x[0], ids))
ret["__str__"] = str(obj)
return ret