182 lines
5.6 KiB
Python
182 lines
5.6 KiB
Python
import json
|
|
from functools import reduce
|
|
from pathlib import Path
|
|
from typing import TypedDict
|
|
|
|
import toml
|
|
|
|
import heurams.kernel.particles as pt
|
|
|
|
from heurams.kernel.auxiliary.lict import Lict
|
|
|
|
|
|
class RepoManifest(TypedDict):
|
|
title: str
|
|
author: str
|
|
package: str
|
|
desc: str
|
|
|
|
|
|
class Repo:
|
|
"""只维护仓库本身
|
|
上层 API 请访问此对象下粒子对象列表"""
|
|
|
|
file_mapping = {
|
|
"schedule": "schedule.toml",
|
|
"payload": "payload.toml",
|
|
"algodata": "algodata.json",
|
|
"manifest": "manifest.toml",
|
|
"typedef": "typedef.toml",
|
|
}
|
|
|
|
type_mapping = {
|
|
"schedule": "dict",
|
|
"payload": "lict",
|
|
"algodata": "lict",
|
|
"manifest": "dict",
|
|
"typedef": "dict",
|
|
}
|
|
|
|
default_save_list = ["algodata"]
|
|
|
|
def __init__(
|
|
self,
|
|
schedule: dict,
|
|
payload: Lict,
|
|
manifest: dict,
|
|
typedef: dict,
|
|
algodata: Lict,
|
|
source=None,
|
|
) -> None:
|
|
self.schedule: dict = schedule
|
|
self.manifest: RepoManifest = manifest # type: ignore
|
|
self.typedef: dict = typedef
|
|
self.payload: Lict = payload
|
|
self.algodata: Lict = algodata
|
|
self.source: Path | None = source # 若存在, 指向 repo 所在 dir
|
|
self.database = {
|
|
"schedule": self.schedule,
|
|
"payload": self.payload,
|
|
"manifest": self.manifest,
|
|
"typedef": self.typedef,
|
|
"algodata": self.algodata,
|
|
"source": self.source,
|
|
}
|
|
self._generate_particles_data()
|
|
|
|
def _generate_particles_data(self):
|
|
"""生成上层的粒子对象组和 API 交互, 会在 init 后自动调用"""
|
|
self.nucleonic_data_lict = Lict(
|
|
initlist=list(map(self._nucleonic_proc, self.payload))
|
|
)
|
|
self.orbitic_data = self.schedule
|
|
self.data_length = len(self.nucleonic_data_lict)
|
|
self.ident_index = self.nucleonic_data_lict.keys()
|
|
for i in self.ident_index:
|
|
self.algodata.append_new((i, {}))
|
|
self.electronic_data_lict = self.algodata
|
|
|
|
def _nucleonic_proc(self, unit):
|
|
ident = unit[0]
|
|
common = self.typedef["common"]
|
|
return (ident, (unit[1], common))
|
|
|
|
def __len__(self):
|
|
return len(self.payload)
|
|
|
|
def __repr__(self):
|
|
from pprint import pformat
|
|
|
|
s = pformat(self.database, indent=4)
|
|
return s
|
|
|
|
def persist_to_repodir(
|
|
self, save_list: list | None = None, source: Path | None = None
|
|
):
|
|
"""保存单元集数据到目录"""
|
|
if save_list == None:
|
|
save_list = self.default_save_list
|
|
if self.source != None and source == None:
|
|
source = self.source
|
|
if source == None:
|
|
raise FileNotFoundError("不存在仓库到文件的映射")
|
|
source.mkdir(parents=True, exist_ok=True)
|
|
for keyname in save_list:
|
|
filename = self.file_mapping[keyname]
|
|
with open(source / filename, "w") as f:
|
|
try:
|
|
dict_data = self.database[keyname].dicted_data
|
|
except:
|
|
dict_data = dict(self.database[keyname])
|
|
if filename.endswith("toml"):
|
|
toml.dump(dict_data, f)
|
|
elif filename.endswith("json"):
|
|
json.dump(dict_data, f, ensure_ascii=False, indent=4)
|
|
else:
|
|
raise ValueError(f"不支持的文件类型: {filename}")
|
|
|
|
def export_to_dict(self):
|
|
"""导出至单个字典"""
|
|
return self.database
|
|
|
|
@classmethod
|
|
def create_new_repo(cls, source=None):
|
|
"""创建新的空单元集"""
|
|
default_database = {
|
|
"schedule": {},
|
|
"payload": Lict([]),
|
|
"algodata": Lict([]),
|
|
"manifest": {},
|
|
"typedef": {},
|
|
"source": source,
|
|
}
|
|
return Repo(**default_database)
|
|
|
|
@classmethod
|
|
def from_repodir(cls, source: Path):
|
|
"""从目录创建单元集"""
|
|
database = {}
|
|
for keyname, filename in cls.file_mapping.items():
|
|
with open(source / filename, "r") as f:
|
|
loaded: dict
|
|
if filename.endswith("toml"):
|
|
loaded = toml.load(f)
|
|
elif filename.endswith("json"):
|
|
loaded = json.load(f)
|
|
else:
|
|
raise ValueError(f"不支持的文件类型: {filename}")
|
|
if cls.type_mapping[keyname] == "lict":
|
|
database[keyname] = Lict(list(loaded.items()))
|
|
elif cls.type_mapping[keyname] == "dict":
|
|
database[keyname] = loaded
|
|
else:
|
|
raise ValueError(f"不支持的数据容器: {cls.type_mapping[keyname]}")
|
|
database["source"] = source
|
|
return Repo(**database)
|
|
|
|
@classmethod
|
|
def from_dict(cls, dictdata, source: Path | None = None):
|
|
"""从单一字典创建单元集"""
|
|
database = dictdata
|
|
database["source"] = source
|
|
return Repo(**database)
|
|
|
|
@classmethod
|
|
def check_repodir(cls, source: Path):
|
|
"""检测单元集目录合法性"""
|
|
try:
|
|
cls.from_repodir(source)
|
|
return True
|
|
except:
|
|
return False
|
|
|
|
@classmethod
|
|
def probe_valid_repos_in_dir(cls, folder: Path):
|
|
"""返回一个合法的子目录 Path() 列表"""
|
|
lst = list()
|
|
for i in folder.iterdir():
|
|
if i.is_dir():
|
|
if cls.check_repodir(i):
|
|
lst.append(i)
|
|
return lst
|