├── .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 | [![Subscibe](https://img.shields.io/badge/Telegram-Subscribe-blue.svg)](https://t.me/OnePlusUpdatesTracker) 4 | 5 | [![Open Source Love](https://badges.frapsoft.com/os/v1/open-source.png?v=103)](https://github.com/ellerbrock/open-source-badges/) 6 | [![made-with-python](https://img.shields.io/badge/Made%20with-Python-1f425f.svg)](https://www.python.org/) 7 | 8 | [![PayPal](https://img.shields.io/badge/PayPal-Donate-blue.svg)](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 | --------------------------------------------------------------------------------