diff --git a/.gitignore b/.gitignore index ed8ebf5..fbe2907 100644 --- a/.gitignore +++ b/.gitignore @@ -1 +1,5 @@ -__pycache__ \ No newline at end of file +__pycache__ + +build +dist +thousandhands.spec \ No newline at end of file diff --git a/api/api.py b/api/api.py index 8b13789..e11d5d4 100644 --- a/api/api.py +++ b/api/api.py @@ -1 +1,13 @@ +# from flask import Flask + +# app = Flask(__name__, static_folder="./web/dist/", static_url_path="") + + +# @app.route("/") +# def main(): +# return app.send_static_file("index.html") + + +# def appRun(): +# pass \ No newline at end of file diff --git a/app.py b/app.py index ef0fa1c..1e43ecc 100644 --- a/app.py +++ b/app.py @@ -1,3 +1,10 @@ +''' +******************************************** +* @Date: 2024 10 22 +* @Description: flask app入口,在开发模式运行,在生产环境应该运行start.py +******************************************** +''' + from flask import Flask app = Flask(__name__, static_folder="./web/dist/", static_url_path="") @@ -9,4 +16,4 @@ def main(): if __name__ == "__main__": - app.run(debug=True) + app.run() diff --git a/core/objects/backup_object.py b/core/objects/backup_object.py index e322e05..fc05f10 100644 --- a/core/objects/backup_object.py +++ b/core/objects/backup_object.py @@ -6,12 +6,16 @@ """ import os +import asyncio from typing import List, Union from .tree_object import TreeObject from .blob_object import BlobObject +from utils.log import Logger + +logger = Logger("BackupObject") -class BackupObject(ObjectBase): +class BackupObject: """ 备份对象,负责管理存储对象(tree、blob)对象、备份信息、恢复等操作 """ @@ -21,19 +25,27 @@ class BackupObject(ObjectBase): backup_time_create: str backup_size: int backup_version_number: int - backup_tree: List[Union[TreeObject, BlobObject]] + backup_trees: List[Union[TreeObject, BlobObject]] = [] new_backup_flag: bool def __init__(self, backup_name: str, backup_base_path: str): self.backup_name = backup_name - backup_path = os.path.join(backup_base_path, backup_name) - if not os.path.exists(backup_path): + # 获取备份路径的绝对路径 + if os.path.isabs(backup_base_path): + self.backup_path = os.path.join(backup_base_path, backup_name) + else: + self.backup_path = os.path.join( + os.path.abspath(backup_base_path), backup_name + ) + + logger.debug(f"Backup path: {self.backup_path}") + + if not os.path.exists(self.backup_path): self.new_backup_flag = True else: self.new_backup_flag = False - pass #TODO 读取备份信息 - + pass # TODO 读取备份信息 def createNewBackup(self, backup_dirs: List[str]): """ @@ -43,12 +55,26 @@ class BackupObject(ObjectBase): """ for backup_dir in backup_dirs: if os.path.isdir(backup_dir): - self.backup_tree.append(TreeObject(backup_dir)) + self.backup_trees.append(TreeObject(backup_dir)) else: - self.backup_tree.append(BlobObject(backup_dir)) + self.backup_trees.append(BlobObject(backup_dir)) + + logger.info("New backup created successfully.") + logger.debug(f"Backup trees: {self.backup_trees}") def backup(self): - pass + asyncio.run(self.__writeBlobs()) + + async def __writeBlobs(self): + """ + 写入所有对象到备份路径 + """ + obj_save_path = os.path.join(self.backup_path, "objects") + for obj in self.backup_trees: + if isinstance(obj, TreeObject): + await obj.writeBlobs(obj_save_path) # 如果是TreeObject,则调用writeBlobs方法 + else: + await obj.writeBlob(obj_save_path) # 如果是BlobObject,则调用writeBlob方法 def recover(self, recover_path: str): """ diff --git a/core/objects/blob_object.py b/core/objects/blob_object.py index 7deb68e..c78442a 100644 --- a/core/objects/blob_object.py +++ b/core/objects/blob_object.py @@ -8,6 +8,7 @@ import hashlib import os import zlib +import aiofiles class BlobObject: @@ -16,45 +17,45 @@ class BlobObject: def __init__(self, file_path: str) -> None: self.file_path = file_path - self.object_id = self.contentSha1() + self.object_id = self.__contentSha1() - def writeBlob(self, base_path) -> None: - Folder = base_path + "/" + self.__getFolderName() + async def writeBlob(self, base_path: str) -> None: + Folder = base_path + "/" + self.object_id[:2] if not os.path.exists(Folder): os.makedirs(Folder) - self.__compressFile( + await self.__compressFile( self.file_path, - base_path + "/" + self.__getFolderName() + "/" + self.__getFileName(), + base_path + "/" + self.object_id[:2] + "/" + self.object_id[2:], ) - def __compressFile(self, file_path, save_path) -> None: - compresser = zlib.compressobj(9) - write_file = open(save_path, "wb") - with open(file_path, "rb") as f: - while True: - data = f.read(self.__buff_size) - if not data: - break - compressedData = compresser.compress(data) - write_file.write(compressedData) - write_file.write(compresser.flush()) - write_file.close() - - def __getFolderName(self) -> str: - return self.object_id[:2] - - def __getFileName(self) -> str: - return self.object_id[2:] - def getBlobAbsPath(self) -> str: """ 获取blob的绝对路径,此相对路径是基于存储路径的,不是相对当前文件的路径 :return 相对路径,格式为xx/xxxx... """ - return self.__getFolderName() + "/" + self.__getFileName() + return self.object_id[:2] + "/" + self.object_id[2:] - def contentSha1(self) -> str: + async def __compressFile(self, file_path: str, save_path: str) -> None: + """ + 压缩文件 + + :param file_path: 要压缩的文件路径 + :param save_path: 保存路径 + :return + """ + compresser = zlib.compressobj(9) + async with aiofiles.open(save_path, mode='wb') as wf, aiofiles.open(file_path, mode='rb') as rf: + while True: + data = await rf.read(self.__buff_size) + if not data: + break + compressedData = compresser.compress(data) + await wf.write(compressedData) + await wf.write(compresser.flush()) + + + def __contentSha1(self) -> str: """ 计算文件内容的sha1值 @@ -68,3 +69,7 @@ class BlobObject: break sha1.update(data) return sha1.hexdigest() + + +def newBlobObject(file_path: str) -> BlobObject: + return BlobObject(file_path) \ No newline at end of file diff --git a/core/objects/tree_object.py b/core/objects/tree_object.py index cc40aa4..a3b679c 100644 --- a/core/objects/tree_object.py +++ b/core/objects/tree_object.py @@ -8,7 +8,7 @@ from typing import Dict, List from .blob_object import BlobObject import os - +import asyncio class TreeObject: __children: List[BlobObject] = [] # 备份树节点列表,节点为BlobObject备份存储对象 @@ -26,6 +26,8 @@ class TreeObject: self.__children.append(blob) self.__file_map[blob_path] = blob.object_id - def writeBlobs(self, base_path: str): + async def writeBlobs(self, base_path: str): + tasks = [] for child in self.__children: - child.writeBlob(base_path) + tasks.append(asyncio.create_task(child.writeBlob(base_path))) + asyncio.gather(*tasks) diff --git a/core/plan/Plan.py b/core/plan/Plan.py index 9338e2c..a6a0c76 100644 --- a/core/plan/Plan.py +++ b/core/plan/Plan.py @@ -5,4 +5,7 @@ class Plan: pass def getPriorityFactor(self) -> float: + ''' + 获取优先级因子 + ''' return self.__priority_factor diff --git a/tests/core/objects/test_back_object.py b/core/plan/PlanBackup.py similarity index 100% rename from tests/core/objects/test_back_object.py rename to core/plan/PlanBackup.py diff --git a/core/target/targetBaidu.py b/core/target/targetBaidu.py new file mode 100644 index 0000000..e69de29 diff --git a/core/target/targetBase.py b/core/target/targetBase.py index c68f1ea..779714a 100644 --- a/core/target/targetBase.py +++ b/core/target/targetBase.py @@ -6,8 +6,7 @@ """ from abc import ABC, abstractmethod -from core.objects.back_object import BackObject - +from core.objects.backup_object import BackupObject class TargetBase(ABC): @abstractmethod @@ -21,5 +20,5 @@ class TargetBase(ABC): pass @abstractmethod - def back(self, back_object: BackObject): + def back(self, back_object: BackupObject): pass diff --git a/core/target/targetFTP.py b/core/target/targetFTP.py new file mode 100644 index 0000000..adad678 --- /dev/null +++ b/core/target/targetFTP.py @@ -0,0 +1,5 @@ +# from .targetBase import TargetBase + + +# class TargetFTP(TargetBase): +# pass \ No newline at end of file diff --git a/core/target/targetNFS.py b/core/target/targetNFS.py new file mode 100644 index 0000000..e69de29 diff --git a/core/timer/timer.py b/core/timer/timer.py new file mode 100644 index 0000000..4a6e6d9 --- /dev/null +++ b/core/timer/timer.py @@ -0,0 +1,11 @@ +from typing import Callable + + +class Timer: + def __init__(self, name: str, interval: float, callback: Callable): + self.name = name + self.interval = interval + self.callback = callback + + def start(self): + pass diff --git a/pyinstaller.py b/pyinstaller.py new file mode 100644 index 0000000..1095d3b --- /dev/null +++ b/pyinstaller.py @@ -0,0 +1,14 @@ +import PyInstaller.__main__ + +if __name__ == "__main__": + PyInstaller.__main__.run([ + '--add-data', + 'web/dist/:./web/dist', + '--hidden-import=gunicorn.glogging', + '--hidden-import=gunicorn.workers.sync', + '-y', + '--clean', + '--name', + 'thousandhands', + 'start.py' + ]) \ No newline at end of file diff --git a/pyproject.toml b/pyproject.toml index 0634ffb..71b53f9 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,11 +1,14 @@ [project] name = "thousandhands" version = "0.1.0" -description = "Add your description here" +description = "千手备份" readme = "README.md" requires-python = ">=3.12" dependencies = [ + "aiofiles>=24.1.0", "flask>=3.0.3", + "gunicorn>=23.0.0", + "pyinstaller>=6.11", "pytest>=8.3.3", ] diff --git a/start.py b/start.py new file mode 100644 index 0000000..24a458a --- /dev/null +++ b/start.py @@ -0,0 +1,24 @@ +''' +******************************************** +* @Date: 2024 10 22 +* @Description: gunicorn启动文件,用于在生产环境下启动flask应用 +******************************************** +''' + +from gunicorn.app.base import BaseApplication +from gunicorn.config import Config + +class Application(BaseApplication): + def load_config(self): + if isinstance(self.cfg, Config): + self.cfg.set('bind', "0.0.0.0:8080") + self.cfg.set('workers', 1) + self.cfg.set('timeout', 30) + self.cfg.set('access_log_format', '%(t)s %(h)s "%(r)s" %(s)s %(b)s %(D)s "%(a)s"') + + def load(self): + from app import app + return app + +if __name__ == '__main__': + Application().run() \ No newline at end of file diff --git a/test.py b/test.py index 5681f79..ed2ac31 100644 --- a/test.py +++ b/test.py @@ -1,4 +1,4 @@ import pytest if __name__ == "__main__": - pytest.main(["tests/", "-v"]) + pytest.main(["tests/", "-v", "-s"]) \ No newline at end of file diff --git a/tests/core/objects/test_backup_object.py b/tests/core/objects/test_backup_object.py new file mode 100644 index 0000000..0395baf --- /dev/null +++ b/tests/core/objects/test_backup_object.py @@ -0,0 +1,32 @@ + +from core.objects.backup_object import BackupObject +import tempfile +import os + + +class TestBackup(): + def createTempDataDir(self): + temp_data_dir = tempfile.TemporaryDirectory() + for i in range(100): + f = open(temp_data_dir.name + "/testFile" + str(i), "w") + f.write("test data"+str(i)) + f.close() + os.mkdir(temp_data_dir.name + "/testDir") + for i in range(100): + f = open(temp_data_dir.name + "/testDir/testFile" + str(i), "w") + f.write("test data1"*10000+str(i)) + f.close() + return temp_data_dir + + def test_BackupWriteBlob(self): + ''' + 测试备份写入blob + ''' + with tempfile.TemporaryDirectory() as tmpdir: + temp_data_dir = self.createTempDataDir() + b = BackupObject("testBackup", tmpdir) + b.createNewBackup([temp_data_dir.name]) + b.backup() + temp_data_dir.cleanup() + assert os.path.exists(b.backup_path+"/objects") + diff --git a/tests/core/objects/test_blob_object.py b/tests/core/objects/test_blob_object.py index 61ee7a9..773afe4 100644 --- a/tests/core/objects/test_blob_object.py +++ b/tests/core/objects/test_blob_object.py @@ -1,3 +1,4 @@ +import asyncio from core.objects.blob_object import BlobObject import hashlib import tempfile @@ -11,7 +12,8 @@ class Test_Blob: newBlob.object_id == hashlib.sha1(open("README.md", "rb").read()).hexdigest() ) + def test_writeBlob(self): newBlob = BlobObject("README.md") with tempfile.TemporaryDirectory() as tmpdirname: - newBlob.writeBlob(tmpdirname) \ No newline at end of file + asyncio.run(newBlob.writeBlob(tmpdirname)) \ No newline at end of file diff --git a/utils/log.py b/utils/log.py index 4392010..ac49bf2 100644 --- a/utils/log.py +++ b/utils/log.py @@ -2,5 +2,30 @@ import logging class Logger: + __instance = None + + def __new__(cls, *args, **kwargs): + if cls.__instance is None: + cls.__instance = object.__new__(cls) + return cls.__instance + def __init__(self, name): self.logger = logging.getLogger(name) + logging.basicConfig( + level=logging.DEBUG, + format="%(asctime)s %(message)s", + datefmt="%Y-%m-%d %H:%M:%S", + ) + print(f"Logger initialized for {name}") + + def debug(self, msg): + self.logger.debug(msg) + + def info(self, msg): + self.logger.info(msg) + + def warning(self, msg): + self.logger.warning(msg) + + def error(self, msg): + self.logger.error(msg) diff --git a/uv.lock b/uv.lock index 6a74674..3547846 100644 --- a/uv.lock +++ b/uv.lock @@ -2,12 +2,30 @@ version = 1 requires-python = ">=3.12" [[package]] -name = "blinker" -version = "1.8.2" +name = "aiofiles" +version = "24.1.0" source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } -sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/1e/57/a6a1721eff09598fb01f3c7cda070c1b6a0f12d63c83236edf79a440abcc/blinker-1.8.2.tar.gz", hash = "sha256:8f77b09d3bf7c795e969e9486f39c2c5e9c39d4ee07424be2bc594ece9642d83", size = 23161 } +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/0b/03/a88171e277e8caa88a4c77808c20ebb04ba74cc4681bf1e9416c862de237/aiofiles-24.1.0.tar.gz", hash = "sha256:22a075c9e5a3810f0c2e48f3008c94d68c65d763b9b03857924c99e57355166c", size = 30247 } wheels = [ - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/bb/2a/10164ed1f31196a2f7f3799368a821765c62851ead0e630ab52b8e14b4d0/blinker-1.8.2-py3-none-any.whl", hash = "sha256:1779309f71bf239144b9399d06ae925637cf6634cf6bd131104184531bf67c01", size = 9456 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/a5/45/30bb92d442636f570cb5651bc661f52b610e2eec3f891a5dc3a4c3667db0/aiofiles-24.1.0-py3-none-any.whl", hash = "sha256:b4ec55f4195e3eb5d7abd1bf7e061763e864dd4954231fb8539a0ef8bb8260e5", size = 15896 }, +] + +[[package]] +name = "altgraph" +version = "0.17.4" +source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/de/a8/7145824cf0b9e3c28046520480f207df47e927df83aa9555fb47f8505922/altgraph-0.17.4.tar.gz", hash = "sha256:1b5afbb98f6c4dcadb2e2ae6ab9fa994bbb8c1d75f4fa96d340f9437ae454406" } +wheels = [ + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/4d/3f/3bc3f1d83f6e4a7fcb834d3720544ca597590425be5ba9db032b2bf322a2/altgraph-0.17.4-py2.py3-none-any.whl", hash = "sha256:642743b4750de17e655e6711601b077bc6598dbfa3ba5fa2b2a35ce12b508dff" }, +] + +[[package]] +name = "blinker" +version = "1.9.0" +source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/21/28/9b3f50ce0e048515135495f198351908d99540d69bfdc8c1d15b73dc55ce/blinker-1.9.0.tar.gz", hash = "sha256:b4ce2265a7abece45e7cc896e98dbebe6cead56bcf805a3d23136d145f5445bf", size = 22460 } +wheels = [ + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/10/cb/f2ad4230dc2eb1a74edf38f1a38b9b52277f75bef262d8908e60d957e13c/blinker-1.9.0-py3-none-any.whl", hash = "sha256:ba0efaa9080b619ff2f3459d1d500c57bddea4a6b424b60a91141db6fd2f08bc", size = 8458 }, ] [[package]] @@ -47,6 +65,18 @@ wheels = [ { url = "https://pypi.tuna.tsinghua.edu.cn/packages/61/80/ffe1da13ad9300f87c93af113edd0638c75138c42a0994becfacac078c06/flask-3.0.3-py3-none-any.whl", hash = "sha256:34e815dfaa43340d1d15a5c3a02b8476004037eb4840b34910c6e21679d288f3", size = 101735 }, ] +[[package]] +name = "gunicorn" +version = "23.0.0" +source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } +dependencies = [ + { name = "packaging" }, +] +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/34/72/9614c465dc206155d93eff0ca20d42e1e35afc533971379482de953521a4/gunicorn-23.0.0.tar.gz", hash = "sha256:f014447a0101dc57e294f6c18ca6b40227a4c90e9bdb586042628030cba004ec", size = 375031 } +wheels = [ + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/cb/7d/6dac2a6e1eba33ee43f318edbed4ff29151a49b5d37f080aad1e6469bca4/gunicorn-23.0.0-py3-none-any.whl", hash = "sha256:ec400d38950de4dfd418cff8328b2c8faed0edb0d517d3394e457c317908ca4d", size = 85029 }, +] + [[package]] name = "iniconfig" version = "2.0.0" @@ -78,30 +108,71 @@ wheels = [ ] [[package]] -name = "markupsafe" -version = "2.1.5" +name = "macholib" +version = "1.16.3" source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } -sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/87/5b/aae44c6655f3801e81aa3eef09dbbf012431987ba564d7231722f68df02d/MarkupSafe-2.1.5.tar.gz", hash = "sha256:d283d37a890ba4c1ae73ffadf8046435c76e7bc2247bbb63c00bd1a709c6544b", size = 19384 } +dependencies = [ + { name = "altgraph" }, +] +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/95/ee/af1a3842bdd5902ce133bd246eb7ffd4375c38642aeb5dc0ae3a0329dfa2/macholib-1.16.3.tar.gz", hash = "sha256:07ae9e15e8e4cd9a788013d81f5908b3609aa76f9b1421bae9c4d7606ec86a30" } wheels = [ - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/53/bd/583bf3e4c8d6a321938c13f49d44024dbe5ed63e0a7ba127e454a66da974/MarkupSafe-2.1.5-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:8dec4936e9c3100156f8a2dc89c4b88d5c435175ff03413b443469c7c8c5f4d1", size = 18215 }, - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/48/d6/e7cd795fc710292c3af3a06d80868ce4b02bfbbf370b7cee11d282815a2a/MarkupSafe-2.1.5-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:3c6b973f22eb18a789b1460b4b91bf04ae3f0c4234a0a6aa6b0a92f6f7b951d4", size = 14069 }, - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/51/b5/5d8ec796e2a08fc814a2c7d2584b55f889a55cf17dd1a90f2beb70744e5c/MarkupSafe-2.1.5-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ac07bad82163452a6884fe8fa0963fb98c2346ba78d779ec06bd7a6262132aee", size = 29452 }, - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/0a/0d/2454f072fae3b5a137c119abf15465d1771319dfe9e4acbb31722a0fff91/MarkupSafe-2.1.5-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f5dfb42c4604dddc8e4305050aa6deb084540643ed5804d7455b5df8fe16f5e5", size = 28462 }, - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/2d/75/fd6cb2e68780f72d47e6671840ca517bda5ef663d30ada7616b0462ad1e3/MarkupSafe-2.1.5-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ea3d8a3d18833cf4304cd2fc9cbb1efe188ca9b5efef2bdac7adc20594a0e46b", size = 27869 }, - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/b0/81/147c477391c2750e8fc7705829f7351cf1cd3be64406edcf900dc633feb2/MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:d050b3361367a06d752db6ead6e7edeb0009be66bc3bae0ee9d97fb326badc2a", size = 33906 }, - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/8b/ff/9a52b71839d7a256b563e85d11050e307121000dcebc97df120176b3ad93/MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:bec0a414d016ac1a18862a519e54b2fd0fc8bbfd6890376898a6c0891dd82e9f", size = 32296 }, - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/88/07/2dc76aa51b481eb96a4c3198894f38b480490e834479611a4053fbf08623/MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:58c98fee265677f63a4385256a6d7683ab1832f3ddd1e66fe948d5880c21a169", size = 33038 }, - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/96/0c/620c1fb3661858c0e37eb3cbffd8c6f732a67cd97296f725789679801b31/MarkupSafe-2.1.5-cp312-cp312-win32.whl", hash = "sha256:8590b4ae07a35970728874632fed7bd57b26b0102df2d2b233b6d9d82f6c62ad", size = 16572 }, - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/3f/14/c3554d512d5f9100a95e737502f4a2323a1959f6d0d01e0d0997b35f7b10/MarkupSafe-2.1.5-cp312-cp312-win_amd64.whl", hash = "sha256:823b65d8706e32ad2df51ed89496147a42a2a6e01c13cfb6ffb8b1e92bc910bb", size = 17127 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/d1/5d/c059c180c84f7962db0aeae7c3b9303ed1d73d76f2bfbc32bc231c8be314/macholib-1.16.3-py2.py3-none-any.whl", hash = "sha256:0e315d7583d38b8c77e815b1ecbdbf504a8258d8b3e17b61165c6feb60d18f2c" }, +] + +[[package]] +name = "markupsafe" +version = "3.0.2" +source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/b2/97/5d42485e71dfc078108a86d6de8fa46db44a1a9295e89c5d6d4a06e23a62/markupsafe-3.0.2.tar.gz", hash = "sha256:ee55d3edf80167e48ea11a923c7386f4669df67d7994554387f84e7d8b0a2bf0", size = 20537 } +wheels = [ + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/22/09/d1f21434c97fc42f09d290cbb6350d44eb12f09cc62c9476effdb33a18aa/MarkupSafe-3.0.2-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:9778bd8ab0a994ebf6f84c2b949e65736d5575320a17ae8984a77fab08db94cf", size = 14274 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/6b/b0/18f76bba336fa5aecf79d45dcd6c806c280ec44538b3c13671d49099fdd0/MarkupSafe-3.0.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:846ade7b71e3536c4e56b386c2a47adf5741d2d8b94ec9dc3e92e5e1ee1e2225", size = 12348 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/e0/25/dd5c0f6ac1311e9b40f4af06c78efde0f3b5cbf02502f8ef9501294c425b/MarkupSafe-3.0.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1c99d261bd2d5f6b59325c92c73df481e05e57f19837bdca8413b9eac4bd8028", size = 24149 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/f3/f0/89e7aadfb3749d0f52234a0c8c7867877876e0a20b60e2188e9850794c17/MarkupSafe-3.0.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e17c96c14e19278594aa4841ec148115f9c7615a47382ecb6b82bd8fea3ab0c8", size = 23118 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/d5/da/f2eeb64c723f5e3777bc081da884b414671982008c47dcc1873d81f625b6/MarkupSafe-3.0.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:88416bd1e65dcea10bc7569faacb2c20ce071dd1f87539ca2ab364bf6231393c", size = 22993 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/da/0e/1f32af846df486dce7c227fe0f2398dc7e2e51d4a370508281f3c1c5cddc/MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:2181e67807fc2fa785d0592dc2d6206c019b9502410671cc905d132a92866557", size = 24178 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/c4/f6/bb3ca0532de8086cbff5f06d137064c8410d10779c4c127e0e47d17c0b71/MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:52305740fe773d09cffb16f8ed0427942901f00adedac82ec8b67752f58a1b22", size = 23319 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/a2/82/8be4c96ffee03c5b4a034e60a31294daf481e12c7c43ab8e34a1453ee48b/MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:ad10d3ded218f1039f11a75f8091880239651b52e9bb592ca27de44eed242a48", size = 23352 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/51/ae/97827349d3fcffee7e184bdf7f41cd6b88d9919c80f0263ba7acd1bbcb18/MarkupSafe-3.0.2-cp312-cp312-win32.whl", hash = "sha256:0f4ca02bea9a23221c0182836703cbf8930c5e9454bacce27e767509fa286a30", size = 15097 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/c1/80/a61f99dc3a936413c3ee4e1eecac96c0da5ed07ad56fd975f1a9da5bc630/MarkupSafe-3.0.2-cp312-cp312-win_amd64.whl", hash = "sha256:8e06879fc22a25ca47312fbe7c8264eb0b662f6db27cb2d3bbbc74b1df4b9b87", size = 15601 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/83/0e/67eb10a7ecc77a0c2bbe2b0235765b98d164d81600746914bebada795e97/MarkupSafe-3.0.2-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:ba9527cdd4c926ed0760bc301f6728ef34d841f405abf9d4f959c478421e4efd", size = 14274 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/2b/6d/9409f3684d3335375d04e5f05744dfe7e9f120062c9857df4ab490a1031a/MarkupSafe-3.0.2-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:f8b3d067f2e40fe93e1ccdd6b2e1d16c43140e76f02fb1319a05cf2b79d99430", size = 12352 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/d2/f5/6eadfcd3885ea85fe2a7c128315cc1bb7241e1987443d78c8fe712d03091/MarkupSafe-3.0.2-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:569511d3b58c8791ab4c2e1285575265991e6d8f8700c7be0e88f86cb0672094", size = 24122 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/0c/91/96cf928db8236f1bfab6ce15ad070dfdd02ed88261c2afafd4b43575e9e9/MarkupSafe-3.0.2-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:15ab75ef81add55874e7ab7055e9c397312385bd9ced94920f2802310c930396", size = 23085 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/c2/cf/c9d56af24d56ea04daae7ac0940232d31d5a8354f2b457c6d856b2057d69/MarkupSafe-3.0.2-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f3818cb119498c0678015754eba762e0d61e5b52d34c8b13d770f0719f7b1d79", size = 22978 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/2a/9f/8619835cd6a711d6272d62abb78c033bda638fdc54c4e7f4272cf1c0962b/MarkupSafe-3.0.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:cdb82a876c47801bb54a690c5ae105a46b392ac6099881cdfb9f6e95e4014c6a", size = 24208 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/f9/bf/176950a1792b2cd2102b8ffeb5133e1ed984547b75db47c25a67d3359f77/MarkupSafe-3.0.2-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:cabc348d87e913db6ab4aa100f01b08f481097838bdddf7c7a84b7575b7309ca", size = 23357 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/ce/4f/9a02c1d335caabe5c4efb90e1b6e8ee944aa245c1aaaab8e8a618987d816/MarkupSafe-3.0.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:444dcda765c8a838eaae23112db52f1efaf750daddb2d9ca300bcae1039adc5c", size = 23344 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/ee/55/c271b57db36f748f0e04a759ace9f8f759ccf22b4960c270c78a394f58be/MarkupSafe-3.0.2-cp313-cp313-win32.whl", hash = "sha256:bcf3e58998965654fdaff38e58584d8937aa3096ab5354d493c77d1fdd66d7a1", size = 15101 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/29/88/07df22d2dd4df40aba9f3e402e6dc1b8ee86297dddbad4872bd5e7b0094f/MarkupSafe-3.0.2-cp313-cp313-win_amd64.whl", hash = "sha256:e6a2a455bd412959b57a172ce6328d2dd1f01cb2135efda2e4576e8a23fa3b0f", size = 15603 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/62/6a/8b89d24db2d32d433dffcd6a8779159da109842434f1dd2f6e71f32f738c/MarkupSafe-3.0.2-cp313-cp313t-macosx_10_13_universal2.whl", hash = "sha256:b5a6b3ada725cea8a5e634536b1b01c30bcdcd7f9c6fff4151548d5bf6b3a36c", size = 14510 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/7a/06/a10f955f70a2e5a9bf78d11a161029d278eeacbd35ef806c3fd17b13060d/MarkupSafe-3.0.2-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:a904af0a6162c73e3edcb969eeeb53a63ceeb5d8cf642fade7d39e7963a22ddb", size = 12486 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/34/cf/65d4a571869a1a9078198ca28f39fba5fbb910f952f9dbc5220afff9f5e6/MarkupSafe-3.0.2-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4aa4e5faecf353ed117801a068ebab7b7e09ffb6e1d5e412dc852e0da018126c", size = 25480 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/0c/e3/90e9651924c430b885468b56b3d597cabf6d72be4b24a0acd1fa0e12af67/MarkupSafe-3.0.2-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c0ef13eaeee5b615fb07c9a7dadb38eac06a0608b41570d8ade51c56539e509d", size = 23914 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/66/8c/6c7cf61f95d63bb866db39085150df1f2a5bd3335298f14a66b48e92659c/MarkupSafe-3.0.2-cp313-cp313t-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d16a81a06776313e817c951135cf7340a3e91e8c1ff2fac444cfd75fffa04afe", size = 23796 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/bb/35/cbe9238ec3f47ac9a7c8b3df7a808e7cb50fe149dc7039f5f454b3fba218/MarkupSafe-3.0.2-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:6381026f158fdb7c72a168278597a5e3a5222e83ea18f543112b2662a9b699c5", size = 25473 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/e6/32/7621a4382488aa283cc05e8984a9c219abad3bca087be9ec77e89939ded9/MarkupSafe-3.0.2-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:3d79d162e7be8f996986c064d1c7c817f6df3a77fe3d6859f6f9e7be4b8c213a", size = 24114 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/0d/80/0985960e4b89922cb5a0bac0ed39c5b96cbc1a536a99f30e8c220a996ed9/MarkupSafe-3.0.2-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:131a3c7689c85f5ad20f9f6fb1b866f402c445b220c19fe4308c0b147ccd2ad9", size = 24098 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/82/78/fedb03c7d5380df2427038ec8d973587e90561b2d90cd472ce9254cf348b/MarkupSafe-3.0.2-cp313-cp313t-win32.whl", hash = "sha256:ba8062ed2cf21c07a9e295d5b8a2a5ce678b913b45fdf68c32d95d6c1291e0b6", size = 15208 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/4f/65/6079a46068dfceaeabb5dcad6d674f5f5c61a6fa5673746f42a9f4c233b3/MarkupSafe-3.0.2-cp313-cp313t-win_amd64.whl", hash = "sha256:e444a31f8db13eb18ada366ab3cf45fd4b31e4db1236a4448f68778c1d1a5a2f", size = 15739 }, ] [[package]] name = "packaging" -version = "24.1" +version = "24.2" source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } -sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/51/65/50db4dda066951078f0a96cf12f4b9ada6e4b811516bf0262c0f4f7064d4/packaging-24.1.tar.gz", hash = "sha256:026ed72c8ed3fcce5bf8950572258698927fd1dbda10a5e981cdf0ac37f4f002", size = 148788 } +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/d0/63/68dbb6eb2de9cb10ee4c9c14a0148804425e13c4fb20d61cce69f53106da/packaging-24.2.tar.gz", hash = "sha256:c228a6dc5e932d346bc5739379109d49e8853dd8223571c7c5b55260edc0b97f", size = 163950 } wheels = [ - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/08/aa/cc0199a5f0ad350994d660967a8efb233fe0416e4639146c089643407ce6/packaging-24.1-py3-none-any.whl", hash = "sha256:5b8f2217dbdbd2f7f384c41c628544e6d52f2d0f53c6d0c3ea61aa5d1d7ff124", size = 53985 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/88/ef/eb23f262cca3c0c4eb7ab1933c3b1f03d021f2c48f54763065b6f0e321be/packaging-24.2-py3-none-any.whl", hash = "sha256:09abb1bccd265c01f4a3aa3f7a7db064b36514d2cba19a2f694fe6150451a759", size = 65451 }, +] + +[[package]] +name = "pefile" +version = "2023.2.7" +source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/78/c5/3b3c62223f72e2360737fd2a57c30e5b2adecd85e70276879609a7403334/pefile-2023.2.7.tar.gz", hash = "sha256:82e6114004b3d6911c77c3953e3838654b04511b8b66e8583db70c65998017dc", size = 74854 } +wheels = [ + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/55/26/d0ad8b448476d0a1e8d3ea5622dc77b916db84c6aa3cb1e1c0965af948fc/pefile-2023.2.7-py3-none-any.whl", hash = "sha256:da185cd2af68c08a6cd4481f7325ed600a88f6a813bad9dea07ab3ef73d8d8d6", size = 71791 }, ] [[package]] @@ -113,6 +184,47 @@ wheels = [ { url = "https://pypi.tuna.tsinghua.edu.cn/packages/88/5f/e351af9a41f866ac3f1fac4ca0613908d9a41741cfcf2228f4ad853b697d/pluggy-1.5.0-py3-none-any.whl", hash = "sha256:44e1ad92c8ca002de6377e165f3e0f1be63266ab4d554740532335b9d75ea669", size = 20556 }, ] +[[package]] +name = "pyinstaller" +version = "6.11.1" +source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } +dependencies = [ + { name = "altgraph" }, + { name = "macholib", marker = "sys_platform == 'darwin'" }, + { name = "packaging" }, + { name = "pefile", marker = "sys_platform == 'win32'" }, + { name = "pyinstaller-hooks-contrib" }, + { name = "pywin32-ctypes", marker = "sys_platform == 'win32'" }, + { name = "setuptools" }, +] +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/55/d4/54f5f5c73b803e6256ea97ffc6ba8a305d9a5f57f85f9b00b282512bf18a/pyinstaller-6.11.1.tar.gz", hash = "sha256:491dfb4d9d5d1d9650d9507daec1ff6829527a254d8e396badd60a0affcb72ef", size = 4249772 } +wheels = [ + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/96/15/b0f1c0985ee32fcd2f6ad9a486ef94e4db3fef9af025a3655e76cb708009/pyinstaller-6.11.1-py3-none-macosx_10_13_universal2.whl", hash = "sha256:44e36172de326af6d4e7663b12f71dbd34e2e3e02233e181e457394423daaf03", size = 991780 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/fd/0f/9f54cb18abe2b1d89051bc9214c0cb40d7b5f4049c151c315dacc067f4a2/pyinstaller-6.11.1-py3-none-manylinux2014_aarch64.whl", hash = "sha256:6d12c45a29add78039066a53fb05967afaa09a672426072b13816fe7676abfc4", size = 711739 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/32/f7/79d10830780eff8339bfa793eece1df4b2459e35a712fc81983e8536cc29/pyinstaller-6.11.1-py3-none-manylinux2014_i686.whl", hash = "sha256:ddc0fddd75f07f7e423da1f0822e389a42af011f9589e0269b87e0d89aa48c1f", size = 714053 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/25/f7/9961ef02cdbd2dbb1b1a215292656bd0ea72a83aafd8fb6373513849711e/pyinstaller-6.11.1-py3-none-manylinux2014_ppc64le.whl", hash = "sha256:0d6475559c4939f0735122989611d7f739ed3bf02f666ce31022928f7a7e4fda", size = 719133 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/6f/4d/7f854842a1ce798de762a0b0bc5d5a4fc26ad06164a98575dc3c54abed1f/pyinstaller-6.11.1-py3-none-manylinux2014_s390x.whl", hash = "sha256:e21c7806e34f40181e7606926a14579f848bfb1dc52cbca7eea66eccccbfe977", size = 709591 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/7f/e0/00d29fc90c3ba50620c61554e26ebb4d764569507be7cd1c8794aa696f9a/pyinstaller-6.11.1-py3-none-manylinux2014_x86_64.whl", hash = "sha256:32c742a24fe65d0702958fadf4040f76de85859c26bec0008766e5dbabc5b68f", size = 710068 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/3e/57/d14b44a69f068d2caaee49d15e45f9fa0f37c6a2d2ad778c953c1722a1ca/pyinstaller-6.11.1-py3-none-musllinux_1_1_aarch64.whl", hash = "sha256:208c0ef6dab0837a0a273ea32d1a3619a208e3d1fe3fec3785eea71a77fd00ce", size = 714439 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/88/01/256824bb57ca208099c86c2fb289f888ca7732580e91ced48fa14e5903b2/pyinstaller-6.11.1-py3-none-musllinux_1_1_x86_64.whl", hash = "sha256:ad84abf465bcda363c1d54eafa76745d77b6a8a713778348377dc98d12a452f7", size = 710457 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/7c/f0/98c9138f5f0ff17462f1ad6d712dcfa643b9a283d6238d464d8145bc139d/pyinstaller-6.11.1-py3-none-win32.whl", hash = "sha256:2e8365276c5131c9bef98e358fbc305e4022db8bedc9df479629d6414021956a", size = 1280261 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/7d/08/f43080614b3e8bce481d4dfd580e579497c7dcdaf87656d9d2ad912e5796/pyinstaller-6.11.1-py3-none-win_amd64.whl", hash = "sha256:7ac83c0dc0e04357dab98c487e74ad2adb30e7eb186b58157a8faf46f1fa796f", size = 1340482 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/ed/56/953c6594cb66e249563854c9cc04ac5a055c6c99d1614298feeaeaa9b87e/pyinstaller-6.11.1-py3-none-win_arm64.whl", hash = "sha256:35e6b8077d240600bb309ed68bb0b1453fd2b7ab740b66d000db7abae6244423", size = 1267519 }, +] + +[[package]] +name = "pyinstaller-hooks-contrib" +version = "2024.10" +source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } +dependencies = [ + { name = "packaging" }, + { name = "setuptools" }, +] +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/73/6a/9d0057e312b85fbd17a79e1c1955d115fd9bbc78b85bab757777c8ef2307/pyinstaller_hooks_contrib-2024.10.tar.gz", hash = "sha256:8a46655e5c5b0186b5e527399118a9b342f10513eb1425c483fa4f6d02e8800c", size = 140592 } +wheels = [ + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/a9/64/445861ee7a5fd32874c0f6cfe8222aacc8feda22539332e0d8ff50dadec6/pyinstaller_hooks_contrib-2024.10-py3-none-any.whl", hash = "sha256:ad47db0e153683b4151e10d231cb91f2d93c85079e78d76d9e0f57ac6c8a5e10", size = 338417 }, +] + [[package]] name = "pytest" version = "8.3.3" @@ -128,29 +240,53 @@ wheels = [ { url = "https://pypi.tuna.tsinghua.edu.cn/packages/6b/77/7440a06a8ead44c7757a64362dd22df5760f9b12dc5f11b6188cd2fc27a0/pytest-8.3.3-py3-none-any.whl", hash = "sha256:a6853c7375b2663155079443d2e45de913a911a11d669df02a50814944db57b2", size = 342341 }, ] +[[package]] +name = "pywin32-ctypes" +version = "0.2.3" +source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/85/9f/01a1a99704853cb63f253eea009390c88e7131c67e66a0a02099a8c917cb/pywin32-ctypes-0.2.3.tar.gz", hash = "sha256:d162dc04946d704503b2edc4d55f3dba5c1d539ead017afa00142c38b9885755", size = 29471 } +wheels = [ + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/de/3d/8161f7711c017e01ac9f008dfddd9410dff3674334c233bde66e7ba65bbf/pywin32_ctypes-0.2.3-py3-none-any.whl", hash = "sha256:8a1513379d709975552d202d942d9837758905c8d01eb82b8bcc30918929e7b8", size = 30756 }, +] + +[[package]] +name = "setuptools" +version = "75.3.0" +source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/ed/22/a438e0caa4576f8c383fa4d35f1cc01655a46c75be358960d815bfbb12bd/setuptools-75.3.0.tar.gz", hash = "sha256:fba5dd4d766e97be1b1681d98712680ae8f2f26d7881245f2ce9e40714f1a686", size = 1351577 } +wheels = [ + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/90/12/282ee9bce8b58130cb762fbc9beabd531549952cac11fc56add11dcb7ea0/setuptools-75.3.0-py3-none-any.whl", hash = "sha256:f2504966861356aa38616760c0f66568e535562374995367b4e69c7143cf6bcd", size = 1251070 }, +] + [[package]] name = "thousandhands" version = "0.1.0" source = { virtual = "." } dependencies = [ + { name = "aiofiles" }, { name = "flask" }, + { name = "gunicorn" }, + { name = "pyinstaller" }, { name = "pytest" }, ] [package.metadata] requires-dist = [ + { name = "aiofiles", specifier = ">=24.1.0" }, { name = "flask", specifier = ">=3.0.3" }, + { name = "gunicorn", specifier = ">=23.0.0" }, + { name = "pyinstaller", specifier = ">=6.11" }, { name = "pytest", specifier = ">=8.3.3" }, ] [[package]] name = "werkzeug" -version = "3.0.4" +version = "3.1.3" source = { registry = "https://pypi.tuna.tsinghua.edu.cn/simple" } dependencies = [ { name = "markupsafe" }, ] -sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/0f/e2/6dbcaab07560909ff8f654d3a2e5a60552d937c909455211b1b36d7101dc/werkzeug-3.0.4.tar.gz", hash = "sha256:34f2371506b250df4d4f84bfe7b0921e4762525762bbd936614909fe25cd7306", size = 803966 } +sdist = { url = "https://pypi.tuna.tsinghua.edu.cn/packages/9f/69/83029f1f6300c5fb2471d621ab06f6ec6b3324685a2ce0f9777fd4a8b71e/werkzeug-3.1.3.tar.gz", hash = "sha256:60723ce945c19328679790e3282cc758aa4a6040e4bb330f53d30fa546d44746", size = 806925 } wheels = [ - { url = "https://pypi.tuna.tsinghua.edu.cn/packages/4b/84/997bbf7c2bf2dc3f09565c6d0b4959fefe5355c18c4096cfd26d83e0785b/werkzeug-3.0.4-py3-none-any.whl", hash = "sha256:02c9eb92b7d6c06f31a782811505d2157837cea66aaede3e217c7c27c039476c", size = 227554 }, + { url = "https://pypi.tuna.tsinghua.edu.cn/packages/52/24/ab44c871b0f07f491e5d2ad12c9bd7358e527510618cb1b803a88e986db1/werkzeug-3.1.3-py3-none-any.whl", hash = "sha256:54b78bf3716d19a65be4fceccc0d1d7b89e608834989dfae50ea87564639213e", size = 224498 }, ]