2024-03-13 20:29:38 +08:00
|
|
|
import os
|
|
|
|
from collections import OrderedDict
|
|
|
|
from collections.abc import Callable
|
2024-06-20 15:16:21 +08:00
|
|
|
from typing import Any
|
2024-03-13 20:29:38 +08:00
|
|
|
|
2024-05-24 12:08:12 +08:00
|
|
|
from core.tools.utils.yaml_utils import load_yaml_file
|
2024-03-13 20:29:38 +08:00
|
|
|
|
|
|
|
|
2024-06-20 15:16:21 +08:00
|
|
|
def get_position_map(folder_path: str, *, file_name: str = "_position.yaml") -> dict[str, int]:
|
2024-03-13 20:29:38 +08:00
|
|
|
"""
|
2024-08-15 23:51:00 +08:00
|
|
|
Get the mapping from name to index from a YAML file
|
2024-03-13 20:29:38 +08:00
|
|
|
:param folder_path:
|
|
|
|
:param file_name: the YAML file name, default to '_position.yaml'
|
|
|
|
:return: a dict with name as key and index as value
|
|
|
|
"""
|
2024-07-29 10:32:11 +08:00
|
|
|
position_file_path = os.path.join(folder_path, file_name)
|
|
|
|
yaml_content = load_yaml_file(file_path=position_file_path, default_value=[])
|
|
|
|
positions = [item.strip() for item in yaml_content if item and isinstance(item, str) and item.strip()]
|
|
|
|
return {name: index for index, name in enumerate(positions)}
|
2024-03-13 20:29:38 +08:00
|
|
|
|
|
|
|
|
|
|
|
def sort_by_position_map(
|
|
|
|
position_map: dict[str, int],
|
|
|
|
data: list[Any],
|
|
|
|
name_func: Callable[[Any], str],
|
|
|
|
) -> list[Any]:
|
|
|
|
"""
|
|
|
|
Sort the objects by the position map.
|
|
|
|
If the name of the object is not in the position map, it will be put at the end.
|
|
|
|
:param position_map: the map holding positions in the form of {name: index}
|
|
|
|
:param name_func: the function to get the name of the object
|
|
|
|
:param data: the data to be sorted
|
|
|
|
:return: the sorted objects
|
|
|
|
"""
|
|
|
|
if not position_map or not data:
|
|
|
|
return data
|
|
|
|
|
2024-08-15 23:51:00 +08:00
|
|
|
return sorted(data, key=lambda x: position_map.get(name_func(x), float('inf')))
|
2024-03-13 20:29:38 +08:00
|
|
|
|
|
|
|
|
|
|
|
def sort_to_dict_by_position_map(
|
|
|
|
position_map: dict[str, int],
|
|
|
|
data: list[Any],
|
|
|
|
name_func: Callable[[Any], str],
|
|
|
|
) -> OrderedDict[str, Any]:
|
|
|
|
"""
|
|
|
|
Sort the objects into a ordered dict by the position map.
|
|
|
|
If the name of the object is not in the position map, it will be put at the end.
|
|
|
|
:param position_map: the map holding positions in the form of {name: index}
|
|
|
|
:param name_func: the function to get the name of the object
|
|
|
|
:param data: the data to be sorted
|
|
|
|
:return: an OrderedDict with the sorted pairs of name and object
|
|
|
|
"""
|
|
|
|
sorted_items = sort_by_position_map(position_map, data, name_func)
|
|
|
|
return OrderedDict([(name_func(item), item) for item in sorted_items])
|