pax_global_header00006660000000000000000000000064150244775220014522gustar00rootroot0000000000000052 comment=b8564f7e3da36bae6070a079c696c0789cce164d a2wsgi-1.10.10/000077500000000000000000000000001502447752200130565ustar00rootroot00000000000000a2wsgi-1.10.10/.coveragerc000066400000000000000000000006601502447752200152010ustar00rootroot00000000000000[run] omit = .venv/* */tests/* [report] # Regexes for lines to exclude from consideration exclude_lines = # Don't complain about missing debug-only code: def __repr__ # Don't complain if tests don't hit defensive assertion code: raise AssertionError raise NotImplementedError # Don't complain if non-runnable code isn't run: if False: assert False show_missing = true skip_covered = true a2wsgi-1.10.10/.gitattributes000066400000000000000000000000141502447752200157440ustar00rootroot00000000000000* text=auto a2wsgi-1.10.10/.github/000077500000000000000000000000001502447752200144165ustar00rootroot00000000000000a2wsgi-1.10.10/.github/workflows/000077500000000000000000000000001502447752200164535ustar00rootroot00000000000000a2wsgi-1.10.10/.github/workflows/ci.yml000066400000000000000000000023021502447752200175660ustar00rootroot00000000000000name: CI/CD on: [push, pull_request] jobs: tests: name: "Python ${{ matrix.python-version }} ${{ matrix.os }}" runs-on: "${{ matrix.os }}" strategy: matrix: python-version: [3.8, 3.9, "3.10", "3.11", "3.12"] os: [ubuntu-latest, windows-latest, macos-13] steps: - uses: actions/checkout@v4 - uses: pdm-project/setup-pdm@v4 name: Setup Python and PDM with: python-version: ${{ matrix.python-version }} architecture: x64 version: 2.20.1 - name: Install dependencies run: | pdm sync -v -dG dev -dG test --no-self - name: Tests run: pdm run pytest tests -o log_cli=true -o log_cli_level=DEBUG env: PYTHONASYNCIODEBUG: 1 publish: needs: tests if: startsWith(github.ref, 'refs/tags/') runs-on: ubuntu-latest steps: - uses: actions/checkout@v4 - uses: pdm-project/setup-pdm@v4 name: Setup Python and PDM with: python-version: "3.10" architecture: x64 version: 2.20.1 - name: Publish run: | pdm publish --username __token__ --password ${{ secrets.PYPI_API_TOKEN }} a2wsgi-1.10.10/.gitignore000066400000000000000000000022051502447752200150450ustar00rootroot00000000000000# Byte-compiled / optimized / DLL files __pycache__/ *.py[cod] *$py.class # C extensions *.so # Distribution / packaging .Python env/ build/ develop-eggs/ dist/ downloads/ eggs/ .eggs/ lib/ lib64/ parts/ sdist/ var/ *.egg-info/ .installed.cfg *.egg # PyInstaller # Usually these files are written by a python script from a template # before PyInstaller builds the exe, so as to inject date/other infos into it. *.manifest *.spec # Installer logs pip-log.txt pip-delete-this-directory.txt # Unit test / coverage reports htmlcov/ .tox/ .coverage .coverage.* .cache nosetests.xml coverage.xml *,cover .hypothesis/ # Translations *.mo *.pot # Django stuff: *.log local_settings.py # Flask stuff: instance/ .webassets-cache # Scrapy stuff: .scrapy # Sphinx documentation docs/_build/ # PyBuilder target/ # IPython Notebook .ipynb_checkpoints # pyenv .python-version # celery beat schedule file celerybeat-schedule # dotenv .env # virtualenv venv/ ENV/ .venv/ # Spyder project settings .spyderproject # Rope project settings .ropeproject *.npy *.pkl # mypy .mypy_cache/ # VSCode .vscode/ # PyCharm .idea/ # mkdocs site/ # PDM .pdm-python a2wsgi-1.10.10/LICENSE000066400000000000000000000261151502447752200140700ustar00rootroot00000000000000 Apache License Version 2.0, January 2004 http://www.apache.org/licenses/ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 1. Definitions. "License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. "Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. "Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. "You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. "Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. "Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. "Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). "Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. "Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." "Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. 2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. 3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. 4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: (a) You must give any other recipients of the Work or Derivative Works a copy of this License; and (b) You must cause any modified files to carry prominent notices stating that You changed the files; and (c) You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and (d) If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. 5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. 6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. 7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. 8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. 9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. END OF TERMS AND CONDITIONS APPENDIX: How to apply the Apache License to your work. To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets "[]" replaced with your own identifying information. (Don't include the brackets!) The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same "printed page" as the copyright notice for easier identification within third-party archives. Copyright 2022 abersheeran Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. a2wsgi-1.10.10/README.md000066400000000000000000000067221502447752200143440ustar00rootroot00000000000000# a2wsgi Convert WSGI app to ASGI app or ASGI app to WSGI app. Pure Python. Only depend on the standard library. Compared with other converters, the advantage is that a2wsgi will not accumulate the requested content or response content in the memory, so you don't have to worry about the memory limit caused by a2wsgi. This problem exists in converters implemented by uvicorn/startlette or hypercorn. ## Install ``` pip install a2wsgi ``` ## How to use ### `WSGIMiddleware` Convert WSGI app to ASGI app: ```python from a2wsgi import WSGIMiddleware ASGI_APP = WSGIMiddleware(WSGI_APP) ``` WSGIMiddleware executes WSGI applications with a thread pool of up to 10 threads by default. If you want to increase or decrease this number, just like `WSGIMiddleware(..., workers=15)`. WSGIMiddleware utilizes a queue to direct traffic from the WSGI App to the client. To adjust the queue size, simply specify the send_queue_size parameter (default to `10`) during initialization, like so: WSGIMiddleware(..., send_queue_size=15). This enable developers to balance memory usage and application responsiveness. ### `ASGIMiddleware` Convert ASGI app to WSGI app: ```python from a2wsgi import ASGIMiddleware WSGI_APP = ASGIMiddleware(ASGI_APP) ``` `ASGIMiddleware` will wait for the ASGI application's Background Task to complete before returning the last null byte. But sometimes you may not want to wait indefinitely for the execution of the Background Task of the ASGI application, then you only need to give the parameter `ASGIMiddleware(..., wait_time=5.0)`, after the time exceeds, the ASGI task corresponding to the request will be tried to cancel, and the last null byte will be returned. You can also specify your own event loop through the `loop` parameter instead of the default event loop. Like `ASGIMiddleware(..., loop=faster_loop)` ### Access the original `Scope`/`Environ` Sometimes you may need to access the original WSGI Environ in the ASGI application, just use `scope["wsgi_environ"]`; it is also easy to access the ASGI Scope in the WSGI Application, use `environ["asgi.scope"]`. ## Benchmark Run `pytest ./benchmark.py -s` to compare the performance of `a2wsgi` and `uvicorn.middleware.wsgi.WSGIMiddleware` / `asgiref.wsgi.WsgiToAsgi`. ## Why a2wsgi ### Convert WSGI app to ASGI app You can convert an existing WSGI project to an ASGI project to make it easier to migrate from WSGI applications to ASGI applications. ### Convert ASGI app to WSGI app There is a lot of support for WSGI. Converting ASGI to WSGI, you will be able to use many existing services to deploy ASGI applications. ## Compatibility list This list quickly demonstrates the compatibility of some common frameworks for users who are unfamiliar with the WSGI and ASGI protocols. - WSGI: [Django(wsgi)](https://docs.djangoproject.com/en/3.0/howto/deployment/wsgi/)/[Kuí(wsgi)](https://kui.aber.sh/wsgi/)/[Falcon(wsgi)](https://falcon.readthedocs.io/en/stable/api/app.html#wsgi-app)/[Pyramid](https://trypyramid.com/)/[Bottle](https://bottlepy.org/)/[Flask](https://flask.palletsprojects.com/) - ASGI: [Django(asgi)](https://docs.djangoproject.com/en/3.0/howto/deployment/asgi/)/[Kuí(asgi)](https://kui.aber.sh/asgi/)/[Falcon(asgi)](https://falcon.readthedocs.io/en/stable/api/app.html#asgi-app)/[Starlette](https://www.starlette.io/)/[FastAPI](https://fastapi.tiangolo.com/)/[Sanic](https://sanic.readthedocs.io/en/stable/)/[Quart](https://pgjones.gitlab.io/quart/) - **Unsupport**: [aiohttp](https://docs.aiohttp.org/en/stable/) a2wsgi-1.10.10/a2wsgi/000077500000000000000000000000001502447752200142525ustar00rootroot00000000000000a2wsgi-1.10.10/a2wsgi/__init__.py000066400000000000000000000002711502447752200163630ustar00rootroot00000000000000from .asgi import ASGIMiddleware from .wsgi import WSGIMiddleware VERSION = (1, 10, 10) __version__: str = ".".join(map(str, VERSION)) __all__ = ("WSGIMiddleware", "ASGIMiddleware") a2wsgi-1.10.10/a2wsgi/asgi.py000066400000000000000000000241731502447752200155560ustar00rootroot00000000000000import asyncio import collections import threading from http import HTTPStatus from io import BytesIO from typing import Any, Coroutine, Deque, Iterable, Optional, TypeVar from typing import cast as typing_cast from .asgi_typing import HTTPScope, ASGIApp, ReceiveEvent, SendEvent from .wsgi_typing import Environ, StartResponse, IterableChunks T = TypeVar("T") class defaultdict(dict): def __init__(self, default_factory, *args, **kwargs) -> None: self.default_factory = default_factory super().__init__(*args, **kwargs) def __missing__(self, key): return self.default_factory(key) StatusStringMapping = defaultdict( lambda status: f"{status} Unknown Status Code", {status.value: f"{status.value} {status.phrase}" for status in HTTPStatus}, ) class AsyncEvent: def __init__(self, loop: asyncio.AbstractEventLoop) -> None: self.loop = loop self.__waiters: Deque[asyncio.Future] = collections.deque() self.__nowait = False def _set(self, message: Any) -> None: for future in filter(lambda f: not f.done(), self.__waiters): future.set_result(message) def set(self, message: Any) -> None: self.loop.call_soon_threadsafe(self._set, message) async def wait(self) -> Any: if self.__nowait: return None future = self.loop.create_future() self.__waiters.append(future) try: result = await future return result finally: self.__waiters.remove(future) def set_nowait(self) -> None: self.__nowait = True class SyncEvent: def __init__(self) -> None: self.__write_event = threading.Event() self.__message: Any = None def set(self, message: Any) -> None: self.__message = message self.__write_event.set() def wait(self) -> Any: self.__write_event.wait() self.__write_event.clear() message, self.__message = self.__message, None return message def build_scope(environ: Environ) -> HTTPScope: headers = [ ( (key[5:] if key.startswith("HTTP_") else key) .lower() .replace("_", "-") .encode("latin-1"), value.encode("latin-1"), # type: ignore ) for key, value in environ.items() if ( key.startswith("HTTP_") and key not in ("HTTP_CONTENT_TYPE", "HTTP_CONTENT_LENGTH") ) or key in ("CONTENT_TYPE", "CONTENT_LENGTH") ] root_path = environ.get("SCRIPT_NAME", "").encode("latin1").decode("utf8") path = root_path + environ.get("PATH_INFO", "").encode("latin1").decode("utf8") scope: HTTPScope = { "wsgi_environ": environ, # type: ignore a2wsgi "type": "http", "asgi": {"version": "3.0", "spec_version": "2.5"}, "http_version": environ.get("SERVER_PROTOCOL", "http/1.0").split("/")[1], "method": environ["REQUEST_METHOD"], "scheme": environ.get("wsgi.url_scheme", "http"), "path": path, "query_string": environ.get("QUERY_STRING", "").encode("ascii"), "root_path": root_path, "server": (environ["SERVER_NAME"], int(environ["SERVER_PORT"])), "headers": headers, "extensions": {}, } if environ.get("REMOTE_ADDR") and environ.get("REMOTE_PORT"): client = (environ.get("REMOTE_ADDR", ""), int(environ.get("REMOTE_PORT", "0"))) scope["client"] = client return scope class ASGIMiddleware: """ Convert ASGIApp to WSGIApp. wait_time: After the http response ends, the maximum time to wait for the ASGI app to run. """ def __init__( self, app: ASGIApp, wait_time: Optional[float] = None, loop: Optional[asyncio.AbstractEventLoop] = None, ) -> None: self.app = app if loop is None: loop = asyncio.new_event_loop() loop_threading = threading.Thread(target=loop.run_forever, daemon=True) loop_threading.start() self.loop = loop self.wait_time = wait_time def __call__( self, environ: Environ, start_response: StartResponse ) -> Iterable[bytes]: return ASGIResponder(self.app, self.loop, self.wait_time)( environ, start_response ) class ASGIResponder: def __init__( self, app: ASGIApp, loop: asyncio.AbstractEventLoop, wait_time: Optional[float] = None, ) -> None: self.app = app self.loop = loop self.wait_time = wait_time self.sync_event = SyncEvent() self.sync_event_set_lock: asyncio.Lock self.receive_event = AsyncEvent(loop) self.send_event = AsyncEvent(loop) def _init_async_lock(): self.sync_event_set_lock = asyncio.Lock() loop.call_soon_threadsafe(_init_async_lock) self.asgi_done = threading.Event() self.wsgi_should_stop: bool = False async def asgi_receive(self) -> ReceiveEvent: await self.sync_event_set_lock.acquire() self.sync_event.set({"type": "receive"}) return await self.receive_event.wait() async def asgi_send(self, message: SendEvent) -> None: await self.sync_event_set_lock.acquire() self.sync_event.set(message) await self.send_event.wait() def asgi_done_callback(self, future: asyncio.Future) -> None: try: exception = future.exception() except asyncio.CancelledError: pass else: if exception is not None: task = asyncio.create_task(self.sync_event_set_lock.acquire()) task.add_done_callback( lambda _: self.sync_event.set( { "type": "a2wsgi.error", "exception": ( type(exception), exception, exception.__traceback__, ), } ) ) finally: self.asgi_done.set() async def start_asgi_app(self, environ: Environ) -> asyncio.Task: run_asgi: asyncio.Task = self.loop.create_task( typing_cast( Coroutine[None, None, None], self.app(build_scope(environ), self.asgi_receive, self.asgi_send), ) ) run_asgi.add_done_callback(self.asgi_done_callback) return run_asgi def execute_in_loop(self, coro: Coroutine[None, None, T]) -> T: return asyncio.run_coroutine_threadsafe(coro, self.loop).result() def __call__( self, environ: Environ, start_response: StartResponse ) -> IterableChunks: read_count: int = 0 body = environ["wsgi.input"] or BytesIO() content_length = int(environ.get("CONTENT_LENGTH", None) or 0) receive_eof = False body_sent = False asgi_task = self.execute_in_loop(self.start_asgi_app(environ)) # activate loop self.loop.call_soon_threadsafe(lambda: None) while True: message = self.sync_event.wait() self.loop.call_soon_threadsafe(self.sync_event_set_lock.release) message_type = message["type"] if message_type == "http.response.start": start_response( StatusStringMapping[message["status"]], [ ( name.strip().decode("latin1"), value.strip().decode("latin1"), ) for name, value in message["headers"] ], None, ) self.send_event.set(None) elif message_type == "http.response.body": yield message.get("body", b"") body_sent = True self.wsgi_should_stop = not message.get("more_body", False) self.send_event.set(None) elif message_type == "http.response.disconnect": self.wsgi_should_stop = True self.send_event.set(None) # ASGI application error elif message_type == "a2wsgi.error": if body_sent: raise message["exception"][1].with_traceback( message["exception"][2] ) start_response( "500 Internal Server Error", [ ("Content-Type", "text/plain; charset=utf-8"), ("Content-Length", "28"), ], message["exception"], ) yield b"Server got itself in trouble" self.wsgi_should_stop = True elif message_type == "receive": read_size = min(65536, content_length - read_count) if read_size == 0: # No more body, so don't read anymore if not receive_eof: self.receive_event.set( {"type": "http.request", "body": b"", "more_body": False} ) receive_eof = True else: pass # let `await receive()` wait else: data: bytes = body.read(read_size) read_count += len(data) more_body = read_count < content_length self.receive_event.set( {"type": "http.request", "body": data, "more_body": more_body} ) if more_body is False: receive_eof = True else: raise RuntimeError(f"Unknown message type: {message_type}") if self.wsgi_should_stop: self.receive_event.set({"type": "http.disconnect"}) break if self.asgi_done.is_set(): break # HTTP response ends, wait for run_asgi's background tasks self.asgi_done.wait(self.wait_time) self.loop.call_soon_threadsafe(asgi_task.cancel) yield b"" a2wsgi-1.10.10/a2wsgi/asgi_typing.py000066400000000000000000000100261502447752200171400ustar00rootroot00000000000000""" https://asgi.readthedocs.io/en/latest/specs/index.html """ import sys from typing import ( Any, Awaitable, Callable, Dict, Iterable, Literal, Optional, Tuple, TypedDict, Union, ) if sys.version_info >= (3, 11): from typing import NotRequired else: from typing_extensions import NotRequired class ASGIVersions(TypedDict): spec_version: str version: Literal["3.0"] class HTTPScope(TypedDict): type: Literal["http"] asgi: ASGIVersions http_version: str method: str scheme: str path: str raw_path: NotRequired[bytes] query_string: bytes root_path: str headers: Iterable[Tuple[bytes, bytes]] client: NotRequired[Tuple[str, int]] server: NotRequired[Tuple[str, Optional[int]]] state: NotRequired[Dict[str, Any]] extensions: NotRequired[Dict[str, Dict[object, object]]] class WebSocketScope(TypedDict): type: Literal["websocket"] asgi: ASGIVersions http_version: str scheme: str path: str raw_path: bytes query_string: bytes root_path: str headers: Iterable[Tuple[bytes, bytes]] client: NotRequired[Tuple[str, int]] server: NotRequired[Tuple[str, Optional[int]]] subprotocols: Iterable[str] state: NotRequired[Dict[str, Any]] extensions: NotRequired[Dict[str, Dict[object, object]]] class LifespanScope(TypedDict): type: Literal["lifespan"] asgi: ASGIVersions state: NotRequired[Dict[str, Any]] WWWScope = Union[HTTPScope, WebSocketScope] Scope = Union[HTTPScope, WebSocketScope, LifespanScope] class HTTPRequestEvent(TypedDict): type: Literal["http.request"] body: bytes more_body: NotRequired[bool] class HTTPResponseStartEvent(TypedDict): type: Literal["http.response.start"] status: int headers: NotRequired[Iterable[Tuple[bytes, bytes]]] trailers: NotRequired[bool] class HTTPResponseBodyEvent(TypedDict): type: Literal["http.response.body"] body: NotRequired[bytes] more_body: NotRequired[bool] class HTTPDisconnectEvent(TypedDict): type: Literal["http.disconnect"] class WebSocketConnectEvent(TypedDict): type: Literal["websocket.connect"] class WebSocketAcceptEvent(TypedDict): type: Literal["websocket.accept"] subprotocol: NotRequired[str] headers: NotRequired[Iterable[Tuple[bytes, bytes]]] class WebSocketReceiveEvent(TypedDict): type: Literal["websocket.receive"] bytes: NotRequired[bytes] text: NotRequired[str] class WebSocketSendEvent(TypedDict): type: Literal["websocket.send"] bytes: NotRequired[bytes] text: NotRequired[str] class WebSocketDisconnectEvent(TypedDict): type: Literal["websocket.disconnect"] code: int class WebSocketCloseEvent(TypedDict): type: Literal["websocket.close"] code: NotRequired[int] reason: NotRequired[str] class LifespanStartupEvent(TypedDict): type: Literal["lifespan.startup"] class LifespanShutdownEvent(TypedDict): type: Literal["lifespan.shutdown"] class LifespanStartupCompleteEvent(TypedDict): type: Literal["lifespan.startup.complete"] class LifespanStartupFailedEvent(TypedDict): type: Literal["lifespan.startup.failed"] message: str class LifespanShutdownCompleteEvent(TypedDict): type: Literal["lifespan.shutdown.complete"] class LifespanShutdownFailedEvent(TypedDict): type: Literal["lifespan.shutdown.failed"] message: str ReceiveEvent = Union[ HTTPRequestEvent, HTTPDisconnectEvent, WebSocketConnectEvent, WebSocketReceiveEvent, WebSocketDisconnectEvent, LifespanStartupEvent, LifespanShutdownEvent, ] SendEvent = Union[ HTTPResponseStartEvent, HTTPResponseBodyEvent, HTTPDisconnectEvent, WebSocketAcceptEvent, WebSocketSendEvent, WebSocketCloseEvent, LifespanStartupCompleteEvent, LifespanStartupFailedEvent, LifespanShutdownCompleteEvent, LifespanShutdownFailedEvent, ] Receive = Callable[[], Awaitable[ReceiveEvent]] Send = Callable[[SendEvent], Awaitable[None]] ASGIApp = Callable[[Scope, Receive, Send], Awaitable[None]] a2wsgi-1.10.10/a2wsgi/py.typed000066400000000000000000000000001502447752200157370ustar00rootroot00000000000000a2wsgi-1.10.10/a2wsgi/wsgi.py000066400000000000000000000222261502447752200156010ustar00rootroot00000000000000import asyncio import contextvars import functools import os import sys import typing from concurrent.futures import ThreadPoolExecutor from .asgi_typing import HTTPScope, Scope, Receive, Send, SendEvent from .wsgi_typing import Environ, StartResponse, ExceptionInfo, WSGIApp, WriteCallable class Body: def __init__(self, loop: asyncio.AbstractEventLoop, receive: Receive) -> None: self.buffer = bytearray() self.loop = loop self.receive = receive self._has_more = True @property def has_more(self) -> bool: if self._has_more or self.buffer: return True return False def _receive_more_data(self) -> bytes: if not self._has_more: return b"" future = asyncio.run_coroutine_threadsafe(self.receive(), loop=self.loop) message = future.result() self._has_more = message.get("more_body", False) return message.get("body", b"") def read(self, size: int = -1) -> bytes: while size == -1 or size > len(self.buffer): self.buffer.extend(self._receive_more_data()) if not self._has_more: break if size == -1: result = bytes(self.buffer) self.buffer.clear() else: result = bytes(self.buffer[:size]) del self.buffer[:size] return result def readline(self, limit: int = -1) -> bytes: while True: lf_index = self.buffer.find(b"\n", 0, limit if limit > -1 else None) if lf_index != -1: result = bytes(self.buffer[: lf_index + 1]) del self.buffer[: lf_index + 1] return result elif limit != -1: result = bytes(self.buffer[:limit]) del self.buffer[:limit] return result if not self._has_more: break self.buffer.extend(self._receive_more_data()) result = bytes(self.buffer) self.buffer.clear() return result def readlines(self, hint: int = -1) -> typing.List[bytes]: if not self.has_more: return [] if hint == -1: raw_data = self.read(-1) bytelist = raw_data.split(b"\n") if raw_data[-1] == 10: # 10 -> b"\n" bytelist.pop(len(bytelist) - 1) return [line + b"\n" for line in bytelist] return [self.readline() for _ in range(hint)] def __iter__(self) -> typing.Generator[bytes, None, None]: while self.has_more: yield self.readline() ENC, ESC = sys.getfilesystemencoding(), "surrogateescape" def unicode_to_wsgi(u): """Convert an environment variable to a WSGI "bytes-as-unicode" string""" return u.encode(ENC, ESC).decode("iso-8859-1") def build_environ(scope: HTTPScope, body: Body) -> Environ: """ Builds a scope and request body into a WSGI environ object. """ script_name = scope.get("root_path", "").encode("utf8").decode("latin1") path_info = scope["path"].encode("utf8").decode("latin1") if path_info.startswith(script_name): path_info = path_info[len(script_name) :] script_name_environ_var = os.environ.get("SCRIPT_NAME", "") if script_name_environ_var: script_name = unicode_to_wsgi(script_name_environ_var) environ: Environ = { "asgi.scope": scope, # type: ignore a2wsgi "REQUEST_METHOD": scope["method"], "SCRIPT_NAME": script_name, "PATH_INFO": path_info, "QUERY_STRING": scope["query_string"].decode("ascii"), "SERVER_PROTOCOL": f"HTTP/{scope['http_version']}", "wsgi.version": (1, 0), "wsgi.url_scheme": scope.get("scheme", "http"), "wsgi.input": body, "wsgi.errors": sys.stdout, "wsgi.multithread": True, "wsgi.multiprocess": True, "wsgi.run_once": False, } # Get server name and port - required in WSGI, not in ASGI server_addr, server_port = scope.get("server") or ("localhost", 80) environ["SERVER_NAME"] = server_addr environ["SERVER_PORT"] = str(server_port or 0) # Get client IP address client = scope.get("client") if client is not None: addr, port = client environ["REMOTE_ADDR"] = addr environ["REMOTE_PORT"] = str(port) # Go through headers and make them into environ entries for name, value in scope.get("headers", []): name = name.decode("latin1") if name == "content-length": corrected_name = "CONTENT_LENGTH" elif name == "content-type": corrected_name = "CONTENT_TYPE" else: corrected_name = f"HTTP_{name}".upper().replace("-", "_") # HTTPbis say only ASCII chars are allowed in headers, but we latin1 just in case value = value.decode("latin1") if corrected_name in environ: value = environ[corrected_name] + "," + value environ[corrected_name] = value return environ class WSGIMiddleware: """ Convert WSGIApp to ASGIApp. """ def __init__( self, app: WSGIApp, workers: int = 10, send_queue_size: int = 10 ) -> None: self.app = app self.send_queue_size = send_queue_size self.executor = ThreadPoolExecutor( thread_name_prefix="WSGI", max_workers=workers ) async def __call__(self, scope: Scope, receive: Receive, send: Send) -> None: if scope["type"] == "http": responder = WSGIResponder(self.app, self.executor, self.send_queue_size) return await responder(scope, receive, send) if scope["type"] == "websocket": await send({"type": "websocket.close", "code": 1000}) return if scope["type"] == "lifespan": message = await receive() assert message["type"] == "lifespan.startup" await send({"type": "lifespan.startup.complete"}) message = await receive() assert message["type"] == "lifespan.shutdown" await send({"type": "lifespan.shutdown.complete"}) return class WSGIResponder: def __init__( self, app: WSGIApp, executor: ThreadPoolExecutor, send_queue_size: int ) -> None: self.app = app self.executor = executor self.loop = asyncio.get_event_loop() self.send_queue = asyncio.Queue(send_queue_size) self.response_started = False self.exc_info: typing.Any = None async def __call__(self, scope: HTTPScope, receive: Receive, send: Send) -> None: body = Body(self.loop, receive) environ = build_environ(scope, body) sender = None try: sender = self.loop.create_task(self.sender(send)) context = contextvars.copy_context() func = functools.partial(context.run, self.wsgi) await self.loop.run_in_executor( self.executor, func, environ, self.start_response ) await self.send_queue.put(None) # Sender may raise an exception, so we need to await it # dont await send_queue.join() because it will never finish await sender if self.exc_info is not None: raise self.exc_info[0].with_traceback( self.exc_info[1], self.exc_info[2] ) finally: if sender and not sender.done(): sender.cancel() # pragma: no cover def send(self, message: typing.Optional[SendEvent]) -> None: future = asyncio.run_coroutine_threadsafe( self.send_queue.put(message), loop=self.loop ) future.result() async def sender(self, send: Send) -> None: while True: message = await self.send_queue.get() if message is None: break await send(message) self.send_queue.task_done() def start_response( self, status: str, response_headers: typing.List[typing.Tuple[str, str]], exc_info: typing.Optional[ExceptionInfo] = None, ) -> WriteCallable: self.exc_info = exc_info if not self.response_started: self.response_started = True status_code_string, _ = status.split(" ", 1) status_code = int(status_code_string) headers = [ (name.strip().encode("latin1").lower(), value.strip().encode("latin1")) for name, value in response_headers ] self.send( { "type": "http.response.start", "status": status_code, "headers": headers, } ) return lambda chunk: self.send( {"type": "http.response.body", "body": chunk, "more_body": True} ) def wsgi(self, environ: Environ, start_response: StartResponse) -> None: iterable = self.app(environ, start_response) try: for chunk in iterable: self.send( {"type": "http.response.body", "body": chunk, "more_body": True} ) self.send({"type": "http.response.body", "body": b""}) finally: getattr(iterable, "close", lambda: None)() a2wsgi-1.10.10/a2wsgi/wsgi_typing.py000066400000000000000000000173401502447752200171740ustar00rootroot00000000000000""" https://peps.python.org/pep-3333/ """ from types import TracebackType from typing import ( Any, Callable, Iterable, List, Optional, Protocol, Tuple, Type, TypedDict, ) CGIRequiredDefined = TypedDict( "CGIRequiredDefined", { # The HTTP request method, such as GET or POST. This cannot ever be an # empty string, and so is always required. "REQUEST_METHOD": str, # When HTTP_HOST is not set, these variables can be combined to determine # a default. # SERVER_NAME and SERVER_PORT are required strings and must never be empty. "SERVER_NAME": str, "SERVER_PORT": str, # The version of the protocol the client used to send the request. # Typically this will be something like "HTTP/1.0" or "HTTP/1.1" and # may be used by the application to determine how to treat any HTTP # request headers. (This variable should probably be called REQUEST_PROTOCOL, # since it denotes the protocol used in the request, and is not necessarily # the protocol that will be used in the server's response. However, for # compatibility with CGI we have to keep the existing name.) "SERVER_PROTOCOL": str, }, ) CGIOptionalDefined = TypedDict( "CGIOptionalDefined", { "REQUEST_URI": str, "REMOTE_ADDR": str, "REMOTE_PORT": str, # The initial portion of the request URL’s “path” that corresponds to the # application object, so that the application knows its virtual “location”. # This may be an empty string, if the application corresponds to the “root” # of the server. "SCRIPT_NAME": str, # The remainder of the request URL’s “path”, designating the virtual # “location” of the request’s target within the application. This may be an # empty string, if the request URL targets the application root and does # not have a trailing slash. "PATH_INFO": str, # The portion of the request URL that follows the “?”, if any. May be empty # or absent. "QUERY_STRING": str, # The contents of any Content-Type fields in the HTTP request. May be empty # or absent. "CONTENT_TYPE": str, # The contents of any Content-Length fields in the HTTP request. May be empty # or absent. "CONTENT_LENGTH": str, }, total=False, ) class InputStream(Protocol): """ An input stream (file-like object) from which the HTTP request body bytes can be read. (The server or gateway may perform reads on-demand as requested by the application, or it may pre- read the client's request body and buffer it in-memory or on disk, or use any other technique for providing such an input stream, according to its preference.) """ def read(self, size: int = -1, /) -> bytes: """ The server is not required to read past the client's specified Content-Length, and should simulate an end-of-file condition if the application attempts to read past that point. The application should not attempt to read more data than is specified by the CONTENT_LENGTH variable. A server should allow read() to be called without an argument, and return the remainder of the client's input stream. A server should return empty bytestrings from any attempt to read from an empty or exhausted input stream. """ raise NotImplementedError def readline(self, limit: int = -1, /) -> bytes: """ Servers should support the optional "size" argument to readline(), but as in WSGI 1.0, they are allowed to omit support for it. (In WSGI 1.0, the size argument was not supported, on the grounds that it might have been complex to implement, and was not often used in practice... but then the cgi module started using it, and so practical servers had to start supporting it anyway!) """ raise NotImplementedError def readlines(self, hint: int = -1, /) -> List[bytes]: """ Note that the hint argument to readlines() is optional for both caller and implementer. The application is free not to supply it, and the server or gateway is free to ignore it. """ raise NotImplementedError class ErrorStream(Protocol): """ An output stream (file-like object) to which error output can be written, for the purpose of recording program or other errors in a standardized and possibly centralized location. This should be a "text mode" stream; i.e., applications should use "\n" as a line ending, and assume that it will be converted to the correct line ending by the server/gateway. (On platforms where the str type is unicode, the error stream should accept and log arbitrary unicode without raising an error; it is allowed, however, to substitute characters that cannot be rendered in the stream's encoding.) For many servers, wsgi.errors will be the server's main error log. Alternatively, this may be sys.stderr, or a log file of some sort. The server's documentation should include an explanation of how to configure this or where to find the recorded output. A server or gateway may supply different error streams to different applications, if this is desired. """ def flush(self) -> None: """ Since the errors stream may not be rewound, servers and gateways are free to forward write operations immediately, without buffering. In this case, the flush() method may be a no-op. Portable applications, however, cannot assume that output is unbuffered or that flush() is a no-op. They must call flush() if they need to ensure that output has in fact been written. (For example, to minimize intermingling of data from multiple processes writing to the same error log.) """ raise NotImplementedError def write(self, s: str, /) -> Any: raise NotImplementedError def writelines(self, seq: List[str], /) -> Any: raise NotImplementedError WSGIDefined = TypedDict( "WSGIDefined", { "wsgi.version": Tuple[int, int], # e.g. (1, 0) "wsgi.url_scheme": str, # e.g. "http" or "https" "wsgi.input": InputStream, "wsgi.errors": ErrorStream, # This value should evaluate true if the application object may be simultaneously # invoked by another thread in the same process, and should evaluate false otherwise. "wsgi.multithread": bool, # This value should evaluate true if an equivalent application object may be # simultaneously invoked by another process, and should evaluate false otherwise. "wsgi.multiprocess": bool, # This value should evaluate true if the server or gateway expects (but does # not guarantee!) that the application will only be invoked this one time during # the life of its containing process. Normally, this will only be true for a # gateway based on CGI (or something similar). "wsgi.run_once": bool, }, ) class Environ(CGIRequiredDefined, CGIOptionalDefined, WSGIDefined): """ WSGI Environ """ ExceptionInfo = Tuple[Type[BaseException], BaseException, Optional[TracebackType]] # https://peps.python.org/pep-3333/#the-write-callable WriteCallable = Callable[[bytes], None] class StartResponse(Protocol): def __call__( self, status: str, response_headers: List[Tuple[str, str]], exc_info: Optional[ExceptionInfo] = None, /, ) -> WriteCallable: raise NotImplementedError IterableChunks = Iterable[bytes] WSGIApp = Callable[[Environ, StartResponse], IterableChunks] a2wsgi-1.10.10/benchmark.py000066400000000000000000000057461502447752200153760ustar00rootroot00000000000000""" **Need Python3.7+** """ import asyncio import time import httpx import pytest from asgiref.wsgi import WsgiToAsgi from uvicorn.middleware.wsgi import WSGIMiddleware as UvicornWSGIMiddleware from a2wsgi import ASGIMiddleware, WSGIMiddleware try: import uvloop uvloop.install() except ImportError: pass async def asgi_echo(scope, receive, send): assert scope["type"] == "http" await send( { "type": "http.response.start", "status": 200, "headers": [[b"content-type", b"text/plain"]], } ) while True: message = await receive() more_body = message.get("more_body", False) await send( { "type": "http.response.body", "body": message.get("body", b""), "more_body": more_body, } ) if not more_body: break def wsgi_echo(environ, start_response): status = "200 OK" body = environ["wsgi.input"].read() headers = [ ("Content-Type", "text/plain; charset=utf-8"), ("Content-Length", str(len(body))), ] start_response(status, headers) return [body] @pytest.fixture(scope="module", autouse=True) def print_title(): print(f"\n{'Name':^30}", "Average Time", end="", flush=True) @pytest.mark.parametrize( "app, name", [ (asgi_echo, "pure-ASGI"), (WSGIMiddleware(wsgi_echo), "a2wsgi-WSGIMiddleware"), (UvicornWSGIMiddleware(wsgi_echo), "uvicorn-WSGIMiddleware"), (WsgiToAsgi(wsgi_echo), "asgiref-WsgiToAsgi"), ], ) @pytest.mark.asyncio async def test_convert_wsgi_to_asgi(app, name): async with httpx.AsyncClient(app=app, base_url="http://testserver") as client: start_time = time.time_ns() await asyncio.gather( *[client.post("/", content=b"hello world") for _ in range(100)] ) time_count_100 = time.time_ns() - start_time start_time = time.time_ns() await asyncio.gather( *[client.post("/", content=b"hello world") for _ in range(10100)] ) time_count_100100 = time.time_ns() - start_time print( f"\n{name:^30}", (time_count_100100 - time_count_100) / 10000 / 10**9, end="", ) @pytest.mark.parametrize( "app, name", [(wsgi_echo, "pure-WSGI"), (ASGIMiddleware(asgi_echo), "a2wsgi-ASGIMiddleware")], ) def test_convert_asgi_to_wsgi(app, name): with httpx.Client(app=app, base_url="http://testserver") as client: start_time = time.time_ns() for _ in range(100): client.post("/", content=b"hello world") time_count_100 = time.time_ns() - start_time start_time = time.time_ns() for _ in range(10100): client.post("/", content=b"hello world") time_count_100100 = time.time_ns() - start_time print( f"\n{name:^30}", (time_count_100100 - time_count_100) / 10000 / 10**9, end="", ) a2wsgi-1.10.10/pdm.lock000066400000000000000000001036451502447752200145210ustar00rootroot00000000000000# This file is @generated by PDM. # It is not intended for manual editing. [metadata] groups = ["default", "benchmark", "dev", "test"] strategy = ["cross_platform"] lock_version = "4.5.0" content_hash = "sha256:ff279e2e363824b4e92d958603ba5e378c0d3d95a62692082dcae085d660d3d5" [[metadata.targets]] requires_python = ">=3.8.0" [[package]] name = "anyio" version = "3.6.1" requires_python = ">=3.6.2" summary = "High level compatibility layer for multiple asynchronous event loop implementations" dependencies = [ "contextvars; python_version < \"3.7\"", "dataclasses; python_version < \"3.7\"", "idna>=2.8", "sniffio>=1.1", "typing-extensions; python_version < \"3.8\"", ] files = [ {file = "anyio-3.6.1-py3-none-any.whl", hash = "sha256:cb29b9c70620506a9a8f87a309591713446953302d7d995344d0d7c6c0c9a7be"}, {file = "anyio-3.6.1.tar.gz", hash = "sha256:413adf95f93886e442aea925f3ee43baa5a765a64a0f52c6081894f9992fdd0b"}, ] [[package]] name = "asgiref" version = "3.8.1" requires_python = ">=3.8" summary = "ASGI specs, helper code, and adapters" dependencies = [ "typing-extensions>=4; python_version < \"3.11\"", ] files = [ {file = "asgiref-3.8.1-py3-none-any.whl", hash = "sha256:3e1e3ecc849832fe52ccf2cb6686b7a55f82bb1d6aee72a58826471390335e47"}, {file = "asgiref-3.8.1.tar.gz", hash = "sha256:c343bd80a0bec947a9860adb4c432ffa7db769836c64238fc34bdc3fec84d590"}, ] [[package]] name = "baize" version = "0.22.2" requires_python = ">=3.7" summary = "Powerful and exquisite WSGI/ASGI framework/toolkit." dependencies = [ "typing-extensions>=4.1.1; python_version < \"3.8\"", ] files = [ {file = "baize-0.22.2-cp310-cp310-macosx_12_0_x86_64.whl", hash = "sha256:c757d6bde857c3df334253132d9bbe25bd6ce63f5bf0fc8d8c174a7ee145d32f"}, {file = "baize-0.22.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f6178431dd933d88b493d8cf00136efef2e8f0eba67aea169a528e89008029f0"}, {file = "baize-0.22.2-cp310-cp310-win_amd64.whl", hash = "sha256:de7e3ea2297484e45014789c3027f62f733dd1e73e957e60f9e5f64ff20c6a2e"}, {file = "baize-0.22.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:ff4fe0cfb073624dffae63174e643acfefdb4cf5f3603cdb5a1329c2c9a89034"}, {file = "baize-0.22.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:84d275aa8bdd93c2f5891ab384bfa5dafae3885c4b460a7fc759b212e0e14c67"}, {file = "baize-0.22.2-cp311-cp311-win_amd64.whl", hash = "sha256:62b97faa098dd8cfee66eb5fdba013c0bb2e951622bd9d9efcf93840a23d080e"}, {file = "baize-0.22.2-cp38-cp38-macosx_12_0_x86_64.whl", hash = "sha256:0a8142146bc2cd4434f0877f2fd4d0b153ede9ba790dda1cd0f4ec14044f90cd"}, {file = "baize-0.22.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6ea64c1d4ef4df0660e1f958838757f98f173367cd3c82d2514da08bb07e79b0"}, {file = "baize-0.22.2-cp38-cp38-win_amd64.whl", hash = "sha256:025952d8b369106f98c749d0f3c78df9865ed6207762c83b4ec939d097b9a972"}, {file = "baize-0.22.2-cp39-cp39-macosx_12_0_x86_64.whl", hash = "sha256:32de25983633954feec459b64546e52e6abd67637e093a83b4e25205ab1b0564"}, {file = "baize-0.22.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:378018c48684f8a1144eef8ad4480e1113b02de61f565731fbef8573092a9020"}, {file = "baize-0.22.2-cp39-cp39-win_amd64.whl", hash = "sha256:1509681642a2295e761cc2eade2ddfae7fe7b772778fa437b434c3019172b535"}, {file = "baize-0.22.2-py3-none-any.whl", hash = "sha256:caa7eea6fd438a383b654475f5ae0af600261c596f5b69e9a5522503da889878"}, {file = "baize-0.22.2.tar.gz", hash = "sha256:27e97c36c493a4287b530b3e669e392d790b10106eaea3ac3abf088a4ffdb266"}, ] [[package]] name = "black" version = "24.8.0" requires_python = ">=3.8" summary = "The uncompromising code formatter." dependencies = [ "click>=8.0.0", "mypy-extensions>=0.4.3", "packaging>=22.0", "pathspec>=0.9.0", "platformdirs>=2", "tomli>=1.1.0; python_version < \"3.11\"", "typing-extensions>=4.0.1; python_version < \"3.11\"", ] files = [ {file = "black-24.8.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:09cdeb74d494ec023ded657f7092ba518e8cf78fa8386155e4a03fdcc44679e6"}, {file = "black-24.8.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:81c6742da39f33b08e791da38410f32e27d632260e599df7245cccee2064afeb"}, {file = "black-24.8.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:707a1ca89221bc8a1a64fb5e15ef39cd755633daa672a9db7498d1c19de66a42"}, {file = "black-24.8.0-cp310-cp310-win_amd64.whl", hash = "sha256:d6417535d99c37cee4091a2f24eb2b6d5ec42b144d50f1f2e436d9fe1916fe1a"}, {file = "black-24.8.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:fb6e2c0b86bbd43dee042e48059c9ad7830abd5c94b0bc518c0eeec57c3eddc1"}, {file = "black-24.8.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:837fd281f1908d0076844bc2b801ad2d369c78c45cf800cad7b61686051041af"}, {file = "black-24.8.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:62e8730977f0b77998029da7971fa896ceefa2c4c4933fcd593fa599ecbf97a4"}, {file = "black-24.8.0-cp311-cp311-win_amd64.whl", hash = "sha256:72901b4913cbac8972ad911dc4098d5753704d1f3c56e44ae8dce99eecb0e3af"}, {file = "black-24.8.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:7c046c1d1eeb7aea9335da62472481d3bbf3fd986e093cffd35f4385c94ae368"}, {file = "black-24.8.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:649f6d84ccbae73ab767e206772cc2d7a393a001070a4c814a546afd0d423aed"}, {file = "black-24.8.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2b59b250fdba5f9a9cd9d0ece6e6d993d91ce877d121d161e4698af3eb9c1018"}, {file = "black-24.8.0-cp312-cp312-win_amd64.whl", hash = "sha256:6e55d30d44bed36593c3163b9bc63bf58b3b30e4611e4d88a0c3c239930ed5b2"}, {file = "black-24.8.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:505289f17ceda596658ae81b61ebbe2d9b25aa78067035184ed0a9d855d18afd"}, {file = "black-24.8.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:b19c9ad992c7883ad84c9b22aaa73562a16b819c1d8db7a1a1a49fb7ec13c7d2"}, {file = "black-24.8.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:1f13f7f386f86f8121d76599114bb8c17b69d962137fc70efe56137727c7047e"}, {file = "black-24.8.0-cp38-cp38-win_amd64.whl", hash = "sha256:f490dbd59680d809ca31efdae20e634f3fae27fba3ce0ba3208333b713bc3920"}, {file = "black-24.8.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:eab4dd44ce80dea27dc69db40dab62d4ca96112f87996bca68cd75639aeb2e4c"}, {file = "black-24.8.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:3c4285573d4897a7610054af5a890bde7c65cb466040c5f0c8b732812d7f0e5e"}, {file = "black-24.8.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:9e84e33b37be070ba135176c123ae52a51f82306def9f7d063ee302ecab2cf47"}, {file = "black-24.8.0-cp39-cp39-win_amd64.whl", hash = "sha256:73bbf84ed136e45d451a260c6b73ed674652f90a2b3211d6a35e78054563a9bb"}, {file = "black-24.8.0-py3-none-any.whl", hash = "sha256:972085c618ee94f402da1af548a4f218c754ea7e5dc70acb168bfaca4c2542ed"}, {file = "black-24.8.0.tar.gz", hash = "sha256:2500945420b6784c38b9ee885af039f5e7471ef284ab03fa35ecdde4688cd83f"}, ] [[package]] name = "certifi" version = "2022.6.15.1" requires_python = ">=3.6" summary = "Python package for providing Mozilla's CA Bundle." files = [ {file = "certifi-2022.6.15.1-py3-none-any.whl", hash = "sha256:43dadad18a7f168740e66944e4fa82c6611848ff9056ad910f8f7a3e46ab89e0"}, {file = "certifi-2022.6.15.1.tar.gz", hash = "sha256:cffdcd380919da6137f76633531a5817e3a9f268575c128249fb637e4f9e73fb"}, ] [[package]] name = "click" version = "8.0.4" requires_python = ">=3.6" summary = "Composable command line interface toolkit" dependencies = [ "colorama; platform_system == \"Windows\"", "importlib-metadata; python_version < \"3.8\"", ] files = [ {file = "click-8.0.4-py3-none-any.whl", hash = "sha256:6a7a62563bbfabfda3a38f3023a1db4a35978c0abd76f6c9605ecd6554d6d9b1"}, {file = "click-8.0.4.tar.gz", hash = "sha256:8458d7b1287c5fb128c90e23381cf99dcde74beaf6c7ff6384ce84d6fe090adb"}, ] [[package]] name = "colorama" version = "0.4.5" requires_python = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" summary = "Cross-platform colored terminal text." files = [ {file = "colorama-0.4.5-py2.py3-none-any.whl", hash = "sha256:854bf444933e37f5824ae7bfc1e98d5bce2ebe4160d46b5edf346a89358e99da"}, {file = "colorama-0.4.5.tar.gz", hash = "sha256:e6c6b4334fc50988a639d9b98aa429a0b57da6e17b9a44f0451f930b6967b7a4"}, ] [[package]] name = "coverage" version = "6.2" requires_python = ">=3.6" summary = "Code coverage measurement for Python" files = [ {file = "coverage-6.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:6dbc1536e105adda7a6312c778f15aaabe583b0e9a0b0a324990334fd458c94b"}, {file = "coverage-6.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:174cf9b4bef0db2e8244f82059a5a72bd47e1d40e71c68ab055425172b16b7d0"}, {file = "coverage-6.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:92b8c845527eae547a2a6617d336adc56394050c3ed8a6918683646328fbb6da"}, {file = "coverage-6.2-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:c7912d1526299cb04c88288e148c6c87c0df600eca76efd99d84396cfe00ef1d"}, {file = "coverage-6.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:d5d2033d5db1d58ae2d62f095e1aefb6988af65b4b12cb8987af409587cc0739"}, {file = "coverage-6.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:3feac4084291642165c3a0d9eaebedf19ffa505016c4d3db15bfe235718d4971"}, {file = "coverage-6.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:276651978c94a8c5672ea60a2656e95a3cce2a3f31e9fb2d5ebd4c215d095840"}, {file = "coverage-6.2-cp310-cp310-win32.whl", hash = "sha256:f506af4f27def639ba45789fa6fde45f9a217da0be05f8910458e4557eed020c"}, {file = "coverage-6.2-cp310-cp310-win_amd64.whl", hash = "sha256:3f7c17209eef285c86f819ff04a6d4cbee9b33ef05cbcaae4c0b4e8e06b3ec8f"}, {file = "coverage-6.2-cp311-cp311-macosx_10_14_x86_64.whl", hash = "sha256:13362889b2d46e8d9f97c421539c97c963e34031ab0cb89e8ca83a10cc71ac76"}, {file = "coverage-6.2-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:22e60a3ca5acba37d1d4a2ee66e051f5b0e1b9ac950b5b0cf4aa5366eda41d47"}, {file = "coverage-6.2-cp311-cp311-win_amd64.whl", hash = "sha256:b637c57fdb8be84e91fac60d9325a66a5981f8086c954ea2772efe28425eaf64"}, {file = "coverage-6.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:f9afb5b746781fc2abce26193d1c817b7eb0e11459510fba65d2bd77fe161d9e"}, {file = "coverage-6.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:edcada2e24ed68f019175c2b2af2a8b481d3d084798b8c20d15d34f5c733fa58"}, {file = "coverage-6.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:a9c8c4283e17690ff1a7427123ffb428ad6a52ed720d550e299e8291e33184dc"}, {file = "coverage-6.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f614fc9956d76d8a88a88bb41ddc12709caa755666f580af3a688899721efecd"}, {file = "coverage-6.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:9365ed5cce5d0cf2c10afc6add145c5037d3148585b8ae0e77cc1efdd6aa2953"}, {file = "coverage-6.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:8bdfe9ff3a4ea37d17f172ac0dff1e1c383aec17a636b9b35906babc9f0f5475"}, {file = "coverage-6.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:63c424e6f5b4ab1cf1e23a43b12f542b0ec2e54f99ec9f11b75382152981df57"}, {file = "coverage-6.2-cp38-cp38-win32.whl", hash = "sha256:49dbff64961bc9bdd2289a2bda6a3a5a331964ba5497f694e2cbd540d656dc1c"}, {file = "coverage-6.2-cp38-cp38-win_amd64.whl", hash = "sha256:9a29311bd6429be317c1f3fe4bc06c4c5ee45e2fa61b2a19d4d1d6111cb94af2"}, {file = "coverage-6.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:03b20e52b7d31be571c9c06b74746746d4eb82fc260e594dc662ed48145e9efd"}, {file = "coverage-6.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:215f8afcc02a24c2d9a10d3790b21054b58d71f4b3c6f055d4bb1b15cecce685"}, {file = "coverage-6.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:a4bdeb0a52d1d04123b41d90a4390b096f3ef38eee35e11f0b22c2d031222c6c"}, {file = "coverage-6.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:c332d8f8d448ded473b97fefe4a0983265af21917d8b0cdcb8bb06b2afe632c3"}, {file = "coverage-6.2-cp39-cp39-win32.whl", hash = "sha256:6e1394d24d5938e561fbeaa0cd3d356207579c28bd1792f25a068743f2d5b282"}, {file = "coverage-6.2-cp39-cp39-win_amd64.whl", hash = "sha256:86f2e78b1eff847609b1ca8050c9e1fa3bd44ce755b2ec30e70f2d3ba3844644"}, {file = "coverage-6.2-pp36.pp37.pp38-none-any.whl", hash = "sha256:5829192582c0ec8ca4a2532407bc14c2f338d9878a10442f5d03804a95fac9de"}, {file = "coverage-6.2.tar.gz", hash = "sha256:e2cad8093172b7d1595b4ad66f24270808658e11acf43a8f95b41276162eb5b8"}, ] [[package]] name = "coverage" version = "6.2" extras = ["toml"] requires_python = ">=3.6" summary = "Code coverage measurement for Python" dependencies = [ "coverage==6.2", "tomli", ] files = [ {file = "coverage-6.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:6dbc1536e105adda7a6312c778f15aaabe583b0e9a0b0a324990334fd458c94b"}, {file = "coverage-6.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:174cf9b4bef0db2e8244f82059a5a72bd47e1d40e71c68ab055425172b16b7d0"}, {file = "coverage-6.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:92b8c845527eae547a2a6617d336adc56394050c3ed8a6918683646328fbb6da"}, {file = "coverage-6.2-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:c7912d1526299cb04c88288e148c6c87c0df600eca76efd99d84396cfe00ef1d"}, {file = "coverage-6.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:d5d2033d5db1d58ae2d62f095e1aefb6988af65b4b12cb8987af409587cc0739"}, {file = "coverage-6.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:3feac4084291642165c3a0d9eaebedf19ffa505016c4d3db15bfe235718d4971"}, {file = "coverage-6.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:276651978c94a8c5672ea60a2656e95a3cce2a3f31e9fb2d5ebd4c215d095840"}, {file = "coverage-6.2-cp310-cp310-win32.whl", hash = "sha256:f506af4f27def639ba45789fa6fde45f9a217da0be05f8910458e4557eed020c"}, {file = "coverage-6.2-cp310-cp310-win_amd64.whl", hash = "sha256:3f7c17209eef285c86f819ff04a6d4cbee9b33ef05cbcaae4c0b4e8e06b3ec8f"}, {file = "coverage-6.2-cp311-cp311-macosx_10_14_x86_64.whl", hash = "sha256:13362889b2d46e8d9f97c421539c97c963e34031ab0cb89e8ca83a10cc71ac76"}, {file = "coverage-6.2-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:22e60a3ca5acba37d1d4a2ee66e051f5b0e1b9ac950b5b0cf4aa5366eda41d47"}, {file = "coverage-6.2-cp311-cp311-win_amd64.whl", hash = "sha256:b637c57fdb8be84e91fac60d9325a66a5981f8086c954ea2772efe28425eaf64"}, {file = "coverage-6.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:f9afb5b746781fc2abce26193d1c817b7eb0e11459510fba65d2bd77fe161d9e"}, {file = "coverage-6.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:edcada2e24ed68f019175c2b2af2a8b481d3d084798b8c20d15d34f5c733fa58"}, {file = "coverage-6.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:a9c8c4283e17690ff1a7427123ffb428ad6a52ed720d550e299e8291e33184dc"}, {file = "coverage-6.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f614fc9956d76d8a88a88bb41ddc12709caa755666f580af3a688899721efecd"}, {file = "coverage-6.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:9365ed5cce5d0cf2c10afc6add145c5037d3148585b8ae0e77cc1efdd6aa2953"}, {file = "coverage-6.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:8bdfe9ff3a4ea37d17f172ac0dff1e1c383aec17a636b9b35906babc9f0f5475"}, {file = "coverage-6.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:63c424e6f5b4ab1cf1e23a43b12f542b0ec2e54f99ec9f11b75382152981df57"}, {file = "coverage-6.2-cp38-cp38-win32.whl", hash = "sha256:49dbff64961bc9bdd2289a2bda6a3a5a331964ba5497f694e2cbd540d656dc1c"}, {file = "coverage-6.2-cp38-cp38-win_amd64.whl", hash = "sha256:9a29311bd6429be317c1f3fe4bc06c4c5ee45e2fa61b2a19d4d1d6111cb94af2"}, {file = "coverage-6.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:03b20e52b7d31be571c9c06b74746746d4eb82fc260e594dc662ed48145e9efd"}, {file = "coverage-6.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:215f8afcc02a24c2d9a10d3790b21054b58d71f4b3c6f055d4bb1b15cecce685"}, {file = "coverage-6.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:a4bdeb0a52d1d04123b41d90a4390b096f3ef38eee35e11f0b22c2d031222c6c"}, {file = "coverage-6.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:c332d8f8d448ded473b97fefe4a0983265af21917d8b0cdcb8bb06b2afe632c3"}, {file = "coverage-6.2-cp39-cp39-win32.whl", hash = "sha256:6e1394d24d5938e561fbeaa0cd3d356207579c28bd1792f25a068743f2d5b282"}, {file = "coverage-6.2-cp39-cp39-win_amd64.whl", hash = "sha256:86f2e78b1eff847609b1ca8050c9e1fa3bd44ce755b2ec30e70f2d3ba3844644"}, {file = "coverage-6.2-pp36.pp37.pp38-none-any.whl", hash = "sha256:5829192582c0ec8ca4a2532407bc14c2f338d9878a10442f5d03804a95fac9de"}, {file = "coverage-6.2.tar.gz", hash = "sha256:e2cad8093172b7d1595b4ad66f24270808658e11acf43a8f95b41276162eb5b8"}, ] [[package]] name = "exceptiongroup" version = "1.1.0" requires_python = ">=3.7" summary = "Backport of PEP 654 (exception groups)" files = [ {file = "exceptiongroup-1.1.0-py3-none-any.whl", hash = "sha256:327cbda3da756e2de031a3107b81ab7b3770a602c4d16ca618298c526f4bec1e"}, {file = "exceptiongroup-1.1.0.tar.gz", hash = "sha256:bcb67d800a4497e1b404c2dd44fca47d3b7a5e5433dbab67f96c1a685cdfdf23"}, ] [[package]] name = "flake8" version = "5.0.4" requires_python = ">=3.6.1" summary = "the modular source code checker: pep8 pyflakes and co" dependencies = [ "importlib-metadata<4.3,>=1.1.0; python_version < \"3.8\"", "mccabe<0.8.0,>=0.7.0", "pycodestyle<2.10.0,>=2.9.0", "pyflakes<2.6.0,>=2.5.0", ] files = [ {file = "flake8-5.0.4-py2.py3-none-any.whl", hash = "sha256:7a1cf6b73744f5806ab95e526f6f0d8c01c66d7bbe349562d22dfca20610b248"}, {file = "flake8-5.0.4.tar.gz", hash = "sha256:6fbe320aad8d6b95cec8b8e47bc933004678dc63095be98528b7bdd2a9f510db"}, ] [[package]] name = "h11" version = "0.14.0" requires_python = ">=3.7" summary = "A pure-Python, bring-your-own-I/O implementation of HTTP/1.1" dependencies = [ "typing-extensions; python_version < \"3.8\"", ] files = [ {file = "h11-0.14.0-py3-none-any.whl", hash = "sha256:e3fe4ac4b851c468cc8363d500db52c2ead036020723024a109d37346efaa761"}, {file = "h11-0.14.0.tar.gz", hash = "sha256:8f19fbbe99e72420ff35c00b27a34cb9937e902a8b810e2c88300c6f0a3b699d"}, ] [[package]] name = "httpcore" version = "1.0.4" requires_python = ">=3.8" summary = "A minimal low-level HTTP client." dependencies = [ "certifi", "h11<0.15,>=0.13", ] files = [ {file = "httpcore-1.0.4-py3-none-any.whl", hash = "sha256:ac418c1db41bade2ad53ae2f3834a3a0f5ae76b56cf5aa497d2d033384fc7d73"}, {file = "httpcore-1.0.4.tar.gz", hash = "sha256:cb2839ccfcba0d2d3c1131d3c3e26dfc327326fbe7a5dc0dbfe9f6c9151bb022"}, ] [[package]] name = "httpx" version = "0.28.1" requires_python = ">=3.8" summary = "The next generation HTTP client." dependencies = [ "anyio", "certifi", "httpcore==1.*", "idna", ] files = [ {file = "httpx-0.28.1-py3-none-any.whl", hash = "sha256:d909fcccc110f8c7faf814ca82a9a4d816bc5a6dbfea25d6591d6985b8ba59ad"}, {file = "httpx-0.28.1.tar.gz", hash = "sha256:75e98c5f16b0f35b567856f597f06ff2270a374470a5c2392242528e3e3e42fc"}, ] [[package]] name = "idna" version = "3.3" requires_python = ">=3.5" summary = "Internationalized Domain Names in Applications (IDNA)" files = [ {file = "idna-3.3-py3-none-any.whl", hash = "sha256:84d9dd047ffa80596e0f246e2eab0b391788b0503584e8945f2368256d2735ff"}, {file = "idna-3.3.tar.gz", hash = "sha256:9d643ff0a55b762d5cdb124b8eaa99c66322e2157b69160bc32796e824360e6d"}, ] [[package]] name = "iniconfig" version = "1.1.1" summary = "iniconfig: brain-dead simple config-ini parsing" files = [ {file = "iniconfig-1.1.1-py2.py3-none-any.whl", hash = "sha256:011e24c64b7f47f6ebd835bb12a743f2fbe9a26d4cecaa7f53bc4f35ee9da8b3"}, {file = "iniconfig-1.1.1.tar.gz", hash = "sha256:bc3af051d7d14b2ee5ef9969666def0cd1a000e121eaea580d4a313df4b37f32"}, ] [[package]] name = "mccabe" version = "0.7.0" requires_python = ">=3.6" summary = "McCabe checker, plugin for flake8" files = [ {file = "mccabe-0.7.0-py2.py3-none-any.whl", hash = "sha256:6c2d30ab6be0e4a46919781807b4f0d834ebdd6c6e3dca0bda5a15f863427b6e"}, {file = "mccabe-0.7.0.tar.gz", hash = "sha256:348e0240c33b60bbdf4e523192ef919f28cb2c3d7d5c7794f74009290f236325"}, ] [[package]] name = "mypy" version = "1.14.1" requires_python = ">=3.8" summary = "Optional static typing for Python" dependencies = [ "mypy-extensions>=1.0.0", "tomli>=1.1.0; python_version < \"3.11\"", "typing-extensions>=4.6.0", ] files = [ {file = "mypy-1.14.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:52686e37cf13d559f668aa398dd7ddf1f92c5d613e4f8cb262be2fb4fedb0fcb"}, {file = "mypy-1.14.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:1fb545ca340537d4b45d3eecdb3def05e913299ca72c290326be19b3804b39c0"}, {file = "mypy-1.14.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:90716d8b2d1f4cd503309788e51366f07c56635a3309b0f6a32547eaaa36a64d"}, {file = "mypy-1.14.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2ae753f5c9fef278bcf12e1a564351764f2a6da579d4a81347e1d5a15819997b"}, {file = "mypy-1.14.1-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:e0fe0f5feaafcb04505bcf439e991c6d8f1bf8b15f12b05feeed96e9e7bf1427"}, {file = "mypy-1.14.1-cp310-cp310-win_amd64.whl", hash = "sha256:7d54bd85b925e501c555a3227f3ec0cfc54ee8b6930bd6141ec872d1c572f81f"}, {file = "mypy-1.14.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:f995e511de847791c3b11ed90084a7a0aafdc074ab88c5a9711622fe4751138c"}, {file = "mypy-1.14.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:d64169ec3b8461311f8ce2fd2eb5d33e2d0f2c7b49116259c51d0d96edee48d1"}, {file = "mypy-1.14.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ba24549de7b89b6381b91fbc068d798192b1b5201987070319889e93038967a8"}, {file = "mypy-1.14.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:183cf0a45457d28ff9d758730cd0210419ac27d4d3f285beda038c9083363b1f"}, {file = "mypy-1.14.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f2a0ecc86378f45347f586e4163d1769dd81c5a223d577fe351f26b179e148b1"}, {file = "mypy-1.14.1-cp311-cp311-win_amd64.whl", hash = "sha256:ad3301ebebec9e8ee7135d8e3109ca76c23752bac1e717bc84cd3836b4bf3eae"}, {file = "mypy-1.14.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:30ff5ef8519bbc2e18b3b54521ec319513a26f1bba19a7582e7b1f58a6e69f14"}, {file = "mypy-1.14.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:cb9f255c18052343c70234907e2e532bc7e55a62565d64536dbc7706a20b78b9"}, {file = "mypy-1.14.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:8b4e3413e0bddea671012b063e27591b953d653209e7a4fa5e48759cda77ca11"}, {file = "mypy-1.14.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:553c293b1fbdebb6c3c4030589dab9fafb6dfa768995a453d8a5d3b23784af2e"}, {file = "mypy-1.14.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:fad79bfe3b65fe6a1efaed97b445c3d37f7be9fdc348bdb2d7cac75579607c89"}, {file = "mypy-1.14.1-cp312-cp312-win_amd64.whl", hash = "sha256:8fa2220e54d2946e94ab6dbb3ba0a992795bd68b16dc852db33028df2b00191b"}, {file = "mypy-1.14.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:92c3ed5afb06c3a8e188cb5da4984cab9ec9a77ba956ee419c68a388b4595255"}, {file = "mypy-1.14.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:dbec574648b3e25f43d23577309b16534431db4ddc09fda50841f1e34e64ed34"}, {file = "mypy-1.14.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:8c6d94b16d62eb3e947281aa7347d78236688e21081f11de976376cf010eb31a"}, {file = "mypy-1.14.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:d4b19b03fdf54f3c5b2fa474c56b4c13c9dbfb9a2db4370ede7ec11a2c5927d9"}, {file = "mypy-1.14.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:0c911fde686394753fff899c409fd4e16e9b294c24bfd5e1ea4675deae1ac6fd"}, {file = "mypy-1.14.1-cp313-cp313-win_amd64.whl", hash = "sha256:8b21525cb51671219f5307be85f7e646a153e5acc656e5cebf64bfa076c50107"}, {file = "mypy-1.14.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:7084fb8f1128c76cd9cf68fe5971b37072598e7c31b2f9f95586b65c741a9d31"}, {file = "mypy-1.14.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:8f845a00b4f420f693f870eaee5f3e2692fa84cc8514496114649cfa8fd5e2c6"}, {file = "mypy-1.14.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:44bf464499f0e3a2d14d58b54674dee25c031703b2ffc35064bd0df2e0fac319"}, {file = "mypy-1.14.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c99f27732c0b7dc847adb21c9d47ce57eb48fa33a17bc6d7d5c5e9f9e7ae5bac"}, {file = "mypy-1.14.1-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:bce23c7377b43602baa0bd22ea3265c49b9ff0b76eb315d6c34721af4cdf1d9b"}, {file = "mypy-1.14.1-cp38-cp38-win_amd64.whl", hash = "sha256:8edc07eeade7ebc771ff9cf6b211b9a7d93687ff892150cb5692e4f4272b0837"}, {file = "mypy-1.14.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:3888a1816d69f7ab92092f785a462944b3ca16d7c470d564165fe703b0970c35"}, {file = "mypy-1.14.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:46c756a444117c43ee984bd055db99e498bc613a70bbbc120272bd13ca579fbc"}, {file = "mypy-1.14.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:27fc248022907e72abfd8e22ab1f10e903915ff69961174784a3900a8cba9ad9"}, {file = "mypy-1.14.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:499d6a72fb7e5de92218db961f1a66d5f11783f9ae549d214617edab5d4dbdbb"}, {file = "mypy-1.14.1-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:57961db9795eb566dc1d1b4e9139ebc4c6b0cb6e7254ecde69d1552bf7613f60"}, {file = "mypy-1.14.1-cp39-cp39-win_amd64.whl", hash = "sha256:07ba89fdcc9451f2ebb02853deb6aaaa3d2239a236669a63ab3801bbf923ef5c"}, {file = "mypy-1.14.1-py3-none-any.whl", hash = "sha256:b66a60cc4073aeb8ae00057f9c1f64d49e90f918fbcef9a977eb121da8b8f1d1"}, {file = "mypy-1.14.1.tar.gz", hash = "sha256:7ec88144fe9b510e8475ec2f5f251992690fcf89ccb4500b214b4226abcd32d6"}, ] [[package]] name = "mypy-extensions" version = "1.0.0" requires_python = ">=3.5" summary = "Type system extensions for programs checked with the mypy type checker." files = [ {file = "mypy_extensions-1.0.0-py3-none-any.whl", hash = "sha256:4392f6c0eb8a5668a69e23d168ffa70f0be9ccfd32b5cc2d26a34ae5b844552d"}, {file = "mypy_extensions-1.0.0.tar.gz", hash = "sha256:75dbf8955dc00442a438fc4d0666508a9a97b6bd41aa2f0ffe9d2f2725af0782"}, ] [[package]] name = "packaging" version = "23.2" requires_python = ">=3.7" summary = "Core utilities for Python packages" files = [ {file = "packaging-23.2-py3-none-any.whl", hash = "sha256:8c491190033a9af7e1d931d0b5dacc2ef47509b34dd0de67ed209b5203fc88c7"}, {file = "packaging-23.2.tar.gz", hash = "sha256:048fb0e9405036518eaaf48a55953c750c11e1a1b68e0dd1a9d62ed0c092cfc5"}, ] [[package]] name = "pathspec" version = "0.9.0" requires_python = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" summary = "Utility library for gitignore style pattern matching of file paths." files = [ {file = "pathspec-0.9.0-py2.py3-none-any.whl", hash = "sha256:7d15c4ddb0b5c802d161efc417ec1a2558ea2653c2e8ad9c19098201dc1c993a"}, {file = "pathspec-0.9.0.tar.gz", hash = "sha256:e564499435a2673d586f6b2130bb5b95f04a3ba06f81b8f895b651a3c76aabb1"}, ] [[package]] name = "platformdirs" version = "2.4.0" requires_python = ">=3.6" summary = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." files = [ {file = "platformdirs-2.4.0-py3-none-any.whl", hash = "sha256:8868bbe3c3c80d42f20156f22e7131d2fb321f5bc86a2a345375c6481a67021d"}, {file = "platformdirs-2.4.0.tar.gz", hash = "sha256:367a5e80b3d04d2428ffa76d33f124cf11e8fff2acdaa9b43d545f5c7d661ef2"}, ] [[package]] name = "pluggy" version = "1.0.0" requires_python = ">=3.6" summary = "plugin and hook calling mechanisms for python" dependencies = [ "importlib-metadata>=0.12; python_version < \"3.8\"", ] files = [ {file = "pluggy-1.0.0-py2.py3-none-any.whl", hash = "sha256:74134bbf457f031a36d68416e1509f34bd5ccc019f0bcc952c7b909d06b37bd3"}, {file = "pluggy-1.0.0.tar.gz", hash = "sha256:4224373bacce55f955a878bf9cfa763c1e360858e330072059e10bad68531159"}, ] [[package]] name = "pycodestyle" version = "2.9.1" requires_python = ">=3.6" summary = "Python style guide checker" files = [ {file = "pycodestyle-2.9.1-py2.py3-none-any.whl", hash = "sha256:d1735fc58b418fd7c5f658d28d943854f8a849b01a5d0a1e6f3f3fdd0166804b"}, {file = "pycodestyle-2.9.1.tar.gz", hash = "sha256:2c9607871d58c76354b697b42f5d57e1ada7d261c261efac224b664affdc5785"}, ] [[package]] name = "pyflakes" version = "2.5.0" requires_python = ">=3.6" summary = "passive checker of Python programs" files = [ {file = "pyflakes-2.5.0-py2.py3-none-any.whl", hash = "sha256:4579f67d887f804e67edb544428f264b7b24f435b263c4614f384135cea553d2"}, {file = "pyflakes-2.5.0.tar.gz", hash = "sha256:491feb020dca48ccc562a8c0cbe8df07ee13078df59813b83959cbdada312ea3"}, ] [[package]] name = "pytest" version = "7.4.4" requires_python = ">=3.7" summary = "pytest: simple powerful testing with Python" dependencies = [ "colorama; sys_platform == \"win32\"", "exceptiongroup>=1.0.0rc8; python_version < \"3.11\"", "importlib-metadata>=0.12; python_version < \"3.8\"", "iniconfig", "packaging", "pluggy<2.0,>=0.12", "tomli>=1.0.0; python_version < \"3.11\"", ] files = [ {file = "pytest-7.4.4-py3-none-any.whl", hash = "sha256:b090cdf5ed60bf4c45261be03239c2c1c22df034fbffe691abe93cd80cea01d8"}, {file = "pytest-7.4.4.tar.gz", hash = "sha256:2cf0005922c6ace4a3e2ec8b4080eb0d9753fdc93107415332f50ce9e7994280"}, ] [[package]] name = "pytest-asyncio" version = "0.23.8" requires_python = ">=3.8" summary = "Pytest support for asyncio" dependencies = [ "pytest<9,>=7.0.0", ] files = [ {file = "pytest_asyncio-0.23.8-py3-none-any.whl", hash = "sha256:50265d892689a5faefb84df80819d1ecef566eb3549cf915dfb33569359d1ce2"}, {file = "pytest_asyncio-0.23.8.tar.gz", hash = "sha256:759b10b33a6dc61cce40a8bd5205e302978bbbcc00e279a8b61d9a6a3c82e4d3"}, ] [[package]] name = "pytest-cov" version = "3.0.0" requires_python = ">=3.6" summary = "Pytest plugin for measuring coverage." dependencies = [ "coverage[toml]>=5.2.1", "pytest>=4.6", ] files = [ {file = "pytest-cov-3.0.0.tar.gz", hash = "sha256:e7f0f5b1617d2210a2cabc266dfe2f4c75a8d32fb89eafb7ad9d06f6d076d470"}, {file = "pytest_cov-3.0.0-py3-none-any.whl", hash = "sha256:578d5d15ac4a25e5f961c938b85a05b09fdaae9deef3bb6de9a6e766622ca7a6"}, ] [[package]] name = "sniffio" version = "1.2.0" requires_python = ">=3.5" summary = "Sniff out which async library your code is running under" dependencies = [ "contextvars>=2.1; python_version < \"3.7\"", ] files = [ {file = "sniffio-1.2.0-py3-none-any.whl", hash = "sha256:471b71698eac1c2112a40ce2752bb2f4a4814c22a54a3eed3676bc0f5ca9f663"}, {file = "sniffio-1.2.0.tar.gz", hash = "sha256:c4666eecec1d3f50960c6bdf61ab7bc350648da6c126e3cf6898d8cd4ddcd3de"}, ] [[package]] name = "starlette" version = "0.44.0" requires_python = ">=3.8" summary = "The little ASGI library that shines." dependencies = [ "anyio<5,>=3.4.0", "typing-extensions>=3.10.0; python_version < \"3.10\"", ] files = [ {file = "starlette-0.44.0-py3-none-any.whl", hash = "sha256:19edeb75844c16dcd4f9dd72f22f9108c1539f3fc9c4c88885654fef64f85aea"}, {file = "starlette-0.44.0.tar.gz", hash = "sha256:e35166950a3ccccc701962fe0711db0bc14f2ecd37c6f9fe5e3eae0cbaea8715"}, ] [[package]] name = "tomli" version = "1.2.3" requires_python = ">=3.6" summary = "A lil' TOML parser" files = [ {file = "tomli-1.2.3-py3-none-any.whl", hash = "sha256:e3069e4be3ead9668e21cb9b074cd948f7b3113fd9c8bba083f48247aab8b11c"}, {file = "tomli-1.2.3.tar.gz", hash = "sha256:05b6166bff487dc068d322585c7ea4ef78deed501cc124060e0f238e89a9231f"}, ] [[package]] name = "typing-extensions" version = "4.12.2" requires_python = ">=3.8" summary = "Backported and Experimental Type Hints for Python 3.8+" files = [ {file = "typing_extensions-4.12.2-py3-none-any.whl", hash = "sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d"}, {file = "typing_extensions-4.12.2.tar.gz", hash = "sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8"}, ] [[package]] name = "uvicorn" version = "0.33.0" requires_python = ">=3.8" summary = "The lightning-fast ASGI server." dependencies = [ "click>=7.0", "h11>=0.8", "typing-extensions>=4.0; python_version < \"3.11\"", ] files = [ {file = "uvicorn-0.33.0-py3-none-any.whl", hash = "sha256:2c30de4aeea83661a520abab179b24084a0019c0c1bbe137e5409f741cbde5f8"}, {file = "uvicorn-0.33.0.tar.gz", hash = "sha256:3577119f82b7091cf4d3d4177bfda0bae4723ed92ab1439e8d779de880c9cc59"}, ] a2wsgi-1.10.10/pyproject.toml000066400000000000000000000015761502447752200160030ustar00rootroot00000000000000[project] authors = [{ name = "abersheeran", email = "me@abersheeran.com" }] classifiers = ["Programming Language :: Python :: 3"] dependencies = ["typing_extensions; python_version<'3.11'"] description = "Convert WSGI app to ASGI app or ASGI app to WSGI app." license = { text = "Apache-2.0" } name = "a2wsgi" readme = "README.md" requires-python = ">=3.8.0" version = "1.10.10" [project.urls] homepage = "https://github.com/abersheeran/a2wsgi" repository = "https://github.com/abersheeran/a2wsgi" [tool.pdm.dev-dependencies] dev = [ "black", "flake8", "mypy", "httpx<1.0.0,>=0.22.0", ] benchmark = ["uvicorn>=0.16.0", "asgiref>=3.4.1"] test = [ "pytest>=7.0.1", "pytest-cov>=3.0.0", "pytest-asyncio>=0.11.0", "starlette>=0.37.2", "baize>=0.20.8", ] [tool.pdm.build] includes = ["a2wsgi"] [build-system] build-backend = "pdm.backend" requires = ["pdm-backend"] a2wsgi-1.10.10/script/000077500000000000000000000000001502447752200143625ustar00rootroot00000000000000a2wsgi-1.10.10/script/version.py000066400000000000000000000012461502447752200164240ustar00rootroot00000000000000import importlib import os import subprocess import sys here = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) def get_version() -> str: """ Return version. """ sys.path.insert(0, here) return importlib.import_module("a2wsgi").__version__ os.chdir(here) subprocess.check_call(f"pdm version {get_version()}", shell=True) subprocess.check_call("git add a2wsgi/__init__.py pyproject.toml", shell=True) subprocess.check_call(f'git commit -m "v{get_version()}"', shell=True) subprocess.check_call("git push", shell=True) subprocess.check_call("git tag v{0}".format(get_version()), shell=True) subprocess.check_call("git push --tags", shell=True) a2wsgi-1.10.10/tests/000077500000000000000000000000001502447752200142205ustar00rootroot00000000000000a2wsgi-1.10.10/tests/__init__.py000066400000000000000000000000001502447752200163170ustar00rootroot00000000000000a2wsgi-1.10.10/tests/test_asgi.py000066400000000000000000000161671502447752200165670ustar00rootroot00000000000000import asyncio import concurrent.futures from collections import Counter import httpx import pytest from a2wsgi.asgi import ASGIMiddleware, build_scope async def hello_world(scope, receive, send): assert scope["type"] == "http" await send( { "type": "http.response.start", "status": 200, "headers": [ [b"content-type", b"text/plain"], [b"content-length", b"13"], ], } ) await send({"type": "http.response.body", "body": b"Hello, world!"}) async def echo_body(scope, receive, send): assert scope["type"] == "http" body = b"" more_body = True while more_body: msg = await receive() body += msg.get("body", b"") more_body = msg.get("more_body", False) await send( { "type": "http.response.start", "status": 200, "headers": [ (b"content-type", b"text/plain"), (b"Content-Length", str(len(body)).encode("latin1")), ], } ) await send({"type": "http.response.body", "body": body}) async def raise_exception(scope, receive, send): raise RuntimeError("Something went wrong") async def background_tasks(scope, receive, send): await hello_world(scope, receive, send) await asyncio.sleep(10) async def concurrent_rw(scope, receive, send): async def listen_for_disconnect() -> None: while True: message = await receive() if message["type"] == "http.disconnect": break async def stream_response() -> None: await send( { "type": "http.response.start", "status": 200, "headers": [], } ) for chunk in range(10): await send( { "type": "http.response.body", "body": chunk.to_bytes(4, "big"), "more_body": True, } ) await send({"type": "http.response.body", "body": b"", "more_body": False}) done, pending = await asyncio.wait( [ asyncio.create_task(listen_for_disconnect()), asyncio.create_task(stream_response()), ], return_when=asyncio.ALL_COMPLETED, ) [task.cancel() for task in pending] [task.result() for task in done] def test_asgi_get(): app = ASGIMiddleware(hello_world) with httpx.Client( transport=httpx.WSGITransport(app=app), base_url="http://testserver:80" ) as client: response = client.get("/") assert response.status_code == 200 assert response.text == "Hello, world!" def test_asgi_post(): app = ASGIMiddleware(echo_body) with httpx.Client( transport=httpx.WSGITransport(app=app), base_url="http://testserver:80" ) as client: response = client.post("/", content="hi boy") assert response.status_code == 200 assert response.text == "hi boy" def test_asgi_exception(): app = ASGIMiddleware(raise_exception) with httpx.Client( transport=httpx.WSGITransport(app=app), base_url="http://testserver:80" ) as client: with pytest.raises(RuntimeError): client.get("/") def test_asgi_exception_info(): app = ASGIMiddleware(raise_exception) with httpx.Client( transport=httpx.WSGITransport(app, raise_app_exceptions=False), base_url="http://testserver:80", ) as client: response = client.get("/") assert response.status_code == 500 assert response.text == "Server got itself in trouble" def test_background_app(): executor = concurrent.futures.ThreadPoolExecutor() def _(): app = ASGIMiddleware(background_tasks) with httpx.Client( transport=httpx.WSGITransport(app=app), base_url="http://testserver:80" ) as client: response = client.get("/") assert response.status_code == 200 assert response.text == "Hello, world!" future = executor.submit(_) with pytest.raises(concurrent.futures.TimeoutError): future.result(1) future.cancel() def test_background_app_wait_time(): executor = concurrent.futures.ThreadPoolExecutor() def _(): app = ASGIMiddleware(background_tasks, wait_time=1) with httpx.Client( transport=httpx.WSGITransport(app=app), base_url="http://testserver:80" ) as client: response = client.get("/") assert response.status_code == 200 assert response.text == "Hello, world!" future = executor.submit(_) future.result(2) def test_concurrent_rw(): app = ASGIMiddleware(concurrent_rw) with httpx.Client( transport=httpx.WSGITransport(app=app), base_url="http://testserver:80" ) as client: response = client.get("/") assert response.status_code == 200 def test_http_content_headers(): content_type = "application/json" content_length = "5" environ = { "REQUEST_METHOD": "POST", "QUERY_STRING": "", "PATH_INFO": "/foo", "SERVER_NAME": "foo.invalid", "SERVER_PORT": "80", "CONTENT_TYPE": content_type, "HTTP_CONTENT_TYPE": content_type, "CONTENT_LENGTH": content_length, "HTTP_CONTENT_LENGTH": content_length, } scope = build_scope(environ) counter = Counter(scope["headers"]) assert counter[(b"content-type", content_type.encode())] == 1 assert counter[(b"content-length", content_length.encode())] == 1 def test_starlette_stream_response(): from starlette.responses import StreamingResponse app = ASGIMiddleware(StreamingResponse(content=map(str, range(10)))) with httpx.Client( transport=httpx.WSGITransport(app=app), base_url="http://testserver:80" ) as client: response = client.get("/") assert response.status_code == 200 assert response.text == "0123456789" def test_starlette_base_http_middleware(): from starlette.responses import JSONResponse from starlette.middleware.base import BaseHTTPMiddleware class Middleware(BaseHTTPMiddleware): async def dispatch(self, request, call_next): response = await call_next(request) response.headers["x-middleware"] = "true" return response app = ASGIMiddleware(Middleware(JSONResponse({"hello": "world"}))) with httpx.Client( transport=httpx.WSGITransport(app=app), base_url="http://testserver:80" ) as client: response = client.get("/") assert response.status_code == 200 assert response.text == '{"hello":"world"}' assert response.headers["x-middleware"] == "true" def test_baize_stream_response(): from baize.asgi import StreamResponse async def stream(): for i in range(10): yield str(i).encode() app = ASGIMiddleware(StreamResponse(stream())) with httpx.Client( transport=httpx.WSGITransport(app=app), base_url="http://testserver:80" ) as client: response = client.get("/") assert response.status_code == 200 assert response.text == "0123456789" a2wsgi-1.10.10/tests/test_wsgi.py000066400000000000000000000163731502447752200166140ustar00rootroot00000000000000import asyncio import os import sys import threading import httpx import pytest from a2wsgi.wsgi import Body, WSGIMiddleware, build_environ def test_body(): event_loop = asyncio.new_event_loop() threading.Thread(target=event_loop.run_forever, daemon=True).start() async def receive(): return { "type": "http.request.body", "body": b"""This is a body test. Why do this? To prevent memory leaks. And cancel pre-reading. Newline.0 Newline.1 Newline.2 Newline.3 """, } body = Body(event_loop, receive) assert body.readline() == b"This is a body test.\n" assert body.read(4) == b"Why " assert body.readline(2) == b"do" assert body.readline(20) == b" this?\n" assert body.readlines(2) == [ b"To prevent memory leaks.\n", b"And cancel pre-reading.\n", ] for index, line in enumerate(body): assert line == b"Newline." + str(index).encode("utf8") + b"\n" if index == 1: break assert body.readlines() == [ b"Newline.2\n", b"Newline.3\n", ] assert body.readlines() == [] assert body.readline() == b"" assert body.read() == b"" for line in body: assert False def hello_world(environ, start_response): status = "200 OK" output = b"Hello World!\n" headers = [ ("Content-Type", "text/plain; charset=utf-8"), ("Content-Length", str(len(output))), ] start_response(status, headers) return [output] def echo_body(environ, start_response): status = "200 OK" output = environ["wsgi.input"].read() headers = [ ("Content-Type", "text/plain; charset=utf-8"), ("Content-Length", str(len(output))), ] start_response(status, headers) return [output] def raise_exception(environ, start_response): raise RuntimeError("Something went wrong") def return_exc_info(environ, start_response): try: raise RuntimeError("Something went wrong") except RuntimeError: status = "500 Internal Server Error" output = b"Internal Server Error" headers = [ ("Content-Type", "text/plain; charset=utf-8"), ("Content-Length", str(len(output))), ] start_response(status, headers, exc_info=sys.exc_info()) return [output] @pytest.mark.asyncio async def test_wsgi_get(): app = WSGIMiddleware(hello_world) async with httpx.AsyncClient( transport=httpx.ASGITransport(app=app), base_url="http://testserver" ) as client: response = await client.get("/") assert response.status_code == 200 assert response.text == "Hello World!\n" @pytest.mark.asyncio async def test_wsgi_post(): app = WSGIMiddleware(echo_body) async with httpx.AsyncClient( transport=httpx.ASGITransport(app=app), base_url="http://testserver" ) as client: response = await client.post("/", json={"example": 123}) assert response.status_code == 200 assert response.json() == {"example": 123} @pytest.mark.asyncio async def test_wsgi_exception(): # Note that we're testing the WSGI app directly here. # The HTTP protocol implementations would catch this error and return 500. app = WSGIMiddleware(raise_exception) async with httpx.AsyncClient( transport=httpx.ASGITransport(app=app), base_url="http://testserver" ) as client: with pytest.raises(RuntimeError): await client.get("/") @pytest.mark.asyncio async def test_wsgi_exc_info(): # Note that we're testing the WSGI app directly here. # The HTTP protocol implementations would catch this error and return 500. app = WSGIMiddleware(return_exc_info) async with httpx.AsyncClient( transport=httpx.ASGITransport(app=app), base_url="http://testserver" ) as client: with pytest.raises(RuntimeError): response = await client.get("/") app = WSGIMiddleware(return_exc_info) async with httpx.AsyncClient( transport=httpx.ASGITransport(app, raise_app_exceptions=False), base_url="http://testserver", ) as client: response = await client.get("/") assert response.status_code == 500 assert response.text == "Internal Server Error" @pytest.mark.asyncio async def test_build_environ(): scope = { "type": "http", "http_version": "1.1", "method": "GET", "scheme": "https", "path": "/中文", "query_string": b"a=123&b=456", "headers": [ (b"host", b"www.example.org"), (b"content-type", b"application/json"), (b"content-length", b"18"), (b"accept", b"application/json"), (b"accept", b"text/plain"), ], "client": ("134.56.78.4", 1453), "server": ("www.example.org", 443), } async def receive(): raise NotImplementedError environ = build_environ(scope, Body(asyncio.get_event_loop(), receive)) environ.pop("wsgi.input") environ.pop("asgi.scope") assert environ == { "CONTENT_LENGTH": "18", "CONTENT_TYPE": "application/json", "HTTP_ACCEPT": "application/json,text/plain", "HTTP_HOST": "www.example.org", "PATH_INFO": "/中文".encode("utf8").decode("latin-1"), "QUERY_STRING": "a=123&b=456", "REMOTE_ADDR": "134.56.78.4", "REMOTE_PORT": "1453", "REQUEST_METHOD": "GET", "SCRIPT_NAME": "", "SERVER_NAME": "www.example.org", "SERVER_PORT": "443", "SERVER_PROTOCOL": "HTTP/1.1", "wsgi.errors": sys.stdout, "wsgi.multiprocess": True, "wsgi.multithread": True, "wsgi.run_once": False, "wsgi.url_scheme": "https", "wsgi.version": (1, 0), } @pytest.mark.asyncio async def test_build_environ_with_env(): os.environ["SCRIPT_NAME"] = "/urlprefix" scope = { "type": "http", "http_version": "1.1", "method": "GET", "scheme": "https", "path": "/中文", "query_string": b"a=123&b=456", "headers": [ (b"host", b"www.example.org"), (b"content-type", b"application/json"), (b"content-length", b"18"), (b"accept", b"application/json"), (b"accept", b"text/plain"), ], "client": ("134.56.78.4", 1453), "server": ("www.example.org", 443), } async def receive(): raise NotImplementedError environ = build_environ(scope, Body(asyncio.get_event_loop(), receive)) environ.pop("wsgi.input") environ.pop("asgi.scope") assert environ == { "CONTENT_LENGTH": "18", "CONTENT_TYPE": "application/json", "HTTP_ACCEPT": "application/json,text/plain", "HTTP_HOST": "www.example.org", "PATH_INFO": "/中文".encode("utf8").decode("latin-1"), "QUERY_STRING": "a=123&b=456", "REMOTE_ADDR": "134.56.78.4", "REMOTE_PORT": "1453", "REQUEST_METHOD": "GET", "SCRIPT_NAME": "/urlprefix", "SERVER_NAME": "www.example.org", "SERVER_PORT": "443", "SERVER_PROTOCOL": "HTTP/1.1", "wsgi.errors": sys.stdout, "wsgi.multiprocess": True, "wsgi.multithread": True, "wsgi.run_once": False, "wsgi.url_scheme": "https", "wsgi.version": (1, 0), }