diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml index 9a8d9bb6c..e4e156315 100644 --- a/.github/workflows/publish.yml +++ b/.github/workflows/publish.yml @@ -14,7 +14,7 @@ jobs: name: deploy steps: - - uses: "actions/checkout@v3" + - uses: "actions/checkout@v4" - uses: "actions/setup-python@v4" with: python-version: "3.11" diff --git a/.github/workflows/test-suite.yml b/.github/workflows/test-suite.yml index e04af7b51..c3ad08f14 100644 --- a/.github/workflows/test-suite.yml +++ b/.github/workflows/test-suite.yml @@ -17,7 +17,7 @@ jobs: python-version: ["3.8", "3.9", "3.10", "3.11", "3.12"] steps: - - uses: "actions/checkout@v3" + - uses: "actions/checkout@v4" - uses: "actions/setup-python@v4" with: python-version: "${{ matrix.python-version }}" diff --git a/docs/js/chat.js b/docs/js/chat.js deleted file mode 100644 index c82454918..000000000 --- a/docs/js/chat.js +++ /dev/null @@ -1,3 +0,0 @@ -((window.gitter = {}).chat = {}).options = { - room: 'encode/community' -}; diff --git a/docs/js/sidecar-1.5.0.js b/docs/js/sidecar-1.5.0.js deleted file mode 100644 index 44899c4b3..000000000 --- a/docs/js/sidecar-1.5.0.js +++ /dev/null @@ -1,6 +0,0 @@ -/*! - * Gitter Sidecar v1.5.0 - * https://sidecar.gitter.im/ - */ -var sidecar=function(t){function e(r){if(i[r])return i[r].exports;var n=i[r]={exports:{},id:r,loaded:!1};return t[r].call(n.exports,n,n.exports,e),n.loaded=!0,n.exports}var i={};return e.m=t,e.c=i,e.p="",e(0)}([function(t,e,i){"use strict";function r(t){return t&&t.__esModule?t:{default:t}}Object.defineProperty(e,"__esModule",{value:!0});var n=Object.assign||function(t){for(var e=1;e1?i-1:0),n=1;n0&&void 0!==arguments[0]?arguments[0]:{};o(this,t),this[C]=new L.default,this[S]=[],this[I]=v,this[A]=u({},this[I],e),this[O]()}return a(t,[{key:O,value:function(){var t=this,e=this[A];e.useStyles&&this[C].add(h()),e.targetElement=(0,x.default)(e.targetElement||function(){var e=t[C].createElement("aside");return e.classList.add("gitter-chat-embed"),e.classList.add("is-collapsed"),m.appendChild(e),e}()),e.targetElement.forEach(function(e){var i=t[C].createElement("div");i.classList.add("gitter-chat-embed-loading-wrapper"),i.innerHTML='\n
\n ',e.insertBefore(i,e.firstChild)}),p(this),e.preload&&this.toggleChat(!1),e.showChatByDefault?this.toggleChat(!0):(void 0===e.activationElement||e.activationElement===!0?e.activationElement=(0,x.default)(function(){var i=t[C].createElement("a");return i.href=""+e.host+e.room,i.innerHTML="Open Chat",i.classList.add("gitter-open-chat-button"),m.appendChild(i),i}()):e.activationElement&&(e.activationElement=(0,x.default)(e.activationElement)),e.activationElement&&(z(e.activationElement,function(e){t.toggleChat(!0),e.preventDefault()}),e.targetElement.forEach(function(t){x.on(t,"gitter-chat-toggle",function(t){var i=t.detail.state;e.activationElement.forEach(function(t){x.toggleClass(t,"is-collapsed",i)})})})));var i=z((0,x.default)(".js-gitter-toggle-chat-button"),function(e){var i=T(e.target.getAttribute("data-gitter-toggle-chat-state"));t.toggleChat(null!==i?i:"toggle"),e.preventDefault()});this[S].push(i),e.targetElement.forEach(function(e){var i=new l.default("gitter-chat-started",{detail:{chat:t}});e.dispatchEvent(i)});var r=new l.default("gitter-sidecar-instance-started",{detail:{chat:this}});document.dispatchEvent(r)}},{key:U,value:function(){if(!this[k]){var t=this[A],e=w(t);this[C].add(e)}this[k]=!0}},{key:Y,value:function(t){var e=this[A];e.targetElement||console.warn("Gitter Sidecar: No chat embed elements to toggle visibility on");var i=e.targetElement;i.forEach(function(e){"toggle"===t?x.toggleClass(e,"is-collapsed"):x.toggleClass(e,"is-collapsed",!t);var i=new l.default("gitter-chat-toggle",{detail:{state:t}});e.dispatchEvent(i)})}},{key:"toggleChat",value:function(t){var e=this,i=this[A];if(t&&!this[k]){var r=i.targetElement;r.forEach(function(t){t.classList.add("is-loading")}),setTimeout(function(){e[U](),e[Y](t),r.forEach(function(t){t.classList.remove("is-loading")})},300)}else this[U](),this[Y](t)}},{key:"destroy",value:function(){this[S].forEach(function(t){t()}),this[C].destroy()}},{key:"options",get:function(){return(0,j.default)(this[A])}}]),t}();e.default=Q},function(t,e){"use strict";function i(t){if(Array.isArray(t)){for(var e=0,i=Array(t.length);eiframe{box-sizing:border-box;-ms-flex:1;flex:1;width:100%;height:100%;border:0}.gitter-chat-embed-loading-wrapper{box-sizing:border-box;position:absolute;top:0;left:0;bottom:0;right:0;display:none;-ms-flex-pack:center;justify-content:center;-ms-flex-align:center;align-items:center}.is-loading .gitter-chat-embed-loading-wrapper{box-sizing:border-box;display:-ms-flexbox;display:flex}.gitter-chat-embed-loading-indicator{box-sizing:border-box;opacity:.75;background-image:url();animation:spin 2s infinite linear}@keyframes spin{0%{box-sizing:border-box;transform:rotate(0deg)}to{box-sizing:border-box;transform:rotate(359.9deg)}}.gitter-chat-embed-action-bar{box-sizing:border-box;position:absolute;top:0;left:0;right:0;display:-ms-flexbox;display:flex;-ms-flex-pack:end;justify-content:flex-end;padding-bottom:.7em;background:linear-gradient(180deg,#fff 0,#fff 50%,hsla(0,0%,100%,0))}.gitter-chat-embed-action-bar-item{box-sizing:border-box;display:-ms-flexbox;display:flex;-ms-flex-pack:center;justify-content:center;-ms-flex-align:center;align-items:center;width:40px;height:40px;padding-left:0;padding-right:0;opacity:.65;background:none;background-position:50%;background-repeat:no-repeat;background-size:22px 22px;border:0;outline:none;cursor:pointer;cursor:hand;transition:all .2s ease}.gitter-chat-embed-action-bar-item:focus,.gitter-chat-embed-action-bar-item:hover{box-sizing:border-box;opacity:1}.gitter-chat-embed-action-bar-item:active{box-sizing:border-box;filter:hue-rotate(80deg) saturate(150)}.gitter-chat-embed-action-bar-item-pop-out{box-sizing:border-box;margin-right:-4px;background-image:url()}.gitter-chat-embed-action-bar-item-collapse-chat{box-sizing:border-box;background-image:url()}.gitter-open-chat-button{z-index:100;position:fixed;bottom:0;right:10px;padding:1em 3em;background-color:#36bc98;border:0;border-top-left-radius:.5em;border-top-right-radius:.5em;font-family:sans-serif;font-size:12px;letter-spacing:1px;text-transform:uppercase;text-align:center;text-decoration:none;cursor:pointer;cursor:hand;transition:all .3s ease}.gitter-open-chat-button,.gitter-open-chat-button:visited{box-sizing:border-box;color:#fff}.gitter-open-chat-button:focus,.gitter-open-chat-button:hover{box-sizing:border-box;background-color:#3ea07f;color:#fff}.gitter-open-chat-button:focus{box-sizing:border-box;box-shadow:0 0 8px rgba(62,160,127,.6);outline:none}.gitter-open-chat-button:active{box-sizing:border-box;color:#eee}.gitter-open-chat-button.is-collapsed{box-sizing:border-box;transform:translateY(120%)}',""])},function(t,e){t.exports=function(){var t=[];return t.toString=function(){for(var t=[],e=0;e None: else: self.send({"type": "websocket.receive", "bytes": text.encode("utf-8")}) - def close(self, code: int = 1000) -> None: - self.send({"type": "websocket.disconnect", "code": code}) + def close(self, code: int = 1000, reason: typing.Union[str, None] = None) -> None: + self.send({"type": "websocket.disconnect", "code": code, "reason": reason}) def receive(self) -> Message: message = self._send_queue.get() diff --git a/starlette/websockets.py b/starlette/websockets.py index 4704dff72..859560857 100644 --- a/starlette/websockets.py +++ b/starlette/websockets.py @@ -102,7 +102,7 @@ async def accept( def _raise_on_disconnect(self, message: Message) -> None: if message["type"] == "websocket.disconnect": - raise WebSocketDisconnect(message["code"]) + raise WebSocketDisconnect(message["code"], message.get("reason")) async def receive_text(self) -> str: if self.application_state != WebSocketState.CONNECTED: diff --git a/tests/middleware/test_base.py b/tests/middleware/test_base.py index cf4780cce..650f4aee1 100644 --- a/tests/middleware/test_base.py +++ b/tests/middleware/test_base.py @@ -396,7 +396,18 @@ async def send(message): def test_app_receives_http_disconnect_while_sending_if_discarded(test_client_factory): class DiscardingMiddleware(BaseHTTPMiddleware): async def dispatch(self, request, call_next): - await call_next(request) + # As a matter of ordering, this test targets the case where the downstream + # app response is discarded while it is sending a response body. + # We need to wait for the downstream app to begin sending a response body + # before sending the middleware response that will overwrite the downstream + # response. + downstream_app_response = await call_next(request) + body_generator = downstream_app_response.body_iterator + try: + await body_generator.__anext__() + finally: + await body_generator.aclose() + return PlainTextResponse("Custom") async def downstream_app(scope, receive, send): @@ -411,17 +422,21 @@ async def downstream_app(scope, receive, send): ) async with anyio.create_task_group() as task_group: - async def cancel_on_disconnect(): + async def cancel_on_disconnect(*, task_status=anyio.TASK_STATUS_IGNORED): + task_status.started() while True: message = await receive() if message["type"] == "http.disconnect": task_group.cancel_scope.cancel() break - task_group.start_soon(cancel_on_disconnect) + # Using start instead of start_soon to ensure that + # cancel_on_disconnect is scheduled by the event loop + # before we start returning the body + await task_group.start(cancel_on_disconnect) # A timeout is set for 0.1 second in order to ensure that - # cancel_on_disconnect is scheduled by the event loop + # we never deadlock the test run in an infinite loop with anyio.move_on_after(0.1): while True: await send( diff --git a/tests/test_websockets.py b/tests/test_websockets.py index 71bccd455..65e16671d 100644 --- a/tests/test_websockets.py +++ b/tests/test_websockets.py @@ -1,11 +1,12 @@ import sys -from typing import Any, MutableMapping +from typing import Any, Callable, MutableMapping import anyio import pytest from anyio.abc import ObjectReceiveStream, ObjectSendStream from starlette import status +from starlette.testclient import TestClient from starlette.types import Receive, Scope, Send from starlette.websockets import WebSocket, WebSocketDisconnect, WebSocketState @@ -209,22 +210,25 @@ async def app(scope: Scope, receive: Receive, send: Send) -> None: assert data == {"hello": "world"} -def test_client_close(test_client_factory): +def test_client_close(test_client_factory: Callable[..., TestClient]): close_code = None + close_reason = None async def app(scope: Scope, receive: Receive, send: Send) -> None: - nonlocal close_code + nonlocal close_code, close_reason websocket = WebSocket(scope, receive=receive, send=send) await websocket.accept() try: await websocket.receive_text() except WebSocketDisconnect as exc: close_code = exc.code + close_reason = exc.reason client = test_client_factory(app) with client.websocket_connect("/") as websocket: - websocket.close(code=status.WS_1001_GOING_AWAY) + websocket.close(code=status.WS_1001_GOING_AWAY, reason="Going Away") assert close_code == status.WS_1001_GOING_AWAY + assert close_reason == "Going Away" def test_application_close(test_client_factory):