diff options
| author | 2021-12-30 10:16:41 +0100 | |
|---|---|---|
| committer | 2022-01-09 13:00:48 +0100 | |
| commit | 46af86f8ace136dd1d1d94590d3423e6b12e3f7b (patch) | |
| tree | f190663cd8202c7bc7034adbf9dabd7da293082c /ttun_server/endpoints.py | |
| download | server-46af86f8ace136dd1d1d94590d3423e6b12e3f7b.tar.gz server-46af86f8ace136dd1d1d94590d3423e6b12e3f7b.tar.bz2 server-46af86f8ace136dd1d1d94590d3423e6b12e3f7b.zip | |
Prepare for githubv1.0.0
Diffstat (limited to 'ttun_server/endpoints.py')
| -rw-r--r-- | ttun_server/endpoints.py | 98 |
1 files changed, 98 insertions, 0 deletions
diff --git a/ttun_server/endpoints.py b/ttun_server/endpoints.py new file mode 100644 index 0000000..d59cb7c --- /dev/null +++ b/ttun_server/endpoints.py | |||
| @@ -0,0 +1,98 @@ | |||
| 1 | import asyncio | ||
| 2 | import os | ||
| 3 | from asyncio import Queue | ||
| 4 | from base64 import b64decode, b64encode | ||
| 5 | from typing import Optional, Any | ||
| 6 | from uuid import uuid4 | ||
| 7 | |||
| 8 | from starlette.endpoints import HTTPEndpoint, WebSocketEndpoint | ||
| 9 | from starlette.requests import Request | ||
| 10 | from starlette.responses import Response | ||
| 11 | from starlette.types import Scope, Receive, Send | ||
| 12 | from starlette.websockets import WebSocket | ||
| 13 | |||
| 14 | from ttun_server.types import Connection, RequestData, Config, ResponseData | ||
| 15 | |||
| 16 | from ttun_server.connections import connections | ||
| 17 | |||
| 18 | |||
| 19 | class Proxy(HTTPEndpoint): | ||
| 20 | async def dispatch(self) -> None: | ||
| 21 | request = Request(self.scope, self.receive) | ||
| 22 | |||
| 23 | [subdomain, *_] = request.headers['host'].split('.') | ||
| 24 | response = Response(content='Not Found', status_code=404) | ||
| 25 | |||
| 26 | if subdomain in connections: | ||
| 27 | connection = connections[subdomain] | ||
| 28 | |||
| 29 | await connection['requests'].put(RequestData( | ||
| 30 | method=request.method, | ||
| 31 | path=str(request.url).replace(str(request.base_url), '/'), | ||
| 32 | headers=dict(request.headers), | ||
| 33 | cookies=dict(request.cookies), | ||
| 34 | body=b64encode(await request.body()).decode() | ||
| 35 | )) | ||
| 36 | |||
| 37 | _response = await connection['responses'].get() | ||
| 38 | response = Response( | ||
| 39 | status_code=_response['status'], | ||
| 40 | headers=_response['headers'], | ||
| 41 | content=b64decode(_response['body'].encode()) | ||
| 42 | ) | ||
| 43 | |||
| 44 | await response(self.scope, self.receive, self.send) | ||
| 45 | |||
| 46 | |||
| 47 | class Tunnel(WebSocketEndpoint): | ||
| 48 | encoding = 'json' | ||
| 49 | |||
| 50 | def __init__(self, scope: Scope, receive: Receive, send: Send): | ||
| 51 | super().__init__(scope, receive, send) | ||
| 52 | self.request_task = None | ||
| 53 | self.config: Optional[Config] = None | ||
| 54 | |||
| 55 | @property | ||
| 56 | def requests(self) -> Queue[RequestData]: | ||
| 57 | return connections[self.config['subdomain']]['requests'] | ||
| 58 | |||
| 59 | @property | ||
| 60 | def responses(self) -> Queue[ResponseData]: | ||
| 61 | return connections[self.config['subdomain']]['responses'] | ||
| 62 | |||
| 63 | async def handle_requests(self, websocket: WebSocket): | ||
| 64 | while request := await self.requests.get(): | ||
| 65 | await websocket.send_json(request) | ||
| 66 | |||
| 67 | async def on_connect(self, websocket: WebSocket) -> None: | ||
| 68 | await websocket.accept() | ||
| 69 | self.config = await websocket.receive_json() | ||
| 70 | |||
| 71 | if self.config['subdomain'] is None \ | ||
| 72 | or self.config['subdomain'] in connections: | ||
| 73 | self.config['subdomain'] = uuid4().hex | ||
| 74 | |||
| 75 | |||
| 76 | connections[self.config['subdomain']] = Connection( | ||
| 77 | requests=Queue(), | ||
| 78 | responses=Queue(), | ||
| 79 | ) | ||
| 80 | |||
| 81 | hostname = os.environ.get("TUNNEL_DOMAIN") | ||
| 82 | protocol = "https" if os.environ.get("SECURE", False) else "http" | ||
| 83 | |||
| 84 | await websocket.send_json({ | ||
| 85 | 'url': f'{protocol}://{self.config["subdomain"]}.{hostname}' | ||
| 86 | }) | ||
| 87 | |||
| 88 | self.request_task = asyncio.create_task(self.handle_requests(websocket)) | ||
| 89 | |||
| 90 | async def on_receive(self, websocket: WebSocket, data: Any) -> None: | ||
| 91 | await self.responses.put(data) | ||
| 92 | |||
| 93 | async def on_disconnect(self, websocket: WebSocket, close_code: int) -> None: | ||
| 94 | if self.config is not None and self.config['subdomain'] in connections: | ||
| 95 | del connections[self.config['subdomain']] | ||
| 96 | |||
| 97 | if self.request_task is not None: | ||
| 98 | self.request_task.cancel() | ||
