| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247 |
- from typing import Any
- from app.api.v1.module_system.auth.schema import AuthSchema
- from app.core.base_schema import BatchSetAvailable
- from app.core.exceptions import CustomException
- from app.utils.common_util import (
- get_child_id_map,
- get_child_recursion,
- get_parent_id_map,
- get_parent_recursion,
- traversal_to_tree,
- )
- from .crud import MenuCRUD
- from .schema import (
- MenuCreateSchema,
- MenuOutSchema,
- MenuQueryParam,
- MenuUpdateSchema,
- )
- class MenuService:
- """
- 菜单模块服务层
- """
- @classmethod
- async def _validate_parent_child_type(
- cls, auth: AuthSchema, parent_id: int | None, child_type: int
- ) -> None:
- """
- 父子类型约束:目录下仅允许目录/菜单/外链;菜单下仅允许按钮;按钮与外链下不可挂子级。
- 无父级时仅允许目录、菜单、外链(与前端一致)。
- """
- if parent_id is None:
- if child_type not in (1, 2, 4):
- raise CustomException(msg="顶级菜单仅允许目录、菜单或外链类型")
- return
- parent = await MenuCRUD(auth).get_by_id_crud(id=parent_id)
- if not parent:
- raise CustomException(msg="父级菜单不存在")
- pt = parent.type
- if pt == 1:
- if child_type not in (1, 2, 4):
- raise CustomException(msg="目录下仅允许新增目录、菜单或外链")
- elif pt == 2:
- if child_type != 3:
- raise CustomException(msg="菜单下仅允许新增按钮")
- else:
- raise CustomException(msg="菜单或链接类型下不允许新增子菜单")
- @classmethod
- async def get_menu_detail_service(cls, auth: AuthSchema, id: int) -> dict:
- """
- 获取菜单详情。
- 参数:
- - auth (AuthSchema): 认证对象。
- - id (int): 菜单ID。
- 返回:
- - dict: 菜单详情对象。
- """
- menu = await MenuCRUD(auth).get_by_id_crud(id=id)
- # 创建实例后再设置parent_name属性
- menu_out = MenuOutSchema.model_validate(menu)
- if menu and menu.parent_id:
- parent = await MenuCRUD(auth).get_by_id_crud(id=menu.parent_id)
- if parent:
- menu_out.parent_name = parent.name
- return menu_out.model_dump()
- @classmethod
- async def get_menu_tree_service(
- cls,
- auth: AuthSchema,
- search: MenuQueryParam | None = None,
- order_by: list[dict] | None = None,
- ) -> list[dict]:
- """
- 获取菜单树形列表。
- 参数:
- - auth (AuthSchema): 认证对象。
- - search (MenuQueryParam | None): 查询参数对象。
- - order_by (list[dict] | None): 排序参数列表。
- 返回:
- - list[dict]: 菜单树形列表对象。
- """
- # 使用树形结构查询,预加载children关系
- menu_list = await MenuCRUD(auth).get_tree_list_crud(
- search=search.__dict__, order_by=order_by
- )
- # 转换为字典列表
- menu_dict_list = [MenuOutSchema.model_validate(menu).model_dump() for menu in menu_list]
- # 使用traversal_to_tree构建树形结构
- tree = traversal_to_tree(menu_dict_list)
- # 对外链菜单(type=4)转换路由数据
- MenuService._transform_external_menus(tree)
- return tree
- @staticmethod
- def _transform_external_menus(items: list[dict]) -> None:
- """递归转换外链菜单的路由和组件路径"""
- for item in items:
- if item.get("type") == 4:
- original_url = item.get("route_path", "")
- route_name = item.get("route_name") or "external"
- item["route_path"] = f"/{route_name}"
- item["component_path"] = "module_system/menu/ExternalLink"
- # 把原始 URL 存入 params,供 ExternalLink 组件读取
- item["params"] = [{"key": "url", "value": original_url}]
- if item.get("children"):
- MenuService._transform_external_menus(item["children"])
- @classmethod
- async def create_menu_service(cls, auth: AuthSchema, data: MenuCreateSchema) -> dict:
- """
- 创建菜单。
- 参数:
- - auth (AuthSchema): 认证对象。
- - data (MenuCreateSchema): 创建参数对象。
- 返回:
- - dict: 创建的菜单对象。
- """
- search: dict[str, Any] = {"title": data.title}
- if data.parent_id is not None:
- search["parent_id"] = data.parent_id
- menu = await MenuCRUD(auth).get(**search)
- if menu:
- raise CustomException(msg="创建失败,该菜单已存在")
- await cls._validate_parent_child_type(auth, data.parent_id, data.type)
- new_menu = await MenuCRUD(auth).create(data=data)
- new_menu_dict = MenuOutSchema.model_validate(new_menu).model_dump()
- return new_menu_dict
- @classmethod
- async def update_menu_service(cls, auth: AuthSchema, id: int, data: MenuUpdateSchema) -> dict:
- """
- 更新菜单。
- 参数:
- - auth (AuthSchema): 认证对象。
- - id (int): 菜单ID。
- - data (MenuUpdateSchema): 更新参数对象。
- 返回:
- - dict: 更新的菜单对象。
- """
- menu = await MenuCRUD(auth).get_by_id_crud(id=id)
- if not menu:
- raise CustomException(msg="更新失败,该菜单不存在")
- await cls._validate_parent_child_type(auth, data.parent_id, data.type)
- search: dict[str, Any] = {"title": data.title}
- if data.parent_id is not None:
- search["parent_id"] = data.parent_id
- exist_menu = await MenuCRUD(auth).get(**search)
- if exist_menu and exist_menu.id != id:
- raise CustomException(msg="更新失败,菜单标题重复")
- if data.parent_id:
- parent_menu = await MenuCRUD(auth).get_by_id_crud(id=data.parent_id)
- if not parent_menu:
- raise CustomException(msg="更新失败,父级菜单不存在")
- data.parent_name = parent_menu.title
- new_menu = await MenuCRUD(auth).update(id=id, data=data)
- await cls.set_menu_available_service(
- auth=auth, data=BatchSetAvailable(ids=[id], status=data.status)
- )
- new_menu_dict = MenuOutSchema.model_validate(new_menu).model_dump()
- return new_menu_dict
- @classmethod
- async def delete_menu_service(cls, auth: AuthSchema, ids: list[int]) -> None:
- """
- 删除菜单。
- 参数:
- - auth (AuthSchema): 认证对象。
- - ids (list[int]): 菜单ID列表。
- 返回:
- - None
- """
- if len(ids) < 1:
- raise CustomException(msg="删除失败,删除对象不能为空")
- # 获取所有菜单列表,用于构建树形关系
- all_menus = await MenuCRUD(auth).get_list_crud()
- # 构建子菜单ID映射
- child_id_map = get_child_id_map(model_list=all_menus)
- # 收集所有需要删除的菜单ID,包括直接指定的ID和它们的所有子菜单ID
- delete_ids_set = set()
- for id in ids:
- # 递归获取该ID的所有子菜单ID
- all_descendants = get_child_recursion(id=id, id_map=child_id_map)
- delete_ids_set.update(all_descendants)
- # 将集合转换为列表
- delete_ids = list(delete_ids_set)
- # 执行批量删除操作
- await MenuCRUD(auth).delete(ids=delete_ids)
- @classmethod
- async def set_menu_available_service(cls, auth: AuthSchema, data: BatchSetAvailable) -> None:
- """
- 递归获取所有父、子级菜单,然后批量修改菜单可用状态。
- 参数:
- - auth (AuthSchema): 认证对象。
- - data (BatchSetAvailable): 批量设置可用参数对象。
- 返回:
- - None
- """
- menu_list = await MenuCRUD(auth).get_list_crud()
- total_ids = []
- if data.status == "0":
- # 激活,则需要把所有父级菜单都激活
- id_map = get_parent_id_map(model_list=menu_list)
- for menu_id in data.ids:
- enable_ids = get_parent_recursion(id=menu_id, id_map=id_map)
- total_ids.extend(enable_ids)
- else:
- # 禁止,则需要把所有子级菜单都禁止
- id_map = get_child_id_map(model_list=menu_list)
- for menu_id in data.ids:
- disable_ids = get_child_recursion(id=menu_id, id_map=id_map)
- total_ids.extend(disable_ids)
- await MenuCRUD(auth).set_available_crud(ids=total_ids, status=data.status)
|