effectual 0.1.0__tar.gz

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,158 @@
1
+ # vscode specific
2
+
3
+ .vscode/
4
+
5
+ # ruff
6
+
7
+ .ruff_cache/
8
+
9
+ # bundled file
10
+
11
+ out/**
12
+ out/bundle.py
13
+
14
+ # testing files
15
+
16
+ testing.py
17
+ test.py
18
+
19
+ # Byte-compiled / optimized / DLL files
20
+ __pycache__/
21
+ *.py[cod]
22
+ *$py.class
23
+
24
+ # C extensions
25
+ *.so
26
+
27
+ # Distribution / packaging
28
+ .Python
29
+ build/
30
+ develop-eggs/
31
+ dist/
32
+ downloads/
33
+ eggs/
34
+ .eggs/
35
+ lib/
36
+ lib64/
37
+ parts/
38
+ sdist/
39
+ var/
40
+ wheels/
41
+ share/python-wheels/
42
+ *.egg-info/
43
+ .installed.cfg
44
+ *.egg
45
+ MANIFEST
46
+
47
+ # PyInstaller
48
+ # Usually these files are written by a python script from a template
49
+ # before PyInstaller builds the exe, so as to inject date/other infos into it.
50
+ *.manifest
51
+ *.spec
52
+
53
+ # Installer logs
54
+ pip-log.txt
55
+ pip-delete-this-directory.txt
56
+
57
+ # Unit test / coverage reports
58
+ htmlcov/
59
+ .tox/
60
+ .nox/
61
+ .coverage
62
+ .coverage.*
63
+ .cache
64
+ nosetests.xml
65
+ coverage.xml
66
+ *.cover
67
+ *.py,cover
68
+ .hypothesis/
69
+ .pytest_cache/
70
+ cover/
71
+
72
+ # Translations
73
+ *.mo
74
+ *.pot
75
+
76
+ # Django stuff:
77
+ *.log
78
+ local_settings.py
79
+ db.sqlite3
80
+ db.sqlite3-journal
81
+
82
+ # Flask stuff:
83
+ instance/
84
+ .webassets-cache
85
+
86
+ # Scrapy stuff:
87
+ .scrapy
88
+
89
+ # Sphinx documentation
90
+ docs/_build/
91
+
92
+ # PyBuilder
93
+ .pybuilder/
94
+ target/
95
+
96
+ # Jupyter Notebook
97
+ .ipynb_checkpoints
98
+
99
+ # IPython
100
+ profile_default/
101
+ ipython_config.py
102
+
103
+ # poetry
104
+ # Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control.
105
+ # This is especially recommended for binary packages to ensure reproducibility, and is more
106
+ # commonly ignored for libraries.
107
+ # https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control
108
+ #poetry.lock
109
+
110
+ # PEP 582; used by e.g. github.com/David-OConnor/pyflow
111
+ __pypackages__/
112
+
113
+ # Celery stuff
114
+ celerybeat-schedule
115
+ celerybeat.pid
116
+
117
+ # SageMath parsed files
118
+ *.sage.py
119
+
120
+ # Environments
121
+ .env
122
+ .venv
123
+ env/
124
+ venv/
125
+ ENV/
126
+ env.bak/
127
+ venv.bak/
128
+
129
+ # Spyder project settings
130
+ .spyderproject
131
+ .spyproject
132
+
133
+ # Rope project settings
134
+ .ropeproject
135
+
136
+ # mkdocs documentation
137
+ /site
138
+
139
+ # mypy
140
+ .mypy_cache/
141
+ .dmypy.json
142
+ dmypy.json
143
+
144
+ # Pyre type checker
145
+ .pyre/
146
+
147
+ # pytype static type analyzer
148
+ .pytype/
149
+
150
+ # Cython debug symbols
151
+ cython_debug/
152
+
153
+ # PyCharm
154
+ # JetBrains specific template is maintainted in a separate JetBrains.gitignore that can
155
+ # be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
156
+ # and can be added to the global gitignore or merged into this file. For a more nuclear
157
+ # option (not recommended) you can uncomment the following to ignore the entire idea folder.
158
+ #.idea/
@@ -0,0 +1 @@
1
+ 3.11
@@ -0,0 +1,373 @@
1
+ Mozilla Public License Version 2.0
2
+ ==================================
3
+
4
+ 1. Definitions
5
+ --------------
6
+
7
+ 1.1. "Contributor"
8
+ means each individual or legal entity that creates, contributes to
9
+ the creation of, or owns Covered Software.
10
+
11
+ 1.2. "Contributor Version"
12
+ means the combination of the Contributions of others (if any) used
13
+ by a Contributor and that particular Contributor's Contribution.
14
+
15
+ 1.3. "Contribution"
16
+ means Covered Software of a particular Contributor.
17
+
18
+ 1.4. "Covered Software"
19
+ means Source Code Form to which the initial Contributor has attached
20
+ the notice in Exhibit A, the Executable Form of such Source Code
21
+ Form, and Modifications of such Source Code Form, in each case
22
+ including portions thereof.
23
+
24
+ 1.5. "Incompatible With Secondary Licenses"
25
+ means
26
+
27
+ (a) that the initial Contributor has attached the notice described
28
+ in Exhibit B to the Covered Software; or
29
+
30
+ (b) that the Covered Software was made available under the terms of
31
+ version 1.1 or earlier of the License, but not also under the
32
+ terms of a Secondary License.
33
+
34
+ 1.6. "Executable Form"
35
+ means any form of the work other than Source Code Form.
36
+
37
+ 1.7. "Larger Work"
38
+ means a work that combines Covered Software with other material, in
39
+ a separate file or files, that is not Covered Software.
40
+
41
+ 1.8. "License"
42
+ means this document.
43
+
44
+ 1.9. "Licensable"
45
+ means having the right to grant, to the maximum extent possible,
46
+ whether at the time of the initial grant or subsequently, any and
47
+ all of the rights conveyed by this License.
48
+
49
+ 1.10. "Modifications"
50
+ means any of the following:
51
+
52
+ (a) any file in Source Code Form that results from an addition to,
53
+ deletion from, or modification of the contents of Covered
54
+ Software; or
55
+
56
+ (b) any new file in Source Code Form that contains any Covered
57
+ Software.
58
+
59
+ 1.11. "Patent Claims" of a Contributor
60
+ means any patent claim(s), including without limitation, method,
61
+ process, and apparatus claims, in any patent Licensable by such
62
+ Contributor that would be infringed, but for the grant of the
63
+ License, by the making, using, selling, offering for sale, having
64
+ made, import, or transfer of either its Contributions or its
65
+ Contributor Version.
66
+
67
+ 1.12. "Secondary License"
68
+ means either the GNU General Public License, Version 2.0, the GNU
69
+ Lesser General Public License, Version 2.1, the GNU Affero General
70
+ Public License, Version 3.0, or any later versions of those
71
+ licenses.
72
+
73
+ 1.13. "Source Code Form"
74
+ means the form of the work preferred for making modifications.
75
+
76
+ 1.14. "You" (or "Your")
77
+ means an individual or a legal entity exercising rights under this
78
+ License. For legal entities, "You" includes any entity that
79
+ controls, is controlled by, or is under common control with You. For
80
+ purposes of this definition, "control" means (a) the power, direct
81
+ or indirect, to cause the direction or management of such entity,
82
+ whether by contract or otherwise, or (b) ownership of more than
83
+ fifty percent (50%) of the outstanding shares or beneficial
84
+ ownership of such entity.
85
+
86
+ 2. License Grants and Conditions
87
+ --------------------------------
88
+
89
+ 2.1. Grants
90
+
91
+ Each Contributor hereby grants You a world-wide, royalty-free,
92
+ non-exclusive license:
93
+
94
+ (a) under intellectual property rights (other than patent or trademark)
95
+ Licensable by such Contributor to use, reproduce, make available,
96
+ modify, display, perform, distribute, and otherwise exploit its
97
+ Contributions, either on an unmodified basis, with Modifications, or
98
+ as part of a Larger Work; and
99
+
100
+ (b) under Patent Claims of such Contributor to make, use, sell, offer
101
+ for sale, have made, import, and otherwise transfer either its
102
+ Contributions or its Contributor Version.
103
+
104
+ 2.2. Effective Date
105
+
106
+ The licenses granted in Section 2.1 with respect to any Contribution
107
+ become effective for each Contribution on the date the Contributor first
108
+ distributes such Contribution.
109
+
110
+ 2.3. Limitations on Grant Scope
111
+
112
+ The licenses granted in this Section 2 are the only rights granted under
113
+ this License. No additional rights or licenses will be implied from the
114
+ distribution or licensing of Covered Software under this License.
115
+ Notwithstanding Section 2.1(b) above, no patent license is granted by a
116
+ Contributor:
117
+
118
+ (a) for any code that a Contributor has removed from Covered Software;
119
+ or
120
+
121
+ (b) for infringements caused by: (i) Your and any other third party's
122
+ modifications of Covered Software, or (ii) the combination of its
123
+ Contributions with other software (except as part of its Contributor
124
+ Version); or
125
+
126
+ (c) under Patent Claims infringed by Covered Software in the absence of
127
+ its Contributions.
128
+
129
+ This License does not grant any rights in the trademarks, service marks,
130
+ or logos of any Contributor (except as may be necessary to comply with
131
+ the notice requirements in Section 3.4).
132
+
133
+ 2.4. Subsequent Licenses
134
+
135
+ No Contributor makes additional grants as a result of Your choice to
136
+ distribute the Covered Software under a subsequent version of this
137
+ License (see Section 10.2) or under the terms of a Secondary License (if
138
+ permitted under the terms of Section 3.3).
139
+
140
+ 2.5. Representation
141
+
142
+ Each Contributor represents that the Contributor believes its
143
+ Contributions are its original creation(s) or it has sufficient rights
144
+ to grant the rights to its Contributions conveyed by this License.
145
+
146
+ 2.6. Fair Use
147
+
148
+ This License is not intended to limit any rights You have under
149
+ applicable copyright doctrines of fair use, fair dealing, or other
150
+ equivalents.
151
+
152
+ 2.7. Conditions
153
+
154
+ Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted
155
+ in Section 2.1.
156
+
157
+ 3. Responsibilities
158
+ -------------------
159
+
160
+ 3.1. Distribution of Source Form
161
+
162
+ All distribution of Covered Software in Source Code Form, including any
163
+ Modifications that You create or to which You contribute, must be under
164
+ the terms of this License. You must inform recipients that the Source
165
+ Code Form of the Covered Software is governed by the terms of this
166
+ License, and how they can obtain a copy of this License. You may not
167
+ attempt to alter or restrict the recipients' rights in the Source Code
168
+ Form.
169
+
170
+ 3.2. Distribution of Executable Form
171
+
172
+ If You distribute Covered Software in Executable Form then:
173
+
174
+ (a) such Covered Software must also be made available in Source Code
175
+ Form, as described in Section 3.1, and You must inform recipients of
176
+ the Executable Form how they can obtain a copy of such Source Code
177
+ Form by reasonable means in a timely manner, at a charge no more
178
+ than the cost of distribution to the recipient; and
179
+
180
+ (b) You may distribute such Executable Form under the terms of this
181
+ License, or sublicense it under different terms, provided that the
182
+ license for the Executable Form does not attempt to limit or alter
183
+ the recipients' rights in the Source Code Form under this License.
184
+
185
+ 3.3. Distribution of a Larger Work
186
+
187
+ You may create and distribute a Larger Work under terms of Your choice,
188
+ provided that You also comply with the requirements of this License for
189
+ the Covered Software. If the Larger Work is a combination of Covered
190
+ Software with a work governed by one or more Secondary Licenses, and the
191
+ Covered Software is not Incompatible With Secondary Licenses, this
192
+ License permits You to additionally distribute such Covered Software
193
+ under the terms of such Secondary License(s), so that the recipient of
194
+ the Larger Work may, at their option, further distribute the Covered
195
+ Software under the terms of either this License or such Secondary
196
+ License(s).
197
+
198
+ 3.4. Notices
199
+
200
+ You may not remove or alter the substance of any license notices
201
+ (including copyright notices, patent notices, disclaimers of warranty,
202
+ or limitations of liability) contained within the Source Code Form of
203
+ the Covered Software, except that You may alter any license notices to
204
+ the extent required to remedy known factual inaccuracies.
205
+
206
+ 3.5. Application of Additional Terms
207
+
208
+ You may choose to offer, and to charge a fee for, warranty, support,
209
+ indemnity or liability obligations to one or more recipients of Covered
210
+ Software. However, You may do so only on Your own behalf, and not on
211
+ behalf of any Contributor. You must make it absolutely clear that any
212
+ such warranty, support, indemnity, or liability obligation is offered by
213
+ You alone, and You hereby agree to indemnify every Contributor for any
214
+ liability incurred by such Contributor as a result of warranty, support,
215
+ indemnity or liability terms You offer. You may include additional
216
+ disclaimers of warranty and limitations of liability specific to any
217
+ jurisdiction.
218
+
219
+ 4. Inability to Comply Due to Statute or Regulation
220
+ ---------------------------------------------------
221
+
222
+ If it is impossible for You to comply with any of the terms of this
223
+ License with respect to some or all of the Covered Software due to
224
+ statute, judicial order, or regulation then You must: (a) comply with
225
+ the terms of this License to the maximum extent possible; and (b)
226
+ describe the limitations and the code they affect. Such description must
227
+ be placed in a text file included with all distributions of the Covered
228
+ Software under this License. Except to the extent prohibited by statute
229
+ or regulation, such description must be sufficiently detailed for a
230
+ recipient of ordinary skill to be able to understand it.
231
+
232
+ 5. Termination
233
+ --------------
234
+
235
+ 5.1. The rights granted under this License will terminate automatically
236
+ if You fail to comply with any of its terms. However, if You become
237
+ compliant, then the rights granted under this License from a particular
238
+ Contributor are reinstated (a) provisionally, unless and until such
239
+ Contributor explicitly and finally terminates Your grants, and (b) on an
240
+ ongoing basis, if such Contributor fails to notify You of the
241
+ non-compliance by some reasonable means prior to 60 days after You have
242
+ come back into compliance. Moreover, Your grants from a particular
243
+ Contributor are reinstated on an ongoing basis if such Contributor
244
+ notifies You of the non-compliance by some reasonable means, this is the
245
+ first time You have received notice of non-compliance with this License
246
+ from such Contributor, and You become compliant prior to 30 days after
247
+ Your receipt of the notice.
248
+
249
+ 5.2. If You initiate litigation against any entity by asserting a patent
250
+ infringement claim (excluding declaratory judgment actions,
251
+ counter-claims, and cross-claims) alleging that a Contributor Version
252
+ directly or indirectly infringes any patent, then the rights granted to
253
+ You by any and all Contributors for the Covered Software under Section
254
+ 2.1 of this License shall terminate.
255
+
256
+ 5.3. In the event of termination under Sections 5.1 or 5.2 above, all
257
+ end user license agreements (excluding distributors and resellers) which
258
+ have been validly granted by You or Your distributors under this License
259
+ prior to termination shall survive termination.
260
+
261
+ ************************************************************************
262
+ * *
263
+ * 6. Disclaimer of Warranty *
264
+ * ------------------------- *
265
+ * *
266
+ * Covered Software is provided under this License on an "as is" *
267
+ * basis, without warranty of any kind, either expressed, implied, or *
268
+ * statutory, including, without limitation, warranties that the *
269
+ * Covered Software is free of defects, merchantable, fit for a *
270
+ * particular purpose or non-infringing. The entire risk as to the *
271
+ * quality and performance of the Covered Software is with You. *
272
+ * Should any Covered Software prove defective in any respect, You *
273
+ * (not any Contributor) assume the cost of any necessary servicing, *
274
+ * repair, or correction. This disclaimer of warranty constitutes an *
275
+ * essential part of this License. No use of any Covered Software is *
276
+ * authorized under this License except under this disclaimer. *
277
+ * *
278
+ ************************************************************************
279
+
280
+ ************************************************************************
281
+ * *
282
+ * 7. Limitation of Liability *
283
+ * -------------------------- *
284
+ * *
285
+ * Under no circumstances and under no legal theory, whether tort *
286
+ * (including negligence), contract, or otherwise, shall any *
287
+ * Contributor, or anyone who distributes Covered Software as *
288
+ * permitted above, be liable to You for any direct, indirect, *
289
+ * special, incidental, or consequential damages of any character *
290
+ * including, without limitation, damages for lost profits, loss of *
291
+ * goodwill, work stoppage, computer failure or malfunction, or any *
292
+ * and all other commercial damages or losses, even if such party *
293
+ * shall have been informed of the possibility of such damages. This *
294
+ * limitation of liability shall not apply to liability for death or *
295
+ * personal injury resulting from such party's negligence to the *
296
+ * extent applicable law prohibits such limitation. Some *
297
+ * jurisdictions do not allow the exclusion or limitation of *
298
+ * incidental or consequential damages, so this exclusion and *
299
+ * limitation may not apply to You. *
300
+ * *
301
+ ************************************************************************
302
+
303
+ 8. Litigation
304
+ -------------
305
+
306
+ Any litigation relating to this License may be brought only in the
307
+ courts of a jurisdiction where the defendant maintains its principal
308
+ place of business and such litigation shall be governed by laws of that
309
+ jurisdiction, without reference to its conflict-of-law provisions.
310
+ Nothing in this Section shall prevent a party's ability to bring
311
+ cross-claims or counter-claims.
312
+
313
+ 9. Miscellaneous
314
+ ----------------
315
+
316
+ This License represents the complete agreement concerning the subject
317
+ matter hereof. If any provision of this License is held to be
318
+ unenforceable, such provision shall be reformed only to the extent
319
+ necessary to make it enforceable. Any law or regulation which provides
320
+ that the language of a contract shall be construed against the drafter
321
+ shall not be used to construe this License against a Contributor.
322
+
323
+ 10. Versions of the License
324
+ ---------------------------
325
+
326
+ 10.1. New Versions
327
+
328
+ Mozilla Foundation is the license steward. Except as provided in Section
329
+ 10.3, no one other than the license steward has the right to modify or
330
+ publish new versions of this License. Each version will be given a
331
+ distinguishing version number.
332
+
333
+ 10.2. Effect of New Versions
334
+
335
+ You may distribute the Covered Software under the terms of the version
336
+ of the License under which You originally received the Covered Software,
337
+ or under the terms of any subsequent version published by the license
338
+ steward.
339
+
340
+ 10.3. Modified Versions
341
+
342
+ If you create software not governed by this License, and you want to
343
+ create a new license for such software, you may create and use a
344
+ modified version of this License if you rename the license and remove
345
+ any references to the name of the license steward (except to note that
346
+ such modified license differs from this License).
347
+
348
+ 10.4. Distributing Source Code Form that is Incompatible With Secondary
349
+ Licenses
350
+
351
+ If You choose to distribute Source Code Form that is Incompatible With
352
+ Secondary Licenses under the terms of this version of the License, the
353
+ notice described in Exhibit B of this License must be attached.
354
+
355
+ Exhibit A - Source Code Form License Notice
356
+ -------------------------------------------
357
+
358
+ This Source Code Form is subject to the terms of the Mozilla Public
359
+ License, v. 2.0. If a copy of the MPL was not distributed with this
360
+ file, You can obtain one at http://mozilla.org/MPL/2.0/.
361
+
362
+ If it is not possible or desirable to put the notice in a particular
363
+ file, then You may include the notice in a location (such as a LICENSE
364
+ file in a relevant directory) where a recipient would be likely to look
365
+ for such a notice.
366
+
367
+ You may add additional accurate notices of copyright ownership.
368
+
369
+ Exhibit B - "Incompatible With Secondary Licenses" Notice
370
+ ---------------------------------------------------------
371
+
372
+ This Source Code Form is "Incompatible With Secondary Licenses", as
373
+ defined by the Mozilla Public License, v. 2.0.
@@ -0,0 +1,19 @@
1
+ Metadata-Version: 2.3
2
+ Name: effectual
3
+ Version: 0.1.0
4
+ Summary: Add your description here
5
+ Author-email: jake <jakewdr@proton.me>
6
+ Classifier: License :: OSI Approved :: MIT License
7
+ Classifier: Programming Language :: Python :: 3.8
8
+ Classifier: Programming Language :: Python :: 3.9
9
+ Classifier: Programming Language :: Python :: 3.10
10
+ Classifier: Programming Language :: Python :: 3.11
11
+ Classifier: Programming Language :: Python :: 3.12
12
+ Classifier: Programming Language :: Python :: 3.13
13
+ Classifier: Programming Language :: Python :: Implementation :: CPython
14
+ Requires-Python: >=3.11
15
+ Requires-Dist: click>=8.1.7
16
+ Requires-Dist: python-minifier>=2.11.3
17
+ Requires-Dist: rtoml>=0.11.0
18
+ Requires-Dist: ruff>=0.8.0
19
+ Requires-Dist: termcolor>=2.4.0
File without changes
@@ -0,0 +1,38 @@
1
+ [project]
2
+ name = "effectual"
3
+ version = "0.1.0"
4
+ description = "Add your description here"
5
+ readme = "README.md"
6
+ authors = [
7
+ { name = "jake", email = "jakewdr@proton.me" }
8
+ ]
9
+ requires-python = ">=3.11"
10
+ classifiers = [
11
+ "License :: OSI Approved :: MIT License",
12
+ "Programming Language :: Python :: 3.8",
13
+ "Programming Language :: Python :: 3.9",
14
+ "Programming Language :: Python :: 3.10",
15
+ "Programming Language :: Python :: 3.11",
16
+ "Programming Language :: Python :: 3.12",
17
+ "Programming Language :: Python :: 3.13",
18
+ "Programming Language :: Python :: Implementation :: CPython",
19
+ ]
20
+
21
+ environments = [
22
+ "implementation_name == 'cpython'"
23
+ ]
24
+
25
+ dependencies = [
26
+ "click>=8.1.7",
27
+ "python-minifier>=2.11.3",
28
+ "rtoml>=0.11.0",
29
+ "ruff>=0.8.0",
30
+ "termcolor>=2.4.0",
31
+ ]
32
+
33
+ [project.scripts]
34
+ efec = "effectual:main"
35
+
36
+ [build-system]
37
+ requires = ["hatchling"]
38
+ build-backend = "hatchling.build"
@@ -0,0 +1,23 @@
1
+ import click
2
+
3
+ @click.group()
4
+ def main():
5
+ """Effectual CLI - A simple command-line interface."""
6
+ pass
7
+
8
+ @click.command("dist")
9
+ def dist():
10
+ """Bundles your source directory."""
11
+ from . import build
12
+ build.main()
13
+
14
+ @click.command("dev")
15
+ def dev():
16
+ """Bundles your source directory."""
17
+ from . import dev
18
+ dev.main()
19
+
20
+ main.add_command(dist)
21
+
22
+ if __name__ == "__main__":
23
+ main()
@@ -0,0 +1,162 @@
1
+ import os
2
+ import shutil
3
+ import zipfile
4
+ from pathlib import Path
5
+ from time import perf_counter
6
+
7
+ import rtoml
8
+ from .colors import completeColor, fileColor, folderColor, tagColor
9
+ from .config import loadConfig
10
+ from .fileHash import getFilehash
11
+ from .minifier import minifyFile, minifyToString
12
+
13
+
14
+ def bundleFiles(
15
+ sourceDirectory: Path,
16
+ outputDirectory: Path,
17
+ outputFileName: str,
18
+ compressionLevel: int,
19
+ minification: bool,
20
+ ) -> None:
21
+ """Bundles dependencies and scripts into a single .py archive
22
+
23
+ Args:
24
+ sourceDirectory (Path): Source directory which must contain a __main__.py script
25
+ outputDirectory (Path): Output directory for the bundle
26
+ outputFileName (str): Name of the output bundle
27
+ compressionLevel (int): Compression level for the bundle from 0-9
28
+ minification (bool): If the dependencies and scripts should be minified
29
+ """
30
+ outputDirectory.mkdir(parents=True, exist_ok=True)
31
+ outputPath: Path = outputDirectory / outputFileName
32
+
33
+ if outputPath.exists():
34
+ outputPath.unlink()
35
+
36
+ with zipfile.ZipFile(
37
+ outputPath,
38
+ "w",
39
+ compresslevel=compressionLevel,
40
+ compression=zipfile.ZIP_DEFLATED,
41
+ ) as bundler:
42
+ cachePath: Path = Path("./.effectual_cache/cachedPackages")
43
+
44
+ totalSize: int = int(0)
45
+ for cachedFile in cachePath.rglob("*"):
46
+ if cachedFile.is_dir() and not any(cachedFile.iterdir()):
47
+ continue
48
+ totalSize += cachedFile.stat().st_size
49
+ arcName = cachedFile.relative_to(cachePath)
50
+ bundler.write(cachedFile, arcname=arcName)
51
+
52
+ print(f"{tagColor('bundling')} || uv dependencies {folderColor(totalSize)}")
53
+
54
+ for pyFile in sourceDirectory.rglob("*.py"):
55
+ print(f"{tagColor('bundling')} || {pyFile.name} {fileColor(pyFile)}")
56
+ if minification:
57
+ fileContents = minifyToString(pyFile)
58
+ bundler.writestr(zinfo_or_arcname=pyFile.name, data=fileContents)
59
+ else:
60
+ bundler.write(pyFile, arcname=pyFile.name)
61
+
62
+ print(f"{tagColor('OUTPUT')} || {outputFileName} {fileColor(outputPath)}")
63
+
64
+
65
+ def dependencies(minify: bool) -> None:
66
+ with open("./pyproject.toml", "r", encoding="utf-8") as file:
67
+ packages: list[str] = dict(rtoml.load(file)).get("project").get("dependencies")
68
+
69
+ arguments: list[str] = ["--no-compile", "--quiet", "--no-binary=none", "--no-cache"]
70
+
71
+ pathToInstallTo: str = "./.effectual_cache/cachedPackages"
72
+ argumentString: str = " ".join(arguments)
73
+
74
+ if Path(pathToInstallTo).exists():
75
+ shutil.rmtree(pathToInstallTo)
76
+
77
+ for key in packages:
78
+ print(f"{tagColor('installing')} || {key}")
79
+ os.system(f'uv pip install "{key}" {argumentString} --target {pathToInstallTo}')
80
+
81
+ print(f"{tagColor('optimizing')} || {', '.join(packages)}")
82
+
83
+ import multiprocessing
84
+
85
+ with multiprocessing.Pool(processes=multiprocessing.cpu_count()) as pool:
86
+ pool.map(optimizeDependencies, Path(pathToInstallTo).rglob("*"))
87
+
88
+
89
+ def optimizeDependencies(file: Path) -> None:
90
+ if (
91
+ file.suffix in (".pyc", ".pyd", ".exe", ".typed")
92
+ or "__pycache__" in str(file)
93
+ or ".dist-info" in str(file)
94
+ or ".lock" in str(file)
95
+ ):
96
+ try:
97
+ file.unlink()
98
+ except PermissionError:
99
+ pass
100
+ elif file.suffix == ".py":
101
+ minifyFile(file)
102
+
103
+
104
+ def main() -> None:
105
+ """Entrypoint
106
+
107
+ Raises:
108
+ RuntimeError: In the event there is no source directory
109
+ """
110
+
111
+ configData: dict = loadConfig("./pyproject.toml")
112
+
113
+ sourceDirectory: Path = Path(configData.get("sourceDirectory", "src/"))
114
+ outputDirectory: Path = Path(configData.get("outputDirectory", "out/"))
115
+ outputFileName: str = configData.get("outputFileName", "bundle.pyz")
116
+ compressionLevel: int = max(
117
+ 0, min(9, configData.get("compressionLevel", 5))
118
+ ) # Default level if not set
119
+ global minification
120
+ minification = configData.get("minification", True)
121
+
122
+ if not sourceDirectory.is_dir():
123
+ raise RuntimeError(
124
+ f"Source directory {sourceDirectory} does not exist or is not a directory."
125
+ )
126
+
127
+ uvHashPath: Path = Path("./.effectual_cache/pyprojectHash.toml")
128
+ currentHash: dict[str] = dict()
129
+
130
+ startTime = perf_counter()
131
+
132
+ Path("./.effectual_cache/").mkdir(parents=True, exist_ok=True)
133
+ currentHash["hashes"]: dict[dict] = dict()
134
+ currentHash["hashes"]["pyproject"] = getFilehash("./pyproject.toml")
135
+ currentHash["hashes"]["lock"] = getFilehash("./uv.lock")
136
+
137
+ if uvHashPath.exists():
138
+ with open(uvHashPath, "r") as file:
139
+ lastHash: dict = dict(rtoml.load(file)).get("hashes")
140
+ if currentHash["hashes"] != lastHash:
141
+ with open(uvHashPath, "w") as file:
142
+ rtoml.dump(currentHash, file)
143
+ dependencies(minify=minification)
144
+ else:
145
+ with open(uvHashPath, "x") as file:
146
+ rtoml.dump(currentHash, file)
147
+ dependencies(minify=minification)
148
+
149
+ bundleFiles(
150
+ sourceDirectory,
151
+ outputDirectory,
152
+ outputFileName,
153
+ compressionLevel,
154
+ minification,
155
+ )
156
+ endTime = perf_counter()
157
+
158
+ print(completeColor(f"Completed in {endTime - startTime:.4f}s"))
159
+
160
+
161
+ if "__main__" in __name__:
162
+ main()
@@ -0,0 +1,63 @@
1
+ from pathlib import Path
2
+
3
+ from termcolor import colored
4
+
5
+
6
+ def fileColor(filePath: Path) -> str:
7
+ """Creates a yellow string with the size of a file
8
+
9
+ Args:
10
+ filePath (Path): Path to the file
11
+
12
+ Returns:
13
+ str: Output string
14
+ """
15
+ return colored(f"{str(round(filePath.stat().st_size / 1024, 3))}kB", "yellow")
16
+
17
+
18
+ def tagColor(nameOfTag: str) -> str:
19
+ """Creates a blue tag with uppercase letters and squared brackets
20
+
21
+ Args:
22
+ nameOfTag (str): What the tag should be called
23
+
24
+ Returns:
25
+ str: Output string
26
+ """
27
+ return colored(f"[{nameOfTag.upper()}]", "blue")
28
+
29
+
30
+ def errorColor(errorString: str) -> str:
31
+ """Makes an error string red
32
+
33
+ Args:
34
+ errorString (str):
35
+
36
+ Returns:
37
+ str: Output string
38
+ """
39
+ return colored(errorString, "red")
40
+
41
+
42
+ def folderColor(sizeOfFolder: int) -> str:
43
+ """Writes out the size of a folder
44
+
45
+ Args:
46
+ sizeOfFolder (int): Size of the folder in bytes
47
+
48
+ Returns:
49
+ str: Output string
50
+ """
51
+ return colored(f"{round((sizeOfFolder / 1024), 3)}kB", "yellow")
52
+
53
+
54
+ def completeColor(completeString: str) -> str:
55
+ """Makes a string light magenta
56
+
57
+ Args:
58
+ completeString (str): String to be shown at end of process
59
+
60
+ Returns:
61
+ str: Output string
62
+ """
63
+ return colored(completeString, "light_magenta")
@@ -0,0 +1,27 @@
1
+ import rtoml
2
+
3
+
4
+ def loadConfig(configPath: str) -> dict:
5
+ """Loads effectual config from a file
6
+
7
+ Args:
8
+ configPath (str): Path to the config file
9
+
10
+ Raises:
11
+ RuntimeError: Invalid TOML format
12
+ RuntimeError: No configuration file found
13
+
14
+ Returns:
15
+ dict: _description_
16
+ """
17
+ try:
18
+ with open(configPath, "r", encoding="utf-8") as file:
19
+ tomlFile: dict = dict(rtoml.load(file))
20
+ configData: dict = tomlFile.get("tool").get("effectual")
21
+
22
+ except ValueError as e:
23
+ raise RuntimeError(f"Invalid TOML in {configPath}: {e}")
24
+ except FileNotFoundError:
25
+ raise RuntimeError(f"Configuration file {configPath} not found.")
26
+
27
+ return configData
@@ -0,0 +1,60 @@
1
+ import subprocess
2
+ import time
3
+ import zipfile
4
+ from pathlib import Path
5
+
6
+ from .colors import completeColor, fileColor, tagColor
7
+ from .config import loadConfig
8
+ from .fileHash import getAllHashes
9
+
10
+
11
+ def bundle(sourceDirectory: Path, outputFile: Path) -> None:
12
+ """Bundles scripts into a single .py archive
13
+
14
+ Args:
15
+ sourceDirectory (Path): Path to the original python scripts
16
+ """
17
+ startTime = time.perf_counter()
18
+
19
+ with zipfile.ZipFile(outputFile, "w") as bundler:
20
+ for pyFile in sourceDirectory.rglob("*.py"):
21
+ print(f"{tagColor('bundling')} || {pyFile.name} {fileColor(pyFile)}")
22
+ bundler.write(pyFile, arcname=pyFile.name)
23
+ endTime = time.perf_counter()
24
+
25
+ print(completeColor(f"Completed in {endTime - startTime:.4f}s"))
26
+
27
+
28
+ def main() -> None:
29
+ """Super fast bundling for the 'task dev' command"""
30
+
31
+ configData: dict = loadConfig("./pyproject.toml")
32
+ sourceDirectory: Path = Path(configData.get("sourceDirectory", "src/"))
33
+ outputFileName: str = Path(configData.get("outputFileName", "bundle.pyz"))
34
+ devBundlePath: Path = Path("./.effectual_cache/dev/")
35
+ devBundlePath.mkdir(parents=True, exist_ok=True)
36
+
37
+ outputFile: Path = devBundlePath / outputFileName
38
+
39
+ bundle(sourceDirectory, outputFile)
40
+
41
+ runCommand = subprocess.Popen(["uv", "run", outputFile], shell=True)
42
+
43
+ lastHashList: list[str] = getAllHashes(sourceDirectory)
44
+
45
+ while True:
46
+ currentHashList: list[str] = getAllHashes(sourceDirectory)
47
+ if currentHashList != lastHashList:
48
+ lastHashList = currentHashList
49
+ runCommand.kill()
50
+ runCommand.wait()
51
+ outputFile.unlink()
52
+ print(f"{tagColor('reloaded')} || file change detected")
53
+ bundle(sourceDirectory, outputFile)
54
+ runCommand = subprocess.Popen(["uv", "run", outputFile], shell=True)
55
+ else:
56
+ time.sleep(0.1)
57
+
58
+
59
+ if __name__ == "__main__":
60
+ main()
@@ -0,0 +1,32 @@
1
+ import hashlib
2
+ from multiprocessing import Pool
3
+ from pathlib import Path
4
+
5
+
6
+ def getFilehash(filePath: Path) -> str:
7
+ """Gets the file hash of a single python script
8
+
9
+ Args:
10
+ filePath (Path): Path to the python script
11
+
12
+ Returns:
13
+ str: Hash of the python script
14
+ """
15
+ with open(filePath, "rb") as file:
16
+ fileHash = hashlib.sha256(file.read()).hexdigest()
17
+ return fileHash
18
+
19
+
20
+ def getAllHashes(sourceDirectory: Path) -> list[str]:
21
+ """Gets all hashes in directory
22
+
23
+ Args:
24
+ sourceDirectory (Path): Path to the python scripts
25
+
26
+ Returns:
27
+ dict[str]: Dictionary containing paths and hashes
28
+ """
29
+
30
+ with Pool() as pool:
31
+ hashList: list[str] = pool.map(getFilehash, sourceDirectory.glob("*.py"))
32
+ return hashList
@@ -0,0 +1,51 @@
1
+ from pathlib import Path
2
+
3
+ from python_minifier import minify
4
+
5
+
6
+ def minifyFile(filePath: Path) -> None:
7
+ """Minifies a file from a certain path
8
+
9
+ Args:
10
+ filePath (Path): Path to the file to minify
11
+
12
+ Raises:
13
+ RuntimeError: In the event the file cannot be found or an error has occurred
14
+ """
15
+ try:
16
+ with filePath.open("r+", encoding="utf-8") as fileRW:
17
+ minifiedCode = minify(
18
+ fileRW.read(),
19
+ hoist_literals=False,
20
+ remove_literal_statements=True,
21
+ remove_debug=True,
22
+ )
23
+
24
+ fileRW.seek(0)
25
+ fileRW.write(minifiedCode)
26
+ fileRW.truncate()
27
+
28
+ except Exception as e:
29
+ raise RuntimeError(f"Failed to minify {filePath}: {e}")
30
+
31
+
32
+ def minifyToString(filePath: Path) -> str:
33
+ """Minifies string of a python file
34
+
35
+ Args:
36
+ filePath (Path): Path to file to minify
37
+
38
+ Returns:
39
+ str: Minified string
40
+ """
41
+ with filePath.open("r", encoding="utf-8") as fileR:
42
+ minifiedCode: str = str(
43
+ minify(
44
+ fileR.read(),
45
+ hoist_literals=False,
46
+ remove_literal_statements=True,
47
+ remove_debug=True,
48
+ )
49
+ ).encode("utf-8")
50
+
51
+ return minifiedCode
@@ -0,0 +1,121 @@
1
+ version = 1
2
+ requires-python = ">=3.11"
3
+
4
+ [[package]]
5
+ name = "click"
6
+ version = "8.1.7"
7
+ source = { registry = "https://pypi.org/simple" }
8
+ dependencies = [
9
+ { name = "colorama", marker = "platform_system == 'Windows'" },
10
+ ]
11
+ sdist = { url = "https://files.pythonhosted.org/packages/96/d3/f04c7bfcf5c1862a2a5b845c6b2b360488cf47af55dfa79c98f6a6bf98b5/click-8.1.7.tar.gz", hash = "sha256:ca9853ad459e787e2192211578cc907e7594e294c7ccc834310722b41b9ca6de", size = 336121 }
12
+ wheels = [
13
+ { url = "https://files.pythonhosted.org/packages/00/2e/d53fa4befbf2cfa713304affc7ca780ce4fc1fd8710527771b58311a3229/click-8.1.7-py3-none-any.whl", hash = "sha256:ae74fb96c20a0277a1d615f1e4d73c8414f5a98db8b799a7931d1582f3390c28", size = 97941 },
14
+ ]
15
+
16
+ [[package]]
17
+ name = "colorama"
18
+ version = "0.4.6"
19
+ source = { registry = "https://pypi.org/simple" }
20
+ sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697 }
21
+ wheels = [
22
+ { url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335 },
23
+ ]
24
+
25
+ [[package]]
26
+ name = "effectual"
27
+ version = "0.1.0"
28
+ source = { editable = "." }
29
+ dependencies = [
30
+ { name = "click" },
31
+ { name = "python-minifier" },
32
+ { name = "rtoml" },
33
+ { name = "ruff" },
34
+ { name = "termcolor" },
35
+ ]
36
+
37
+ [package.metadata]
38
+ requires-dist = [
39
+ { name = "click", specifier = ">=8.1.7" },
40
+ { name = "python-minifier", specifier = ">=2.11.3" },
41
+ { name = "rtoml", specifier = ">=0.11.0" },
42
+ { name = "ruff", specifier = ">=0.8.0" },
43
+ { name = "termcolor", specifier = ">=2.4.0" },
44
+ ]
45
+
46
+ [[package]]
47
+ name = "python-minifier"
48
+ version = "2.11.3"
49
+ source = { registry = "https://pypi.org/simple" }
50
+ sdist = { url = "https://files.pythonhosted.org/packages/39/63/403fb2d6394b3e455e046d91f64b96072803aaf119027a26e716ed94d63c/python_minifier-2.11.3.tar.gz", hash = "sha256:489133b91212ec9658a7b64d243eb9eb67d7e53faf2ac5166a33301c61b3dcab", size = 64438 }
51
+ wheels = [
52
+ { url = "https://files.pythonhosted.org/packages/53/32/61d20860d18afb81cb7258bb02d4eaf4b09170383c2374514f6aef384fa9/python_minifier-2.11.3-py3-none-any.whl", hash = "sha256:37e10e9e318be701eecb48764942426be73ae9f562d75bea4e29c5f66945ce97", size = 56172 },
53
+ ]
54
+
55
+ [[package]]
56
+ name = "rtoml"
57
+ version = "0.11.0"
58
+ source = { registry = "https://pypi.org/simple" }
59
+ sdist = { url = "https://files.pythonhosted.org/packages/c8/67/ca8b92b2b8aa231d427999609dd289e6bc8511b4a6871e2b533871c9b3e3/rtoml-0.11.0.tar.gz", hash = "sha256:a1d1ec36261c47169934a6c0f713a6cdf917604b3f72a72a809c3a68384255d4", size = 23759 }
60
+ wheels = [
61
+ { url = "https://files.pythonhosted.org/packages/53/bd/f42f86a1c0ec136a521ef59c790f08c3ed49bb794c5c97ff5bd212ef6740/rtoml-0.11.0-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:abf9a0f729040f99c7b1734c433919b19314843d0f1d597f2d82e448e210c60f", size = 315120 },
62
+ { url = "https://files.pythonhosted.org/packages/f7/44/bbf2ac61444fa4986bb45e46a84b6ea41723234ee2c370b93cafde3015e0/rtoml-0.11.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:48a3504d58d9a0178947cf988841f2a771b63bb90155ad20ba4da93318ca2195", size = 307563 },
63
+ { url = "https://files.pythonhosted.org/packages/7a/4c/4c063610404bb8200af18255e86174f0736cff0cc5bdae53cbbd1aafa830/rtoml-0.11.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:56421e722d61ff09df0ae1d447ed1f8f109877281a091da1062165093bfe6291", size = 348665 },
64
+ { url = "https://files.pythonhosted.org/packages/0f/b4/46badd4b99f1f1c7714cf7004112bccac32811540b18a5ccbc80f3066aec/rtoml-0.11.0-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:a1be1b24c51a0a79beaf13e69603a556a165f27c92c93d284408e44686f82fbc", size = 355889 },
65
+ { url = "https://files.pythonhosted.org/packages/f5/1a/0bfc06fbccc899110a8b07063d5c1659f288a993f5a98215927faf2ed9cd/rtoml-0.11.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ec5837a2e0e0e3084d243914c11aff602f6840e5c07a7b2e1ca0c89464ed7079", size = 373119 },
66
+ { url = "https://files.pythonhosted.org/packages/04/4a/95a9e9380cab681e004be5eb1f001d2990e541f6813b5064ba9186c40fda/rtoml-0.11.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:765bb5f226f7694d66af39f23d79b65f6b30744ee1c1396acff303b772f4ba1d", size = 498623 },
67
+ { url = "https://files.pythonhosted.org/packages/be/69/a53e893a19b497ab96af1a2d7aa871c148a738beb885f2d9f5de2a02d6a5/rtoml-0.11.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd6dfbc1830055290e8ee9f7ffb7465106536f54c1465e08c980c38147e3bcf4", size = 341272 },
68
+ { url = "https://files.pythonhosted.org/packages/0f/06/010f5b6f4fccc75fd18578382245d81afcab44580b04b601e3ed121e6aac/rtoml-0.11.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:068131643b18ee839f1d5798ef5b347d04e101b8b9398b774fc839e6231c63c7", size = 352252 },
69
+ { url = "https://files.pythonhosted.org/packages/a2/77/fc9c77b513da7148bb74336bde4c55fa47388840199d385984e7d08d74e8/rtoml-0.11.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:042fc83c225b1075b6e717d156c4c7ecdab255127f875f6188fb062d72e59c5f", size = 527041 },
70
+ { url = "https://files.pythonhosted.org/packages/3c/61/d37e521ad5caf015fa00dbe0011d31e2d128f5d75b1f7abf5144f86d34a7/rtoml-0.11.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:bcab62bbf210434dcab736c658911d0182afe33b3920b051978c04ca442504a9", size = 512088 },
71
+ { url = "https://files.pythonhosted.org/packages/6a/fc/61ddc4c5ffbc10ac57c5f5b1ee038c34a2919e00a5fbc6d7a492cf96eb10/rtoml-0.11.0-cp311-none-win32.whl", hash = "sha256:8edbf3f0498287557a0a170c9a9331dfb69058729968578c37aad19977ba5722", size = 221161 },
72
+ { url = "https://files.pythonhosted.org/packages/4c/6d/48ce15a3919a5c07a19ae3c80b9fadbe1e13a5e475198143965d3de6e60b/rtoml-0.11.0-cp311-none-win_amd64.whl", hash = "sha256:d5e3d8d73a220d06bf2359b13572bdbd198e66f45aeac1e84058e448dc34f85e", size = 230578 },
73
+ { url = "https://files.pythonhosted.org/packages/f6/be/323bb12dc30abb5451bb20f4f2b55bef4dffc7f78fafbd51fd8f904e123e/rtoml-0.11.0-cp311-none-win_arm64.whl", hash = "sha256:806e3893555657012fe89774e62999e8cac50df53a7bd27f7c838f606397c3f7", size = 222165 },
74
+ { url = "https://files.pythonhosted.org/packages/d1/36/69c9c078d8821b0801e14898b2fc92da53185db2f683dd0faf78c66fad7e/rtoml-0.11.0-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:34c4fd6e6465e6de84ac499e19ae3e4b0d4cd3f9763d1a72a75bd2ef4d0e466a", size = 315058 },
75
+ { url = "https://files.pythonhosted.org/packages/5a/ac/25f7b10168fa7ed25a3b28450c66a8be7e37c019721bc5f228237474c6a0/rtoml-0.11.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:0095bc482788d39c6c3e2f9f3a6f7a8149ee1cefe624129e2afc90512d8f16b7", size = 307112 },
76
+ { url = "https://files.pythonhosted.org/packages/26/43/f1cac9c12a752e3a83f66317f7947f37f1ec5828848cbbb99e37d84a7915/rtoml-0.11.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d90b06a36325c9ae6e273669d811aa0e6b8ae41fa9ffe3949f02ef1870dc0ad5", size = 348241 },
77
+ { url = "https://files.pythonhosted.org/packages/39/ea/b6ef0c18b635c6332929ac2198186afeae36d9d33a8d68513af7cd1a0ccb/rtoml-0.11.0-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:902148f2bd13b4c64d2c5ed576910c5a0a2c86dc052f4e31ca1855131ee157c8", size = 355577 },
78
+ { url = "https://files.pythonhosted.org/packages/00/d3/ac534011c12df090d44cce2cbaf023de90c0eba0dc83e74e375989c8cb2c/rtoml-0.11.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d84bbdad28e24f66b8ad632fb7cbec64b921fdff48bf57f4f7179648726f7e54", size = 372476 },
79
+ { url = "https://files.pythonhosted.org/packages/25/c9/db4cebaed3789c8d4333d362ed327dd1ba791f4004d6e440470d3239899e/rtoml-0.11.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4b1d4d4b8d765cf5228e5def9c39837d5b70c6e51422e05b975898c8a765fd1e", size = 497329 },
80
+ { url = "https://files.pythonhosted.org/packages/8b/7e/28f5e4fa9a3aee864b34d774cf4130c84f5c3f075b56ce6116bb2ed3df29/rtoml-0.11.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8b900dcdf2068105b81b1ded3342dad952e3ea7d4a949473a690c686116a1730", size = 341121 },
81
+ { url = "https://files.pythonhosted.org/packages/84/10/05b794e781c05b5bce4837e4f9a99cbc69f9f1da3b13cf36db37b26d5108/rtoml-0.11.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:4a491ad282b918292812518a61f432d6becf1b4a965d9c6595b7f00b1b722e61", size = 351976 },
82
+ { url = "https://files.pythonhosted.org/packages/94/e0/67e8dabd5253ae8364150e475c60e09caddbb31e3f43a51e3dbf3982ff37/rtoml-0.11.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:d5ccbe43af82f17efed273b517e29febbd8225dab9a314f5728465b61a34781a", size = 526442 },
83
+ { url = "https://files.pythonhosted.org/packages/e9/89/de40a48005025c20b392b5fba5d88427b92ebcbc09892ec37cc51daa8d14/rtoml-0.11.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:4125260584290461825abac23dafb34048dcbedaf9fafea913b5f0b1691c1ad0", size = 512134 },
84
+ { url = "https://files.pythonhosted.org/packages/27/d3/20e26214e4736c10653477b1bd616c5f2a79571495c28dec725bbbae7cfb/rtoml-0.11.0-cp312-none-win32.whl", hash = "sha256:6e42385f510458f68587051a533cd0161653e1f2b381400a4873eab6f706940b", size = 217810 },
85
+ { url = "https://files.pythonhosted.org/packages/df/7b/8d908a634ba31c27f8fe38a030e9406064960e1b87053bd45da683166044/rtoml-0.11.0-cp312-none-win_amd64.whl", hash = "sha256:fdce3405e50ba73a45acdf36a793e3563da04fc913c53d54852c5d6cf1eb6d25", size = 226747 },
86
+ { url = "https://files.pythonhosted.org/packages/87/53/01fcd15027bee67947d1965bf610f7f68b9c037ffb833eaf91ec16a394fb/rtoml-0.11.0-cp312-none-win_arm64.whl", hash = "sha256:affc854919b46555f37d1408cb67f06f7453744bf1c5c1c53b69ed8b5227a28d", size = 218370 },
87
+ ]
88
+
89
+ [[package]]
90
+ name = "ruff"
91
+ version = "0.8.1"
92
+ source = { registry = "https://pypi.org/simple" }
93
+ sdist = { url = "https://files.pythonhosted.org/packages/95/d0/8ff5b189d125f4260f2255d143bf2fa413b69c2610c405ace7a0a8ec81ec/ruff-0.8.1.tar.gz", hash = "sha256:3583db9a6450364ed5ca3f3b4225958b24f78178908d5c4bc0f46251ccca898f", size = 3313222 }
94
+ wheels = [
95
+ { url = "https://files.pythonhosted.org/packages/a2/d6/1a6314e568db88acdbb5121ed53e2c52cebf3720d3437a76f82f923bf171/ruff-0.8.1-py3-none-linux_armv6l.whl", hash = "sha256:fae0805bd514066f20309f6742f6ee7904a773eb9e6c17c45d6b1600ca65c9b5", size = 10532605 },
96
+ { url = "https://files.pythonhosted.org/packages/89/a8/a957a8812e31facffb6a26a30be0b5b4af000a6e30c7d43a22a5232a3398/ruff-0.8.1-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:b8a4f7385c2285c30f34b200ca5511fcc865f17578383db154e098150ce0a087", size = 10278243 },
97
+ { url = "https://files.pythonhosted.org/packages/a8/23/9db40fa19c453fabf94f7a35c61c58f20e8200b4734a20839515a19da790/ruff-0.8.1-py3-none-macosx_11_0_arm64.whl", hash = "sha256:cd054486da0c53e41e0086e1730eb77d1f698154f910e0cd9e0d64274979a209", size = 9917739 },
98
+ { url = "https://files.pythonhosted.org/packages/e2/a0/6ee2d949835d5701d832fc5acd05c0bfdad5e89cfdd074a171411f5ccad5/ruff-0.8.1-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2029b8c22da147c50ae577e621a5bfbc5d1fed75d86af53643d7a7aee1d23871", size = 10779153 },
99
+ { url = "https://files.pythonhosted.org/packages/7a/25/9c11dca9404ef1eb24833f780146236131a3c7941de394bc356912ef1041/ruff-0.8.1-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2666520828dee7dfc7e47ee4ea0d928f40de72056d929a7c5292d95071d881d1", size = 10304387 },
100
+ { url = "https://files.pythonhosted.org/packages/c8/b9/84c323780db1b06feae603a707d82dbbd85955c8c917738571c65d7d5aff/ruff-0.8.1-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:333c57013ef8c97a53892aa56042831c372e0bb1785ab7026187b7abd0135ad5", size = 11360351 },
101
+ { url = "https://files.pythonhosted.org/packages/6b/e1/9d4bbb2ace7aad14ded20e4674a48cda5b902aed7a1b14e6b028067060c4/ruff-0.8.1-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:288326162804f34088ac007139488dcb43de590a5ccfec3166396530b58fb89d", size = 12022879 },
102
+ { url = "https://files.pythonhosted.org/packages/75/28/752ff6120c0e7f9981bc4bc275d540c7f36db1379ba9db9142f69c88db21/ruff-0.8.1-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b12c39b9448632284561cbf4191aa1b005882acbc81900ffa9f9f471c8ff7e26", size = 11610354 },
103
+ { url = "https://files.pythonhosted.org/packages/ba/8c/967b61c2cc8ebd1df877607fbe462bc1e1220b4a30ae3352648aec8c24bd/ruff-0.8.1-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:364e6674450cbac8e998f7b30639040c99d81dfb5bbc6dfad69bc7a8f916b3d1", size = 12813976 },
104
+ { url = "https://files.pythonhosted.org/packages/7f/29/e059f945d6bd2d90213387b8c360187f2fefc989ddcee6bbf3c241329b92/ruff-0.8.1-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b22346f845fec132aa39cd29acb94451d030c10874408dbf776af3aaeb53284c", size = 11154564 },
105
+ { url = "https://files.pythonhosted.org/packages/55/47/cbd05e5a62f3fb4c072bc65c1e8fd709924cad1c7ec60a1000d1e4ee8307/ruff-0.8.1-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:b2f2f7a7e7648a2bfe6ead4e0a16745db956da0e3a231ad443d2a66a105c04fa", size = 10760604 },
106
+ { url = "https://files.pythonhosted.org/packages/bb/ee/4c3981c47147c72647a198a94202633130cfda0fc95cd863a553b6f65c6a/ruff-0.8.1-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:adf314fc458374c25c5c4a4a9270c3e8a6a807b1bec018cfa2813d6546215540", size = 10391071 },
107
+ { url = "https://files.pythonhosted.org/packages/6b/e6/083eb61300214590b188616a8ac6ae1ef5730a0974240fb4bec9c17de78b/ruff-0.8.1-py3-none-musllinux_1_2_i686.whl", hash = "sha256:a885d68342a231b5ba4d30b8c6e1b1ee3a65cf37e3d29b3c74069cdf1ee1e3c9", size = 10896657 },
108
+ { url = "https://files.pythonhosted.org/packages/77/bd/aacdb8285d10f1b943dbeb818968efca35459afc29f66ae3bd4596fbf954/ruff-0.8.1-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:d2c16e3508c8cc73e96aa5127d0df8913d2290098f776416a4b157657bee44c5", size = 11228362 },
109
+ { url = "https://files.pythonhosted.org/packages/39/72/fcb7ad41947f38b4eaa702aca0a361af0e9c2bf671d7fd964480670c297e/ruff-0.8.1-py3-none-win32.whl", hash = "sha256:93335cd7c0eaedb44882d75a7acb7df4b77cd7cd0d2255c93b28791716e81790", size = 8803476 },
110
+ { url = "https://files.pythonhosted.org/packages/e4/ea/cae9aeb0f4822c44651c8407baacdb2e5b4dcd7b31a84e1c5df33aa2cc20/ruff-0.8.1-py3-none-win_amd64.whl", hash = "sha256:2954cdbe8dfd8ab359d4a30cd971b589d335a44d444b6ca2cb3d1da21b75e4b6", size = 9614463 },
111
+ { url = "https://files.pythonhosted.org/packages/eb/76/fbb4bd23dfb48fa7758d35b744413b650a9fd2ddd93bca77e30376864414/ruff-0.8.1-py3-none-win_arm64.whl", hash = "sha256:55873cc1a473e5ac129d15eccb3c008c096b94809d693fc7053f588b67822737", size = 8959621 },
112
+ ]
113
+
114
+ [[package]]
115
+ name = "termcolor"
116
+ version = "2.5.0"
117
+ source = { registry = "https://pypi.org/simple" }
118
+ sdist = { url = "https://files.pythonhosted.org/packages/37/72/88311445fd44c455c7d553e61f95412cf89054308a1aa2434ab835075fc5/termcolor-2.5.0.tar.gz", hash = "sha256:998d8d27da6d48442e8e1f016119076b690d962507531df4890fcd2db2ef8a6f", size = 13057 }
119
+ wheels = [
120
+ { url = "https://files.pythonhosted.org/packages/7f/be/df630c387a0a054815d60be6a97eb4e8f17385d5d6fe660e1c02750062b4/termcolor-2.5.0-py3-none-any.whl", hash = "sha256:37b17b5fc1e604945c2642c872a3764b5d547a48009871aea3edd3afa180afb8", size = 7755 },
121
+ ]