From 0699bc53269ad0a81384825ccd2dcade61d24320 Mon Sep 17 00:00:00 2001 From: Lachlan Date: Mon, 13 Jun 2022 22:11:31 +1000 Subject: [PATCH] Use static exceptions for closing websocket flushers and in ContentProducer (#8155) * Use StaticException class in jetty-util for websocket flushers. * Use StaticException class for ContentProducer recycle and consumeAll Signed-off-by: Lachlan Roberts Signed-off-by: Ludovic Orban Co-authored-by: Ludovic Orban --- .../jetty/server/AsyncContentProducer.java | 18 +++---- .../eclipse/jetty/util/StaticException.java | 49 +++++++++++++++++++ .../core/internal/DemandingFlusher.java | 15 ++++++ .../websocket/core/internal/FrameFlusher.java | 12 ++--- .../internal/PerMessageDeflateExtension.java | 14 +----- .../core/internal/TransformingFlusher.java | 11 ++++- 6 files changed, 85 insertions(+), 34 deletions(-) create mode 100644 jetty-util/src/main/java/org/eclipse/jetty/util/StaticException.java diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContentProducer.java b/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContentProducer.java index 0f854124a10..6bce49c378c 100644 --- a/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContentProducer.java +++ b/jetty-server/src/main/java/org/eclipse/jetty/server/AsyncContentProducer.java @@ -19,6 +19,7 @@ import java.util.concurrent.locks.Condition; import org.eclipse.jetty.http.BadMessageException; import org.eclipse.jetty.http.HttpStatus; +import org.eclipse.jetty.util.StaticException; import org.eclipse.jetty.util.component.Destroyable; import org.eclipse.jetty.util.thread.AutoLock; import org.slf4j.Logger; @@ -31,15 +32,8 @@ import org.slf4j.LoggerFactory; class AsyncContentProducer implements ContentProducer { private static final Logger LOG = LoggerFactory.getLogger(AsyncContentProducer.class); - private static final HttpInput.ErrorContent RECYCLED_ERROR_CONTENT = new HttpInput.ErrorContent(new IllegalStateException("ContentProducer has been recycled")); - private static final Throwable UNCONSUMED_CONTENT_EXCEPTION = new IOException("Unconsumed content") - { - @Override - public Throwable fillInStackTrace() - { - return this; - } - }; + private static final HttpInput.ErrorContent RECYCLED_ERROR_CONTENT = new HttpInput.ErrorContent(new StaticException("ContentProducer has been recycled")); + private static final Throwable UNCONSUMED_CONTENT_EXCEPTION = new StaticException("Unconsumed content"); private final AutoLock _lock = new AutoLock(); private final HttpChannel _httpChannel; @@ -188,10 +182,10 @@ class AsyncContentProducer implements ContentProducer { assertLocked(); Throwable x = UNCONSUMED_CONTENT_EXCEPTION; - if (LOG.isDebugEnabled()) + if (LOG.isTraceEnabled()) { - x = new IOException("Unconsumed content"); - LOG.debug("consumeAll {}", this, x); + x = new StaticException("Unconsumed content", true); + LOG.trace("consumeAll {}", this, x); } failCurrentContent(x); // A specific HttpChannel mechanism must be used as the following code diff --git a/jetty-util/src/main/java/org/eclipse/jetty/util/StaticException.java b/jetty-util/src/main/java/org/eclipse/jetty/util/StaticException.java new file mode 100644 index 00000000000..0af3a506cd4 --- /dev/null +++ b/jetty-util/src/main/java/org/eclipse/jetty/util/StaticException.java @@ -0,0 +1,49 @@ +// +// ======================================================================== +// Copyright (c) 1995-2022 Mort Bay Consulting Pty Ltd and others. +// +// This program and the accompanying materials are made available under the +// terms of the Eclipse Public License v. 2.0 which is available at +// https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0 +// which is available at https://www.apache.org/licenses/LICENSE-2.0. +// +// SPDX-License-Identifier: EPL-2.0 OR Apache-2.0 +// ======================================================================== +// + +package org.eclipse.jetty.util; + +/** + * This exception can safely be stored in a static variable as suppressed exceptions are disabled, + * meaning calling {@link #addSuppressed(Throwable)} has no effect. + * This prevents potential memory leaks where a statically-stored exception would accumulate + * suppressed exceptions added to them. + */ +public class StaticException extends Exception +{ + /** + * Create an instance with writable stack trace and suppression disabled. + * + * @param message – the detail message + * + * @see Throwable#Throwable(String, Throwable, boolean, boolean) + */ + public StaticException(String message) + { + this(message, false); + } + + /** + * Create an instance with suppression disabled. + * + * @param message – the detail message + * @param writableStackTrace whether or not the stack trace should be writable + * + * @see Throwable#Throwable(String, Throwable, boolean, boolean) + */ + public StaticException(String message, boolean writableStackTrace) + { + // Make sure to call the super constructor that disables suppressed exception. + super(message, null, false, writableStackTrace); + } +} diff --git a/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/DemandingFlusher.java b/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/DemandingFlusher.java index 3c14c25d915..e14ec84451b 100644 --- a/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/DemandingFlusher.java +++ b/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/DemandingFlusher.java @@ -20,6 +20,7 @@ import java.util.function.LongConsumer; import org.eclipse.jetty.util.Callback; import org.eclipse.jetty.util.CountingCallback; import org.eclipse.jetty.util.IteratingCallback; +import org.eclipse.jetty.util.StaticException; import org.eclipse.jetty.websocket.core.Extension; import org.eclipse.jetty.websocket.core.Frame; import org.eclipse.jetty.websocket.core.IncomingFrames; @@ -38,6 +39,8 @@ import org.eclipse.jetty.websocket.core.IncomingFrames; */ public abstract class DemandingFlusher extends IteratingCallback implements DemandChain { + private static final Throwable SENTINEL_CLOSE_EXCEPTION = new StaticException("Closed"); + private final IncomingFrames _emitFrame; private final AtomicLong _demand = new AtomicLong(); private final AtomicReference _failure = new AtomicReference<>(); @@ -101,6 +104,18 @@ public abstract class DemandingFlusher extends IteratingCallback implements Dema succeeded(); } + /** + * Used to close this flusher when there is no explicit failure. + */ + public void closeFlusher() + { + if (_failure.compareAndSet(null, SENTINEL_CLOSE_EXCEPTION)) + { + failed(SENTINEL_CLOSE_EXCEPTION); + iterate(); + } + } + /** * Used to fail this flusher possibly from an external event such as a callback. * @param t the failure. diff --git a/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/FrameFlusher.java b/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/FrameFlusher.java index e157ce65177..0f2199375c4 100644 --- a/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/FrameFlusher.java +++ b/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/FrameFlusher.java @@ -29,6 +29,7 @@ import org.eclipse.jetty.io.EndPoint; import org.eclipse.jetty.util.BufferUtil; import org.eclipse.jetty.util.Callback; import org.eclipse.jetty.util.IteratingCallback; +import org.eclipse.jetty.util.StaticException; import org.eclipse.jetty.util.TypeUtil; import org.eclipse.jetty.util.thread.AutoLock; import org.eclipse.jetty.util.thread.Scheduler; @@ -44,6 +45,7 @@ public class FrameFlusher extends IteratingCallback { public static final Frame FLUSH_FRAME = new Frame(OpCode.BINARY); private static final Logger LOG = LoggerFactory.getLogger(FrameFlusher.class); + private static final Throwable CLOSED_CHANNEL = new StaticException("Closed"); private final AutoLock lock = new AutoLock(); private final LongAdder messagesOut = new LongAdder(); @@ -184,15 +186,7 @@ public class FrameFlusher extends IteratingCallback { try (AutoLock l = lock.lock()) { - // TODO: find a way to not create exception if cause is null. - closedCause = cause == null ? new ClosedChannelException() - { - @Override - public Throwable fillInStackTrace() - { - return this; - } - } : cause; + closedCause = cause == null ? CLOSED_CHANNEL : cause; } iterate(); } diff --git a/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/PerMessageDeflateExtension.java b/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/PerMessageDeflateExtension.java index a3aa23628ea..beac35ad459 100644 --- a/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/PerMessageDeflateExtension.java +++ b/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/PerMessageDeflateExtension.java @@ -14,7 +14,6 @@ package org.eclipse.jetty.websocket.core.internal; import java.nio.ByteBuffer; -import java.nio.channels.ClosedChannelException; import java.util.HashMap; import java.util.Map; import java.util.concurrent.atomic.AtomicReference; @@ -150,17 +149,8 @@ public class PerMessageDeflateExtension extends AbstractExtension implements Dem @Override public void close() { - // TODO: use IteratingCallback.close() instead of creating exception with failFlusher methods. - ClosedChannelException exception = new ClosedChannelException() - { - @Override - public Throwable fillInStackTrace() - { - return this; - } - }; - incomingFlusher.failFlusher(exception); - outgoingFlusher.failFlusher(exception); + incomingFlusher.closeFlusher(); + outgoingFlusher.closeFlusher(); releaseInflater(); releaseDeflater(); } diff --git a/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/TransformingFlusher.java b/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/TransformingFlusher.java index 5bed90a3d44..4792cd43470 100644 --- a/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/TransformingFlusher.java +++ b/jetty-websocket/websocket-core-common/src/main/java/org/eclipse/jetty/websocket/core/internal/TransformingFlusher.java @@ -18,6 +18,7 @@ import java.util.Queue; import org.eclipse.jetty.util.Callback; import org.eclipse.jetty.util.IteratingCallback; +import org.eclipse.jetty.util.StaticException; import org.eclipse.jetty.util.thread.AutoLock; import org.eclipse.jetty.websocket.core.Frame; import org.slf4j.Logger; @@ -33,6 +34,7 @@ import org.slf4j.LoggerFactory; public abstract class TransformingFlusher { private final Logger log = LoggerFactory.getLogger(this.getClass()); + private static final Throwable SENTINEL_CLOSE_EXCEPTION = new StaticException("Closed"); private final AutoLock lock = new AutoLock(); private final Queue entries = new ArrayDeque<>(); @@ -77,13 +79,20 @@ public abstract class TransformingFlusher notifyCallbackFailure(callback, failure); } + /** + * Used to close this flusher when there is no explicit failure. + */ + public void closeFlusher() + { + failFlusher(SENTINEL_CLOSE_EXCEPTION); + } + /** * Used to fail this flusher possibly from an external event such as a callback. * @param t the failure. */ public void failFlusher(Throwable t) { - // TODO: find a way to close the flusher in non error case without exception. boolean failed = false; try (AutoLock l = lock.lock()) {