├── .gitignore
├── .pylintrc
├── LICENSE
├── README.md
├── config.yml.example
├── op_tracker
├── __init__.py
├── __main__.py
├── common
│ ├── __init__.py
│ ├── api_client
│ │ ├── __init__.py
│ │ └── common_scraper.py
│ └── database
│ │ ├── __init__.py
│ │ ├── database.py
│ │ ├── helpers.py
│ │ └── models
│ │ ├── __init__.py
│ │ └── update.py
├── data
│ ├── latest.yml
│ └── official
│ │ ├── cn
│ │ └── cn.yml
│ │ ├── global
│ │ └── global.yml
│ │ ├── in
│ │ └── in.yml
│ │ ├── regions.yml
│ │ ├── uk
│ │ └── uk.yml
│ │ └── us
│ │ └── us.yml
├── official
│ ├── __init__.py
│ ├── api_client
│ │ ├── __init__.py
│ │ └── api_client.py
│ └── models
│ │ ├── __init__.py
│ │ └── device.py
├── tracker_official.py
└── utils
│ ├── __init__.py
│ ├── data_manager.py
│ ├── git.py
│ ├── helpers.py
│ ├── merger.py
│ └── telegram.py
├── poetry.lock
└── pyproject.toml
/.gitignore:
--------------------------------------------------------------------------------
1 | # Byte-compiled / optimized / DLL files
2 | __pycache__/
3 | *.py[cod]
4 | *$py.class
5 |
6 | # C extensions
7 | *.so
8 |
9 | # Distribution / packaging
10 | .Python
11 | build/
12 | develop-eggs/
13 | dist/
14 | downloads/
15 | eggs/
16 | .eggs/
17 | lib/
18 | lib64/
19 | parts/
20 | sdist/
21 | var/
22 | wheels/
23 | pip-wheel-metadata/
24 | share/python-wheels/
25 | *.egg-info/
26 | .installed.cfg
27 | *.egg
28 | MANIFEST
29 |
30 | # PyInstaller
31 | # Usually these files are written by a python script from a template
32 | # before PyInstaller builds the exe, so as to inject date/other infos into it.
33 | *.manifest
34 | *.spec
35 |
36 | # Installer logs
37 | pip-log.txt
38 | pip-delete-this-directory.txt
39 |
40 | # Unit test / coverage reports
41 | htmlcov/
42 | .tox/
43 | .nox/
44 | .coverage
45 | .coverage.*
46 | .cache
47 | nosetests.xml
48 | coverage.xml
49 | *.cover
50 | *.py,cover
51 | .hypothesis/
52 | .pytest_cache/
53 |
54 | # Translations
55 | *.mo
56 | *.pot
57 |
58 | # Django stuff:
59 | *.log
60 | local_settings.py
61 | db.sqlite3
62 | db.sqlite3-journal
63 |
64 | # Flask stuff:
65 | instance/
66 | .webassets-cache
67 |
68 | # Scrapy stuff:
69 | .scrapy
70 |
71 | # Sphinx documentation
72 | docs/_build/
73 |
74 | # PyBuilder
75 | target/
76 |
77 | # Jupyter Notebook
78 | .ipynb_checkpoints
79 |
80 | # IPython
81 | profile_default/
82 | ipython_config.py
83 |
84 | # pyenv
85 | .python-version
86 |
87 | # pipenv
88 | # According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
89 | # However, in case of collaboration, if having platform-specific dependencies or dependencies
90 | # having no cross-platform support, pipenv may install dependencies that don't work, or not
91 | # install all needed dependencies.
92 | #Pipfile.lock
93 |
94 | # PEP 582; used by e.g. github.com/David-OConnor/pyflow
95 | __pypackages__/
96 |
97 | # Celery stuff
98 | celerybeat-schedule
99 | celerybeat.pid
100 |
101 | # SageMath parsed files
102 | *.sage.py
103 |
104 | # Environments
105 | .env
106 | .venv
107 | env/
108 | venv/
109 | ENV/
110 | env.bak/
111 | venv.bak/
112 |
113 | # Spyder project settings
114 | .spyderproject
115 | .spyproject
116 |
117 | # Rope project settings
118 | .ropeproject
119 |
120 | # mkdocs documentation
121 | /site
122 |
123 | # mypy
124 | .mypy_cache/
125 | .dmypy.json
126 | dmypy.json
127 |
128 | # Pyre type checker
129 | .pyre/
130 |
131 |
132 | *log*
133 | *tmp*
134 | *__pycache__*
135 | config.*
136 | .idea/
137 | *.bak
138 | *_updater.py
139 | *.db
140 | *-wal
141 | *-shm
142 | updater/
143 |
--------------------------------------------------------------------------------
/.pylintrc:
--------------------------------------------------------------------------------
1 | [LOGGING]
2 | disable=logging-fstring-interpolation
3 |
4 | [CODE]
5 | disable=too-many-locals, too-many-instance-attributes, too-few-public-methods
6 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | GNU GENERAL PUBLIC LICENSE
2 | Version 3, 29 June 2007
3 |
4 | Copyright (C) 2007 Free Software Foundation, Inc.
5 | Everyone is permitted to copy and distribute verbatim copies
6 | of this license document, but changing it is not allowed.
7 |
8 | Preamble
9 |
10 | The GNU General Public License is a free, copyleft license for
11 | software and other kinds of works.
12 |
13 | The licenses for most software and other practical works are designed
14 | to take away your freedom to share and change the works. By contrast,
15 | the GNU General Public License is intended to guarantee your freedom to
16 | share and change all versions of a program--to make sure it remains free
17 | software for all its users. We, the Free Software Foundation, use the
18 | GNU General Public License for most of our software; it applies also to
19 | any other work released this way by its authors. You can apply it to
20 | your programs, too.
21 |
22 | When we speak of free software, we are referring to freedom, not
23 | price. Our General Public Licenses are designed to make sure that you
24 | have the freedom to distribute copies of free software (and charge for
25 | them if you wish), that you receive source code or can get it if you
26 | want it, that you can change the software or use pieces of it in new
27 | free programs, and that you know you can do these things.
28 |
29 | To protect your rights, we need to prevent others from denying you
30 | these rights or asking you to surrender the rights. Therefore, you have
31 | certain responsibilities if you distribute copies of the software, or if
32 | you modify it: responsibilities to respect the freedom of others.
33 |
34 | For example, if you distribute copies of such a program, whether
35 | gratis or for a fee, you must pass on to the recipients the same
36 | freedoms that you received. You must make sure that they, too, receive
37 | or can get the source code. And you must show them these terms so they
38 | know their rights.
39 |
40 | Developers that use the GNU GPL protect your rights with two steps:
41 | (1) assert copyright on the software, and (2) offer you this License
42 | giving you legal permission to copy, distribute and/or modify it.
43 |
44 | For the developers' and authors' protection, the GPL clearly explains
45 | that there is no warranty for this free software. For both users' and
46 | authors' sake, the GPL requires that modified versions be marked as
47 | changed, so that their problems will not be attributed erroneously to
48 | authors of previous versions.
49 |
50 | Some devices are designed to deny users access to install or run
51 | modified versions of the software inside them, although the manufacturer
52 | can do so. This is fundamentally incompatible with the aim of
53 | protecting users' freedom to change the software. The systematic
54 | pattern of such abuse occurs in the area of products for individuals to
55 | use, which is precisely where it is most unacceptable. Therefore, we
56 | have designed this version of the GPL to prohibit the practice for those
57 | products. If such problems arise substantially in other domains, we
58 | stand ready to extend this provision to those domains in future versions
59 | of the GPL, as needed to protect the freedom of users.
60 |
61 | Finally, every program is threatened constantly by software patents.
62 | States should not allow patents to restrict development and use of
63 | software on general-purpose computers, but in those that do, we wish to
64 | avoid the special danger that patents applied to a free program could
65 | make it effectively proprietary. To prevent this, the GPL assures that
66 | patents cannot be used to render the program non-free.
67 |
68 | The precise terms and conditions for copying, distribution and
69 | modification follow.
70 |
71 | TERMS AND CONDITIONS
72 |
73 | 0. Definitions.
74 |
75 | "This License" refers to version 3 of the GNU General Public License.
76 |
77 | "Copyright" also means copyright-like laws that apply to other kinds of
78 | works, such as semiconductor masks.
79 |
80 | "The Program" refers to any copyrightable work licensed under this
81 | License. Each licensee is addressed as "you". "Licensees" and
82 | "recipients" may be individuals or organizations.
83 |
84 | To "modify" a work means to copy from or adapt all or part of the work
85 | in a fashion requiring copyright permission, other than the making of an
86 | exact copy. The resulting work is called a "modified version" of the
87 | earlier work or a work "based on" the earlier work.
88 |
89 | A "covered work" means either the unmodified Program or a work based
90 | on the Program.
91 |
92 | To "propagate" a work means to do anything with it that, without
93 | permission, would make you directly or secondarily liable for
94 | infringement under applicable copyright law, except executing it on a
95 | computer or modifying a private copy. Propagation includes copying,
96 | distribution (with or without modification), making available to the
97 | public, and in some countries other activities as well.
98 |
99 | To "convey" a work means any kind of propagation that enables other
100 | parties to make or receive copies. Mere interaction with a user through
101 | a computer network, with no transfer of a copy, is not conveying.
102 |
103 | An interactive user interface displays "Appropriate Legal Notices"
104 | to the extent that it includes a convenient and prominently visible
105 | feature that (1) displays an appropriate copyright notice, and (2)
106 | tells the user that there is no warranty for the work (except to the
107 | extent that warranties are provided), that licensees may convey the
108 | work under this License, and how to view a copy of this License. If
109 | the interface presents a list of user commands or options, such as a
110 | menu, a prominent item in the list meets this criterion.
111 |
112 | 1. Source Code.
113 |
114 | The "source code" for a work means the preferred form of the work
115 | for making modifications to it. "Object code" means any non-source
116 | form of a work.
117 |
118 | A "Standard Interface" means an interface that either is an official
119 | standard defined by a recognized standards body, or, in the case of
120 | interfaces specified for a particular programming language, one that
121 | is widely used among developers working in that language.
122 |
123 | The "System Libraries" of an executable work include anything, other
124 | than the work as a whole, that (a) is included in the normal form of
125 | packaging a Major Component, but which is not part of that Major
126 | Component, and (b) serves only to enable use of the work with that
127 | Major Component, or to implement a Standard Interface for which an
128 | implementation is available to the public in source code form. A
129 | "Major Component", in this context, means a major essential component
130 | (kernel, window system, and so on) of the specific operating system
131 | (if any) on which the executable work runs, or a compiler used to
132 | produce the work, or an object code interpreter used to run it.
133 |
134 | The "Corresponding Source" for a work in object code form means all
135 | the source code needed to generate, install, and (for an executable
136 | work) run the object code and to modify the work, including scripts to
137 | control those activities. However, it does not include the work's
138 | System Libraries, or general-purpose tools or generally available free
139 | programs which are used unmodified in performing those activities but
140 | which are not part of the work. For example, Corresponding Source
141 | includes interface definition files associated with source files for
142 | the work, and the source code for shared libraries and dynamically
143 | linked subprograms that the work is specifically designed to require,
144 | such as by intimate data communication or control flow between those
145 | subprograms and other parts of the work.
146 |
147 | The Corresponding Source need not include anything that users
148 | can regenerate automatically from other parts of the Corresponding
149 | Source.
150 |
151 | The Corresponding Source for a work in source code form is that
152 | same work.
153 |
154 | 2. Basic Permissions.
155 |
156 | All rights granted under this License are granted for the term of
157 | copyright on the Program, and are irrevocable provided the stated
158 | conditions are met. This License explicitly affirms your unlimited
159 | permission to run the unmodified Program. The output from running a
160 | covered work is covered by this License only if the output, given its
161 | content, constitutes a covered work. This License acknowledges your
162 | rights of fair use or other equivalent, as provided by copyright law.
163 |
164 | You may make, run and propagate covered works that you do not
165 | convey, without conditions so long as your license otherwise remains
166 | in force. You may convey covered works to others for the sole purpose
167 | of having them make modifications exclusively for you, or provide you
168 | with facilities for running those works, provided that you comply with
169 | the terms of this License in conveying all material for which you do
170 | not control copyright. Those thus making or running the covered works
171 | for you must do so exclusively on your behalf, under your direction
172 | and control, on terms that prohibit them from making any copies of
173 | your copyrighted material outside their relationship with you.
174 |
175 | Conveying under any other circumstances is permitted solely under
176 | the conditions stated below. Sublicensing is not allowed; section 10
177 | makes it unnecessary.
178 |
179 | 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
180 |
181 | No covered work shall be deemed part of an effective technological
182 | measure under any applicable law fulfilling obligations under article
183 | 11 of the WIPO copyright treaty adopted on 20 December 1996, or
184 | similar laws prohibiting or restricting circumvention of such
185 | measures.
186 |
187 | When you convey a covered work, you waive any legal power to forbid
188 | circumvention of technological measures to the extent such circumvention
189 | is effected by exercising rights under this License with respect to
190 | the covered work, and you disclaim any intention to limit operation or
191 | modification of the work as a means of enforcing, against the work's
192 | users, your or third parties' legal rights to forbid circumvention of
193 | technological measures.
194 |
195 | 4. Conveying Verbatim Copies.
196 |
197 | You may convey verbatim copies of the Program's source code as you
198 | receive it, in any medium, provided that you conspicuously and
199 | appropriately publish on each copy an appropriate copyright notice;
200 | keep intact all notices stating that this License and any
201 | non-permissive terms added in accord with section 7 apply to the code;
202 | keep intact all notices of the absence of any warranty; and give all
203 | recipients a copy of this License along with the Program.
204 |
205 | You may charge any price or no price for each copy that you convey,
206 | and you may offer support or warranty protection for a fee.
207 |
208 | 5. Conveying Modified Source Versions.
209 |
210 | You may convey a work based on the Program, or the modifications to
211 | produce it from the Program, in the form of source code under the
212 | terms of section 4, provided that you also meet all of these conditions:
213 |
214 | a) The work must carry prominent notices stating that you modified
215 | it, and giving a relevant date.
216 |
217 | b) The work must carry prominent notices stating that it is
218 | released under this License and any conditions added under section
219 | 7. This requirement modifies the requirement in section 4 to
220 | "keep intact all notices".
221 |
222 | c) You must license the entire work, as a whole, under this
223 | License to anyone who comes into possession of a copy. This
224 | License will therefore apply, along with any applicable section 7
225 | additional terms, to the whole of the work, and all its parts,
226 | regardless of how they are packaged. This License gives no
227 | permission to license the work in any other way, but it does not
228 | invalidate such permission if you have separately received it.
229 |
230 | d) If the work has interactive user interfaces, each must display
231 | Appropriate Legal Notices; however, if the Program has interactive
232 | interfaces that do not display Appropriate Legal Notices, your
233 | work need not make them do so.
234 |
235 | A compilation of a covered work with other separate and independent
236 | works, which are not by their nature extensions of the covered work,
237 | and which are not combined with it such as to form a larger program,
238 | in or on a volume of a storage or distribution medium, is called an
239 | "aggregate" if the compilation and its resulting copyright are not
240 | used to limit the access or legal rights of the compilation's users
241 | beyond what the individual works permit. Inclusion of a covered work
242 | in an aggregate does not cause this License to apply to the other
243 | parts of the aggregate.
244 |
245 | 6. Conveying Non-Source Forms.
246 |
247 | You may convey a covered work in object code form under the terms
248 | of sections 4 and 5, provided that you also convey the
249 | machine-readable Corresponding Source under the terms of this License,
250 | in one of these ways:
251 |
252 | a) Convey the object code in, or embodied in, a physical product
253 | (including a physical distribution medium), accompanied by the
254 | Corresponding Source fixed on a durable physical medium
255 | customarily used for software interchange.
256 |
257 | b) Convey the object code in, or embodied in, a physical product
258 | (including a physical distribution medium), accompanied by a
259 | written offer, valid for at least three years and valid for as
260 | long as you offer spare parts or customer support for that product
261 | model, to give anyone who possesses the object code either (1) a
262 | copy of the Corresponding Source for all the software in the
263 | product that is covered by this License, on a durable physical
264 | medium customarily used for software interchange, for a price no
265 | more than your reasonable cost of physically performing this
266 | conveying of source, or (2) access to copy the
267 | Corresponding Source from a network server at no charge.
268 |
269 | c) Convey individual copies of the object code with a copy of the
270 | written offer to provide the Corresponding Source. This
271 | alternative is allowed only occasionally and noncommercially, and
272 | only if you received the object code with such an offer, in accord
273 | with subsection 6b.
274 |
275 | d) Convey the object code by offering access from a designated
276 | place (gratis or for a charge), and offer equivalent access to the
277 | Corresponding Source in the same way through the same place at no
278 | further charge. You need not require recipients to copy the
279 | Corresponding Source along with the object code. If the place to
280 | copy the object code is a network server, the Corresponding Source
281 | may be on a different server (operated by you or a third party)
282 | that supports equivalent copying facilities, provided you maintain
283 | clear directions next to the object code saying where to find the
284 | Corresponding Source. Regardless of what server hosts the
285 | Corresponding Source, you remain obligated to ensure that it is
286 | available for as long as needed to satisfy these requirements.
287 |
288 | e) Convey the object code using peer-to-peer transmission, provided
289 | you inform other peers where the object code and Corresponding
290 | Source of the work are being offered to the general public at no
291 | charge under subsection 6d.
292 |
293 | A separable portion of the object code, whose source code is excluded
294 | from the Corresponding Source as a System Library, need not be
295 | included in conveying the object code work.
296 |
297 | A "User Product" is either (1) a "consumer product", which means any
298 | tangible personal property which is normally used for personal, family,
299 | or household purposes, or (2) anything designed or sold for incorporation
300 | into a dwelling. In determining whether a product is a consumer product,
301 | doubtful cases shall be resolved in favor of coverage. For a particular
302 | product received by a particular user, "normally used" refers to a
303 | typical or common use of that class of product, regardless of the status
304 | of the particular user or of the way in which the particular user
305 | actually uses, or expects or is expected to use, the product. A product
306 | is a consumer product regardless of whether the product has substantial
307 | commercial, industrial or non-consumer uses, unless such uses represent
308 | the only significant mode of use of the product.
309 |
310 | "Installation Information" for a User Product means any methods,
311 | procedures, authorization keys, or other information required to install
312 | and execute modified versions of a covered work in that User Product from
313 | a modified version of its Corresponding Source. The information must
314 | suffice to ensure that the continued functioning of the modified object
315 | code is in no case prevented or interfered with solely because
316 | modification has been made.
317 |
318 | If you convey an object code work under this section in, or with, or
319 | specifically for use in, a User Product, and the conveying occurs as
320 | part of a transaction in which the right of possession and use of the
321 | User Product is transferred to the recipient in perpetuity or for a
322 | fixed term (regardless of how the transaction is characterized), the
323 | Corresponding Source conveyed under this section must be accompanied
324 | by the Installation Information. But this requirement does not apply
325 | if neither you nor any third party retains the ability to install
326 | modified object code on the User Product (for example, the work has
327 | been installed in ROM).
328 |
329 | The requirement to provide Installation Information does not include a
330 | requirement to continue to provide support service, warranty, or updates
331 | for a work that has been modified or installed by the recipient, or for
332 | the User Product in which it has been modified or installed. Access to a
333 | network may be denied when the modification itself materially and
334 | adversely affects the operation of the network or violates the rules and
335 | protocols for communication across the network.
336 |
337 | Corresponding Source conveyed, and Installation Information provided,
338 | in accord with this section must be in a format that is publicly
339 | documented (and with an implementation available to the public in
340 | source code form), and must require no special password or key for
341 | unpacking, reading or copying.
342 |
343 | 7. Additional Terms.
344 |
345 | "Additional permissions" are terms that supplement the terms of this
346 | License by making exceptions from one or more of its conditions.
347 | Additional permissions that are applicable to the entire Program shall
348 | be treated as though they were included in this License, to the extent
349 | that they are valid under applicable law. If additional permissions
350 | apply only to part of the Program, that part may be used separately
351 | under those permissions, but the entire Program remains governed by
352 | this License without regard to the additional permissions.
353 |
354 | When you convey a copy of a covered work, you may at your option
355 | remove any additional permissions from that copy, or from any part of
356 | it. (Additional permissions may be written to require their own
357 | removal in certain cases when you modify the work.) You may place
358 | additional permissions on material, added by you to a covered work,
359 | for which you have or can give appropriate copyright permission.
360 |
361 | Notwithstanding any other provision of this License, for material you
362 | add to a covered work, you may (if authorized by the copyright holders of
363 | that material) supplement the terms of this License with terms:
364 |
365 | a) Disclaiming warranty or limiting liability differently from the
366 | terms of sections 15 and 16 of this License; or
367 |
368 | b) Requiring preservation of specified reasonable legal notices or
369 | author attributions in that material or in the Appropriate Legal
370 | Notices displayed by works containing it; or
371 |
372 | c) Prohibiting misrepresentation of the origin of that material, or
373 | requiring that modified versions of such material be marked in
374 | reasonable ways as different from the original version; or
375 |
376 | d) Limiting the use for publicity purposes of names of licensors or
377 | authors of the material; or
378 |
379 | e) Declining to grant rights under trademark law for use of some
380 | trade names, trademarks, or service marks; or
381 |
382 | f) Requiring indemnification of licensors and authors of that
383 | material by anyone who conveys the material (or modified versions of
384 | it) with contractual assumptions of liability to the recipient, for
385 | any liability that these contractual assumptions directly impose on
386 | those licensors and authors.
387 |
388 | All other non-permissive additional terms are considered "further
389 | restrictions" within the meaning of section 10. If the Program as you
390 | received it, or any part of it, contains a notice stating that it is
391 | governed by this License along with a term that is a further
392 | restriction, you may remove that term. If a license document contains
393 | a further restriction but permits relicensing or conveying under this
394 | License, you may add to a covered work material governed by the terms
395 | of that license document, provided that the further restriction does
396 | not survive such relicensing or conveying.
397 |
398 | If you add terms to a covered work in accord with this section, you
399 | must place, in the relevant source files, a statement of the
400 | additional terms that apply to those files, or a notice indicating
401 | where to find the applicable terms.
402 |
403 | Additional terms, permissive or non-permissive, may be stated in the
404 | form of a separately written license, or stated as exceptions;
405 | the above requirements apply either way.
406 |
407 | 8. Termination.
408 |
409 | You may not propagate or modify a covered work except as expressly
410 | provided under this License. Any attempt otherwise to propagate or
411 | modify it is void, and will automatically terminate your rights under
412 | this License (including any patent licenses granted under the third
413 | paragraph of section 11).
414 |
415 | However, if you cease all violation of this License, then your
416 | license from a particular copyright holder is reinstated (a)
417 | provisionally, unless and until the copyright holder explicitly and
418 | finally terminates your license, and (b) permanently, if the copyright
419 | holder fails to notify you of the violation by some reasonable means
420 | prior to 60 days after the cessation.
421 |
422 | Moreover, your license from a particular copyright holder is
423 | reinstated permanently if the copyright holder notifies you of the
424 | violation by some reasonable means, this is the first time you have
425 | received notice of violation of this License (for any work) from that
426 | copyright holder, and you cure the violation prior to 30 days after
427 | your receipt of the notice.
428 |
429 | Termination of your rights under this section does not terminate the
430 | licenses of parties who have received copies or rights from you under
431 | this License. If your rights have been terminated and not permanently
432 | reinstated, you do not qualify to receive new licenses for the same
433 | material under section 10.
434 |
435 | 9. Acceptance Not Required for Having Copies.
436 |
437 | You are not required to accept this License in order to receive or
438 | run a copy of the Program. Ancillary propagation of a covered work
439 | occurring solely as a consequence of using peer-to-peer transmission
440 | to receive a copy likewise does not require acceptance. However,
441 | nothing other than this License grants you permission to propagate or
442 | modify any covered work. These actions infringe copyright if you do
443 | not accept this License. Therefore, by modifying or propagating a
444 | covered work, you indicate your acceptance of this License to do so.
445 |
446 | 10. Automatic Licensing of Downstream Recipients.
447 |
448 | Each time you convey a covered work, the recipient automatically
449 | receives a license from the original licensors, to run, modify and
450 | propagate that work, subject to this License. You are not responsible
451 | for enforcing compliance by third parties with this License.
452 |
453 | An "entity transaction" is a transaction transferring control of an
454 | organization, or substantially all assets of one, or subdividing an
455 | organization, or merging organizations. If propagation of a covered
456 | work results from an entity transaction, each party to that
457 | transaction who receives a copy of the work also receives whatever
458 | licenses to the work the party's predecessor in interest had or could
459 | give under the previous paragraph, plus a right to possession of the
460 | Corresponding Source of the work from the predecessor in interest, if
461 | the predecessor has it or can get it with reasonable efforts.
462 |
463 | You may not impose any further restrictions on the exercise of the
464 | rights granted or affirmed under this License. For example, you may
465 | not impose a license fee, royalty, or other charge for exercise of
466 | rights granted under this License, and you may not initiate litigation
467 | (including a cross-claim or counterclaim in a lawsuit) alleging that
468 | any patent claim is infringed by making, using, selling, offering for
469 | sale, or importing the Program or any portion of it.
470 |
471 | 11. Patents.
472 |
473 | A "contributor" is a copyright holder who authorizes use under this
474 | License of the Program or a work on which the Program is based. The
475 | work thus licensed is called the contributor's "contributor version".
476 |
477 | A contributor's "essential patent claims" are all patent claims
478 | owned or controlled by the contributor, whether already acquired or
479 | hereafter acquired, that would be infringed by some manner, permitted
480 | by this License, of making, using, or selling its contributor version,
481 | but do not include claims that would be infringed only as a
482 | consequence of further modification of the contributor version. For
483 | purposes of this definition, "control" includes the right to grant
484 | patent sublicenses in a manner consistent with the requirements of
485 | this License.
486 |
487 | Each contributor grants you a non-exclusive, worldwide, royalty-free
488 | patent license under the contributor's essential patent claims, to
489 | make, use, sell, offer for sale, import and otherwise run, modify and
490 | propagate the contents of its contributor version.
491 |
492 | In the following three paragraphs, a "patent license" is any express
493 | agreement or commitment, however denominated, not to enforce a patent
494 | (such as an express permission to practice a patent or covenant not to
495 | sue for patent infringement). To "grant" such a patent license to a
496 | party means to make such an agreement or commitment not to enforce a
497 | patent against the party.
498 |
499 | If you convey a covered work, knowingly relying on a patent license,
500 | and the Corresponding Source of the work is not available for anyone
501 | to copy, free of charge and under the terms of this License, through a
502 | publicly available network server or other readily accessible means,
503 | then you must either (1) cause the Corresponding Source to be so
504 | available, or (2) arrange to deprive yourself of the benefit of the
505 | patent license for this particular work, or (3) arrange, in a manner
506 | consistent with the requirements of this License, to extend the patent
507 | license to downstream recipients. "Knowingly relying" means you have
508 | actual knowledge that, but for the patent license, your conveying the
509 | covered work in a country, or your recipient's use of the covered work
510 | in a country, would infringe one or more identifiable patents in that
511 | country that you have reason to believe are valid.
512 |
513 | If, pursuant to or in connection with a single transaction or
514 | arrangement, you convey, or propagate by procuring conveyance of, a
515 | covered work, and grant a patent license to some of the parties
516 | receiving the covered work authorizing them to use, propagate, modify
517 | or convey a specific copy of the covered work, then the patent license
518 | you grant is automatically extended to all recipients of the covered
519 | work and works based on it.
520 |
521 | A patent license is "discriminatory" if it does not include within
522 | the scope of its coverage, prohibits the exercise of, or is
523 | conditioned on the non-exercise of one or more of the rights that are
524 | specifically granted under this License. You may not convey a covered
525 | work if you are a party to an arrangement with a third party that is
526 | in the business of distributing software, under which you make payment
527 | to the third party based on the extent of your activity of conveying
528 | the work, and under which the third party grants, to any of the
529 | parties who would receive the covered work from you, a discriminatory
530 | patent license (a) in connection with copies of the covered work
531 | conveyed by you (or copies made from those copies), or (b) primarily
532 | for and in connection with specific products or compilations that
533 | contain the covered work, unless you entered into that arrangement,
534 | or that patent license was granted, prior to 28 March 2007.
535 |
536 | Nothing in this License shall be construed as excluding or limiting
537 | any implied license or other defenses to infringement that may
538 | otherwise be available to you under applicable patent law.
539 |
540 | 12. No Surrender of Others' Freedom.
541 |
542 | If conditions are imposed on you (whether by court order, agreement or
543 | otherwise) that contradict the conditions of this License, they do not
544 | excuse you from the conditions of this License. If you cannot convey a
545 | covered work so as to satisfy simultaneously your obligations under this
546 | License and any other pertinent obligations, then as a consequence you may
547 | not convey it at all. For example, if you agree to terms that obligate you
548 | to collect a royalty for further conveying from those to whom you convey
549 | the Program, the only way you could satisfy both those terms and this
550 | License would be to refrain entirely from conveying the Program.
551 |
552 | 13. Use with the GNU Affero General Public License.
553 |
554 | Notwithstanding any other provision of this License, you have
555 | permission to link or combine any covered work with a work licensed
556 | under version 3 of the GNU Affero General Public License into a single
557 | combined work, and to convey the resulting work. The terms of this
558 | License will continue to apply to the part which is the covered work,
559 | but the special requirements of the GNU Affero General Public License,
560 | section 13, concerning interaction through a network will apply to the
561 | combination as such.
562 |
563 | 14. Revised Versions of this License.
564 |
565 | The Free Software Foundation may publish revised and/or new versions of
566 | the GNU General Public License from time to time. Such new versions will
567 | be similar in spirit to the present version, but may differ in detail to
568 | address new problems or concerns.
569 |
570 | Each version is given a distinguishing version number. If the
571 | Program specifies that a certain numbered version of the GNU General
572 | Public License "or any later version" applies to it, you have the
573 | option of following the terms and conditions either of that numbered
574 | version or of any later version published by the Free Software
575 | Foundation. If the Program does not specify a version number of the
576 | GNU General Public License, you may choose any version ever published
577 | by the Free Software Foundation.
578 |
579 | If the Program specifies that a proxy can decide which future
580 | versions of the GNU General Public License can be used, that proxy's
581 | public statement of acceptance of a version permanently authorizes you
582 | to choose that version for the Program.
583 |
584 | Later license versions may give you additional or different
585 | permissions. However, no additional obligations are imposed on any
586 | author or copyright holder as a result of your choosing to follow a
587 | later version.
588 |
589 | 15. Disclaimer of Warranty.
590 |
591 | THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
592 | APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
593 | HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
594 | OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
595 | THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
596 | PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
597 | IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
598 | ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
599 |
600 | 16. Limitation of Liability.
601 |
602 | IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
603 | WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
604 | THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
605 | GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
606 | USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
607 | DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
608 | PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
609 | EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
610 | SUCH DAMAGES.
611 |
612 | 17. Interpretation of Sections 15 and 16.
613 |
614 | If the disclaimer of warranty and limitation of liability provided
615 | above cannot be given local legal effect according to their terms,
616 | reviewing courts shall apply local law that most closely approximates
617 | an absolute waiver of all civil liability in connection with the
618 | Program, unless a warranty or assumption of liability accompanies a
619 | copy of the Program in return for a fee.
620 |
621 | END OF TERMS AND CONDITIONS
622 |
623 | How to Apply These Terms to Your New Programs
624 |
625 | If you develop a new program, and you want it to be of the greatest
626 | possible use to the public, the best way to achieve this is to make it
627 | free software which everyone can redistribute and change under these terms.
628 |
629 | To do so, attach the following notices to the program. It is safest
630 | to attach them to the start of each source file to most effectively
631 | state the exclusion of warranty; and each file should have at least
632 | the "copyright" line and a pointer to where the full notice is found.
633 |
634 |
635 | Copyright (C)
636 |
637 | This program is free software: you can redistribute it and/or modify
638 | it under the terms of the GNU General Public License as published by
639 | the Free Software Foundation, either version 3 of the License, or
640 | (at your option) any later version.
641 |
642 | This program is distributed in the hope that it will be useful,
643 | but WITHOUT ANY WARRANTY; without even the implied warranty of
644 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
645 | GNU General Public License for more details.
646 |
647 | You should have received a copy of the GNU General Public License
648 | along with this program. If not, see .
649 |
650 | Also add information on how to contact you by electronic and paper mail.
651 |
652 | If the program does terminal interaction, make it output a short
653 | notice like this when it starts in an interactive mode:
654 |
655 | Copyright (C)
656 | This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
657 | This is free software, and you are welcome to redistribute it
658 | under certain conditions; type `show c' for details.
659 |
660 | The hypothetical commands `show w' and `show c' should show the appropriate
661 | parts of the General Public License. Of course, your program's commands
662 | might be different; for a GUI interface, you would use an "about box".
663 |
664 | You should also get your employer (if you work as a programmer) or school,
665 | if any, to sign a "copyright disclaimer" for the program, if necessary.
666 | For more information on this, and how to apply and follow the GNU GPL, see
667 | .
668 |
669 | The GNU General Public License does not permit incorporating your program
670 | into proprietary programs. If your program is a subroutine library, you
671 | may consider it more useful to permit linking proprietary applications with
672 | the library. If this is what you want to do, use the GNU Lesser General
673 | Public License instead of this License. But first, please read
674 | .
675 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # OnePlus Updates Tracker
2 |
3 | [](https://t.me/OnePlusUpdatesTracker)
4 |
5 | [](https://github.com/ellerbrock/open-source-badges/)
6 | [](https://www.python.org/)
7 |
8 | [](https://www.paypal.me/yshalsager)
9 |
10 | A Python 3 script that automatically tracks OnePlus Oxygen OS and Hydrogen OS ROMs releases and send messages of new updates to telegram channel to notify users!
11 |
12 | It currently uses OnePlus official websites as the only updates source, and supports all devices.
13 |
14 | #### Running it on your own:
15 |
16 | - You need Python 3.6 at least and pip 19 installed on your device.
17 | - Install the requirements using `pip install .`
18 | - Copy `config.yml.example` to `config.yml` and fill it with the following:
19 |
20 | ```yml
21 | tg_bot_token: # Telegram bot token, get it from @BotFather
22 | tg_chat: # Telegram chat username or ID
23 | git_oauth_token: # GitHub OAuth token
24 | source: "official"
25 | ```
26 |
--------------------------------------------------------------------------------
/config.yml.example:
--------------------------------------------------------------------------------
1 | tg_bot_token:
2 | tg_chat:
3 | git_oauth_token:
4 | source: "official"
5 | db:
--------------------------------------------------------------------------------
/op_tracker/__init__.py:
--------------------------------------------------------------------------------
1 | """OnePlus Updates Tracker initialization"""
2 | import logging
3 | from logging import Formatter
4 | from logging.handlers import TimedRotatingFileHandler
5 | from pathlib import Path
6 | from sys import stderr
7 |
8 | # from sys import stdout
9 | import yaml
10 |
11 | WORK_DIR = Path(__file__).parent
12 | CONF_DIR = Path(__file__).parent.parent
13 |
14 | # read script configuration file
15 | with open(CONF_DIR / "config.yml", "r") as f:
16 | CONFIG = yaml.load(f, Loader=yaml.FullLoader)
17 |
18 | # set logging configuration
19 | LOG_FILE = CONF_DIR / "last_run.log"
20 | LOG_FORMAT: str = (
21 | "%(asctime)s [%(levelname)s] %(name)s [%(module)s.%(funcName)s:%(lineno)d]: "
22 | "%(message)s"
23 | )
24 | FORMATTER: Formatter = logging.Formatter(LOG_FORMAT)
25 | handler = TimedRotatingFileHandler(LOG_FILE, when="d", interval=1, backupCount=2)
26 | logging.basicConfig(filename=LOG_FILE, filemode="a", format=LOG_FORMAT)
27 | # OUT = logging.StreamHandler(stdout)
28 | ERR = logging.StreamHandler(stderr)
29 | # OUT.setFormatter(FORMATTER)
30 | ERR.setFormatter(FORMATTER)
31 | # OUT.setLevel(logging.DEBUG)
32 | ERR.setLevel(logging.WARNING)
33 | LOGGER = logging.getLogger()
34 | # LOGGER.addHandler(OUT)
35 | LOGGER.addHandler(ERR)
36 | LOGGER.addHandler(handler)
37 | LOGGER.setLevel(logging.INFO)
38 |
--------------------------------------------------------------------------------
/op_tracker/__main__.py:
--------------------------------------------------------------------------------
1 | """OnePlus Updates Tracker entry point"""
2 | from importlib import import_module
3 |
4 | from op_tracker import CONFIG
5 | from op_tracker.tracker_official import run as official
6 |
7 | source = CONFIG.get("source")
8 |
9 | if source == "tracker_updater":
10 | from op_tracker.tracker_updater import run as extra_run
11 | else:
12 | try:
13 | script = import_module(f"{__package__}.{source}")
14 | extra_run = script.run()
15 | except ImportError:
16 | raise Exception("Incorrect Scraper has been specified! exiting...")
17 |
18 | if __name__ == "__main__":
19 | if extra_run:
20 | extra_run()
21 | official()
22 |
--------------------------------------------------------------------------------
/op_tracker/common/__init__.py:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/androidtrackers/oneplus-updates-tracker/2485e7670ef576380ff487c826263afd85664f34/op_tracker/common/__init__.py
--------------------------------------------------------------------------------
/op_tracker/common/api_client/__init__.py:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/androidtrackers/oneplus-updates-tracker/2485e7670ef576380ff487c826263afd85664f34/op_tracker/common/api_client/__init__.py
--------------------------------------------------------------------------------
/op_tracker/common/api_client/common_scraper.py:
--------------------------------------------------------------------------------
1 | """
2 | OnePlus Websites common API Client class implementation
3 | """
4 |
5 | from aiohttp import ClientSession
6 |
7 |
8 | class CommonClient:
9 | """
10 | Base Scraper class
11 |
12 | This class is used to scrape OnePlus websites.
13 | It's responsible for interacting with OnePlus websites API in order to:
14 | - Get devices list.
15 | - Get device's updates information
16 | :attr: `session`: ClientSession - aiohttp client session object
17 | :attr: `base_url`: str - Website base URL
18 | :attr: `devices`: list - list of devices available on the website
19 | """
20 |
21 | def __init__(self):
22 | """
23 | Website Class constructor
24 | :param region: OnePlus website region
25 | """
26 | self.session: ClientSession = ClientSession()
27 | self.base_url: str = ""
28 | self.devices: list = []
29 |
30 | async def close(self):
31 | """
32 | Closes the connection of aiohttp client
33 | :return:
34 | """
35 | await self.session.close()
36 |
--------------------------------------------------------------------------------
/op_tracker/common/database/__init__.py:
--------------------------------------------------------------------------------
1 | """OnePlus Updates Tracker Database initialization"""
2 | import logging
3 |
4 | from sqlalchemy import (Column, Integer, MetaData, String, Table,
5 | create_engine, inspect)
6 | from sqlalchemy.engine import Connection, Engine
7 | from sqlalchemy.orm import sessionmaker
8 |
9 | from op_tracker import CONFIG, WORK_DIR
10 |
11 | logger = logging.getLogger(__name__)
12 | engine: Engine = create_engine(
13 | f"sqlite:///{WORK_DIR}/{CONFIG.get('db')}.db",
14 | connect_args={"check_same_thread": False},
15 | )
16 | logger.info(f"Connected to {engine.name} database at {engine.url}")
17 | connection: Connection = engine.connect()
18 |
19 | # Create a MetaData instance
20 | metadata: MetaData = MetaData()
21 | # reflect db schema to MetaData
22 | metadata.reflect(bind=engine)
23 | # check if the table exists
24 | ins = inspect(engine)
25 | if "updates" not in ins.get_table_names():
26 | logger.info("Updates table not found, creating one")
27 | posts = Table(
28 | "updates",
29 | metadata,
30 | Column("id", Integer(), primary_key=True, autoincrement=True),
31 | Column("device", String(), nullable=False),
32 | Column("region", String(), nullable=False),
33 | Column("version", String(), nullable=False),
34 | Column("branch", String(), nullable=False),
35 | Column("type", String(), nullable=False),
36 | Column("size", String(), nullable=False),
37 | Column("md5", String(32), nullable=False, unique=True),
38 | Column("filename", String(), nullable=False, unique=True),
39 | Column("link", String(), nullable=False, unique=True),
40 | Column("date", String(), nullable=False),
41 | Column("changelog", String(), nullable=False),
42 | Column("changelog_link", String()),
43 | Column("product", String()),
44 | Column("insert_date", String()),
45 | )
46 | metadata.create_all(engine)
47 | Session: sessionmaker = sessionmaker(bind=engine)
48 | session: Session = Session()
49 | else:
50 | Session: sessionmaker = sessionmaker(bind=engine)
51 | session = Session()
52 |
--------------------------------------------------------------------------------
/op_tracker/common/database/database.py:
--------------------------------------------------------------------------------
1 | """
2 | Database related functions
3 | """
4 |
5 | from op_tracker.common.database import session
6 | from op_tracker.common.database.models.update import Update
7 |
8 |
9 | def get_devices():
10 | all_devices = (
11 | session.query(
12 | Update.device,
13 | Update.region,
14 | Update.version,
15 | Update.branch,
16 | Update.type,
17 | Update.product,
18 | )
19 | .filter(Update.type == "Full")
20 | .order_by(Update.date.desc())
21 | .distinct(Update.product)
22 | )
23 | return [
24 | i
25 | for i in all_devices
26 | if i.device
27 | not in [
28 | "OnePlus 1",
29 | "OnePlus 2",
30 | "OnePlus 3",
31 | "OnePlus 3T",
32 | "OnePlus 5",
33 | "OnePlus 5T",
34 | "OnePlus X",
35 | ]
36 | ]
37 |
38 |
39 | devices = get_devices()
40 |
41 |
42 | def get_latest() -> list:
43 | latest_updates = (
44 | session.query(Update)
45 | .filter(Update.type == "Full")
46 | .order_by(Update.date.desc())
47 | .distinct(Update.product)
48 | .all()
49 | )
50 | latest = [
51 | {
52 | "changelog": item.changelog,
53 | "device": item.device,
54 | "link": item.link,
55 | "md5": item.md5,
56 | "region": item.region,
57 | "size": item.size,
58 | "branch": item.branch,
59 | "date": item.date,
60 | "version": item.version,
61 | }
62 | for item in latest_updates
63 | ]
64 | return latest
65 |
66 |
67 | def get_incremental(version: str) -> Update:
68 | """
69 | Get incremental update information of a version
70 | :type version: str
71 | :param version: OnePlus software version
72 | """
73 | return (
74 | session.query(Update)
75 | .filter(Update.version == version)
76 | .filter(Update.type == "Incremental")
77 | .one_or_none()
78 | )
79 |
80 |
81 | def get_version(device: str, branch: str) -> str:
82 | """
83 | Get device version example
84 | :param device: device name
85 | :type device: str
86 | :param branch: Update branch
87 | """
88 | return (
89 | session.query(Update.version)
90 | .filter(Update.filename.startswith(device))
91 | .filter(Update.branch == branch)
92 | .first()
93 | )
94 |
95 |
96 | def add_to_db(update: Update):
97 | """Adds an update to the database"""
98 | session.add(update)
99 | session.commit()
100 |
101 |
102 | def already_in_db(md5):
103 | """
104 | Check if an update is already in the database
105 | :param md5: Update file md5
106 | :return: True if the update is already in the database
107 | """
108 | return bool(session.query(Update).filter_by(md5=md5).count() >= 1)
109 |
--------------------------------------------------------------------------------
/op_tracker/common/database/helpers.py:
--------------------------------------------------------------------------------
1 | """
2 | Database helper functions
3 | """
4 | from op_tracker import WORK_DIR
5 | from op_tracker.common.database.database import get_latest
6 | from op_tracker.utils.data_manager import DataManager
7 |
8 |
9 | def export_latest():
10 | """
11 | Export latest updates from the database to YAML file
12 | :return:
13 | """
14 | latest = get_latest()
15 | DataManager.write_file(f"{WORK_DIR}/data/latest.yml", latest)
16 |
--------------------------------------------------------------------------------
/op_tracker/common/database/models/__init__.py:
--------------------------------------------------------------------------------
1 | """OnePlus Updates Tracker Database models"""
2 | from sqlalchemy.ext.declarative import declarative_base
3 |
4 | Base = declarative_base()
5 |
--------------------------------------------------------------------------------
/op_tracker/common/database/models/update.py:
--------------------------------------------------------------------------------
1 | """OnePlus Updates Tracker Database Update model"""
2 | from typing import Union
3 |
4 | from sqlalchemy import Column, Integer, String
5 |
6 | from op_tracker.common.database.models import Base
7 |
8 |
9 | class Update(Base):
10 | """
11 | Update class that represents a device update
12 | """
13 |
14 | __tablename__ = "updates"
15 | id: int = Column(Integer, primary_key=True)
16 | device: str = Column(String)
17 | region: str = Column(String)
18 | version: str = Column(String)
19 | branch: str = Column(String)
20 | type: str = Column(String)
21 | size: str = Column(String)
22 | md5: str = Column(String)
23 | filename: str = Column(String)
24 | link: str = Column(String)
25 | date: str = Column(String)
26 | changelog: str = Column(String)
27 | changelog_link: Union[str, None] = Column(String)
28 | insert_date: str = Column(String)
29 | product: str = Column(String)
30 |
31 | def __repr__(self):
32 | return "" % (
33 | self.device,
34 | self.version,
35 | self.branch,
36 | )
37 |
--------------------------------------------------------------------------------
/op_tracker/data/official/cn/cn.yml:
--------------------------------------------------------------------------------
1 | []
2 |
--------------------------------------------------------------------------------
/op_tracker/data/official/global/global.yml:
--------------------------------------------------------------------------------
1 | - code: PM1629453412331
2 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/NordCE.png
3 | name: OnePlus Nord CE
4 | product: ''
5 | region: ''
6 | - code: PM1617074857015
7 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP9Pro.png
8 | name: OnePlus 9 Pro
9 | product: ''
10 | region: ''
11 | - code: PM1617074715494
12 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP9.png
13 | name: OnePlus 9
14 | product: ''
15 | region: ''
16 | - code: PM1605856872898
17 | image: https://image01.oneplus.net/ebp/202012/24/1-m00-1e-c3-rb8bwl_kytgaurb0aaamf2d35tg077.jpg
18 | name: OnePlus Nord N100
19 | product: ''
20 | region: ''
21 | - code: PM1605596915581
22 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP8T.png
23 | name: OnePlus 8T
24 | product: ''
25 | region: ''
26 | - code: PM1596593883414
27 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8bwl-QTtiAVm9wAAGpr2NioxQ87.jpeg
28 | name: OnePlus Nord
29 | product: ''
30 | region: ''
31 | - code: PM1587123976090
32 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8LB1-QTr6AQjkSAAFqgNqvGog25.jpeg
33 | name: OnePlus 8 Pro
34 | product: ''
35 | region: ''
36 | - code: PM1574162297465
37 | image: https://image01.oneplus.net/shop/202010/22/1-M00-1B-F1-rB8LB1-Q7iWAQMu8AAFGhFTM2xE411.png
38 | name: OnePlus 7T Pro
39 | product: ''
40 | region: ''
41 | - code: PM1574161738631
42 | image: https://image01.oneplus.net/shop/201910/14/1-M00-0A-A8-rB8bwl2kIbaAXMD5AAEotHHdgPs919.jpg
43 | name: OnePlus 7T
44 | product: ''
45 | region: ''
46 | - code: PM1574156267635
47 | image: https://image01.oneplus.net/shop/201905/16/372/ebd496dd6969f9919143673ef580220b.png
48 | name: OnePlus 7Pro
49 | product: ''
50 | region: ''
51 | - code: PM1574156235282
52 | image: https://image01.oneplus.net/shop/201906/13/13/465d3610bc4cfc7714d97ef31949d048.png
53 | name: OnePlus 7
54 | product: ''
55 | region: ''
56 | - code: PM1574156215016
57 | image: https://image01.oneplus.net/shop/201811/06/1634/ba120163978cbef9e23d4c8ed9f8b62e.png
58 | name: OnePlus 6T
59 | product: ''
60 | region: ''
61 | - code: PM1574156173727
62 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op6.jpg
63 | name: OnePlus 6
64 | product: ''
65 | region: ''
66 | - code: PM1574156155944
67 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op5t.png
68 | name: OnePlus 5T
69 | product: ''
70 | region: ''
71 | - code: PM1574156143164
72 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op5.png
73 | name: OnePlus 5
74 | product: ''
75 | region: ''
76 | - code: PM1574156120231
77 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op3t.png
78 | name: OnePlus 3T
79 | product: ''
80 | region: ''
81 | - code: PM1574156097213
82 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op3.png
83 | name: OnePlus 3
84 | product: ''
85 | region: ''
86 | - code: PM1574156082320
87 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/opx.png
88 | name: OnePlus X
89 | product: ''
90 | region: ''
91 | - code: PM1574156053972
92 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op2.png
93 | name: OnePlus 2
94 | product: ''
95 | region: ''
96 | - code: PM1574156029809
97 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP1.png
98 | name: OnePlus 1
99 | product: ''
100 | region: ''
101 |
--------------------------------------------------------------------------------
/op_tracker/data/official/in/in.yml:
--------------------------------------------------------------------------------
1 | - code: PM1652238678972
2 | image: http://download.h2os.com/OnePlus10Pro/OnePlus10Pro.JPG
3 | name: OnePlus10 Pro
4 | product: ''
5 | region: ''
6 | - code: PM1646817751718
7 | image: http://download.h2os.com/photo/Nord CE 2.png
8 | name: OnePlus Nord CE 2
9 | product: ''
10 | region: ''
11 | - code: PM1646119244382
12 | image: http://download.h2os.com/photo/9RT.jpg
13 | name: OnePlus9RT
14 | product: ''
15 | region: ''
16 | - code: PM1629453412331
17 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/NordCE.png
18 | name: OnePlus Nord CE
19 | product: ''
20 | region: ''
21 | - code: PM1620699830843
22 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP9R.png
23 | name: OnePlus 9R
24 | product: ''
25 | region: ''
26 | - code: PM1617074857015
27 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP9Pro.png
28 | name: OnePlus 9 Pro
29 | product: ''
30 | region: ''
31 | - code: PM1617074715494
32 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP9.png
33 | name: OnePlus 9
34 | product: ''
35 | region: ''
36 | - code: PM1605596915581
37 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP8T.png
38 | name: OnePlus 8T
39 | product: ''
40 | region: ''
41 | - code: PM1596593883414
42 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8bwl-QTtiAVm9wAAGpr2NioxQ87.jpeg
43 | name: OnePlus Nord
44 | product: ''
45 | region: ''
46 | - code: PM1587123976090
47 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8LB1-QTr6AQjkSAAFqgNqvGog25.jpeg
48 | name: OnePlus 8 Pro
49 | product: ''
50 | region: ''
51 | - code: PM1586920535300
52 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8bwl-QTqSAfo4qAAFru52nytI97.jpeg
53 | name: OnePlus 8
54 | product: ''
55 | region: ''
56 | - code: PM1574162297465
57 | image: https://image01.oneplus.net/shop/202010/22/1-M00-1B-F1-rB8LB1-Q7iWAQMu8AAFGhFTM2xE411.png
58 | name: OnePlus 7T Pro
59 | product: ''
60 | region: ''
61 | - code: PM1574161738631
62 | image: https://image01.oneplus.net/shop/201910/14/1-M00-0A-A8-rB8bwl2kIbaAXMD5AAEotHHdgPs919.jpg
63 | name: OnePlus 7T
64 | product: ''
65 | region: ''
66 | - code: PM1574156267635
67 | image: https://image01.oneplus.net/shop/201905/16/372/ebd496dd6969f9919143673ef580220b.png
68 | name: OnePlus 7Pro
69 | product: ''
70 | region: ''
71 | - code: PM1574156235282
72 | image: https://image01.oneplus.net/shop/201906/13/13/465d3610bc4cfc7714d97ef31949d048.png
73 | name: OnePlus 7
74 | product: ''
75 | region: ''
76 | - code: PM1574156215016
77 | image: https://image01.oneplus.net/shop/201811/06/1634/ba120163978cbef9e23d4c8ed9f8b62e.png
78 | name: OnePlus 6T
79 | product: ''
80 | region: ''
81 | - code: PM1574156173727
82 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op6.jpg
83 | name: OnePlus 6
84 | product: ''
85 | region: ''
86 | - code: PM1574156155944
87 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op5t.png
88 | name: OnePlus 5T
89 | product: ''
90 | region: ''
91 | - code: PM1574156143164
92 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op5.png
93 | name: OnePlus 5
94 | product: ''
95 | region: ''
96 | - code: PM1574156120231
97 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op3t.png
98 | name: OnePlus 3T
99 | product: ''
100 | region: ''
101 | - code: PM1574156097213
102 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op3.png
103 | name: OnePlus 3
104 | product: ''
105 | region: ''
106 | - code: PM1574156082320
107 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/opx.png
108 | name: OnePlus X
109 | product: ''
110 | region: ''
111 | - code: PM1574156053972
112 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op2.png
113 | name: OnePlus 2
114 | product: ''
115 | region: ''
116 | - code: PM1574156029809
117 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP1.png
118 | name: OnePlus 1
119 | product: ''
120 | region: ''
121 |
--------------------------------------------------------------------------------
/op_tracker/data/official/regions.yml:
--------------------------------------------------------------------------------
1 | - code: cn
2 | name: China
3 | real_name: China
4 | - code: in
5 | name: India
6 | real_name: India
7 | - code: global
8 | name: Global
9 | real_name: Global
10 | - code: uk
11 | name: EEA
12 | real_name: United Kingdom
13 | - code: us
14 | name: Global
15 | real_name: United States
--------------------------------------------------------------------------------
/op_tracker/data/official/uk/uk.yml:
--------------------------------------------------------------------------------
1 | - code: PM1629453412331
2 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/NordCE.png
3 | name: OnePlus Nord CE
4 | product: ''
5 | region: ''
6 | - code: PM1617074857015
7 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP9Pro.png
8 | name: OnePlus 9 Pro
9 | product: ''
10 | region: ''
11 | - code: PM1617074715494
12 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP9.png
13 | name: OnePlus 9
14 | product: ''
15 | region: ''
16 | - code: PM1605856872898
17 | image: https://image01.oneplus.net/ebp/202012/24/1-m00-1e-c3-rb8bwl_kytgaurb0aaamf2d35tg077.jpg
18 | name: OnePlus Nord N100
19 | product: ''
20 | region: ''
21 | - code: PM1605857280263
22 | image: https://image01.oneplus.net/ebp/202012/24/1-m00-1e-c3-rb8bwl_kyrmaxebdaaamuvaownq313.jpg
23 | name: OnePlus Nord N10
24 | product: ''
25 | region: ''
26 | - code: PM1605596915581
27 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP8T.png
28 | name: OnePlus 8T
29 | product: ''
30 | region: ''
31 | - code: PM1596593883414
32 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8bwl-QTtiAVm9wAAGpr2NioxQ87.jpeg
33 | name: OnePlus Nord
34 | product: ''
35 | region: ''
36 | - code: PM1587123976090
37 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8LB1-QTr6AQjkSAAFqgNqvGog25.jpeg
38 | name: OnePlus 8 Pro
39 | product: ''
40 | region: ''
41 | - code: PM1586920535300
42 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8bwl-QTqSAfo4qAAFru52nytI97.jpeg
43 | name: OnePlus 8
44 | product: ''
45 | region: ''
46 | - code: PM1574162297465
47 | image: https://image01.oneplus.net/shop/202010/22/1-M00-1B-F1-rB8LB1-Q7iWAQMu8AAFGhFTM2xE411.png
48 | name: OnePlus 7T Pro
49 | product: ''
50 | region: ''
51 | - code: PM1574161738631
52 | image: https://image01.oneplus.net/shop/201910/14/1-M00-0A-A8-rB8bwl2kIbaAXMD5AAEotHHdgPs919.jpg
53 | name: OnePlus 7T
54 | product: ''
55 | region: ''
56 | - code: PM1574156267635
57 | image: https://image01.oneplus.net/shop/201905/16/372/ebd496dd6969f9919143673ef580220b.png
58 | name: OnePlus 7Pro
59 | product: ''
60 | region: ''
61 | - code: PM1574156235282
62 | image: https://image01.oneplus.net/shop/201906/13/13/465d3610bc4cfc7714d97ef31949d048.png
63 | name: OnePlus 7
64 | product: ''
65 | region: ''
66 | - code: PM1574156215016
67 | image: https://image01.oneplus.net/shop/201811/06/1634/ba120163978cbef9e23d4c8ed9f8b62e.png
68 | name: OnePlus 6T
69 | product: ''
70 | region: ''
71 | - code: PM1574156173727
72 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op6.jpg
73 | name: OnePlus 6
74 | product: ''
75 | region: ''
76 | - code: PM1574156155944
77 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op5t.png
78 | name: OnePlus 5T
79 | product: ''
80 | region: ''
81 | - code: PM1574156143164
82 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op5.png
83 | name: OnePlus 5
84 | product: ''
85 | region: ''
86 | - code: PM1574156120231
87 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op3t.png
88 | name: OnePlus 3T
89 | product: ''
90 | region: ''
91 | - code: PM1574156097213
92 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op3.png
93 | name: OnePlus 3
94 | product: ''
95 | region: ''
96 | - code: PM1574156082320
97 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/opx.png
98 | name: OnePlus X
99 | product: ''
100 | region: ''
101 | - code: PM1574156053972
102 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op2.png
103 | name: OnePlus 2
104 | product: ''
105 | region: ''
106 | - code: PM1574156029809
107 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP1.png
108 | name: OnePlus 1
109 | product: ''
110 | region: ''
111 |
--------------------------------------------------------------------------------
/op_tracker/data/official/us/us.yml:
--------------------------------------------------------------------------------
1 | - code: PM1630638351161
2 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/N200.png
3 | name: OnePlus Nord N200 5G
4 | product: ''
5 | region: ''
6 | - code: PM1629453412331
7 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/NordCE.png
8 | name: OnePlus Nord CE
9 | product: ''
10 | region: ''
11 | - code: PM1617074857015
12 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP9Pro.png
13 | name: OnePlus 9 Pro
14 | product: ''
15 | region: ''
16 | - code: PM1617074715494
17 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP9.png
18 | name: OnePlus 9
19 | product: ''
20 | region: ''
21 | - code: PM1605856872898
22 | image: https://image01.oneplus.net/ebp/202012/24/1-m00-1e-c3-rb8bwl_kytgaurb0aaamf2d35tg077.jpg
23 | name: OnePlus Nord N100
24 | product: ''
25 | region: ''
26 | - code: PM1605857280263
27 | image: https://image01.oneplus.net/ebp/202012/24/1-m00-1e-c3-rb8bwl_kyrmaxebdaaamuvaownq313.jpg
28 | name: OnePlus Nord N10
29 | product: ''
30 | region: ''
31 | - code: PM1605596915581
32 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP8T.png
33 | name: OnePlus 8T
34 | product: ''
35 | region: ''
36 | - code: PM1596593883414
37 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8bwl-QTtiAVm9wAAGpr2NioxQ87.jpeg
38 | name: OnePlus Nord
39 | product: ''
40 | region: ''
41 | - code: PM1587123976090
42 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8LB1-QTr6AQjkSAAFqgNqvGog25.jpeg
43 | name: OnePlus 8 Pro
44 | product: ''
45 | region: ''
46 | - code: PM1586920535300
47 | image: https://image01.oneplus.net/shop/202010/21/1-M00-1B-EB-rB8bwl-QTqSAfo4qAAFru52nytI97.jpeg
48 | name: OnePlus 8
49 | product: ''
50 | region: ''
51 | - code: PM1600329725089
52 | image: https://image01.oneplus.net/shop/202010/22/1-M00-1B-F1-rB8bwl-Q7JuAH2X7AAQpxd402Ok348.png
53 | name: OnePlus 8 Visible
54 | product: ''
55 | region: ''
56 | - code: PM1574162297465
57 | image: https://image01.oneplus.net/shop/202010/22/1-M00-1B-F1-rB8LB1-Q7iWAQMu8AAFGhFTM2xE411.png
58 | name: OnePlus 7T Pro
59 | product: ''
60 | region: ''
61 | - code: PM1574161738631
62 | image: https://image01.oneplus.net/shop/201910/14/1-M00-0A-A8-rB8bwl2kIbaAXMD5AAEotHHdgPs919.jpg
63 | name: OnePlus 7T
64 | product: ''
65 | region: ''
66 | - code: PM1574156267635
67 | image: https://image01.oneplus.net/shop/201905/16/372/ebd496dd6969f9919143673ef580220b.png
68 | name: OnePlus 7Pro
69 | product: ''
70 | region: ''
71 | - code: PM1574156235282
72 | image: https://image01.oneplus.net/shop/201906/13/13/465d3610bc4cfc7714d97ef31949d048.png
73 | name: OnePlus 7
74 | product: ''
75 | region: ''
76 | - code: PM1574156215016
77 | image: https://image01.oneplus.net/shop/201811/06/1634/ba120163978cbef9e23d4c8ed9f8b62e.png
78 | name: OnePlus 6T
79 | product: ''
80 | region: ''
81 | - code: PM1574156173727
82 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op6.jpg
83 | name: OnePlus 6
84 | product: ''
85 | region: ''
86 | - code: PM1574156155944
87 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op5t.png
88 | name: OnePlus 5T
89 | product: ''
90 | region: ''
91 | - code: PM1574156143164
92 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op5.png
93 | name: OnePlus 5
94 | product: ''
95 | region: ''
96 | - code: PM1574156120231
97 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op3t.png
98 | name: OnePlus 3T
99 | product: ''
100 | region: ''
101 | - code: PM1574156097213
102 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op3.png
103 | name: OnePlus 3
104 | product: ''
105 | region: ''
106 | - code: PM1574156082320
107 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/opx.png
108 | name: OnePlus X
109 | product: ''
110 | region: ''
111 | - code: PM1574156053972
112 | image: https://cdn.opstatics.com/store/20170907/assets/images/support/phone/mage/op2.png
113 | name: OnePlus 2
114 | product: ''
115 | region: ''
116 | - code: PM1574156029809
117 | image: https://oasis.opstatics.com/content/dam/support/spare-parts-price/OP1.png
118 | name: OnePlus 1
119 | product: ''
120 | region: ''
121 |
--------------------------------------------------------------------------------
/op_tracker/official/__init__.py:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/androidtrackers/oneplus-updates-tracker/2485e7670ef576380ff487c826263afd85664f34/op_tracker/official/__init__.py
--------------------------------------------------------------------------------
/op_tracker/official/api_client/__init__.py:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/androidtrackers/oneplus-updates-tracker/2485e7670ef576380ff487c826263afd85664f34/op_tracker/official/api_client/__init__.py
--------------------------------------------------------------------------------
/op_tracker/official/api_client/api_client.py:
--------------------------------------------------------------------------------
1 | """
2 | OnePlus Websites Scraper class implementation
3 | """
4 | import json
5 | import logging
6 | from datetime import datetime
7 | from typing import Dict, List
8 |
9 | from aiohttp import ClientResponse
10 |
11 | from op_tracker.common.api_client.common_scraper import CommonClient
12 | from op_tracker.common.database.database import add_to_db, already_in_db
13 | from op_tracker.common.database.models.update import Update
14 | from op_tracker.official.models.device import Device
15 | from op_tracker.utils.helpers import (get_version_from_file,
16 | parse_changelog_from_website)
17 |
18 |
19 | class APIClient(CommonClient):
20 | """
21 | OnePlus Websites API client
22 |
23 | This class is used to get data OnePlus websites API.
24 | It's responsible for interacting with OnePlus websites API in order to:
25 | - Get devices list.
26 | - Get device's updates information
27 | :attr: `region`: str - Website region
28 | :attr: `headers`: dict - HTTP request headers
29 | :meth: `get_devices` - Get all available devices on the website.
30 | :meth: `get_updates` - Get all updates available for a device.
31 | """
32 |
33 | def __init__(self, region):
34 | """
35 | Website Class constructor
36 | :param region: OnePlus website region
37 | """
38 | super().__init__()
39 | self.region: str = region
40 | self.base_url: str = (
41 | "https://storeapi-na.oneplus.com"
42 | if self.region != "cn"
43 | else "https://store.oneplus.com"
44 | )
45 | self.headers: Dict[str, str] = {
46 | # 'authority': 'storeapi-na.oneplus.com',
47 | "authority": self.base_url.split("/")[-1],
48 | "content-type": "application/json",
49 | "origin": "https://www.oneplus.com",
50 | "referer": "https://www.oneplus.com/",
51 | }
52 | self._logger = logging.getLogger(__name__)
53 |
54 | async def get_devices(self):
55 | """
56 | Get all available devices list from the website API.
57 | """
58 | json_data = json.dumps({"storeCode": self.region})
59 | async with self.session.post(
60 | f"{self.base_url}/xman/send-in-repair/find-phone-models",
61 | headers=self.headers,
62 | data=json_data,
63 | ) as response:
64 | if response.status == 200:
65 | self.devices = [
66 | Device.from_response(item)
67 | for item in await self._get_json_response(response)
68 | ]
69 | return self.devices
70 |
71 | async def get_updates(self, device: Device, region: dict) -> list:
72 | """
73 | Get the latest available updates for a device from the website API.
74 | :param region: Website region
75 | :param device: Device - the device object
76 | :return:a list of the device's available updates information
77 | """
78 | updates = []
79 | device.region = region.get("name")
80 | # Get latest
81 | update = await self._fetch(device)
82 | if update:
83 | for item in update:
84 | updates.append(item)
85 | return updates
86 |
87 | async def _request(self, json_data: str) -> list:
88 | """
89 | Perform an OTA request
90 | :param json_data: OnePlus API request data
91 | :return: OTA response dictionary
92 | """
93 | async with self.session.post(
94 | f"{self.base_url}/xman/send-in-repair/find-phone-systems",
95 | headers=self.headers,
96 | data=json_data,
97 | ) as response:
98 | if response.status == 200:
99 | try:
100 | data: dict = json.loads(await response.text())
101 | if data["ret"] == 1 and data["errCode"] == 0:
102 | return data["data"]
103 | except json.JSONDecodeError:
104 | self._logger.warning(f"Cannot decode JSON response of {data}")
105 | return []
106 | else:
107 | self._logger.warning(
108 | f"Not ok response ({response.reason}): "
109 | f"{data}\n{response.content}"
110 | )
111 |
112 | async def _fetch(self, device: Device) -> List[Update]:
113 | """
114 | Fetch an update and add it to the database if new
115 | :param device: device object
116 | :return: Update object
117 | """
118 | json_data = json.dumps({"storeCode": device.region, "phoneCode": device.code})
119 | response: list = await self._request(json_data)
120 | if response:
121 | updates = []
122 | for item in response:
123 | filename = item.get("versionLink").split("/")[-1]
124 | if not already_in_db(item.get("versionSign").lower()):
125 | update = self._parse_response(item, device)
126 | add_to_db(update)
127 | self._logger.info(f"Added {filename} to db")
128 | updates.append(update)
129 | return updates
130 |
131 | def _parse_response(self, response: dict, device: Device) -> Update:
132 | """
133 | Parse the response from th API into an Update object
134 | :param response: API response dictionary
135 | :param device: device object
136 | :return: Update object
137 | """
138 | _changelog = response.get("versionLog")
139 | name = response.get("phoneName")
140 | filename = response.get("versionLink").split("/")[-1]
141 | branch = "Stable" if response.get("versionType") == 1 else "Beta"
142 | version = get_version_from_file(filename, branch)
143 | if not _changelog:
144 | self._logger.warning(f"{name} ({version}) empty changelog!")
145 | return Update(
146 | device=device.name,
147 | changelog=parse_changelog_from_website(_changelog),
148 | changelog_link=None,
149 | link=response.get("versionLink"),
150 | region=device.region,
151 | size=response.get("versionSize"),
152 | md5=response.get("versionSign").lower(),
153 | date=datetime.utcfromtimestamp(
154 | response.get("versionReleaseTime") / 1000
155 | ).strftime("%Y-%m-%d"),
156 | version=version if version else response.get("versionNo"),
157 | type="Full" if "patch" not in filename else "Incremental",
158 | branch=branch,
159 | filename=filename,
160 | insert_date=datetime.now().strftime("%Y-%m-%d %H:%M:%S"),
161 | product=device.get_product(),
162 | )
163 |
164 | @staticmethod
165 | async def _get_json_response(_response: ClientResponse):
166 | """
167 | Get a JSON response from the HTTP response
168 | :param _response: ClientResponse: The API response client object
169 | :return:
170 | """
171 | response: dict = json.loads(await _response.text())
172 | return (
173 | response["data"]
174 | if response["ret"] == 1 and response["errCode"] == 0
175 | else []
176 | )
177 |
--------------------------------------------------------------------------------
/op_tracker/official/models/__init__.py:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/androidtrackers/oneplus-updates-tracker/2485e7670ef576380ff487c826263afd85664f34/op_tracker/official/models/__init__.py
--------------------------------------------------------------------------------
/op_tracker/official/models/device.py:
--------------------------------------------------------------------------------
1 | """
2 | OnePlus Device representation class (Official Website API)
3 | """
4 | import re
5 | from dataclasses import dataclass
6 |
7 |
8 | @dataclass
9 | class Device:
10 | """
11 | A class representing device information
12 | :param name: str - the name of the device
13 | :param code: str - the code of the device (on OnePlus website)
14 | :param image: str - the image of the device
15 | :param region: str - region of the device
16 | :param product: str - product name the device
17 | """
18 |
19 | name: str
20 | code: str
21 | image: str
22 | region: str
23 | product: str
24 |
25 | @classmethod
26 | def from_response(cls, response: dict):
27 | """
28 | Factory method to create an instance of :class:`Device` from OnePlus updates api response
29 | :param response: dict - OnePlus updates api response
30 | :return: :class:`Device` instance
31 | """
32 | if re.match(r"\d\s\w{2,}", response.get("phoneName")):
33 | name = re.sub(r"\d\w{2,}", r"\d\s\w{2,}", response.get("phoneName"))
34 | else:
35 | name = response.get("phoneName")
36 | return cls(name, response.get("phoneCode"), response.get("phoneImage"), "", "")
37 |
38 | def get_product(self):
39 | product = None
40 | device = re.sub(r"OnePlus\s", "", self.name)
41 | if device in ["1", "2", "3", "3T", "5", "6", "6T", "X"]:
42 | product = self.name.replace(" ", "")
43 | elif device == "7" or device == "7 Pro":
44 | if self.region == "China":
45 | product = f"{self.name.replace(' ', '')}_CH"
46 | elif self.region == "EEA":
47 | product = f"{self.name.replace(' ', '')}_EEA"
48 | else:
49 | product = self.name.replace(" ", "")
50 | elif device == "7T" or device == "7T Pro":
51 | if self.region == "China":
52 | product = f"{self.name.replace(' ', '')}_CH"
53 | else:
54 | product = self.name.replace(" ", "")
55 | else:
56 | if self.region == "Global":
57 | product = self.name.replace(" ", "")
58 | elif self.region == "China":
59 | product = f"{self.name.replace(' ', '')}_CH"
60 | elif self.region == "EEA" or self.region == "Europe":
61 | product = f"{self.name.replace(' ', '')}_EEA"
62 | elif self.region == "India":
63 | product = f"{self.name.replace(' ', '')}_IND"
64 | self.product = product
65 | return product
66 |
67 |
68 | def __str__(self):
69 | return f"{self.device} ({self.code})"
70 |
--------------------------------------------------------------------------------
/op_tracker/tracker_official.py:
--------------------------------------------------------------------------------
1 | """
2 | OnePlus Updates Tracker main module
3 | This module is the entry point for the tracker script and contains the controller part
4 | """
5 | import asyncio
6 | import logging
7 | from dataclasses import asdict
8 |
9 | from op_tracker import CONFIG, WORK_DIR
10 | from op_tracker.common.database.helpers import export_latest
11 | from op_tracker.official.api_client.api_client import APIClient
12 | from op_tracker.official.models.device import Device
13 | from op_tracker.utils.data_manager import DataManager
14 | from op_tracker.utils.git import git_commit_push
15 | from op_tracker.utils.telegram import TelegramBot
16 |
17 | logger = logging.getLogger(__name__)
18 |
19 |
20 | async def check_update(device: Device, region, api):
21 | """Asynchronously checks device updates"""
22 | updates: list = await api.get_updates(device, region)
23 | logger.debug(updates)
24 | return [i for i in updates] if updates else None
25 |
26 |
27 | async def main():
28 | """Main function"""
29 | new_updates: list = []
30 | regions = DataManager.read_file(f"{WORK_DIR}/data/official/regions.yml")
31 | for region in regions:
32 | region_code = region.get("code")
33 | logger.info(f"Fetching {region_code}")
34 | api: APIClient = APIClient(region_code)
35 | devices: list = await api.get_devices()
36 | logger.debug(f"{region_code} devices: {devices}")
37 | DataManager.write_file(
38 | f"{WORK_DIR}/data/official/{region_code}/{region_code}.yml",
39 | [asdict(i) for i in devices],
40 | )
41 | tasks = [
42 | asyncio.ensure_future(check_update(device, region, api))
43 | for device in api.devices
44 | ]
45 | results = await asyncio.gather(*tasks)
46 | for result in results:
47 | if result:
48 | new_updates.append(result[0])
49 | await api.close()
50 | if new_updates:
51 | logger.info(f"New updates: {new_updates}")
52 | bot: TelegramBot = TelegramBot(
53 | CONFIG.get("tg_bot_token"), CONFIG.get("tg_chat"), "website"
54 | )
55 | bot.post_updates(new_updates)
56 | export_latest()
57 | await git_commit_push()
58 |
59 |
60 | def run():
61 | """asyncio trigger function"""
62 | event_loop = asyncio.get_event_loop()
63 | event_loop.run_until_complete(main())
64 |
--------------------------------------------------------------------------------
/op_tracker/utils/__init__.py:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/androidtrackers/oneplus-updates-tracker/2485e7670ef576380ff487c826263afd85664f34/op_tracker/utils/__init__.py
--------------------------------------------------------------------------------
/op_tracker/utils/data_manager.py:
--------------------------------------------------------------------------------
1 | """
2 | OnePlus Updates Tracker Data Management class
3 | """
4 | from glob import glob
5 | from pathlib import Path
6 | from shutil import copyfile
7 |
8 | import yaml
9 |
10 | from op_tracker.utils.helpers import is_newer_datetime
11 |
12 |
13 | class DataManager:
14 | """
15 | DataManager class is responsible for managing all data storing, loading, and backing up tasks.
16 |
17 | :attr: `data`: a dictionary of the data
18 | :attr: `file`: the file containing the data path.
19 | :attr: `backup_file`: the backup file path.
20 | :meth: `save` a wrapper function to call `write_file` method with `data` and `file` parameters.`
21 | :meth: `write_file` A method that writes the data to YAML file.
22 | :meth: `read_file` A method that reads the data from YAML file.
23 | :meth: `backup` A method for backing up the `file` into `backup_file`.
24 | :meth: `backup_all` A method for backing up all files in `file` parent directory.
25 | :meth: `is_new_version` A method for checking if data (of update)
26 | is a newer than the backup (old) data.
27 | :meth: `diff_dicts` A method for comparing two directories and return the new changes only.
28 | """
29 |
30 | def __init__(self, data: dict, file):
31 | """
32 | DataManager constructor
33 | :param data: date that will be stored
34 | :param file: file that will be used to store data
35 | """
36 | self.data: dict = data
37 | self.file: Path = Path(file)
38 | self.backup_file: Path = Path(f"{self.file}.bak")
39 | if not self.file.exists():
40 | self.file.parent.mkdir(parents=True, exist_ok=True)
41 | self.file.touch()
42 |
43 | def save(self):
44 | """
45 | Saves the data into the file
46 | :return:
47 | """
48 | self.write_file(self.file, self.data)
49 |
50 | @staticmethod
51 | def write_file(file, data):
52 | """Write data into the file"""
53 | with open(f"{file}", "w", encoding="utf-8") as out:
54 | yaml.dump(data, out, allow_unicode=True)
55 |
56 | @staticmethod
57 | def read_file(file):
58 | """Read data from the file"""
59 | with open(file, "r") as yaml_file:
60 | return yaml.load(yaml_file, yaml.FullLoader)
61 |
62 | def backup(self):
63 | """Backup the file up (copy current to backup one)"""
64 | copyfile(self.file, self.backup_file)
65 | return self.backup_file
66 |
67 | @staticmethod
68 | def backup_all(directory):
69 | """Iterate through all files in a directory and backup them"""
70 | for item in glob(directory):
71 | copyfile(f"{item}", f"{item}.bak")
72 |
73 | def is_new_version(self):
74 | """Check if the version of data is newer than the old data's one"""
75 | if "version" in self.data.keys() and self.backup_file.exists():
76 | old: dict = self.read_file(self.backup_file)
77 | if not old:
78 | return True
79 | return bool(
80 | self.data["version"] != old["version"]
81 | and is_newer_datetime(old["updated"], self.data["updated"])
82 | )
83 |
84 | def diff_dicts(self):
85 | """Diff two dictionaries and return the new changes."""
86 | old: dict = self.read_file(self.backup_file)
87 | if not old:
88 | return {}
89 | return {key: value for key, value in self.data.items() if key not in old.keys()}
90 |
--------------------------------------------------------------------------------
/op_tracker/utils/git.py:
--------------------------------------------------------------------------------
1 | """
2 | Git helper module
3 | """
4 | import logging
5 | from asyncio import create_subprocess_shell
6 | from asyncio.subprocess import PIPE, Process
7 | from datetime import datetime
8 |
9 | from op_tracker import CONFIG, WORK_DIR
10 |
11 |
12 | async def git_commit_push():
13 | """Git helper function that adds, commits, and pushes changes"""
14 | command: str = (
15 | f"git add {WORK_DIR}/data/official/*.yml {WORK_DIR}/data/official/*/*.yml "
16 | f"{WORK_DIR}/data/*.yml -f && "
17 | f'git -c "user.name=CI" -c "user.email=CI@example.com" '
18 | f'commit -m "sync: {datetime.today().strftime("%d-%m-%Y %H:%M:%S")}" && '
19 | f'git push -q https://{CONFIG.get("git_oauth_token")}@'
20 | f"github.com/androidtrackers/"
21 | f"oneplus-updates-tracker.git HEAD:master"
22 | )
23 | process: Process = await create_subprocess_shell(command, stdin=PIPE, stdout=PIPE)
24 | await process.wait()
25 | if process.returncode != 0 and process.returncode != 1:
26 | stdout = await process.stdout.read()
27 | logger = logging.getLogger(__name__)
28 | logger.warning(
29 | f"Cannot commit and push changes! Error code: {process.returncode}\n"
30 | f"Output: {stdout.decode()}"
31 | )
32 |
--------------------------------------------------------------------------------
/op_tracker/utils/helpers.py:
--------------------------------------------------------------------------------
1 | """OnePlus Updates Tracker helper functions"""
2 | import re
3 | from datetime import datetime
4 |
5 | from bs4 import BeautifulSoup
6 |
7 | from op_tracker.common.database.database import get_version
8 |
9 |
10 | def is_newer_datetime(old_datetime: int, new_datetime: int) -> bool:
11 | """
12 | Check if a datetime is newer than another
13 | :param new_datetime: A datetime in posix time format
14 | :param old_datetime: A datetime in posix time format
15 | """
16 | return bool(
17 | datetime.strptime(new_datetime, "%d-%m-%Y")
18 | >= datetime.strptime(old_datetime, "%d-%m-%Y")
19 | )
20 |
21 |
22 | def parse_changelog(changelog: str) -> str:
23 | """
24 | Parse the changelog markdown and return clean string
25 | :param changelog: OnePlus API response changelog
26 | :return: A clean string of changelog response
27 | """
28 | changelog = re.sub(r"#", r"", changelog)
29 | changelog = changelog.replace("[www.oneplus.com]{http://www.oneplus.com/}", "")
30 | changelog = re.sub("\\\\", "", changelog)
31 | changelog = re.sub("\n\n+", r"\n", changelog)
32 | changelog = re.sub(" +", " ", changelog)
33 | changelog = changelog.strip()
34 | return changelog
35 |
36 |
37 | def parse_changelog_from_website(_changelog: str) -> str:
38 | """
39 | Parse the changelog html and return clean string
40 | :param _changelog: OnePlus API response changelog
41 | :return: A clean string of changelog html
42 | """
43 |
44 | changelog: str = BeautifulSoup(_changelog, "html.parser").get_text(separator="\n")
45 | # clean up the text
46 | changelog: str = re.sub(r"\s\s+", r" ", changelog)
47 | changelog: str = re.sub(" •", r"\n•", changelog)
48 | changelog: str = re.sub(r"\s+$", "", changelog)
49 | changelog: str = re.sub(r"\xa0", "", changelog)
50 | return changelog
51 |
52 |
53 | def get_version_letter(version: str, branch: str) -> str:
54 | """
55 | Get version letter of irregular update version
56 | :param version: irregular version from zip name
57 | :param branch: zip file branch
58 | :return:
59 | """
60 | letter = ""
61 | device = version.split("_")[0].replace("Hydrogen", "").replace("Oxygen", "")
62 | if (
63 | device.startswith("OnePlus5") or device.startswith("OnePlus7")
64 | ) and branch == "Stable":
65 | letter = "J"
66 | if device.startswith("OnePlus5") and branch == "Beta":
67 | letter = "T"
68 | if (
69 | device.startswith("OnePlus6") or device.startswith("OnePlus7")
70 | ) and branch == "Stable":
71 | letter = "J"
72 | return letter
73 |
74 |
75 | def get_version_from_file(filename: str, branch: str) -> str:
76 | version: str = ""
77 | pattern = re.search(r"_\d{2}\.\w\.\d{2}", filename)
78 | pattern2 = re.search(r"_\d{2}_OTA_\d{3}_all", filename)
79 | if pattern:
80 | version = re.sub("_OTA", "", filename)
81 | version = re.sub("_all", "", version)
82 | version = re.sub(r"_[a-z0-9]{16}\.zip", "", version)
83 | elif pattern2:
84 | another_version = get_version(filename.split("_")[0], branch)
85 | if another_version:
86 | split_version = re.search(r"_OTA_(\d{3})_all", filename).group(1)
87 | version = re.sub(
88 | r"\.\d{2}", f".{split_version[1:]}", another_version.version
89 | )
90 | version = re.sub(r"_\d{3}_", f"_{split_version}_", version)
91 | version = re.sub(r"_\d{10}_", filename.split("_")[-2], version)
92 | return version
93 |
--------------------------------------------------------------------------------
/op_tracker/utils/merger.py:
--------------------------------------------------------------------------------
1 | """YAML files merge functions"""
2 | from glob import glob
3 |
4 | from op_tracker import WORK_DIR
5 | from op_tracker.utils.data_manager import DataManager
6 |
7 |
8 | def merge_devices(regions: dict) -> list:
9 | """
10 | Merge YAML files of regions information
11 | :param regions: a dictionary of regions
12 | :return: a list of regions
13 | """
14 | devices = set()
15 | for region_code in regions.keys():
16 | data: dict = DataManager.read_file(
17 | f"{WORK_DIR}/data/official/{region_code}/{region_code}.yml"
18 | )
19 | for item in data.keys():
20 | devices.add(item)
21 | devices = sorted(list(devices))
22 | DataManager.write_file(f"{WORK_DIR}/data/official/devices.yml", devices)
23 | return devices
24 |
25 |
26 | def merge_updates(devices: list):
27 | """
28 | Merge updates of a list of regions
29 | :param devices: a list of devices
30 | :return: None
31 | """
32 | for device in devices:
33 | all_updates = []
34 | updates = glob(f"{WORK_DIR}/data/official/*/*/{device}.yml")
35 | for update in updates:
36 | data = DataManager.read_file(update)
37 | all_updates.append(data)
38 | DataManager.write_file(
39 | f"{WORK_DIR}/data/official/latest/{device}.yml", all_updates
40 | )
41 |
--------------------------------------------------------------------------------
/op_tracker/utils/telegram.py:
--------------------------------------------------------------------------------
1 | """Telegram Bot implementation"""
2 | from time import sleep
3 | from typing import List, Union
4 |
5 | from telegram import InlineKeyboardButton, InlineKeyboardMarkup
6 | from telegram.ext import Updater
7 |
8 | from op_tracker.common.database.database import get_incremental
9 | from op_tracker.common.database.models.update import Update
10 |
11 |
12 | class TelegramBot:
13 | """
14 | This class implements telegram bot that is used for sending updates to a telegram chat
15 | :attr:`bot` Telegram bot object
16 | :attr:`updater` Telegram updater object
17 | :attr:`chat` Telegram chat username or id
18 | """
19 |
20 | def __init__(self, bot_token: str, chat: Union[int, str], source: str):
21 | """
22 | TelegramBot class constructor
23 | :param bot_token: Telegram Bot API access token
24 | :param chat: Telegram chat username or id that will be used to send updates to
25 | """
26 | self.updater = Updater(token=bot_token, use_context=True)
27 | self.chat = chat if isinstance(chat, int) else f"@{chat}"
28 | self.source = source
29 |
30 | def post_updates(self, new_updates: List[Update]):
31 | """
32 | Send updates to a Telegram chat
33 | :param new_updates: a list of updates
34 | :return: None
35 | """
36 | for update in new_updates:
37 | message, button = self.generate_message(update)
38 | self.send_telegram_message(message, button)
39 |
40 | def generate_message(self, update: Update) -> (str, InlineKeyboardMarkup):
41 | """
42 | Generate an update message from and `Update` object
43 | :param update: an Update object that contains update's information from official website
44 | :return: A string containing the update's message
45 | and inline keyboard that has download link'
46 | """
47 | message: str = f"New update available!"
48 | if self.source == "website":
49 | message += " (on the official website)\n"
50 | elif self.source == "updater":
51 | message += " (via OTA)\n"
52 | else:
53 | message += "\n"
54 | message += (
55 | f"*Device*: {update.device}\n"
56 | f"*Region*: {update.region}\n"
57 | f"*Type*: {update.branch}\n"
58 | f"*Version*: ```{update.version}```\n"
59 | f"*Release Date*: {update.date}\n"
60 | f"*Size*: {update.size}\n"
61 | f"*MD5*: `{update.md5}`\n"
62 | f"*Changelog*:\n```{update.changelog}```"
63 | )
64 | button: InlineKeyboardButton = InlineKeyboardButton("Full ROM", update.link)
65 | incremental = get_incremental(update.version)
66 | if incremental:
67 | incremental_button: InlineKeyboardButton = InlineKeyboardButton(
68 | "Incremental", incremental.link
69 | )
70 | return message, InlineKeyboardMarkup([[button], [incremental_button]])
71 | return message, InlineKeyboardMarkup([[button]])
72 |
73 | def send_telegram_message(self, message: str, reply_markup: InlineKeyboardMarkup):
74 | """
75 | Send a message to Telegram chat
76 | :param message: A string of the update message to be sent
77 | :param reply_markup: A inline keyboard markup object that contains the update list
78 | :return:
79 | """
80 | self.updater.bot.send_message(
81 | chat_id=self.chat,
82 | text=message,
83 | parse_mode="Markdown",
84 | disable_web_page_preview="yes",
85 | reply_markup=reply_markup,
86 | )
87 | sleep(5)
88 |
--------------------------------------------------------------------------------
/poetry.lock:
--------------------------------------------------------------------------------
1 | [[package]]
2 | name = "aiohttp"
3 | version = "3.8.1"
4 | description = "Async http client/server framework (asyncio)"
5 | category = "main"
6 | optional = false
7 | python-versions = ">=3.6"
8 |
9 | [package.dependencies]
10 | aiosignal = ">=1.1.2"
11 | async-timeout = ">=4.0.0a3,<5.0"
12 | attrs = ">=17.3.0"
13 | charset-normalizer = ">=2.0,<3.0"
14 | frozenlist = ">=1.1.1"
15 | multidict = ">=4.5,<7.0"
16 | yarl = ">=1.0,<2.0"
17 |
18 | [package.extras]
19 | speedups = ["Brotli", "aiodns", "cchardet"]
20 |
21 | [[package]]
22 | name = "aiosignal"
23 | version = "1.2.0"
24 | description = "aiosignal: a list of registered asynchronous callbacks"
25 | category = "main"
26 | optional = false
27 | python-versions = ">=3.6"
28 |
29 | [package.dependencies]
30 | frozenlist = ">=1.1.0"
31 |
32 | [[package]]
33 | name = "async-timeout"
34 | version = "4.0.2"
35 | description = "Timeout context manager for asyncio programs"
36 | category = "main"
37 | optional = false
38 | python-versions = ">=3.6"
39 |
40 | [[package]]
41 | name = "attrs"
42 | version = "22.1.0"
43 | description = "Classes Without Boilerplate"
44 | category = "main"
45 | optional = false
46 | python-versions = ">=3.5"
47 |
48 | [package.extras]
49 | dev = ["cloudpickle", "coverage[toml] (>=5.0.2)", "furo", "hypothesis", "mypy (>=0.900,!=0.940)", "pre-commit", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "sphinx", "sphinx-notfound-page", "zope.interface"]
50 | docs = ["furo", "sphinx", "sphinx-notfound-page", "zope.interface"]
51 | tests = ["cloudpickle", "coverage[toml] (>=5.0.2)", "hypothesis", "mypy (>=0.900,!=0.940)", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "zope.interface"]
52 | tests_no_zope = ["cloudpickle", "coverage[toml] (>=5.0.2)", "hypothesis", "mypy (>=0.900,!=0.940)", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins"]
53 |
54 | [[package]]
55 | name = "beautifulsoup4"
56 | version = "4.11.1"
57 | description = "Screen-scraping library"
58 | category = "main"
59 | optional = false
60 | python-versions = ">=3.6.0"
61 |
62 | [package.dependencies]
63 | soupsieve = ">1.2"
64 |
65 | [package.extras]
66 | html5lib = ["html5lib"]
67 | lxml = ["lxml"]
68 |
69 | [[package]]
70 | name = "black"
71 | version = "22.8.0"
72 | description = "The uncompromising code formatter."
73 | category = "dev"
74 | optional = false
75 | python-versions = ">=3.6.2"
76 |
77 | [package.dependencies]
78 | click = ">=8.0.0"
79 | mypy-extensions = ">=0.4.3"
80 | pathspec = ">=0.9.0"
81 | platformdirs = ">=2"
82 | tomli = { version = ">=1.1.0", markers = "python_full_version < \"3.11.0a7\"" }
83 | typing-extensions = { version = ">=3.10.0.0", markers = "python_version < \"3.10\"" }
84 |
85 | [package.extras]
86 | colorama = ["colorama (>=0.4.3)"]
87 | d = ["aiohttp (>=3.7.4)"]
88 | jupyter = ["ipython (>=7.8.0)", "tokenize-rt (>=3.2.0)"]
89 | uvloop = ["uvloop (>=0.15.2)"]
90 |
91 | [[package]]
92 | name = "certifi"
93 | version = "2022.6.15"
94 | description = "Python package for providing Mozilla's CA Bundle."
95 | category = "main"
96 | optional = false
97 | python-versions = ">=3.6"
98 |
99 | [[package]]
100 | name = "cffi"
101 | version = "1.15.1"
102 | description = "Foreign Function Interface for Python calling C code."
103 | category = "main"
104 | optional = false
105 | python-versions = "*"
106 |
107 | [package.dependencies]
108 | pycparser = "*"
109 |
110 | [[package]]
111 | name = "charset-normalizer"
112 | version = "2.1.1"
113 | description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet."
114 | category = "main"
115 | optional = false
116 | python-versions = ">=3.6.0"
117 |
118 | [package.extras]
119 | unicode_backport = ["unicodedata2"]
120 |
121 | [[package]]
122 | name = "click"
123 | version = "8.1.3"
124 | description = "Composable command line interface toolkit"
125 | category = "dev"
126 | optional = false
127 | python-versions = ">=3.7"
128 |
129 | [package.dependencies]
130 | colorama = { version = "*", markers = "platform_system == \"Windows\"" }
131 |
132 | [[package]]
133 | name = "colorama"
134 | version = "0.4.5"
135 | description = "Cross-platform colored terminal text."
136 | category = "dev"
137 | optional = false
138 | python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
139 |
140 | [[package]]
141 | name = "cryptography"
142 | version = "37.0.4"
143 | description = "cryptography is a package which provides cryptographic recipes and primitives to Python developers."
144 | category = "main"
145 | optional = false
146 | python-versions = ">=3.6"
147 |
148 | [package.dependencies]
149 | cffi = ">=1.12"
150 |
151 | [package.extras]
152 | docs = ["sphinx (>=1.6.5,!=1.8.0,!=3.1.0,!=3.1.1)", "sphinx_rtd_theme"]
153 | docstest = ["pyenchant (>=1.6.11)", "sphinxcontrib-spelling (>=4.0.1)", "twine (>=1.12.0)"]
154 | pep8test = ["black", "flake8", "flake8-import-order", "pep8-naming"]
155 | sdist = ["setuptools_rust (>=0.11.4)"]
156 | ssh = ["bcrypt (>=3.1.5)"]
157 | test = ["hypothesis (>=1.11.4,!=3.79.2)", "iso8601", "pretend", "pytest (>=6.2.0)", "pytest-benchmark", "pytest-cov", "pytest-subtests", "pytest-xdist", "pytz"]
158 |
159 | [[package]]
160 | name = "decorator"
161 | version = "5.1.1"
162 | description = "Decorators for Humans"
163 | category = "main"
164 | optional = false
165 | python-versions = ">=3.5"
166 |
167 | [[package]]
168 | name = "frozenlist"
169 | version = "1.2.0"
170 | description = "A list-like structure which implements collections.abc.MutableSequence"
171 | category = "main"
172 | optional = false
173 | python-versions = ">=3.6"
174 |
175 | [[package]]
176 | name = "greenlet"
177 | version = "1.1.3"
178 | description = "Lightweight in-process concurrent programming"
179 | category = "main"
180 | optional = false
181 | python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*"
182 |
183 | [package.extras]
184 | docs = ["Sphinx"]
185 |
186 | [[package]]
187 | name = "humanize"
188 | version = "2.6.0"
189 | description = "Python humanize utilities"
190 | category = "main"
191 | optional = false
192 | python-versions = ">=3.5"
193 |
194 | [package.extras]
195 | tests = ["freezegun", "pytest", "pytest-cov"]
196 |
197 | [[package]]
198 | name = "idna"
199 | version = "3.3"
200 | description = "Internationalized Domain Names in Applications (IDNA)"
201 | category = "main"
202 | optional = false
203 | python-versions = ">=3.5"
204 |
205 | [[package]]
206 | name = "isort"
207 | version = "5.10.1"
208 | description = "A Python utility / library to sort Python imports."
209 | category = "dev"
210 | optional = false
211 | python-versions = ">=3.6.1,<4.0"
212 |
213 | [package.extras]
214 | colors = ["colorama (>=0.4.3,<0.5.0)"]
215 | pipfile_deprecated_finder = ["pipreqs", "requirementslib"]
216 | plugins = ["setuptools"]
217 | requirements_deprecated_finder = ["pip-api", "pipreqs"]
218 |
219 | [[package]]
220 | name = "multidict"
221 | version = "5.2.0"
222 | description = "multidict implementation"
223 | category = "main"
224 | optional = false
225 | python-versions = ">=3.6"
226 |
227 | [[package]]
228 | name = "mypy-extensions"
229 | version = "0.4.3"
230 | description = "Experimental type system extensions for programs checked with the mypy typechecker."
231 | category = "dev"
232 | optional = false
233 | python-versions = "*"
234 |
235 | [[package]]
236 | name = "pathspec"
237 | version = "0.10.1"
238 | description = "Utility library for gitignore style pattern matching of file paths."
239 | category = "dev"
240 | optional = false
241 | python-versions = ">=3.7"
242 |
243 | [[package]]
244 | name = "platformdirs"
245 | version = "2.5.2"
246 | description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"."
247 | category = "dev"
248 | optional = false
249 | python-versions = ">=3.7"
250 |
251 | [package.extras]
252 | docs = ["furo (>=2021.7.5b38)", "proselint (>=0.10.2)", "sphinx (>=4)", "sphinx-autodoc-typehints (>=1.12)"]
253 | test = ["appdirs (==1.4.4)", "pytest (>=6)", "pytest-cov (>=2.7)", "pytest-mock (>=3.6)"]
254 |
255 | [[package]]
256 | name = "pycparser"
257 | version = "2.21"
258 | description = "C parser in Python"
259 | category = "main"
260 | optional = false
261 | python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
262 |
263 | [[package]]
264 | name = "python-telegram-bot"
265 | version = "12.8"
266 | description = "We have made you a wrapper you can't refuse"
267 | category = "main"
268 | optional = false
269 | python-versions = "*"
270 |
271 | [package.dependencies]
272 | certifi = "*"
273 | cryptography = "*"
274 | decorator = ">=4.4.0"
275 | tornado = ">=5.1"
276 |
277 | [package.extras]
278 | json = ["ujson"]
279 | socks = ["PySocks"]
280 |
281 | [[package]]
282 | name = "PyYAML"
283 | version = "5.4.1"
284 | description = "YAML parser and emitter for Python"
285 | category = "main"
286 | optional = false
287 | python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*"
288 |
289 | [[package]]
290 | name = "soupsieve"
291 | version = "2.3.2.post1"
292 | description = "A modern CSS selector implementation for Beautiful Soup."
293 | category = "main"
294 | optional = false
295 | python-versions = ">=3.6"
296 |
297 | [[package]]
298 | name = "SQLAlchemy"
299 | version = "1.4.40"
300 | description = "Database Abstraction Library"
301 | category = "main"
302 | optional = false
303 | python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7"
304 |
305 | [package.dependencies]
306 | greenlet = { version = "!=0.4.17", markers = "python_version >= \"3\" and (platform_machine == \"aarch64\" or platform_machine == \"ppc64le\" or platform_machine == \"x86_64\" or platform_machine == \"amd64\" or platform_machine == \"AMD64\" or platform_machine == \"win32\" or platform_machine == \"WIN32\")" }
307 |
308 | [package.extras]
309 | aiomysql = ["aiomysql", "greenlet (!=0.4.17)"]
310 | aiosqlite = ["aiosqlite", "greenlet (!=0.4.17)", "typing_extensions (!=3.10.0.1)"]
311 | asyncio = ["greenlet (!=0.4.17)"]
312 | asyncmy = ["asyncmy (>=0.2.3,!=0.2.4)", "greenlet (!=0.4.17)"]
313 | mariadb_connector = ["mariadb (>=1.0.1,!=1.1.2)"]
314 | mssql = ["pyodbc"]
315 | mssql_pymssql = ["pymssql"]
316 | mssql_pyodbc = ["pyodbc"]
317 | mypy = ["mypy (>=0.910)", "sqlalchemy2-stubs"]
318 | mysql = ["mysqlclient (>=1.4.0)", "mysqlclient (>=1.4.0,<2)"]
319 | mysql_connector = ["mysql-connector-python"]
320 | oracle = ["cx_oracle (>=7)", "cx_oracle (>=7,<8)"]
321 | postgresql = ["psycopg2 (>=2.7)"]
322 | postgresql_asyncpg = ["asyncpg", "greenlet (!=0.4.17)"]
323 | postgresql_pg8000 = ["pg8000 (>=1.16.6,!=1.29.0)"]
324 | postgresql_psycopg2binary = ["psycopg2-binary"]
325 | postgresql_psycopg2cffi = ["psycopg2cffi"]
326 | pymysql = ["pymysql", "pymysql (<1)"]
327 | sqlcipher = ["sqlcipher3_binary"]
328 |
329 | [[package]]
330 | name = "tomli"
331 | version = "2.0.1"
332 | description = "A lil' TOML parser"
333 | category = "dev"
334 | optional = false
335 | python-versions = ">=3.7"
336 |
337 | [[package]]
338 | name = "tornado"
339 | version = "6.1"
340 | description = "Tornado is a Python web framework and asynchronous networking library, originally developed at FriendFeed."
341 | category = "main"
342 | optional = false
343 | python-versions = ">= 3.5"
344 |
345 | [[package]]
346 | name = "typing-extensions"
347 | version = "4.3.0"
348 | description = "Backported and Experimental Type Hints for Python 3.7+"
349 | category = "dev"
350 | optional = false
351 | python-versions = ">=3.7"
352 |
353 | [[package]]
354 | name = "yarl"
355 | version = "1.7.2"
356 | description = "Yet another URL library"
357 | category = "main"
358 | optional = false
359 | python-versions = ">=3.6"
360 |
361 | [package.dependencies]
362 | idna = ">=2.0"
363 | multidict = ">=4.0"
364 |
365 | [metadata]
366 | lock-version = "1.1"
367 | python-versions = "^3.9"
368 | content-hash = "64745084db0c99b1eea8da10ae5d4f1e12c720716c49b439947ea6d74d343bf9"
369 |
370 | [metadata.files]
371 | aiohttp = [
372 | { file = "aiohttp-3.8.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:1ed0b6477896559f17b9eaeb6d38e07f7f9ffe40b9f0f9627ae8b9926ae260a8" },
373 | { file = "aiohttp-3.8.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:7dadf3c307b31e0e61689cbf9e06be7a867c563d5a63ce9dca578f956609abf8" },
374 | { file = "aiohttp-3.8.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a79004bb58748f31ae1cbe9fa891054baaa46fb106c2dc7af9f8e3304dc30316" },
375 | { file = "aiohttp-3.8.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:12de6add4038df8f72fac606dff775791a60f113a725c960f2bab01d8b8e6b15" },
376 | { file = "aiohttp-3.8.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:6f0d5f33feb5f69ddd57a4a4bd3d56c719a141080b445cbf18f238973c5c9923" },
377 | { file = "aiohttp-3.8.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:eaba923151d9deea315be1f3e2b31cc39a6d1d2f682f942905951f4e40200922" },
378 | { file = "aiohttp-3.8.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:099ebd2c37ac74cce10a3527d2b49af80243e2a4fa39e7bce41617fbc35fa3c1" },
379 | { file = "aiohttp-3.8.1-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:2e5d962cf7e1d426aa0e528a7e198658cdc8aa4fe87f781d039ad75dcd52c516" },
380 | { file = "aiohttp-3.8.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:fa0ffcace9b3aa34d205d8130f7873fcfefcb6a4dd3dd705b0dab69af6712642" },
381 | { file = "aiohttp-3.8.1-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:61bfc23df345d8c9716d03717c2ed5e27374e0fe6f659ea64edcd27b4b044cf7" },
382 | { file = "aiohttp-3.8.1-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:31560d268ff62143e92423ef183680b9829b1b482c011713ae941997921eebc8" },
383 | { file = "aiohttp-3.8.1-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:01d7bdb774a9acc838e6b8f1d114f45303841b89b95984cbb7d80ea41172a9e3" },
384 | { file = "aiohttp-3.8.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:97ef77eb6b044134c0b3a96e16abcb05ecce892965a2124c566af0fd60f717e2" },
385 | { file = "aiohttp-3.8.1-cp310-cp310-win32.whl", hash = "sha256:c2aef4703f1f2ddc6df17519885dbfa3514929149d3ff900b73f45998f2532fa" },
386 | { file = "aiohttp-3.8.1-cp310-cp310-win_amd64.whl", hash = "sha256:713ac174a629d39b7c6a3aa757b337599798da4c1157114a314e4e391cd28e32" },
387 | { file = "aiohttp-3.8.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:473d93d4450880fe278696549f2e7aed8cd23708c3c1997981464475f32137db" },
388 | { file = "aiohttp-3.8.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:99b5eeae8e019e7aad8af8bb314fb908dd2e028b3cdaad87ec05095394cce632" },
389 | { file = "aiohttp-3.8.1-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3af642b43ce56c24d063325dd2cf20ee012d2b9ba4c3c008755a301aaea720ad" },
390 | { file = "aiohttp-3.8.1-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c3630c3ef435c0a7c549ba170a0633a56e92629aeed0e707fec832dee313fb7a" },
391 | { file = "aiohttp-3.8.1-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:4a4a4e30bf1edcad13fb0804300557aedd07a92cabc74382fdd0ba6ca2661091" },
392 | { file = "aiohttp-3.8.1-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:6f8b01295e26c68b3a1b90efb7a89029110d3a4139270b24fda961893216c440" },
393 | { file = "aiohttp-3.8.1-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:a25fa703a527158aaf10dafd956f7d42ac6d30ec80e9a70846253dd13e2f067b" },
394 | { file = "aiohttp-3.8.1-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:5bfde62d1d2641a1f5173b8c8c2d96ceb4854f54a44c23102e2ccc7e02f003ec" },
395 | { file = "aiohttp-3.8.1-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:51467000f3647d519272392f484126aa716f747859794ac9924a7aafa86cd411" },
396 | { file = "aiohttp-3.8.1-cp36-cp36m-musllinux_1_1_s390x.whl", hash = "sha256:03a6d5349c9ee8f79ab3ff3694d6ce1cfc3ced1c9d36200cb8f08ba06bd3b782" },
397 | { file = "aiohttp-3.8.1-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:102e487eeb82afac440581e5d7f8f44560b36cf0bdd11abc51a46c1cd88914d4" },
398 | { file = "aiohttp-3.8.1-cp36-cp36m-win32.whl", hash = "sha256:4aed991a28ea3ce320dc8ce655875e1e00a11bdd29fe9444dd4f88c30d558602" },
399 | { file = "aiohttp-3.8.1-cp36-cp36m-win_amd64.whl", hash = "sha256:b0e20cddbd676ab8a64c774fefa0ad787cc506afd844de95da56060348021e96" },
400 | { file = "aiohttp-3.8.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:37951ad2f4a6df6506750a23f7cbabad24c73c65f23f72e95897bb2cecbae676" },
401 | { file = "aiohttp-3.8.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5c23b1ad869653bc818e972b7a3a79852d0e494e9ab7e1a701a3decc49c20d51" },
402 | { file = "aiohttp-3.8.1-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:15b09b06dae900777833fe7fc4b4aa426556ce95847a3e8d7548e2d19e34edb8" },
403 | { file = "aiohttp-3.8.1-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:477c3ea0ba410b2b56b7efb072c36fa91b1e6fc331761798fa3f28bb224830dd" },
404 | { file = "aiohttp-3.8.1-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:2f2f69dca064926e79997f45b2f34e202b320fd3782f17a91941f7eb85502ee2" },
405 | { file = "aiohttp-3.8.1-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:ef9612483cb35171d51d9173647eed5d0069eaa2ee812793a75373447d487aa4" },
406 | { file = "aiohttp-3.8.1-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:6d69f36d445c45cda7b3b26afef2fc34ef5ac0cdc75584a87ef307ee3c8c6d00" },
407 | { file = "aiohttp-3.8.1-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:55c3d1072704d27401c92339144d199d9de7b52627f724a949fc7d5fc56d8b93" },
408 | { file = "aiohttp-3.8.1-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:b9d00268fcb9f66fbcc7cd9fe423741d90c75ee029a1d15c09b22d23253c0a44" },
409 | { file = "aiohttp-3.8.1-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:07b05cd3305e8a73112103c834e91cd27ce5b4bd07850c4b4dbd1877d3f45be7" },
410 | { file = "aiohttp-3.8.1-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:c34dc4958b232ef6188c4318cb7b2c2d80521c9a56c52449f8f93ab7bc2a8a1c" },
411 | { file = "aiohttp-3.8.1-cp37-cp37m-win32.whl", hash = "sha256:d2f9b69293c33aaa53d923032fe227feac867f81682f002ce33ffae978f0a9a9" },
412 | { file = "aiohttp-3.8.1-cp37-cp37m-win_amd64.whl", hash = "sha256:6ae828d3a003f03ae31915c31fa684b9890ea44c9c989056fea96e3d12a9fa17" },
413 | { file = "aiohttp-3.8.1-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:0c7ebbbde809ff4e970824b2b6cb7e4222be6b95a296e46c03cf050878fc1785" },
414 | { file = "aiohttp-3.8.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:8b7ef7cbd4fec9a1e811a5de813311ed4f7ac7d93e0fda233c9b3e1428f7dd7b" },
415 | { file = "aiohttp-3.8.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:c3d6a4d0619e09dcd61021debf7059955c2004fa29f48788a3dfaf9c9901a7cd" },
416 | { file = "aiohttp-3.8.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:718626a174e7e467f0558954f94af117b7d4695d48eb980146016afa4b580b2e" },
417 | { file = "aiohttp-3.8.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:589c72667a5febd36f1315aa6e5f56dd4aa4862df295cb51c769d16142ddd7cd" },
418 | { file = "aiohttp-3.8.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2ed076098b171573161eb146afcb9129b5ff63308960aeca4b676d9d3c35e700" },
419 | { file = "aiohttp-3.8.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:086f92daf51a032d062ec5f58af5ca6a44d082c35299c96376a41cbb33034675" },
420 | { file = "aiohttp-3.8.1-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:11691cf4dc5b94236ccc609b70fec991234e7ef8d4c02dd0c9668d1e486f5abf" },
421 | { file = "aiohttp-3.8.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:31d1e1c0dbf19ebccbfd62eff461518dcb1e307b195e93bba60c965a4dcf1ba0" },
422 | { file = "aiohttp-3.8.1-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:11a67c0d562e07067c4e86bffc1553f2cf5b664d6111c894671b2b8712f3aba5" },
423 | { file = "aiohttp-3.8.1-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:bb01ba6b0d3f6c68b89fce7305080145d4877ad3acaed424bae4d4ee75faa950" },
424 | { file = "aiohttp-3.8.1-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:44db35a9e15d6fe5c40d74952e803b1d96e964f683b5a78c3cc64eb177878155" },
425 | { file = "aiohttp-3.8.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:844a9b460871ee0a0b0b68a64890dae9c415e513db0f4a7e3cab41a0f2fedf33" },
426 | { file = "aiohttp-3.8.1-cp38-cp38-win32.whl", hash = "sha256:7d08744e9bae2ca9c382581f7dce1273fe3c9bae94ff572c3626e8da5b193c6a" },
427 | { file = "aiohttp-3.8.1-cp38-cp38-win_amd64.whl", hash = "sha256:04d48b8ce6ab3cf2097b1855e1505181bdd05586ca275f2505514a6e274e8e75" },
428 | { file = "aiohttp-3.8.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:f5315a2eb0239185af1bddb1abf472d877fede3cc8d143c6cddad37678293237" },
429 | { file = "aiohttp-3.8.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:a996d01ca39b8dfe77440f3cd600825d05841088fd6bc0144cc6c2ec14cc5f74" },
430 | { file = "aiohttp-3.8.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:13487abd2f761d4be7c8ff9080de2671e53fff69711d46de703c310c4c9317ca" },
431 | { file = "aiohttp-3.8.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ea302f34477fda3f85560a06d9ebdc7fa41e82420e892fc50b577e35fc6a50b2" },
432 | { file = "aiohttp-3.8.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a2f635ce61a89c5732537a7896b6319a8fcfa23ba09bec36e1b1ac0ab31270d2" },
433 | { file = "aiohttp-3.8.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e999f2d0e12eea01caeecb17b653f3713d758f6dcc770417cf29ef08d3931421" },
434 | { file = "aiohttp-3.8.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0770e2806a30e744b4e21c9d73b7bee18a1cfa3c47991ee2e5a65b887c49d5cf" },
435 | { file = "aiohttp-3.8.1-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:d15367ce87c8e9e09b0f989bfd72dc641bcd04ba091c68cd305312d00962addd" },
436 | { file = "aiohttp-3.8.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:6c7cefb4b0640703eb1069835c02486669312bf2f12b48a748e0a7756d0de33d" },
437 | { file = "aiohttp-3.8.1-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:71927042ed6365a09a98a6377501af5c9f0a4d38083652bcd2281a06a5976724" },
438 | { file = "aiohttp-3.8.1-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:28d490af82bc6b7ce53ff31337a18a10498303fe66f701ab65ef27e143c3b0ef" },
439 | { file = "aiohttp-3.8.1-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:b6613280ccedf24354406caf785db748bebbddcf31408b20c0b48cb86af76866" },
440 | { file = "aiohttp-3.8.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:81e3d8c34c623ca4e36c46524a3530e99c0bc95ed068fd6e9b55cb721d408fb2" },
441 | { file = "aiohttp-3.8.1-cp39-cp39-win32.whl", hash = "sha256:7187a76598bdb895af0adbd2fb7474d7f6025d170bc0a1130242da817ce9e7d1" },
442 | { file = "aiohttp-3.8.1-cp39-cp39-win_amd64.whl", hash = "sha256:1c182cb873bc91b411e184dab7a2b664d4fea2743df0e4d57402f7f3fa644bac" },
443 | { file = "aiohttp-3.8.1.tar.gz", hash = "sha256:fc5471e1a54de15ef71c1bc6ebe80d4dc681ea600e68bfd1cbce40427f0b7578" },
444 | ]
445 | aiosignal = [
446 | { file = "aiosignal-1.2.0-py3-none-any.whl", hash = "sha256:26e62109036cd181df6e6ad646f91f0dcfd05fe16d0cb924138ff2ab75d64e3a" },
447 | { file = "aiosignal-1.2.0.tar.gz", hash = "sha256:78ed67db6c7b7ced4f98e495e572106d5c432a93e1ddd1bf475e1dc05f5b7df2" },
448 | ]
449 | async-timeout = [
450 | { file = "async-timeout-4.0.2.tar.gz", hash = "sha256:2163e1640ddb52b7a8c80d0a67a08587e5d245cc9c553a74a847056bc2976b15" },
451 | { file = "async_timeout-4.0.2-py3-none-any.whl", hash = "sha256:8ca1e4fcf50d07413d66d1a5e416e42cfdf5851c981d679a09851a6853383b3c" },
452 | ]
453 | attrs = [
454 | { file = "attrs-22.1.0-py2.py3-none-any.whl", hash = "sha256:86efa402f67bf2df34f51a335487cf46b1ec130d02b8d39fd248abfd30da551c" },
455 | { file = "attrs-22.1.0.tar.gz", hash = "sha256:29adc2665447e5191d0e7c568fde78b21f9672d344281d0c6e1ab085429b22b6" },
456 | ]
457 | beautifulsoup4 = [
458 | { file = "beautifulsoup4-4.11.1-py3-none-any.whl", hash = "sha256:58d5c3d29f5a36ffeb94f02f0d786cd53014cf9b3b3951d42e0080d8a9498d30" },
459 | { file = "beautifulsoup4-4.11.1.tar.gz", hash = "sha256:ad9aa55b65ef2808eb405f46cf74df7fcb7044d5cbc26487f96eb2ef2e436693" },
460 | ]
461 | black = [
462 | { file = "black-22.8.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:ce957f1d6b78a8a231b18e0dd2d94a33d2ba738cd88a7fe64f53f659eea49fdd" },
463 | { file = "black-22.8.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:5107ea36b2b61917956d018bd25129baf9ad1125e39324a9b18248d362156a27" },
464 | { file = "black-22.8.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:e8166b7bfe5dcb56d325385bd1d1e0f635f24aae14b3ae437102dedc0c186747" },
465 | { file = "black-22.8.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd82842bb272297503cbec1a2600b6bfb338dae017186f8f215c8958f8acf869" },
466 | { file = "black-22.8.0-cp310-cp310-win_amd64.whl", hash = "sha256:d839150f61d09e7217f52917259831fe2b689f5c8e5e32611736351b89bb2a90" },
467 | { file = "black-22.8.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:a05da0430bd5ced89176db098567973be52ce175a55677436a271102d7eaa3fe" },
468 | { file = "black-22.8.0-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4a098a69a02596e1f2a58a2a1c8d5a05d5a74461af552b371e82f9fa4ada8342" },
469 | { file = "black-22.8.0-cp36-cp36m-win_amd64.whl", hash = "sha256:5594efbdc35426e35a7defa1ea1a1cb97c7dbd34c0e49af7fb593a36bd45edab" },
470 | { file = "black-22.8.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:a983526af1bea1e4cf6768e649990f28ee4f4137266921c2c3cee8116ae42ec3" },
471 | { file = "black-22.8.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3b2c25f8dea5e8444bdc6788a2f543e1fb01494e144480bc17f806178378005e" },
472 | { file = "black-22.8.0-cp37-cp37m-win_amd64.whl", hash = "sha256:78dd85caaab7c3153054756b9fe8c611efa63d9e7aecfa33e533060cb14b6d16" },
473 | { file = "black-22.8.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:cea1b2542d4e2c02c332e83150e41e3ca80dc0fb8de20df3c5e98e242156222c" },
474 | { file = "black-22.8.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:5b879eb439094751185d1cfdca43023bc6786bd3c60372462b6f051efa6281a5" },
475 | { file = "black-22.8.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:0a12e4e1353819af41df998b02c6742643cfef58282915f781d0e4dd7a200411" },
476 | { file = "black-22.8.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c3a73f66b6d5ba7288cd5d6dad9b4c9b43f4e8a4b789a94bf5abfb878c663eb3" },
477 | { file = "black-22.8.0-cp38-cp38-win_amd64.whl", hash = "sha256:e981e20ec152dfb3e77418fb616077937378b322d7b26aa1ff87717fb18b4875" },
478 | { file = "black-22.8.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:8ce13ffed7e66dda0da3e0b2eb1bdfc83f5812f66e09aca2b0978593ed636b6c" },
479 | { file = "black-22.8.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:32a4b17f644fc288c6ee2bafdf5e3b045f4eff84693ac069d87b1a347d861497" },
480 | { file = "black-22.8.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:0ad827325a3a634bae88ae7747db1a395d5ee02cf05d9aa7a9bd77dfb10e940c" },
481 | { file = "black-22.8.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:53198e28a1fb865e9fe97f88220da2e44df6da82b18833b588b1883b16bb5d41" },
482 | { file = "black-22.8.0-cp39-cp39-win_amd64.whl", hash = "sha256:bc4d4123830a2d190e9cc42a2e43570f82ace35c3aeb26a512a2102bce5af7ec" },
483 | { file = "black-22.8.0-py3-none-any.whl", hash = "sha256:d2c21d439b2baf7aa80d6dd4e3659259be64c6f49dfd0f32091063db0e006db4" },
484 | { file = "black-22.8.0.tar.gz", hash = "sha256:792f7eb540ba9a17e8656538701d3eb1afcb134e3b45b71f20b25c77a8db7e6e" },
485 | ]
486 | certifi = [
487 | { file = "certifi-2022.6.15-py3-none-any.whl", hash = "sha256:fe86415d55e84719d75f8b69414f6438ac3547d2078ab91b67e779ef69378412" },
488 | { file = "certifi-2022.6.15.tar.gz", hash = "sha256:84c85a9078b11105f04f3036a9482ae10e4621616db313fe045dd24743a0820d" },
489 | ]
490 | cffi = [
491 | { file = "cffi-1.15.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:a66d3508133af6e8548451b25058d5812812ec3798c886bf38ed24a98216fab2" },
492 | { file = "cffi-1.15.1-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:470c103ae716238bbe698d67ad020e1db9d9dba34fa5a899b5e21577e6d52ed2" },
493 | { file = "cffi-1.15.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:9ad5db27f9cabae298d151c85cf2bad1d359a1b9c686a275df03385758e2f914" },
494 | { file = "cffi-1.15.1-cp27-cp27m-win32.whl", hash = "sha256:b3bbeb01c2b273cca1e1e0c5df57f12dce9a4dd331b4fa1635b8bec26350bde3" },
495 | { file = "cffi-1.15.1-cp27-cp27m-win_amd64.whl", hash = "sha256:e00b098126fd45523dd056d2efba6c5a63b71ffe9f2bbe1a4fe1716e1d0c331e" },
496 | { file = "cffi-1.15.1-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:d61f4695e6c866a23a21acab0509af1cdfd2c013cf256bbf5b6b5e2695827162" },
497 | { file = "cffi-1.15.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:ed9cb427ba5504c1dc15ede7d516b84757c3e3d7868ccc85121d9310d27eed0b" },
498 | { file = "cffi-1.15.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:39d39875251ca8f612b6f33e6b1195af86d1b3e60086068be9cc053aa4376e21" },
499 | { file = "cffi-1.15.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:285d29981935eb726a4399badae8f0ffdff4f5050eaa6d0cfc3f64b857b77185" },
500 | { file = "cffi-1.15.1-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3eb6971dcff08619f8d91607cfc726518b6fa2a9eba42856be181c6d0d9515fd" },
501 | { file = "cffi-1.15.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:21157295583fe8943475029ed5abdcf71eb3911894724e360acff1d61c1d54bc" },
502 | { file = "cffi-1.15.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5635bd9cb9731e6d4a1132a498dd34f764034a8ce60cef4f5319c0541159392f" },
503 | { file = "cffi-1.15.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2012c72d854c2d03e45d06ae57f40d78e5770d252f195b93f581acf3ba44496e" },
504 | { file = "cffi-1.15.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd86c085fae2efd48ac91dd7ccffcfc0571387fe1193d33b6394db7ef31fe2a4" },
505 | { file = "cffi-1.15.1-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:fa6693661a4c91757f4412306191b6dc88c1703f780c8234035eac011922bc01" },
506 | { file = "cffi-1.15.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:59c0b02d0a6c384d453fece7566d1c7e6b7bae4fc5874ef2ef46d56776d61c9e" },
507 | { file = "cffi-1.15.1-cp310-cp310-win32.whl", hash = "sha256:cba9d6b9a7d64d4bd46167096fc9d2f835e25d7e4c121fb2ddfc6528fb0413b2" },
508 | { file = "cffi-1.15.1-cp310-cp310-win_amd64.whl", hash = "sha256:ce4bcc037df4fc5e3d184794f27bdaab018943698f4ca31630bc7f84a7b69c6d" },
509 | { file = "cffi-1.15.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3d08afd128ddaa624a48cf2b859afef385b720bb4b43df214f85616922e6a5ac" },
510 | { file = "cffi-1.15.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3799aecf2e17cf585d977b780ce79ff0dc9b78d799fc694221ce814c2c19db83" },
511 | { file = "cffi-1.15.1-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a591fe9e525846e4d154205572a029f653ada1a78b93697f3b5a8f1f2bc055b9" },
512 | { file = "cffi-1.15.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3548db281cd7d2561c9ad9984681c95f7b0e38881201e157833a2342c30d5e8c" },
513 | { file = "cffi-1.15.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:91fc98adde3d7881af9b59ed0294046f3806221863722ba7d8d120c575314325" },
514 | { file = "cffi-1.15.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:94411f22c3985acaec6f83c6df553f2dbe17b698cc7f8ae751ff2237d96b9e3c" },
515 | { file = "cffi-1.15.1-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:03425bdae262c76aad70202debd780501fabeaca237cdfddc008987c0e0f59ef" },
516 | { file = "cffi-1.15.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:cc4d65aeeaa04136a12677d3dd0b1c0c94dc43abac5860ab33cceb42b801c1e8" },
517 | { file = "cffi-1.15.1-cp311-cp311-win32.whl", hash = "sha256:a0f100c8912c114ff53e1202d0078b425bee3649ae34d7b070e9697f93c5d52d" },
518 | { file = "cffi-1.15.1-cp311-cp311-win_amd64.whl", hash = "sha256:04ed324bda3cda42b9b695d51bb7d54b680b9719cfab04227cdd1e04e5de3104" },
519 | { file = "cffi-1.15.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:50a74364d85fd319352182ef59c5c790484a336f6db772c1a9231f1c3ed0cbd7" },
520 | { file = "cffi-1.15.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e263d77ee3dd201c3a142934a086a4450861778baaeeb45db4591ef65550b0a6" },
521 | { file = "cffi-1.15.1-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:cec7d9412a9102bdc577382c3929b337320c4c4c4849f2c5cdd14d7368c5562d" },
522 | { file = "cffi-1.15.1-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4289fc34b2f5316fbb762d75362931e351941fa95fa18789191b33fc4cf9504a" },
523 | { file = "cffi-1.15.1-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:173379135477dc8cac4bc58f45db08ab45d228b3363adb7af79436135d028405" },
524 | { file = "cffi-1.15.1-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:6975a3fac6bc83c4a65c9f9fcab9e47019a11d3d2cf7f3c0d03431bf145a941e" },
525 | { file = "cffi-1.15.1-cp36-cp36m-win32.whl", hash = "sha256:2470043b93ff09bf8fb1d46d1cb756ce6132c54826661a32d4e4d132e1977adf" },
526 | { file = "cffi-1.15.1-cp36-cp36m-win_amd64.whl", hash = "sha256:30d78fbc8ebf9c92c9b7823ee18eb92f2e6ef79b45ac84db507f52fbe3ec4497" },
527 | { file = "cffi-1.15.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:198caafb44239b60e252492445da556afafc7d1e3ab7a1fb3f0584ef6d742375" },
528 | { file = "cffi-1.15.1-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5ef34d190326c3b1f822a5b7a45f6c4535e2f47ed06fec77d3d799c450b2651e" },
529 | { file = "cffi-1.15.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8102eaf27e1e448db915d08afa8b41d6c7ca7a04b7d73af6514df10a3e74bd82" },
530 | { file = "cffi-1.15.1-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5df2768244d19ab7f60546d0c7c63ce1581f7af8b5de3eb3004b9b6fc8a9f84b" },
531 | { file = "cffi-1.15.1-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a8c4917bd7ad33e8eb21e9a5bbba979b49d9a97acb3a803092cbc1133e20343c" },
532 | { file = "cffi-1.15.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0e2642fe3142e4cc4af0799748233ad6da94c62a8bec3a6648bf8ee68b1c7426" },
533 | { file = "cffi-1.15.1-cp37-cp37m-win32.whl", hash = "sha256:e229a521186c75c8ad9490854fd8bbdd9a0c9aa3a524326b55be83b54d4e0ad9" },
534 | { file = "cffi-1.15.1-cp37-cp37m-win_amd64.whl", hash = "sha256:a0b71b1b8fbf2b96e41c4d990244165e2c9be83d54962a9a1d118fd8657d2045" },
535 | { file = "cffi-1.15.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:320dab6e7cb2eacdf0e658569d2575c4dad258c0fcc794f46215e1e39f90f2c3" },
536 | { file = "cffi-1.15.1-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1e74c6b51a9ed6589199c787bf5f9875612ca4a8a0785fb2d4a84429badaf22a" },
537 | { file = "cffi-1.15.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a5c84c68147988265e60416b57fc83425a78058853509c1b0629c180094904a5" },
538 | { file = "cffi-1.15.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3b926aa83d1edb5aa5b427b4053dc420ec295a08e40911296b9eb1b6170f6cca" },
539 | { file = "cffi-1.15.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:87c450779d0914f2861b8526e035c5e6da0a3199d8f1add1a665e1cbc6fc6d02" },
540 | { file = "cffi-1.15.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4f2c9f67e9821cad2e5f480bc8d83b8742896f1242dba247911072d4fa94c192" },
541 | { file = "cffi-1.15.1-cp38-cp38-win32.whl", hash = "sha256:8b7ee99e510d7b66cdb6c593f21c043c248537a32e0bedf02e01e9553a172314" },
542 | { file = "cffi-1.15.1-cp38-cp38-win_amd64.whl", hash = "sha256:00a9ed42e88df81ffae7a8ab6d9356b371399b91dbdf0c3cb1e84c03a13aceb5" },
543 | { file = "cffi-1.15.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:54a2db7b78338edd780e7ef7f9f6c442500fb0d41a5a4ea24fff1c929d5af585" },
544 | { file = "cffi-1.15.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:fcd131dd944808b5bdb38e6f5b53013c5aa4f334c5cad0c72742f6eba4b73db0" },
545 | { file = "cffi-1.15.1-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7473e861101c9e72452f9bf8acb984947aa1661a7704553a9f6e4baa5ba64415" },
546 | { file = "cffi-1.15.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6c9a799e985904922a4d207a94eae35c78ebae90e128f0c4e521ce339396be9d" },
547 | { file = "cffi-1.15.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3bcde07039e586f91b45c88f8583ea7cf7a0770df3a1649627bf598332cb6984" },
548 | { file = "cffi-1.15.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:33ab79603146aace82c2427da5ca6e58f2b3f2fb5da893ceac0c42218a40be35" },
549 | { file = "cffi-1.15.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5d598b938678ebf3c67377cdd45e09d431369c3b1a5b331058c338e201f12b27" },
550 | { file = "cffi-1.15.1-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:db0fbb9c62743ce59a9ff687eb5f4afbe77e5e8403d6697f7446e5f609976f76" },
551 | { file = "cffi-1.15.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:98d85c6a2bef81588d9227dde12db8a7f47f639f4a17c9ae08e773aa9c697bf3" },
552 | { file = "cffi-1.15.1-cp39-cp39-win32.whl", hash = "sha256:40f4774f5a9d4f5e344f31a32b5096977b5d48560c5592e2f3d2c4374bd543ee" },
553 | { file = "cffi-1.15.1-cp39-cp39-win_amd64.whl", hash = "sha256:70df4e3b545a17496c9b3f41f5115e69a4f2e77e94e1d2a8e1070bc0c38c8a3c" },
554 | { file = "cffi-1.15.1.tar.gz", hash = "sha256:d400bfb9a37b1351253cb402671cea7e89bdecc294e8016a707f6d1d8ac934f9" },
555 | ]
556 | charset-normalizer = [
557 | { file = "charset-normalizer-2.1.1.tar.gz", hash = "sha256:5a3d016c7c547f69d6f81fb0db9449ce888b418b5b9952cc5e6e66843e9dd845" },
558 | { file = "charset_normalizer-2.1.1-py3-none-any.whl", hash = "sha256:83e9a75d1911279afd89352c68b45348559d1fc0506b054b346651b5e7fee29f" },
559 | ]
560 | click = [
561 | { file = "click-8.1.3-py3-none-any.whl", hash = "sha256:bb4d8133cb15a609f44e8213d9b391b0809795062913b383c62be0ee95b1db48" },
562 | { file = "click-8.1.3.tar.gz", hash = "sha256:7682dc8afb30297001674575ea00d1814d808d6a36af415a82bd481d37ba7b8e" },
563 | ]
564 | colorama = [
565 | { file = "colorama-0.4.5-py2.py3-none-any.whl", hash = "sha256:854bf444933e37f5824ae7bfc1e98d5bce2ebe4160d46b5edf346a89358e99da" },
566 | { file = "colorama-0.4.5.tar.gz", hash = "sha256:e6c6b4334fc50988a639d9b98aa429a0b57da6e17b9a44f0451f930b6967b7a4" },
567 | ]
568 | cryptography = [
569 | { file = "cryptography-37.0.4-cp36-abi3-macosx_10_10_universal2.whl", hash = "sha256:549153378611c0cca1042f20fd9c5030d37a72f634c9326e225c9f666d472884" },
570 | { file = "cryptography-37.0.4-cp36-abi3-macosx_10_10_x86_64.whl", hash = "sha256:a958c52505c8adf0d3822703078580d2c0456dd1d27fabfb6f76fe63d2971cd6" },
571 | { file = "cryptography-37.0.4-cp36-abi3-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f721d1885ecae9078c3f6bbe8a88bc0786b6e749bf32ccec1ef2b18929a05046" },
572 | { file = "cryptography-37.0.4-cp36-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:3d41b965b3380f10e4611dbae366f6dc3cefc7c9ac4e8842a806b9672ae9add5" },
573 | { file = "cryptography-37.0.4-cp36-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:80f49023dd13ba35f7c34072fa17f604d2f19bf0989f292cedf7ab5770b87a0b" },
574 | { file = "cryptography-37.0.4-cp36-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f2dcb0b3b63afb6df7fd94ec6fbddac81b5492513f7b0436210d390c14d46ee8" },
575 | { file = "cryptography-37.0.4-cp36-abi3-manylinux_2_24_x86_64.whl", hash = "sha256:b7f8dd0d4c1f21759695c05a5ec8536c12f31611541f8904083f3dc582604280" },
576 | { file = "cryptography-37.0.4-cp36-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:30788e070800fec9bbcf9faa71ea6d8068f5136f60029759fd8c3efec3c9dcb3" },
577 | { file = "cryptography-37.0.4-cp36-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:190f82f3e87033821828f60787cfa42bff98404483577b591429ed99bed39d59" },
578 | { file = "cryptography-37.0.4-cp36-abi3-win32.whl", hash = "sha256:b62439d7cd1222f3da897e9a9fe53bbf5c104fff4d60893ad1355d4c14a24157" },
579 | { file = "cryptography-37.0.4-cp36-abi3-win_amd64.whl", hash = "sha256:f7a6de3e98771e183645181b3627e2563dcde3ce94a9e42a3f427d2255190327" },
580 | { file = "cryptography-37.0.4-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6bc95ed67b6741b2607298f9ea4932ff157e570ef456ef7ff0ef4884a134cc4b" },
581 | { file = "cryptography-37.0.4-pp37-pypy37_pp73-manylinux_2_24_x86_64.whl", hash = "sha256:f8c0a6e9e1dd3eb0414ba320f85da6b0dcbd543126e30fcc546e7372a7fbf3b9" },
582 | { file = "cryptography-37.0.4-pp38-pypy38_pp73-macosx_10_10_x86_64.whl", hash = "sha256:e007f052ed10cc316df59bc90fbb7ff7950d7e2919c9757fd42a2b8ecf8a5f67" },
583 | { file = "cryptography-37.0.4-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7bc997818309f56c0038a33b8da5c0bfbb3f1f067f315f9abd6fc07ad359398d" },
584 | { file = "cryptography-37.0.4-pp38-pypy38_pp73-manylinux_2_24_x86_64.whl", hash = "sha256:d204833f3c8a33bbe11eda63a54b1aad7aa7456ed769a982f21ec599ba5fa282" },
585 | { file = "cryptography-37.0.4-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:75976c217f10d48a8b5a8de3d70c454c249e4b91851f6838a4e48b8f41eb71aa" },
586 | { file = "cryptography-37.0.4-pp39-pypy39_pp73-macosx_10_10_x86_64.whl", hash = "sha256:7099a8d55cd49b737ffc99c17de504f2257e3787e02abe6d1a6d136574873441" },
587 | { file = "cryptography-37.0.4-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2be53f9f5505673eeda5f2736bea736c40f051a739bfae2f92d18aed1eb54596" },
588 | { file = "cryptography-37.0.4-pp39-pypy39_pp73-manylinux_2_24_x86_64.whl", hash = "sha256:91ce48d35f4e3d3f1d83e29ef4a9267246e6a3be51864a5b7d2247d5086fa99a" },
589 | { file = "cryptography-37.0.4-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:4c590ec31550a724ef893c50f9a97a0c14e9c851c85621c5650d699a7b88f7ab" },
590 | { file = "cryptography-37.0.4.tar.gz", hash = "sha256:63f9c17c0e2474ccbebc9302ce2f07b55b3b3fcb211ded18a42d5764f5c10a82" },
591 | ]
592 | decorator = [
593 | { file = "decorator-5.1.1-py3-none-any.whl", hash = "sha256:b8c3f85900b9dc423225913c5aace94729fe1fa9763b38939a95226f02d37186" },
594 | { file = "decorator-5.1.1.tar.gz", hash = "sha256:637996211036b6385ef91435e4fae22989472f9d571faba8927ba8253acbc330" },
595 | ]
596 | frozenlist = [
597 | { file = "frozenlist-1.2.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:977a1438d0e0d96573fd679d291a1542097ea9f4918a8b6494b06610dfeefbf9" },
598 | { file = "frozenlist-1.2.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:a8d86547a5e98d9edd47c432f7a14b0c5592624b496ae9880fb6332f34af1edc" },
599 | { file = "frozenlist-1.2.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:181754275d5d32487431a0a29add4f897968b7157204bc1eaaf0a0ce80c5ba7d" },
600 | { file = "frozenlist-1.2.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5df31bb2b974f379d230a25943d9bf0d3bc666b4b0807394b131a28fca2b0e5f" },
601 | { file = "frozenlist-1.2.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4766632cd8a68e4f10f156a12c9acd7b1609941525569dd3636d859d79279ed3" },
602 | { file = "frozenlist-1.2.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:16eef427c51cb1203a7c0ab59d1b8abccaba9a4f58c4bfca6ed278fc896dc193" },
603 | { file = "frozenlist-1.2.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:01d79515ed5aa3d699b05f6bdcf1fe9087d61d6b53882aa599a10853f0479c6c" },
604 | { file = "frozenlist-1.2.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:28e164722ea0df0cf6d48c4d5bdf3d19e87aaa6dfb39b0ba91153f224b912020" },
605 | { file = "frozenlist-1.2.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:e63ad0beef6ece06475d29f47d1f2f29727805376e09850ebf64f90777962792" },
606 | { file = "frozenlist-1.2.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:41de4db9b9501679cf7cddc16d07ac0f10ef7eb58c525a1c8cbff43022bddca4" },
607 | { file = "frozenlist-1.2.0-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:c6a9d84ee6427b65a81fc24e6ef589cb794009f5ca4150151251c062773e7ed2" },
608 | { file = "frozenlist-1.2.0-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:f5f3b2942c3b8b9bfe76b408bbaba3d3bb305ee3693e8b1d631fe0a0d4f93673" },
609 | { file = "frozenlist-1.2.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:c98d3c04701773ad60d9545cd96df94d955329efc7743fdb96422c4b669c633b" },
610 | { file = "frozenlist-1.2.0-cp310-cp310-win32.whl", hash = "sha256:72cfbeab7a920ea9e74b19aa0afe3b4ad9c89471e3badc985d08756efa9b813b" },
611 | { file = "frozenlist-1.2.0-cp310-cp310-win_amd64.whl", hash = "sha256:11ff401951b5ac8c0701a804f503d72c048173208490c54ebb8d7bb7c07a6d00" },
612 | { file = "frozenlist-1.2.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:b46f997d5ed6d222a863b02cdc9c299101ee27974d9bbb2fd1b3c8441311c408" },
613 | { file = "frozenlist-1.2.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:351686ca020d1bcd238596b1fa5c8efcbc21bffda9d0efe237aaa60348421e2a" },
614 | { file = "frozenlist-1.2.0-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:bfbaa08cf1452acad9cb1c1d7b89394a41e712f88df522cea1a0f296b57782a0" },
615 | { file = "frozenlist-1.2.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b2ae2f5e9fa10805fb1c9adbfefaaecedd9e31849434be462c3960a0139ed729" },
616 | { file = "frozenlist-1.2.0-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:6790b8d96bbb74b7a6f4594b6f131bd23056c25f2aa5d816bd177d95245a30e3" },
617 | { file = "frozenlist-1.2.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:41f62468af1bd4e4b42b5508a3fe8cc46a693f0cdd0ca2f443f51f207893d837" },
618 | { file = "frozenlist-1.2.0-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:ec6cf345771cdb00791d271af9a0a6fbfc2b6dd44cb753f1eeaa256e21622adb" },
619 | { file = "frozenlist-1.2.0-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:14a5cef795ae3e28fb504b73e797c1800e9249f950e1c964bb6bdc8d77871161" },
620 | { file = "frozenlist-1.2.0-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:8b54cdd2fda15467b9b0bfa78cee2ddf6dbb4585ef23a16e14926f4b076dfae4" },
621 | { file = "frozenlist-1.2.0-cp36-cp36m-musllinux_1_1_s390x.whl", hash = "sha256:f025f1d6825725b09c0038775acab9ae94264453a696cc797ce20c0769a7b367" },
622 | { file = "frozenlist-1.2.0-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:84e97f59211b5b9083a2e7a45abf91cfb441369e8bb6d1f5287382c1c526def3" },
623 | { file = "frozenlist-1.2.0-cp36-cp36m-win32.whl", hash = "sha256:c5328ed53fdb0a73c8a50105306a3bc013e5ca36cca714ec4f7bd31d38d8a97f" },
624 | { file = "frozenlist-1.2.0-cp36-cp36m-win_amd64.whl", hash = "sha256:9ade70aea559ca98f4b1b1e5650c45678052e76a8ab2f76d90f2ac64180215a2" },
625 | { file = "frozenlist-1.2.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:a0d3ffa8772464441b52489b985d46001e2853a3b082c655ec5fad9fb6a3d618" },
626 | { file = "frozenlist-1.2.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3457f8cf86deb6ce1ba67e120f1b0128fcba1332a180722756597253c465fc1d" },
627 | { file = "frozenlist-1.2.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5a72eecf37eface331636951249d878750db84034927c997d47f7f78a573b72b" },
628 | { file = "frozenlist-1.2.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:acc4614e8d1feb9f46dd829a8e771b8f5c4b1051365d02efb27a3229048ade8a" },
629 | { file = "frozenlist-1.2.0-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:87521e32e18a2223311afc2492ef2d99946337da0779ddcda77b82ee7319df59" },
630 | { file = "frozenlist-1.2.0-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:8b4c7665a17c3a5430edb663e4ad4e1ad457614d1b2f2b7f87052e2ef4fa45ca" },
631 | { file = "frozenlist-1.2.0-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:ed58803563a8c87cf4c0771366cf0ad1aa265b6b0ae54cbbb53013480c7ad74d" },
632 | { file = "frozenlist-1.2.0-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:aa44c4740b4e23fcfa259e9dd52315d2b1770064cde9507457e4c4a65a04c397" },
633 | { file = "frozenlist-1.2.0-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:2de5b931701257d50771a032bba4e448ff958076380b049fd36ed8738fdb375b" },
634 | { file = "frozenlist-1.2.0-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:6e105013fa84623c057a4381dc8ea0361f4d682c11f3816cc80f49a1f3bc17c6" },
635 | { file = "frozenlist-1.2.0-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:705c184b77565955a99dc360f359e8249580c6b7eaa4dc0227caa861ef46b27a" },
636 | { file = "frozenlist-1.2.0-cp37-cp37m-win32.whl", hash = "sha256:a37594ad6356e50073fe4f60aa4187b97d15329f2138124d252a5a19c8553ea4" },
637 | { file = "frozenlist-1.2.0-cp37-cp37m-win_amd64.whl", hash = "sha256:25b358aaa7dba5891b05968dd539f5856d69f522b6de0bf34e61f133e077c1a4" },
638 | { file = "frozenlist-1.2.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:af2a51c8a381d76eabb76f228f565ed4c3701441ecec101dd18be70ebd483cfd" },
639 | { file = "frozenlist-1.2.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:82d22f6e6f2916e837c91c860140ef9947e31194c82aaeda843d6551cec92f19" },
640 | { file = "frozenlist-1.2.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:1cfe6fef507f8bac40f009c85c7eddfed88c1c0d38c75e72fe10476cef94e10f" },
641 | { file = "frozenlist-1.2.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:26f602e380a5132880fa245c92030abb0fc6ff34e0c5500600366cedc6adb06a" },
642 | { file = "frozenlist-1.2.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4ad065b2ebd09f32511ff2be35c5dfafee6192978b5a1e9d279a5c6e121e3b03" },
643 | { file = "frozenlist-1.2.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:bc93f5f62df3bdc1f677066327fc81f92b83644852a31c6aa9b32c2dde86ea7d" },
644 | { file = "frozenlist-1.2.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:89fdfc84c6bf0bff2ff3170bb34ecba8a6911b260d318d377171429c4be18c73" },
645 | { file = "frozenlist-1.2.0-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:47b2848e464883d0bbdcd9493c67443e5e695a84694efff0476f9059b4cb6257" },
646 | { file = "frozenlist-1.2.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:4f52d0732e56906f8ddea4bd856192984650282424049c956857fed43697ea43" },
647 | { file = "frozenlist-1.2.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:16ef7dd5b7d17495404a2e7a49bac1bc13d6d20c16d11f4133c757dd94c4144c" },
648 | { file = "frozenlist-1.2.0-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:1cf63243bc5f5c19762943b0aa9e0d3fb3723d0c514d820a18a9b9a5ef864315" },
649 | { file = "frozenlist-1.2.0-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:54a1e09ab7a69f843cd28fefd2bcaf23edb9e3a8d7680032c8968b8ac934587d" },
650 | { file = "frozenlist-1.2.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:954b154a4533ef28bd3e83ffdf4eadf39deeda9e38fb8feaf066d6069885e034" },
651 | { file = "frozenlist-1.2.0-cp38-cp38-win32.whl", hash = "sha256:cb3957c39668d10e2b486acc85f94153520a23263b6401e8f59422ef65b9520d" },
652 | { file = "frozenlist-1.2.0-cp38-cp38-win_amd64.whl", hash = "sha256:0a7c7cce70e41bc13d7d50f0e5dd175f14a4f1837a8549b0936ed0cbe6170bf9" },
653 | { file = "frozenlist-1.2.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:4c457220468d734e3077580a3642b7f682f5fd9507f17ddf1029452450912cdc" },
654 | { file = "frozenlist-1.2.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:e74f8b4d8677ebb4015ac01fcaf05f34e8a1f22775db1f304f497f2f88fdc697" },
655 | { file = "frozenlist-1.2.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:fbd4844ff111449f3bbe20ba24fbb906b5b1c2384d0f3287c9f7da2354ce6d23" },
656 | { file = "frozenlist-1.2.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f0081a623c886197ff8de9e635528fd7e6a387dccef432149e25c13946cb0cd0" },
657 | { file = "frozenlist-1.2.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9b6e21e5770df2dea06cb7b6323fbc008b13c4a4e3b52cb54685276479ee7676" },
658 | { file = "frozenlist-1.2.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:406aeb340613b4b559db78d86864485f68919b7141dec82aba24d1477fd2976f" },
659 | { file = "frozenlist-1.2.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:878ebe074839d649a1cdb03a61077d05760624f36d196884a5cafb12290e187b" },
660 | { file = "frozenlist-1.2.0-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:1fef737fd1388f9b93bba8808c5f63058113c10f4e3c0763ced68431773f72f9" },
661 | { file = "frozenlist-1.2.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:4a495c3d513573b0b3f935bfa887a85d9ae09f0627cf47cad17d0cc9b9ba5c38" },
662 | { file = "frozenlist-1.2.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:e7d0dd3e727c70c2680f5f09a0775525229809f1a35d8552b92ff10b2b14f2c2" },
663 | { file = "frozenlist-1.2.0-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:66a518731a21a55b7d3e087b430f1956a36793acc15912e2878431c7aec54210" },
664 | { file = "frozenlist-1.2.0-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:94728f97ddf603d23c8c3dd5cae2644fa12d33116e69f49b1644a71bb77b89ae" },
665 | { file = "frozenlist-1.2.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:c1e8e9033d34c2c9e186e58279879d78c94dd365068a3607af33f2bc99357a53" },
666 | { file = "frozenlist-1.2.0-cp39-cp39-win32.whl", hash = "sha256:83334e84a290a158c0c4cc4d22e8c7cfe0bba5b76d37f1c2509dabd22acafe15" },
667 | { file = "frozenlist-1.2.0-cp39-cp39-win_amd64.whl", hash = "sha256:735f386ec522e384f511614c01d2ef9cf799f051353876b4c6fb93ef67a6d1ee" },
668 | { file = "frozenlist-1.2.0.tar.gz", hash = "sha256:68201be60ac56aff972dc18085800b6ee07973c49103a8aba669dee3d71079de" },
669 | ]
670 | greenlet = [
671 | { file = "greenlet-1.1.3-cp27-cp27m-macosx_10_14_x86_64.whl", hash = "sha256:8c287ae7ac921dfde88b1c125bd9590b7ec3c900c2d3db5197f1286e144e712b" },
672 | { file = "greenlet-1.1.3-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:870a48007872d12e95a996fca3c03a64290d3ea2e61076aa35d3b253cf34cd32" },
673 | { file = "greenlet-1.1.3-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:7c5227963409551ae4a6938beb70d56bf1918c554a287d3da6853526212fbe0a" },
674 | { file = "greenlet-1.1.3-cp27-cp27m-win32.whl", hash = "sha256:9fae214f6c43cd47f7bef98c56919b9222481e833be2915f6857a1e9e8a15318" },
675 | { file = "greenlet-1.1.3-cp27-cp27m-win_amd64.whl", hash = "sha256:de431765bd5fe62119e0bc6bc6e7b17ac53017ae1782acf88fcf6b7eae475a49" },
676 | { file = "greenlet-1.1.3-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:510c3b15587afce9800198b4b142202b323bf4b4b5f9d6c79cb9a35e5e3c30d2" },
677 | { file = "greenlet-1.1.3-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:9951dcbd37850da32b2cb6e391f621c1ee456191c6ae5528af4a34afe357c30e" },
678 | { file = "greenlet-1.1.3-cp310-cp310-macosx_10_15_x86_64.whl", hash = "sha256:07c58e169bbe1e87b8bbf15a5c1b779a7616df9fd3e61cadc9d691740015b4f8" },
679 | { file = "greenlet-1.1.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:df02fdec0c533301497acb0bc0f27f479a3a63dcdc3a099ae33a902857f07477" },
680 | { file = "greenlet-1.1.3-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9c88e134d51d5e82315a7c32b914a58751b7353eb5268dbd02eabf020b4c4700" },
681 | { file = "greenlet-1.1.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7b41d19c0cfe5c259fe6c539fd75051cd39a5d33d05482f885faf43f7f5e7d26" },
682 | { file = "greenlet-1.1.3-cp310-cp310-win_amd64.whl", hash = "sha256:6f5d4b2280ceea76c55c893827961ed0a6eadd5a584a7c4e6e6dd7bc10dfdd96" },
683 | { file = "greenlet-1.1.3-cp311-cp311-macosx_10_15_x86_64.whl", hash = "sha256:184416e481295832350a4bf731ba619a92f5689bf5d0fa4341e98b98b1265bd7" },
684 | { file = "greenlet-1.1.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dd0404d154084a371e6d2bafc787201612a1359c2dee688ae334f9118aa0bf47" },
685 | { file = "greenlet-1.1.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7a43bbfa9b6cfdfaeefbd91038dde65ea2c421dc387ed171613df340650874f2" },
686 | { file = "greenlet-1.1.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ce5b64dfe8d0cca407d88b0ee619d80d4215a2612c1af8c98a92180e7109f4b5" },
687 | { file = "greenlet-1.1.3-cp35-cp35m-macosx_10_14_x86_64.whl", hash = "sha256:903fa5716b8fbb21019268b44f73f3748c41d1a30d71b4a49c84b642c2fed5fa" },
688 | { file = "greenlet-1.1.3-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:0118817c9341ef2b0f75f5af79ac377e4da6ff637e5ee4ac91802c0e379dadb4" },
689 | { file = "greenlet-1.1.3-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:466ce0928e33421ee84ae04c4ac6f253a3a3e6b8d600a79bd43fd4403e0a7a76" },
690 | { file = "greenlet-1.1.3-cp35-cp35m-win32.whl", hash = "sha256:65ad1a7a463a2a6f863661329a944a5802c7129f7ad33583dcc11069c17e622c" },
691 | { file = "greenlet-1.1.3-cp35-cp35m-win_amd64.whl", hash = "sha256:7532a46505470be30cbf1dbadb20379fb481244f1ca54207d7df3bf0bbab6a20" },
692 | { file = "greenlet-1.1.3-cp36-cp36m-macosx_10_14_x86_64.whl", hash = "sha256:caff52cb5cd7626872d9696aee5b794abe172804beb7db52eed1fd5824b63910" },
693 | { file = "greenlet-1.1.3-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:db41f3845eb579b544c962864cce2c2a0257fe30f0f1e18e51b1e8cbb4e0ac6d" },
694 | { file = "greenlet-1.1.3-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:e8533f5111704d75de3139bf0b8136d3a6c1642c55c067866fa0a51c2155ee33" },
695 | { file = "greenlet-1.1.3-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9537e4baf0db67f382eb29255a03154fcd4984638303ff9baaa738b10371fa57" },
696 | { file = "greenlet-1.1.3-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f8bfd36f368efe0ab2a6aa3db7f14598aac454b06849fb633b762ddbede1db90" },
697 | { file = "greenlet-1.1.3-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b0877a9a2129a2c56a2eae2da016743db7d9d6a05d5e1c198f1b7808c602a30e" },
698 | { file = "greenlet-1.1.3-cp36-cp36m-win32.whl", hash = "sha256:88b04e12c9b041a1e0bcb886fec709c488192638a9a7a3677513ac6ba81d8e79" },
699 | { file = "greenlet-1.1.3-cp36-cp36m-win_amd64.whl", hash = "sha256:4f166b4aca8d7d489e82d74627a7069ab34211ef5ebb57c300ec4b9337b60fc0" },
700 | { file = "greenlet-1.1.3-cp37-cp37m-macosx_10_15_x86_64.whl", hash = "sha256:cd16a89efe3a003029c87ff19e9fba635864e064da646bc749fc1908a4af18f3" },
701 | { file = "greenlet-1.1.3-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:5b756e6730ea59b2745072e28ad27f4c837084688e6a6b3633c8b1e509e6ae0e" },
702 | { file = "greenlet-1.1.3-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:9b2f7d0408ddeb8ea1fd43d3db79a8cefaccadd2a812f021333b338ed6b10aba" },
703 | { file = "greenlet-1.1.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:44b4817c34c9272c65550b788913620f1fdc80362b209bc9d7dd2f40d8793080" },
704 | { file = "greenlet-1.1.3-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d58a5a71c4c37354f9e0c24c9c8321f0185f6945ef027460b809f4bb474bfe41" },
705 | { file = "greenlet-1.1.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1dd51d2650e70c6c4af37f454737bf4a11e568945b27f74b471e8e2a9fd21268" },
706 | { file = "greenlet-1.1.3-cp37-cp37m-win32.whl", hash = "sha256:048d2bed76c2aa6de7af500ae0ea51dd2267aec0e0f2a436981159053d0bc7cc" },
707 | { file = "greenlet-1.1.3-cp37-cp37m-win_amd64.whl", hash = "sha256:77e41db75f9958f2083e03e9dd39da12247b3430c92267df3af77c83d8ff9eed" },
708 | { file = "greenlet-1.1.3-cp38-cp38-macosx_10_15_x86_64.whl", hash = "sha256:1626185d938d7381631e48e6f7713e8d4b964be246073e1a1d15c2f061ac9f08" },
709 | { file = "greenlet-1.1.3-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:1ec2779774d8e42ed0440cf8bc55540175187e8e934f2be25199bf4ed948cd9e" },
710 | { file = "greenlet-1.1.3-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:f2f908239b7098799b8845e5936c2ccb91d8c2323be02e82f8dcb4a80dcf4a25" },
711 | { file = "greenlet-1.1.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0b181e9aa6cb2f5ec0cacc8cee6e5a3093416c841ba32c185c30c160487f0380" },
712 | { file = "greenlet-1.1.3-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2cf45e339cabea16c07586306a31cfcc5a3b5e1626d365714d283732afed6809" },
713 | { file = "greenlet-1.1.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6200a11f003ec26815f7e3d2ded01b43a3810be3528dd760d2f1fa777490c3cd" },
714 | { file = "greenlet-1.1.3-cp38-cp38-win32.whl", hash = "sha256:db5b25265010a1b3dca6a174a443a0ed4c4ab12d5e2883a11c97d6e6d59b12f9" },
715 | { file = "greenlet-1.1.3-cp38-cp38-win_amd64.whl", hash = "sha256:095a980288fe05adf3d002fbb180c99bdcf0f930e220aa66fcd56e7914a38202" },
716 | { file = "greenlet-1.1.3-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:cbc1eb55342cbac8f7ec159088d54e2cfdd5ddf61c87b8bbe682d113789331b2" },
717 | { file = "greenlet-1.1.3-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:694ffa7144fa5cc526c8f4512665003a39fa09ef00d19bbca5c8d3406db72fbe" },
718 | { file = "greenlet-1.1.3-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:aa741c1a8a8cc25eb3a3a01a62bdb5095a773d8c6a86470bde7f607a447e7905" },
719 | { file = "greenlet-1.1.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a3a669f11289a8995d24fbfc0e63f8289dd03c9aaa0cc8f1eab31d18ca61a382" },
720 | { file = "greenlet-1.1.3-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:76a53bfa10b367ee734b95988bd82a9a5f0038a25030f9f23bbbc005010ca600" },
721 | { file = "greenlet-1.1.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2fb0aa7f6996879551fd67461d5d3ab0c3c0245da98be90c89fcb7a18d437403" },
722 | { file = "greenlet-1.1.3-cp39-cp39-win32.whl", hash = "sha256:5fbe1ab72b998ca77ceabbae63a9b2e2dc2d963f4299b9b278252ddba142d3f1" },
723 | { file = "greenlet-1.1.3-cp39-cp39-win_amd64.whl", hash = "sha256:ffe73f9e7aea404722058405ff24041e59d31ca23d1da0895af48050a07b6932" },
724 | { file = "greenlet-1.1.3.tar.gz", hash = "sha256:bcb6c6dd1d6be6d38d6db283747d07fda089ff8c559a835236560a4410340455" },
725 | ]
726 | humanize = [
727 | {file = "humanize-2.6.0-py3-none-any.whl", hash = "sha256:fd5b32945687443d5b8bc1e02fad027da1d293a9e963b3450122ad98ef534f21"},
728 | {file = "humanize-2.6.0.tar.gz", hash = "sha256:8ee358ea6c23de896b9d1925ebe6a8504bb2ba7e98d5ccf4d07ab7f3b28f3819"},
729 | ]
730 | idna = [
731 | { file = "idna-3.3-py3-none-any.whl", hash = "sha256:84d9dd047ffa80596e0f246e2eab0b391788b0503584e8945f2368256d2735ff" },
732 | { file = "idna-3.3.tar.gz", hash = "sha256:9d643ff0a55b762d5cdb124b8eaa99c66322e2157b69160bc32796e824360e6d" },
733 | ]
734 | isort = [
735 | { file = "isort-5.10.1-py3-none-any.whl", hash = "sha256:6f62d78e2f89b4500b080fe3a81690850cd254227f27f75c3a0c491a1f351ba7" },
736 | { file = "isort-5.10.1.tar.gz", hash = "sha256:e8443a5e7a020e9d7f97f1d7d9cd17c88bcb3bc7e218bf9cf5095fe550be2951" },
737 | ]
738 | multidict = [
739 | { file = "multidict-5.2.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:3822c5894c72e3b35aae9909bef66ec83e44522faf767c0ad39e0e2de11d3b55" },
740 | { file = "multidict-5.2.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:28e6d883acd8674887d7edc896b91751dc2d8e87fbdca8359591a13872799e4e" },
741 | { file = "multidict-5.2.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:b61f85101ef08cbbc37846ac0e43f027f7844f3fade9b7f6dd087178caedeee7" },
742 | { file = "multidict-5.2.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d9b668c065968c5979fe6b6fa6760bb6ab9aeb94b75b73c0a9c1acf6393ac3bf" },
743 | { file = "multidict-5.2.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:517d75522b7b18a3385726b54a081afd425d4f41144a5399e5abd97ccafdf36b" },
744 | { file = "multidict-5.2.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1b4ac3ba7a97b35a5ccf34f41b5a8642a01d1e55454b699e5e8e7a99b5a3acf5" },
745 | { file = "multidict-5.2.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:df23c83398715b26ab09574217ca21e14694917a0c857e356fd39e1c64f8283f" },
746 | { file = "multidict-5.2.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:e58a9b5cc96e014ddf93c2227cbdeca94b56a7eb77300205d6e4001805391747" },
747 | { file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:f76440e480c3b2ca7f843ff8a48dc82446b86ed4930552d736c0bac507498a52" },
748 | { file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:cfde464ca4af42a629648c0b0d79b8f295cf5b695412451716531d6916461628" },
749 | { file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:0fed465af2e0eb6357ba95795d003ac0bdb546305cc2366b1fc8f0ad67cc3fda" },
750 | { file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:b70913cbf2e14275013be98a06ef4b412329fe7b4f83d64eb70dce8269ed1e1a" },
751 | { file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:a5635bcf1b75f0f6ef3c8a1ad07b500104a971e38d3683167b9454cb6465ac86" },
752 | { file = "multidict-5.2.0-cp310-cp310-win32.whl", hash = "sha256:77f0fb7200cc7dedda7a60912f2059086e29ff67cefbc58d2506638c1a9132d7" },
753 | { file = "multidict-5.2.0-cp310-cp310-win_amd64.whl", hash = "sha256:9416cf11bcd73c861267e88aea71e9fcc35302b3943e45e1dbb4317f91a4b34f" },
754 | { file = "multidict-5.2.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:fd77c8f3cba815aa69cb97ee2b2ef385c7c12ada9c734b0f3b32e26bb88bbf1d" },
755 | { file = "multidict-5.2.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:98ec9aea6223adf46999f22e2c0ab6cf33f5914be604a404f658386a8f1fba37" },
756 | { file = "multidict-5.2.0-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e5283c0a00f48e8cafcecadebfa0ed1dac8b39e295c7248c44c665c16dc1138b" },
757 | { file = "multidict-5.2.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5f79c19c6420962eb17c7e48878a03053b7ccd7b69f389d5831c0a4a7f1ac0a1" },
758 | { file = "multidict-5.2.0-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:e4a67f1080123de76e4e97a18d10350df6a7182e243312426d508712e99988d4" },
759 | { file = "multidict-5.2.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:94b117e27efd8e08b4046c57461d5a114d26b40824995a2eb58372b94f9fca02" },
760 | { file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:2e77282fd1d677c313ffcaddfec236bf23f273c4fba7cdf198108f5940ae10f5" },
761 | { file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:116347c63ba049c1ea56e157fa8aa6edaf5e92925c9b64f3da7769bdfa012858" },
762 | { file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:dc3a866cf6c13d59a01878cd806f219340f3e82eed514485e094321f24900677" },
763 | { file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_s390x.whl", hash = "sha256:ac42181292099d91217a82e3fa3ce0e0ddf3a74fd891b7c2b347a7f5aa0edded" },
764 | { file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:f0bb0973f42ffcb5e3537548e0767079420aefd94ba990b61cf7bb8d47f4916d" },
765 | { file = "multidict-5.2.0-cp36-cp36m-win32.whl", hash = "sha256:ea21d4d5104b4f840b91d9dc8cbc832aba9612121eaba503e54eaab1ad140eb9" },
766 | { file = "multidict-5.2.0-cp36-cp36m-win_amd64.whl", hash = "sha256:e6453f3cbeb78440747096f239d282cc57a2997a16b5197c9bc839099e1633d0" },
767 | { file = "multidict-5.2.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:d3def943bfd5f1c47d51fd324df1e806d8da1f8e105cc7f1c76a1daf0f7e17b0" },
768 | { file = "multidict-5.2.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:35591729668a303a02b06e8dba0eb8140c4a1bfd4c4b3209a436a02a5ac1de11" },
769 | { file = "multidict-5.2.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ce8cacda0b679ebc25624d5de66c705bc53dcc7c6f02a7fb0f3ca5e227d80422" },
770 | { file = "multidict-5.2.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:baf1856fab8212bf35230c019cde7c641887e3fc08cadd39d32a421a30151ea3" },
771 | { file = "multidict-5.2.0-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:a43616aec0f0d53c411582c451f5d3e1123a68cc7b3475d6f7d97a626f8ff90d" },
772 | { file = "multidict-5.2.0-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:25cbd39a9029b409167aa0a20d8a17f502d43f2efebfe9e3ac019fe6796c59ac" },
773 | { file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:0a2cbcfbea6dc776782a444db819c8b78afe4db597211298dd8b2222f73e9cd0" },
774 | { file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:3d2d7d1fff8e09d99354c04c3fd5b560fb04639fd45926b34e27cfdec678a704" },
775 | { file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:a37e9a68349f6abe24130846e2f1d2e38f7ddab30b81b754e5a1fde32f782b23" },
776 | { file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:637c1896497ff19e1ee27c1c2c2ddaa9f2d134bbb5e0c52254361ea20486418d" },
777 | { file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:9815765f9dcda04921ba467957be543423e5ec6a1136135d84f2ae092c50d87b" },
778 | { file = "multidict-5.2.0-cp37-cp37m-win32.whl", hash = "sha256:8b911d74acdc1fe2941e59b4f1a278a330e9c34c6c8ca1ee21264c51ec9b67ef" },
779 | { file = "multidict-5.2.0-cp37-cp37m-win_amd64.whl", hash = "sha256:380b868f55f63d048a25931a1632818f90e4be71d2081c2338fcf656d299949a" },
780 | { file = "multidict-5.2.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:e7d81ce5744757d2f05fc41896e3b2ae0458464b14b5a2c1e87a6a9d69aefaa8" },
781 | { file = "multidict-5.2.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:2d1d55cdf706ddc62822d394d1df53573d32a7a07d4f099470d3cb9323b721b6" },
782 | { file = "multidict-5.2.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:a4771d0d0ac9d9fe9e24e33bed482a13dfc1256d008d101485fe460359476065" },
783 | { file = "multidict-5.2.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:da7d57ea65744d249427793c042094c4016789eb2562576fb831870f9c878d9e" },
784 | { file = "multidict-5.2.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:cdd68778f96216596218b4e8882944d24a634d984ee1a5a049b300377878fa7c" },
785 | { file = "multidict-5.2.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ecc99bce8ee42dcad15848c7885197d26841cb24fa2ee6e89d23b8993c871c64" },
786 | { file = "multidict-5.2.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:067150fad08e6f2dd91a650c7a49ba65085303fcc3decbd64a57dc13a2733031" },
787 | { file = "multidict-5.2.0-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:78c106b2b506b4d895ddc801ff509f941119394b89c9115580014127414e6c2d" },
788 | { file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:e6c4fa1ec16e01e292315ba76eb1d012c025b99d22896bd14a66628b245e3e01" },
789 | { file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:b227345e4186809d31f22087d0265655114af7cda442ecaf72246275865bebe4" },
790 | { file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:06560fbdcf22c9387100979e65b26fba0816c162b888cb65b845d3def7a54c9b" },
791 | { file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:7878b61c867fb2df7a95e44b316f88d5a3742390c99dfba6c557a21b30180cac" },
792 | { file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:246145bff76cc4b19310f0ad28bd0769b940c2a49fc601b86bfd150cbd72bb22" },
793 | { file = "multidict-5.2.0-cp38-cp38-win32.whl", hash = "sha256:c30ac9f562106cd9e8071c23949a067b10211917fdcb75b4718cf5775356a940" },
794 | { file = "multidict-5.2.0-cp38-cp38-win_amd64.whl", hash = "sha256:f19001e790013ed580abfde2a4465388950728861b52f0da73e8e8a9418533c0" },
795 | { file = "multidict-5.2.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:c1ff762e2ee126e6f1258650ac641e2b8e1f3d927a925aafcfde943b77a36d24" },
796 | { file = "multidict-5.2.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:bd6c9c50bf2ad3f0448edaa1a3b55b2e6866ef8feca5d8dbec10ec7c94371d21" },
797 | { file = "multidict-5.2.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:fc66d4016f6e50ed36fb39cd287a3878ffcebfa90008535c62e0e90a7ab713ae" },
798 | { file = "multidict-5.2.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a9acb76d5f3dd9421874923da2ed1e76041cb51b9337fd7f507edde1d86535d6" },
799 | { file = "multidict-5.2.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:dfc924a7e946dd3c6360e50e8f750d51e3ef5395c95dc054bc9eab0f70df4f9c" },
800 | { file = "multidict-5.2.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:32fdba7333eb2351fee2596b756d730d62b5827d5e1ab2f84e6cbb287cc67fe0" },
801 | { file = "multidict-5.2.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:b9aad49466b8d828b96b9e3630006234879c8d3e2b0a9d99219b3121bc5cdb17" },
802 | { file = "multidict-5.2.0-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:93de39267c4c676c9ebb2057e98a8138bade0d806aad4d864322eee0803140a0" },
803 | { file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:f9bef5cff994ca3026fcc90680e326d1a19df9841c5e3d224076407cc21471a1" },
804 | { file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:5f841c4f14331fd1e36cbf3336ed7be2cb2a8f110ce40ea253e5573387db7621" },
805 | { file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:38ba256ee9b310da6a1a0f013ef4e422fca30a685bcbec86a969bd520504e341" },
806 | { file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:3bc3b1621b979621cee9f7b09f024ec76ec03cc365e638126a056317470bde1b" },
807 | { file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:6ee908c070020d682e9b42c8f621e8bb10c767d04416e2ebe44e37d0f44d9ad5" },
808 | { file = "multidict-5.2.0-cp39-cp39-win32.whl", hash = "sha256:1c7976cd1c157fa7ba5456ae5d31ccdf1479680dc9b8d8aa28afabc370df42b8" },
809 | { file = "multidict-5.2.0-cp39-cp39-win_amd64.whl", hash = "sha256:c9631c642e08b9fff1c6255487e62971d8b8e821808ddd013d8ac058087591ac" },
810 | { file = "multidict-5.2.0.tar.gz", hash = "sha256:0dd1c93edb444b33ba2274b66f63def8a327d607c6c790772f448a53b6ea59ce" },
811 | ]
812 | mypy-extensions = [
813 | { file = "mypy_extensions-0.4.3-py2.py3-none-any.whl", hash = "sha256:090fedd75945a69ae91ce1303b5824f428daf5a028d2f6ab8a299250a846f15d" },
814 | { file = "mypy_extensions-0.4.3.tar.gz", hash = "sha256:2d82818f5bb3e369420cb3c4060a7970edba416647068eb4c5343488a6c604a8" },
815 | ]
816 | pathspec = [
817 | { file = "pathspec-0.10.1-py3-none-any.whl", hash = "sha256:46846318467efc4556ccfd27816e004270a9eeeeb4d062ce5e6fc7a87c573f93" },
818 | { file = "pathspec-0.10.1.tar.gz", hash = "sha256:7ace6161b621d31e7902eb6b5ae148d12cfd23f4a249b9ffb6b9fee12084323d" },
819 | ]
820 | platformdirs = [
821 | { file = "platformdirs-2.5.2-py3-none-any.whl", hash = "sha256:027d8e83a2d7de06bbac4e5ef7e023c02b863d7ea5d079477e722bb41ab25788" },
822 | { file = "platformdirs-2.5.2.tar.gz", hash = "sha256:58c8abb07dcb441e6ee4b11d8df0ac856038f944ab98b7be6b27b2a3c7feef19" },
823 | ]
824 | pycparser = [
825 | { file = "pycparser-2.21-py2.py3-none-any.whl", hash = "sha256:8ee45429555515e1f6b185e78100aea234072576aa43ab53aefcae078162fca9" },
826 | { file = "pycparser-2.21.tar.gz", hash = "sha256:e644fdec12f7872f86c58ff790da456218b10f863970249516d60a5eaca77206" },
827 | ]
828 | python-telegram-bot = [
829 | { file = "python-telegram-bot-12.8.tar.gz", hash = "sha256:327186c56469216207dcdf8706892e58e0a62e51ef46f5143268e387bbb4edc3" },
830 | { file = "python_telegram_bot-12.8-py2.py3-none-any.whl", hash = "sha256:7eebed539ccacf77896cff9e41d1f68746b8ff3ca4da1e2e59285e9c749cb050" },
831 | ]
832 | PyYAML = [
833 | { file = "PyYAML-5.4.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:3b2b1824fe7112845700f815ff6a489360226a5609b96ec2190a45e62a9fc922" },
834 | { file = "PyYAML-5.4.1-cp27-cp27m-win32.whl", hash = "sha256:129def1b7c1bf22faffd67b8f3724645203b79d8f4cc81f674654d9902cb4393" },
835 | { file = "PyYAML-5.4.1-cp27-cp27m-win_amd64.whl", hash = "sha256:4465124ef1b18d9ace298060f4eccc64b0850899ac4ac53294547536533800c8" },
836 | { file = "PyYAML-5.4.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:bb4191dfc9306777bc594117aee052446b3fa88737cd13b7188d0e7aa8162185" },
837 | { file = "PyYAML-5.4.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:6c78645d400265a062508ae399b60b8c167bf003db364ecb26dcab2bda048253" },
838 | { file = "PyYAML-5.4.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:4e0583d24c881e14342eaf4ec5fbc97f934b999a6828693a99157fde912540cc" },
839 | { file = "PyYAML-5.4.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:72a01f726a9c7851ca9bfad6fd09ca4e090a023c00945ea05ba1638c09dc3347" },
840 | { file = "PyYAML-5.4.1-cp36-cp36m-manylinux2014_s390x.whl", hash = "sha256:895f61ef02e8fed38159bb70f7e100e00f471eae2bc838cd0f4ebb21e28f8541" },
841 | { file = "PyYAML-5.4.1-cp36-cp36m-win32.whl", hash = "sha256:3bd0e463264cf257d1ffd2e40223b197271046d09dadf73a0fe82b9c1fc385a5" },
842 | { file = "PyYAML-5.4.1-cp36-cp36m-win_amd64.whl", hash = "sha256:e4fac90784481d221a8e4b1162afa7c47ed953be40d31ab4629ae917510051df" },
843 | { file = "PyYAML-5.4.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:5accb17103e43963b80e6f837831f38d314a0495500067cb25afab2e8d7a4018" },
844 | { file = "PyYAML-5.4.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:e1d4970ea66be07ae37a3c2e48b5ec63f7ba6804bdddfdbd3cfd954d25a82e63" },
845 | { file = "PyYAML-5.4.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:cb333c16912324fd5f769fff6bc5de372e9e7a202247b48870bc251ed40239aa" },
846 | { file = "PyYAML-5.4.1-cp37-cp37m-manylinux2014_s390x.whl", hash = "sha256:fe69978f3f768926cfa37b867e3843918e012cf83f680806599ddce33c2c68b0" },
847 | { file = "PyYAML-5.4.1-cp37-cp37m-win32.whl", hash = "sha256:dd5de0646207f053eb0d6c74ae45ba98c3395a571a2891858e87df7c9b9bd51b" },
848 | { file = "PyYAML-5.4.1-cp37-cp37m-win_amd64.whl", hash = "sha256:08682f6b72c722394747bddaf0aa62277e02557c0fd1c42cb853016a38f8dedf" },
849 | { file = "PyYAML-5.4.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d2d9808ea7b4af864f35ea216be506ecec180628aced0704e34aca0b040ffe46" },
850 | { file = "PyYAML-5.4.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:8c1be557ee92a20f184922c7b6424e8ab6691788e6d86137c5d93c1a6ec1b8fb" },
851 | { file = "PyYAML-5.4.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:fd7f6999a8070df521b6384004ef42833b9bd62cfee11a09bda1079b4b704247" },
852 | { file = "PyYAML-5.4.1-cp38-cp38-manylinux2014_s390x.whl", hash = "sha256:bfb51918d4ff3d77c1c856a9699f8492c612cde32fd3bcd344af9be34999bfdc" },
853 | { file = "PyYAML-5.4.1-cp38-cp38-win32.whl", hash = "sha256:fa5ae20527d8e831e8230cbffd9f8fe952815b2b7dae6ffec25318803a7528fc" },
854 | { file = "PyYAML-5.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:0f5f5786c0e09baddcd8b4b45f20a7b5d61a7e7e99846e3c799b05c7c53fa696" },
855 | { file = "PyYAML-5.4.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:294db365efa064d00b8d1ef65d8ea2c3426ac366c0c4368d930bf1c5fb497f77" },
856 | { file = "PyYAML-5.4.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:74c1485f7707cf707a7aef42ef6322b8f97921bd89be2ab6317fd782c2d53183" },
857 | { file = "PyYAML-5.4.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:d483ad4e639292c90170eb6f7783ad19490e7a8defb3e46f97dfe4bacae89122" },
858 | { file = "PyYAML-5.4.1-cp39-cp39-manylinux2014_s390x.whl", hash = "sha256:fdc842473cd33f45ff6bce46aea678a54e3d21f1b61a7750ce3c498eedfe25d6" },
859 | { file = "PyYAML-5.4.1-cp39-cp39-win32.whl", hash = "sha256:49d4cdd9065b9b6e206d0595fee27a96b5dd22618e7520c33204a4a3239d5b10" },
860 | { file = "PyYAML-5.4.1-cp39-cp39-win_amd64.whl", hash = "sha256:c20cfa2d49991c8b4147af39859b167664f2ad4561704ee74c1de03318e898db" },
861 | { file = "PyYAML-5.4.1.tar.gz", hash = "sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e" },
862 | ]
863 | soupsieve = [
864 | { file = "soupsieve-2.3.2.post1-py3-none-any.whl", hash = "sha256:3b2503d3c7084a42b1ebd08116e5f81aadfaea95863628c80a3b774a11b7c759" },
865 | { file = "soupsieve-2.3.2.post1.tar.gz", hash = "sha256:fc53893b3da2c33de295667a0e19f078c14bf86544af307354de5fcf12a3f30d" },
866 | ]
867 | SQLAlchemy = [
868 | { file = "SQLAlchemy-1.4.40-cp27-cp27m-macosx_10_14_x86_64.whl", hash = "sha256:b07fc38e6392a65935dc8b486229679142b2ea33c94059366b4d8b56f1e35a97" },
869 | { file = "SQLAlchemy-1.4.40-cp27-cp27m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:fb4edb6c354eac0fcc07cb91797e142f702532dbb16c1d62839d6eec35f814cf" },
870 | { file = "SQLAlchemy-1.4.40-cp27-cp27m-win32.whl", hash = "sha256:2026632051a93997cf8f6fda14360f99230be1725b7ab2ef15be205a4b8a5430" },
871 | { file = "SQLAlchemy-1.4.40-cp27-cp27m-win_amd64.whl", hash = "sha256:f2aa85aebc0ef6b342d5d3542f969caa8c6a63c8d36cf5098769158a9fa2123c" },
872 | { file = "SQLAlchemy-1.4.40-cp27-cp27mu-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:a0b9e3d81f86ba04007f0349e373a5b8c81ec2047aadb8d669caf8c54a092461" },
873 | { file = "SQLAlchemy-1.4.40-cp310-cp310-macosx_10_15_x86_64.whl", hash = "sha256:1ab08141d93de83559f6a7d9a962830f918623a885b3759ec2b9d1a531ff28fe" },
874 | { file = "SQLAlchemy-1.4.40-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:00dd998b43b282c71de46b061627b5edb9332510eb1edfc5017b9e4356ed44ea" },
875 | { file = "SQLAlchemy-1.4.40-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:bb342c0e25cc8f78a0e7c692da3b984f072666b316fbbec2a0e371cb4dfef5f0" },
876 | { file = "SQLAlchemy-1.4.40-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:23b693876ac7963b6bc7b1a5f3a2642f38d2624af834faad5933913928089d1b" },
877 | { file = "SQLAlchemy-1.4.40-cp310-cp310-win32.whl", hash = "sha256:2cf50611ef4221ad587fb7a1708e61ff72966f84330c6317642e08d6db4138fd" },
878 | { file = "SQLAlchemy-1.4.40-cp310-cp310-win_amd64.whl", hash = "sha256:26ee4dbac5dd7abf18bf3cd8f04e51f72c339caf702f68172d308888cd26c6c9" },
879 | { file = "SQLAlchemy-1.4.40-cp36-cp36m-macosx_10_14_x86_64.whl", hash = "sha256:b41b87b929118838bafc4bb18cf3c5cd1b3be4b61cd9042e75174df79e8ac7a2" },
880 | { file = "SQLAlchemy-1.4.40-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:885e11638946472b4a0a7db8e6df604b2cf64d23dc40eedc3806d869fcb18fae" },
881 | { file = "SQLAlchemy-1.4.40-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:b7ff0a8bf0aec1908b92b8dfa1246128bf4f94adbdd3da6730e9c542e112542d" },
882 | { file = "SQLAlchemy-1.4.40-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cfa8ab4ba0c97ab6bcae1f0948497d14c11b6c6ecd1b32b8a79546a0823d8211" },
883 | { file = "SQLAlchemy-1.4.40-cp36-cp36m-win32.whl", hash = "sha256:d259fa08e4b3ed952c01711268bcf6cd2442b0c54866d64aece122f83da77c6d" },
884 | { file = "SQLAlchemy-1.4.40-cp36-cp36m-win_amd64.whl", hash = "sha256:c8d974c991eef0cd29418a5957ae544559dc326685a6f26b3a914c87759bf2f4" },
885 | { file = "SQLAlchemy-1.4.40-cp37-cp37m-macosx_10_15_x86_64.whl", hash = "sha256:28b1791a30d62fc104070965f1a2866699c45bbf5adc0be0cf5f22935edcac58" },
886 | { file = "SQLAlchemy-1.4.40-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b7ccdca6cd167611f4a62a8c2c0c4285c2535640d77108f782ce3f3cccb70f3a" },
887 | { file = "SQLAlchemy-1.4.40-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:69deec3a94de10062080d91e1ba69595efeafeafe68b996426dec9720031fb25" },
888 | { file = "SQLAlchemy-1.4.40-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:63ad778f4e80913fb171247e4fa82123d0068615ae1d51a9791fc4284cb81748" },
889 | { file = "SQLAlchemy-1.4.40-cp37-cp37m-win32.whl", hash = "sha256:9ced2450c9fd016f9232d976661623e54c450679eeefc7aa48a3d29924a63189" },
890 | { file = "SQLAlchemy-1.4.40-cp37-cp37m-win_amd64.whl", hash = "sha256:cdee4d475e35684d210dc6b430ff8ca2ed0636378ac19b457e2f6f350d1f5acc" },
891 | { file = "SQLAlchemy-1.4.40-cp38-cp38-macosx_10_15_x86_64.whl", hash = "sha256:08b47c971327e733ffd6bae2d4f50a7b761793efe69d41067fcba86282819eea" },
892 | { file = "SQLAlchemy-1.4.40-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1cf03d37819dc17a388d313919daf32058d19ba1e592efdf14ce8cbd997e6023" },
893 | { file = "SQLAlchemy-1.4.40-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:a62c0ecbb9976550f26f7bf75569f425e661e7249349487f1483115e5fc893a6" },
894 | { file = "SQLAlchemy-1.4.40-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4ec440990ab00650d0c7ea2c75bc225087afdd7ddcb248e3d934def4dff62762" },
895 | { file = "SQLAlchemy-1.4.40-cp38-cp38-win32.whl", hash = "sha256:2b64955850a14b9d481c17becf0d3f62fb1bb31ac2c45c2caf5ad06d9e811187" },
896 | { file = "SQLAlchemy-1.4.40-cp38-cp38-win_amd64.whl", hash = "sha256:959bf4390766a8696aa01285016c766b4eb676f712878aac5fce956dd49695d9" },
897 | { file = "SQLAlchemy-1.4.40-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:0992f3cc640ec0f88f721e426da884c34ff0a60eb73d3d64172e23dfadfc8a0b" },
898 | { file = "SQLAlchemy-1.4.40-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fa9e0d7832b7511b3b3fd0e67fac85ff11fd752834c143ca2364c9b778c0485a" },
899 | { file = "SQLAlchemy-1.4.40-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:c9d0f1a9538cc5e75f2ea0cb6c3d70155a1b7f18092c052e0d84105622a41b63" },
900 | { file = "SQLAlchemy-1.4.40-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0c956a5d1adb49a35d78ef0fae26717afc48a36262359bb5b0cbd7a3a247c26f" },
901 | { file = "SQLAlchemy-1.4.40-cp39-cp39-win32.whl", hash = "sha256:6b70d02bbe1adbbf715d2249cacf9ac17c6f8d22dfcb3f1a4fbc5bf64364da8a" },
902 | { file = "SQLAlchemy-1.4.40-cp39-cp39-win_amd64.whl", hash = "sha256:bf073c619b5a7f7cd731507d0fdc7329bee14b247a63b0419929e4acd24afea8" },
903 | { file = "SQLAlchemy-1.4.40.tar.gz", hash = "sha256:44a660506080cc975e1dfa5776fe5f6315ddc626a77b50bf0eee18b0389ea265" },
904 | ]
905 | tomli = [
906 | { file = "tomli-2.0.1-py3-none-any.whl", hash = "sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc" },
907 | { file = "tomli-2.0.1.tar.gz", hash = "sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f" },
908 | ]
909 | tornado = [
910 | { file = "tornado-6.1-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:d371e811d6b156d82aa5f9a4e08b58debf97c302a35714f6f45e35139c332e32" },
911 | { file = "tornado-6.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:0d321a39c36e5f2c4ff12b4ed58d41390460f798422c4504e09eb5678e09998c" },
912 | { file = "tornado-6.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:9de9e5188a782be6b1ce866e8a51bc76a0fbaa0e16613823fc38e4fc2556ad05" },
913 | { file = "tornado-6.1-cp35-cp35m-manylinux2010_i686.whl", hash = "sha256:61b32d06ae8a036a6607805e6720ef00a3c98207038444ba7fd3d169cd998910" },
914 | { file = "tornado-6.1-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:3e63498f680547ed24d2c71e6497f24bca791aca2fe116dbc2bd0ac7f191691b" },
915 | { file = "tornado-6.1-cp35-cp35m-manylinux2014_aarch64.whl", hash = "sha256:6c77c9937962577a6a76917845d06af6ab9197702a42e1346d8ae2e76b5e3675" },
916 | { file = "tornado-6.1-cp35-cp35m-win32.whl", hash = "sha256:6286efab1ed6e74b7028327365cf7346b1d777d63ab30e21a0f4d5b275fc17d5" },
917 | { file = "tornado-6.1-cp35-cp35m-win_amd64.whl", hash = "sha256:fa2ba70284fa42c2a5ecb35e322e68823288a4251f9ba9cc77be04ae15eada68" },
918 | {file = "tornado-6.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:0a00ff4561e2929a2c37ce706cb8233b7907e0cdc22eab98888aca5dd3775feb"},
919 | {file = "tornado-6.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:748290bf9112b581c525e6e6d3820621ff020ed95af6f17fedef416b27ed564c"},
920 | {file = "tornado-6.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:e385b637ac3acaae8022e7e47dfa7b83d3620e432e3ecb9a3f7f58f150e50921"},
921 | {file = "tornado-6.1-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:25ad220258349a12ae87ede08a7b04aca51237721f63b1808d39bdb4b2164558"},
922 | {file = "tornado-6.1-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:65d98939f1a2e74b58839f8c4dab3b6b3c1ce84972ae712be02845e65391ac7c"},
923 | {file = "tornado-6.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:e519d64089b0876c7b467274468709dadf11e41d65f63bba207e04217f47c085"},
924 | {file = "tornado-6.1-cp36-cp36m-win32.whl", hash = "sha256:b87936fd2c317b6ee08a5741ea06b9d11a6074ef4cc42e031bc6403f82a32575"},
925 | {file = "tornado-6.1-cp36-cp36m-win_amd64.whl", hash = "sha256:cc0ee35043162abbf717b7df924597ade8e5395e7b66d18270116f8745ceb795"},
926 | {file = "tornado-6.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:7250a3fa399f08ec9cb3f7b1b987955d17e044f1ade821b32e5f435130250d7f"},
927 | {file = "tornado-6.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:ed3ad863b1b40cd1d4bd21e7498329ccaece75db5a5bf58cd3c9f130843e7102"},
928 | {file = "tornado-6.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:dcef026f608f678c118779cd6591c8af6e9b4155c44e0d1bc0c87c036fb8c8c4"},
929 | {file = "tornado-6.1-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:70dec29e8ac485dbf57481baee40781c63e381bebea080991893cd297742b8fd"},
930 | {file = "tornado-6.1-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:d3f7594930c423fd9f5d1a76bee85a2c36fd8b4b16921cae7e965f22575e9c01"},
931 | {file = "tornado-6.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:3447475585bae2e77ecb832fc0300c3695516a47d46cefa0528181a34c5b9d3d"},
932 | {file = "tornado-6.1-cp37-cp37m-win32.whl", hash = "sha256:e7229e60ac41a1202444497ddde70a48d33909e484f96eb0da9baf8dc68541df"},
933 | {file = "tornado-6.1-cp37-cp37m-win_amd64.whl", hash = "sha256:cb5ec8eead331e3bb4ce8066cf06d2dfef1bfb1b2a73082dfe8a161301b76e37"},
934 | {file = "tornado-6.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:20241b3cb4f425e971cb0a8e4ffc9b0a861530ae3c52f2b0434e6c1b57e9fd95"},
935 | {file = "tornado-6.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:c77da1263aa361938476f04c4b6c8916001b90b2c2fdd92d8d535e1af48fba5a"},
936 | {file = "tornado-6.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:fba85b6cd9c39be262fcd23865652920832b61583de2a2ca907dbd8e8a8c81e5"},
937 | {file = "tornado-6.1-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:1e8225a1070cd8eec59a996c43229fe8f95689cb16e552d130b9793cb570a288"},
938 | {file = "tornado-6.1-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:d14d30e7f46a0476efb0deb5b61343b1526f73ebb5ed84f23dc794bdb88f9d9f"},
939 | {file = "tornado-6.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:8f959b26f2634a091bb42241c3ed8d3cedb506e7c27b8dd5c7b9f745318ddbb6"},
940 | {file = "tornado-6.1-cp38-cp38-win32.whl", hash = "sha256:34ca2dac9e4d7afb0bed4677512e36a52f09caa6fded70b4e3e1c89dbd92c326"},
941 | {file = "tornado-6.1-cp38-cp38-win_amd64.whl", hash = "sha256:6196a5c39286cc37c024cd78834fb9345e464525d8991c21e908cc046d1cc02c"},
942 | {file = "tornado-6.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:f0ba29bafd8e7e22920567ce0d232c26d4d47c8b5cf4ed7b562b5db39fa199c5"},
943 | {file = "tornado-6.1-cp39-cp39-manylinux1_i686.whl", hash = "sha256:33892118b165401f291070100d6d09359ca74addda679b60390b09f8ef325ffe"},
944 | {file = "tornado-6.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:7da13da6f985aab7f6f28debab00c67ff9cbacd588e8477034c0652ac141feea"},
945 | {file = "tornado-6.1-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:e0791ac58d91ac58f694d8d2957884df8e4e2f6687cdf367ef7eb7497f79eaa2"},
946 | {file = "tornado-6.1-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:66324e4e1beede9ac79e60f88de548da58b1f8ab4b2f1354d8375774f997e6c0"},
947 | {file = "tornado-6.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:a48900ecea1cbb71b8c71c620dee15b62f85f7c14189bdeee54966fbd9a0c5bd"},
948 | {file = "tornado-6.1-cp39-cp39-win32.whl", hash = "sha256:d3d20ea5782ba63ed13bc2b8c291a053c8d807a8fa927d941bd718468f7b950c"},
949 | {file = "tornado-6.1-cp39-cp39-win_amd64.whl", hash = "sha256:548430be2740e327b3fe0201abe471f314741efcb0067ec4f2d7dcfb4825f3e4"},
950 | {file = "tornado-6.1.tar.gz", hash = "sha256:33c6e81d7bd55b468d2e793517c909b139960b6c790a60b7991b9b6b76fb9791"},
951 | ]
952 | typing-extensions = [
953 | { file = "typing_extensions-4.3.0-py3-none-any.whl", hash = "sha256:25642c956049920a5aa49edcdd6ab1e06d7e5d467fc00e0506c44ac86fbfca02" },
954 | { file = "typing_extensions-4.3.0.tar.gz", hash = "sha256:e6d2677a32f47fc7eb2795db1dd15c1f34eff616bcaf2cfb5e997f854fa1c4a6" },
955 | ]
956 | yarl = [
957 | { file = "yarl-1.7.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:f2a8508f7350512434e41065684076f640ecce176d262a7d54f0da41d99c5a95" },
958 | { file = "yarl-1.7.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:da6df107b9ccfe52d3a48165e48d72db0eca3e3029b5b8cb4fe6ee3cb870ba8b" },
959 | { file = "yarl-1.7.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a1d0894f238763717bdcfea74558c94e3bc34aeacd3351d769460c1a586a8b05" },
960 | { file = "yarl-1.7.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dfe4b95b7e00c6635a72e2d00b478e8a28bfb122dc76349a06e20792eb53a523" },
961 | { file = "yarl-1.7.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c145ab54702334c42237a6c6c4cc08703b6aa9b94e2f227ceb3d477d20c36c63" },
962 | { file = "yarl-1.7.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1ca56f002eaf7998b5fcf73b2421790da9d2586331805f38acd9997743114e98" },
963 | { file = "yarl-1.7.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:1d3d5ad8ea96bd6d643d80c7b8d5977b4e2fb1bab6c9da7322616fd26203d125" },
964 | { file = "yarl-1.7.2-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:167ab7f64e409e9bdd99333fe8c67b5574a1f0495dcfd905bc7454e766729b9e" },
965 | { file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:95a1873b6c0dd1c437fb3bb4a4aaa699a48c218ac7ca1e74b0bee0ab16c7d60d" },
966 | { file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:6152224d0a1eb254f97df3997d79dadd8bb2c1a02ef283dbb34b97d4f8492d23" },
967 | { file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:5bb7d54b8f61ba6eee541fba4b83d22b8a046b4ef4d8eb7f15a7e35db2e1e245" },
968 | { file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:9c1f083e7e71b2dd01f7cd7434a5f88c15213194df38bc29b388ccdf1492b739" },
969 | { file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:f44477ae29025d8ea87ec308539f95963ffdc31a82f42ca9deecf2d505242e72" },
970 | { file = "yarl-1.7.2-cp310-cp310-win32.whl", hash = "sha256:cff3ba513db55cc6a35076f32c4cdc27032bd075c9faef31fec749e64b45d26c" },
971 | { file = "yarl-1.7.2-cp310-cp310-win_amd64.whl", hash = "sha256:c9c6d927e098c2d360695f2e9d38870b2e92e0919be07dbe339aefa32a090265" },
972 | { file = "yarl-1.7.2-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:9b4c77d92d56a4c5027572752aa35082e40c561eec776048330d2907aead891d" },
973 | { file = "yarl-1.7.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c01a89a44bb672c38f42b49cdb0ad667b116d731b3f4c896f72302ff77d71656" },
974 | { file = "yarl-1.7.2-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c19324a1c5399b602f3b6e7db9478e5b1adf5cf58901996fc973fe4fccd73eed" },
975 | { file = "yarl-1.7.2-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3abddf0b8e41445426d29f955b24aeecc83fa1072be1be4e0d194134a7d9baee" },
976 | { file = "yarl-1.7.2-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:6a1a9fe17621af43e9b9fcea8bd088ba682c8192d744b386ee3c47b56eaabb2c" },
977 | { file = "yarl-1.7.2-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:8b0915ee85150963a9504c10de4e4729ae700af11df0dc5550e6587ed7891e92" },
978 | { file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:29e0656d5497733dcddc21797da5a2ab990c0cb9719f1f969e58a4abac66234d" },
979 | { file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:bf19725fec28452474d9887a128e98dd67eee7b7d52e932e6949c532d820dc3b" },
980 | { file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:d6f3d62e16c10e88d2168ba2d065aa374e3c538998ed04996cd373ff2036d64c" },
981 | { file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_s390x.whl", hash = "sha256:ac10bbac36cd89eac19f4e51c032ba6b412b3892b685076f4acd2de18ca990aa" },
982 | { file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:aa32aaa97d8b2ed4e54dc65d241a0da1c627454950f7d7b1f95b13985afd6c5d" },
983 | { file = "yarl-1.7.2-cp36-cp36m-win32.whl", hash = "sha256:87f6e082bce21464857ba58b569370e7b547d239ca22248be68ea5d6b51464a1" },
984 | { file = "yarl-1.7.2-cp36-cp36m-win_amd64.whl", hash = "sha256:ac35ccde589ab6a1870a484ed136d49a26bcd06b6a1c6397b1967ca13ceb3913" },
985 | { file = "yarl-1.7.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:a467a431a0817a292121c13cbe637348b546e6ef47ca14a790aa2fa8cc93df63" },
986 | { file = "yarl-1.7.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6ab0c3274d0a846840bf6c27d2c60ba771a12e4d7586bf550eefc2df0b56b3b4" },
987 | { file = "yarl-1.7.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d260d4dc495c05d6600264a197d9d6f7fc9347f21d2594926202fd08cf89a8ba" },
988 | { file = "yarl-1.7.2-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fc4dd8b01a8112809e6b636b00f487846956402834a7fd59d46d4f4267181c41" },
989 | { file = "yarl-1.7.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c1164a2eac148d85bbdd23e07dfcc930f2e633220f3eb3c3e2a25f6148c2819e" },
990 | { file = "yarl-1.7.2-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:67e94028817defe5e705079b10a8438b8cb56e7115fa01640e9c0bb3edf67332" },
991 | { file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:89ccbf58e6a0ab89d487c92a490cb5660d06c3a47ca08872859672f9c511fc52" },
992 | { file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:8cce6f9fa3df25f55521fbb5c7e4a736683148bcc0c75b21863789e5185f9185" },
993 | { file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:211fcd65c58bf250fb994b53bc45a442ddc9f441f6fec53e65de8cba48ded986" },
994 | { file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:c10ea1e80a697cf7d80d1ed414b5cb8f1eec07d618f54637067ae3c0334133c4" },
995 | { file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:52690eb521d690ab041c3919666bea13ab9fbff80d615ec16fa81a297131276b" },
996 | { file = "yarl-1.7.2-cp37-cp37m-win32.whl", hash = "sha256:695ba021a9e04418507fa930d5f0704edbce47076bdcfeeaba1c83683e5649d1" },
997 | { file = "yarl-1.7.2-cp37-cp37m-win_amd64.whl", hash = "sha256:c17965ff3706beedafd458c452bf15bac693ecd146a60a06a214614dc097a271" },
998 | { file = "yarl-1.7.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:fce78593346c014d0d986b7ebc80d782b7f5e19843ca798ed62f8e3ba8728576" },
999 | { file = "yarl-1.7.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:c2a1ac41a6aa980db03d098a5531f13985edcb451bcd9d00670b03129922cd0d" },
1000 | { file = "yarl-1.7.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:39d5493c5ecd75c8093fa7700a2fb5c94fe28c839c8e40144b7ab7ccba6938c8" },
1001 | { file = "yarl-1.7.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1eb6480ef366d75b54c68164094a6a560c247370a68c02dddb11f20c4c6d3c9d" },
1002 | { file = "yarl-1.7.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5ba63585a89c9885f18331a55d25fe81dc2d82b71311ff8bd378fc8004202ff6" },
1003 | { file = "yarl-1.7.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e39378894ee6ae9f555ae2de332d513a5763276a9265f8e7cbaeb1b1ee74623a" },
1004 | { file = "yarl-1.7.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c0910c6b6c31359d2f6184828888c983d54d09d581a4a23547a35f1d0b9484b1" },
1005 | { file = "yarl-1.7.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:6feca8b6bfb9eef6ee057628e71e1734caf520a907b6ec0d62839e8293e945c0" },
1006 | { file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:8300401dc88cad23f5b4e4c1226f44a5aa696436a4026e456fe0e5d2f7f486e6" },
1007 | { file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:788713c2896f426a4e166b11f4ec538b5736294ebf7d5f654ae445fd44270832" },
1008 | { file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:fd547ec596d90c8676e369dd8a581a21227fe9b4ad37d0dc7feb4ccf544c2d59" },
1009 | { file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:737e401cd0c493f7e3dd4db72aca11cfe069531c9761b8ea474926936b3c57c8" },
1010 | { file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:baf81561f2972fb895e7844882898bda1eef4b07b5b385bcd308d2098f1a767b" },
1011 | { file = "yarl-1.7.2-cp38-cp38-win32.whl", hash = "sha256:ede3b46cdb719c794427dcce9d8beb4abe8b9aa1e97526cc20de9bd6583ad1ef" },
1012 | { file = "yarl-1.7.2-cp38-cp38-win_amd64.whl", hash = "sha256:cc8b7a7254c0fc3187d43d6cb54b5032d2365efd1df0cd1749c0c4df5f0ad45f" },
1013 | { file = "yarl-1.7.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:580c1f15500e137a8c37053e4cbf6058944d4c114701fa59944607505c2fe3a0" },
1014 | { file = "yarl-1.7.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:3ec1d9a0d7780416e657f1e405ba35ec1ba453a4f1511eb8b9fbab81cb8b3ce1" },
1015 | { file = "yarl-1.7.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:3bf8cfe8856708ede6a73907bf0501f2dc4e104085e070a41f5d88e7faf237f3" },
1016 | { file = "yarl-1.7.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1be4bbb3d27a4e9aa5f3df2ab61e3701ce8fcbd3e9846dbce7c033a7e8136746" },
1017 | { file = "yarl-1.7.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:534b047277a9a19d858cde163aba93f3e1677d5acd92f7d10ace419d478540de" },
1018 | { file = "yarl-1.7.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c6ddcd80d79c96eb19c354d9dca95291589c5954099836b7c8d29278a7ec0bda" },
1019 | { file = "yarl-1.7.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:9bfcd43c65fbb339dc7086b5315750efa42a34eefad0256ba114cd8ad3896f4b" },
1020 | { file = "yarl-1.7.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f64394bd7ceef1237cc604b5a89bf748c95982a84bcd3c4bbeb40f685c810794" },
1021 | { file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:044daf3012e43d4b3538562da94a88fb12a6490652dbc29fb19adfa02cf72eac" },
1022 | { file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:368bcf400247318382cc150aaa632582d0780b28ee6053cd80268c7e72796dec" },
1023 | { file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:bab827163113177aee910adb1f48ff7af31ee0289f434f7e22d10baf624a6dfe" },
1024 | { file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:0cba38120db72123db7c58322fa69e3c0efa933040ffb586c3a87c063ec7cae8" },
1025 | { file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:59218fef177296451b23214c91ea3aba7858b4ae3306dde120224cfe0f7a6ee8" },
1026 | { file = "yarl-1.7.2-cp39-cp39-win32.whl", hash = "sha256:1edc172dcca3f11b38a9d5c7505c83c1913c0addc99cd28e993efeaafdfaa18d" },
1027 | { file = "yarl-1.7.2-cp39-cp39-win_amd64.whl", hash = "sha256:797c2c412b04403d2da075fb93c123df35239cd7b4cc4e0cd9e5839b73f52c58" },
1028 | { file = "yarl-1.7.2.tar.gz", hash = "sha256:45399b46d60c253327a460e99856752009fcee5f5d3c80b2f7c0cae1c38d56dd" },
1029 | ]
1030 |
--------------------------------------------------------------------------------
/pyproject.toml:
--------------------------------------------------------------------------------
1 | [tool.poetry]
2 | name = "op_tracker"
3 | version = "0.1.0"
4 | description = ""
5 | authors = ["yshalsager "]
6 |
7 | [tool.poetry.dependencies]
8 | python = "^3.9"
9 | humanize = "^2.4.0"
10 | PyYAML = "^5.3.1"
11 | python-telegram-bot = "^12.7"
12 | aiohttp = "^3.6.2"
13 | beautifulsoup4 = "^4.9.1"
14 | sqlalchemy = "^1.3.17"
15 |
16 | [tool.poetry.group.dev.dependencies]
17 | black = "^22.8.0"
18 | isort = "^5.10.1"
19 |
20 | [build-system]
21 | requires = ["poetry>=0.12"]
22 | build-backend = "poetry.masonry.api"
23 |
--------------------------------------------------------------------------------