effectual 0.1.0__tar.gz
Sign up to get free protection for your applications and to get access to all the features.
- effectual-0.1.0/.gitignore +158 -0
- effectual-0.1.0/.python-version +1 -0
- effectual-0.1.0/LICENSE +373 -0
- effectual-0.1.0/PKG-INFO +19 -0
- effectual-0.1.0/README.md +0 -0
- effectual-0.1.0/pyproject.toml +38 -0
- effectual-0.1.0/src/effectual/__init__.py +23 -0
- effectual-0.1.0/src/effectual/build.py +162 -0
- effectual-0.1.0/src/effectual/colors.py +63 -0
- effectual-0.1.0/src/effectual/config.py +27 -0
- effectual-0.1.0/src/effectual/dev.py +60 -0
- effectual-0.1.0/src/effectual/fileHash.py +32 -0
- effectual-0.1.0/src/effectual/minifier.py +51 -0
- effectual-0.1.0/uv.lock +121 -0
@@ -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
|
effectual-0.1.0/LICENSE
ADDED
@@ -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.
|
effectual-0.1.0/PKG-INFO
ADDED
@@ -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
|
effectual-0.1.0/uv.lock
ADDED
@@ -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
|
+
]
|