From 5a52235464917eadc473b9744708ace623971805 Mon Sep 17 00:00:00 2001 From: Lachlan Roberts Date: Fri, 20 Sep 2019 10:30:31 +1000 Subject: [PATCH] Issue #4104 - WebSocketSession will reject outgoing frames if closed Outgoing frames will now go RemoteEndpoint->Session->ExtensionStack instead of just RemoteEndpoint->ExtensionStack. This will allow the Session to check whether it has been closed before allowing the frame through the ExtensionStack. Signed-off-by: Lachlan Roberts --- .../jetty/websocket/tests/EventSocket.java | 2 +- .../websocket/tests/WriteAfterStopTest.java | 15 +++------- .../client/WebSocketUpgradeRequest.java | 4 +-- .../client/io/WebSocketClientConnection.java | 20 ------------- .../websocket/common/WebSocketSession.java | 28 +++++++++++++++++-- .../io/AbstractWebSocketConnection.java | 13 +++++++++ .../server/WebSocketServerConnection.java | 23 +-------------- 7 files changed, 46 insertions(+), 59 deletions(-) diff --git a/jetty-websocket/jetty-websocket-tests/src/test/java/org/eclipse/jetty/websocket/tests/EventSocket.java b/jetty-websocket/jetty-websocket-tests/src/test/java/org/eclipse/jetty/websocket/tests/EventSocket.java index 3671e3ccf86..8b5c5220536 100644 --- a/jetty-websocket/jetty-websocket-tests/src/test/java/org/eclipse/jetty/websocket/tests/EventSocket.java +++ b/jetty-websocket/jetty-websocket-tests/src/test/java/org/eclipse/jetty/websocket/tests/EventSocket.java @@ -37,7 +37,7 @@ public class EventSocket { private static Logger LOG = Log.getLogger(EventSocket.class); - protected Session session; + public Session session; private String behavior; public volatile Throwable failure = null; public volatile int closeCode = -1; diff --git a/jetty-websocket/jetty-websocket-tests/src/test/java/org/eclipse/jetty/websocket/tests/WriteAfterStopTest.java b/jetty-websocket/jetty-websocket-tests/src/test/java/org/eclipse/jetty/websocket/tests/WriteAfterStopTest.java index 6aca5cbe627..79bf0f40099 100644 --- a/jetty-websocket/jetty-websocket-tests/src/test/java/org/eclipse/jetty/websocket/tests/WriteAfterStopTest.java +++ b/jetty-websocket/jetty-websocket-tests/src/test/java/org/eclipse/jetty/websocket/tests/WriteAfterStopTest.java @@ -19,20 +19,17 @@ package org.eclipse.jetty.websocket.tests; import java.net.URI; -import java.nio.channels.ClosedChannelException; import java.util.concurrent.TimeUnit; import org.eclipse.jetty.server.Server; import org.eclipse.jetty.server.ServerConnector; import org.eclipse.jetty.servlet.ServletContextHandler; import org.eclipse.jetty.servlet.ServletHolder; -import org.eclipse.jetty.util.log.StacklessLogging; import org.eclipse.jetty.websocket.api.Session; import org.eclipse.jetty.websocket.api.StatusCode; +import org.eclipse.jetty.websocket.api.WebSocketException; import org.eclipse.jetty.websocket.client.ClientUpgradeRequest; import org.eclipse.jetty.websocket.client.WebSocketClient; -import org.eclipse.jetty.websocket.common.WebSocketSession; -import org.eclipse.jetty.websocket.common.extensions.compress.CompressExtension; import org.eclipse.jetty.websocket.servlet.WebSocketServlet; import org.eclipse.jetty.websocket.servlet.WebSocketServletFactory; import org.junit.jupiter.api.AfterEach; @@ -105,12 +102,8 @@ public class WriteAfterStopTest assertThat(clientSocket.closeCode, is(StatusCode.NORMAL)); assertThat(serverSocket.closeCode, is(StatusCode.NORMAL)); - ((WebSocketSession)session).stop(); - - try (StacklessLogging stacklessLogging = new StacklessLogging(CompressExtension.class)) - { - assertThrows(ClosedChannelException.class, - () -> session.getRemote().sendString("hello world")); - } + WebSocketException failure = assertThrows(WebSocketException.class, () -> + clientSocket.session.getRemote().sendString("this should fail before ExtensionStack")); + assertThat(failure.getMessage(), is("Session closed")); } } \ No newline at end of file diff --git a/jetty-websocket/websocket-client/src/main/java/org/eclipse/jetty/websocket/client/WebSocketUpgradeRequest.java b/jetty-websocket/websocket-client/src/main/java/org/eclipse/jetty/websocket/client/WebSocketUpgradeRequest.java index c8b6259e019..62867c658c1 100644 --- a/jetty-websocket/websocket-client/src/main/java/org/eclipse/jetty/websocket/client/WebSocketUpgradeRequest.java +++ b/jetty-websocket/websocket-client/src/main/java/org/eclipse/jetty/websocket/client/WebSocketUpgradeRequest.java @@ -601,7 +601,6 @@ public class WebSocketUpgradeRequest extends HttpRequest implements CompleteList session.setUpgradeResponse(new ClientUpgradeResponse(response)); connection.addListener(session); - ExtensionStack extensionStack = new ExtensionStack(getExtensionFactory()); List extensions = new ArrayList<>(); HttpField extField = response.getHeaders().getField(HttpHeader.SEC_WEBSOCKET_EXTENSIONS); if (extField != null) @@ -619,8 +618,9 @@ public class WebSocketUpgradeRequest extends HttpRequest implements CompleteList } } } - extensionStack.negotiate(extensions); + ExtensionStack extensionStack = new ExtensionStack(getExtensionFactory()); + extensionStack.negotiate(extensions); extensionStack.configure(connection.getParser()); extensionStack.configure(connection.getGenerator()); diff --git a/jetty-websocket/websocket-client/src/main/java/org/eclipse/jetty/websocket/client/io/WebSocketClientConnection.java b/jetty-websocket/websocket-client/src/main/java/org/eclipse/jetty/websocket/client/io/WebSocketClientConnection.java index b249eb4f332..780a7edba48 100644 --- a/jetty-websocket/websocket-client/src/main/java/org/eclipse/jetty/websocket/client/io/WebSocketClientConnection.java +++ b/jetty-websocket/websocket-client/src/main/java/org/eclipse/jetty/websocket/client/io/WebSocketClientConnection.java @@ -18,7 +18,6 @@ package org.eclipse.jetty.websocket.client.io; -import java.net.InetSocketAddress; import java.util.concurrent.Executor; import org.eclipse.jetty.io.ByteBufferPool; @@ -28,7 +27,6 @@ import org.eclipse.jetty.websocket.api.BatchMode; import org.eclipse.jetty.websocket.api.WebSocketPolicy; import org.eclipse.jetty.websocket.api.WriteCallback; import org.eclipse.jetty.websocket.api.extensions.Frame; -import org.eclipse.jetty.websocket.api.extensions.IncomingFrames; import org.eclipse.jetty.websocket.client.masks.Masker; import org.eclipse.jetty.websocket.client.masks.RandomMasker; import org.eclipse.jetty.websocket.common.WebSocketFrame; @@ -47,18 +45,6 @@ public class WebSocketClientConnection extends AbstractWebSocketConnection this.masker = new RandomMasker(); } - @Override - public InetSocketAddress getLocalAddress() - { - return getEndPoint().getLocalAddress(); - } - - @Override - public InetSocketAddress getRemoteAddress() - { - return getEndPoint().getRemoteAddress(); - } - /** * Override to set the masker. */ @@ -71,10 +57,4 @@ public class WebSocketClientConnection extends AbstractWebSocketConnection } super.outgoingFrame(frame, callback, batchMode); } - - @Override - public void setNextIncomingFrames(IncomingFrames incoming) - { - getParser().setIncomingFramesHandler(incoming); - } } diff --git a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/WebSocketSession.java b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/WebSocketSession.java index 700ba25a55c..ef7ccc94161 100644 --- a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/WebSocketSession.java +++ b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/WebSocketSession.java @@ -48,7 +48,9 @@ import org.eclipse.jetty.websocket.api.StatusCode; import org.eclipse.jetty.websocket.api.SuspendToken; import org.eclipse.jetty.websocket.api.UpgradeRequest; import org.eclipse.jetty.websocket.api.UpgradeResponse; +import org.eclipse.jetty.websocket.api.WebSocketException; import org.eclipse.jetty.websocket.api.WebSocketPolicy; +import org.eclipse.jetty.websocket.api.WriteCallback; import org.eclipse.jetty.websocket.api.extensions.ExtensionFactory; import org.eclipse.jetty.websocket.api.extensions.Frame; import org.eclipse.jetty.websocket.api.extensions.IncomingFrames; @@ -59,7 +61,7 @@ import org.eclipse.jetty.websocket.common.scopes.WebSocketContainerScope; import org.eclipse.jetty.websocket.common.scopes.WebSocketSessionScope; @ManagedObject("A Jetty WebSocket Session") -public class WebSocketSession extends ContainerLifeCycle implements Session, RemoteEndpointFactory, WebSocketSessionScope, IncomingFrames, Connection.Listener +public class WebSocketSession extends ContainerLifeCycle implements Session, RemoteEndpointFactory, WebSocketSessionScope, IncomingFrames, OutgoingFrames, Connection.Listener { private static final Logger LOG = Log.getLogger(WebSocketSession.class); private final WebSocketContainerScope containerScope; @@ -334,6 +336,26 @@ public class WebSocketSession extends ContainerLifeCycle implements Session, Rem } } + @Override + public void outgoingFrame(Frame frame, WriteCallback callback, BatchMode batchMode) + { + if (onCloseCalled.get()) + { + try + { + if (callback != null) + callback.writeFailed(new WebSocketException("Session closed")); + } + catch (Throwable x) + { + LOG.debug("Exception while notifying failure of callback " + callback, x); + } + return; + } + + outgoingHandler.outgoingFrame(frame, callback, batchMode); + } + @Override public boolean isOpen() { @@ -420,7 +442,7 @@ public class WebSocketSession extends ContainerLifeCycle implements Session, Rem @Override public WebSocketRemoteEndpoint newRemoteEndpoint(LogicalConnection connection, OutgoingFrames outgoingFrames, BatchMode batchMode) { - return new WebSocketRemoteEndpoint(connection, outgoingHandler, getBatchMode()); + return new WebSocketRemoteEndpoint(connection, outgoingFrames, getBatchMode()); } /** @@ -443,7 +465,7 @@ public class WebSocketSession extends ContainerLifeCycle implements Session, Rem if (connection.opening()) { // Connect remote - remote = remoteEndpointFactory.newRemoteEndpoint(connection, outgoingHandler, getBatchMode()); + remote = remoteEndpointFactory.newRemoteEndpoint(connection, this, getBatchMode()); if (LOG.isDebugEnabled()) LOG.debug("[{}] {}.open() remote={}", policy.getBehavior(), this.getClass().getSimpleName(), remote); diff --git a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/io/AbstractWebSocketConnection.java b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/io/AbstractWebSocketConnection.java index 5c872db308b..87d5f3de069 100644 --- a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/io/AbstractWebSocketConnection.java +++ b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/io/AbstractWebSocketConnection.java @@ -49,6 +49,7 @@ import org.eclipse.jetty.websocket.api.WebSocketPolicy; import org.eclipse.jetty.websocket.api.WriteCallback; import org.eclipse.jetty.websocket.api.extensions.ExtensionConfig; import org.eclipse.jetty.websocket.api.extensions.Frame; +import org.eclipse.jetty.websocket.api.extensions.IncomingFrames; import org.eclipse.jetty.websocket.common.CloseInfo; import org.eclipse.jetty.websocket.common.Generator; import org.eclipse.jetty.websocket.common.LogicalConnection; @@ -392,6 +393,12 @@ public abstract class AbstractWebSocketConnection extends AbstractConnection imp return this.policy; } + @Override + public InetSocketAddress getLocalAddress() + { + return getEndPoint().getLocalAddress(); + } + @Override public InetSocketAddress getRemoteAddress() { @@ -649,6 +656,12 @@ public abstract class AbstractWebSocketConnection extends AbstractConnection imp setInitialBuffer(prefilled); } + @Override + public void setNextIncomingFrames(IncomingFrames incoming) + { + getParser().setIncomingFramesHandler(incoming); + } + /** * @return the number of WebSocket frames received over this connection */ diff --git a/jetty-websocket/websocket-server/src/main/java/org/eclipse/jetty/websocket/server/WebSocketServerConnection.java b/jetty-websocket/websocket-server/src/main/java/org/eclipse/jetty/websocket/server/WebSocketServerConnection.java index 4662b24a11c..7c085b89951 100644 --- a/jetty-websocket/websocket-server/src/main/java/org/eclipse/jetty/websocket/server/WebSocketServerConnection.java +++ b/jetty-websocket/websocket-server/src/main/java/org/eclipse/jetty/websocket/server/WebSocketServerConnection.java @@ -18,7 +18,6 @@ package org.eclipse.jetty.websocket.server; -import java.net.InetSocketAddress; import java.util.concurrent.Executor; import org.eclipse.jetty.io.ByteBufferPool; @@ -26,7 +25,6 @@ import org.eclipse.jetty.io.Connection; import org.eclipse.jetty.io.EndPoint; import org.eclipse.jetty.util.thread.Scheduler; import org.eclipse.jetty.websocket.api.WebSocketPolicy; -import org.eclipse.jetty.websocket.api.extensions.IncomingFrames; import org.eclipse.jetty.websocket.common.io.AbstractWebSocketConnection; public class WebSocketServerConnection extends AbstractWebSocketConnection implements Connection.UpgradeTo @@ -34,27 +32,8 @@ public class WebSocketServerConnection extends AbstractWebSocketConnection imple public WebSocketServerConnection(EndPoint endp, Executor executor, Scheduler scheduler, WebSocketPolicy policy, ByteBufferPool bufferPool) { super(endp, executor, scheduler, policy, bufferPool); + if (policy.getIdleTimeout() > 0) - { endp.setIdleTimeout(policy.getIdleTimeout()); - } - } - - @Override - public InetSocketAddress getLocalAddress() - { - return getEndPoint().getLocalAddress(); - } - - @Override - public InetSocketAddress getRemoteAddress() - { - return getEndPoint().getRemoteAddress(); - } - - @Override - public void setNextIncomingFrames(IncomingFrames incoming) - { - getParser().setIncomingFramesHandler(incoming); } }