From 85bed93afaa89e067bb807e83781bffcdadeafc0 Mon Sep 17 00:00:00 2001 From: Ceki Gulcu Date: Wed, 14 Aug 2024 11:41:31 +0200 Subject: [PATCH] fix race condition reported in LOGBACK-1362 Signed-off-by: Ceki Gulcu --- .../ch/qos/logback/core/AppenderBase.java | 4 +- .../logback/core/OutputStreamAppender.java | 18 ++- .../core/UnsynchronizedAppenderBase.java | 4 +- .../core/OutputStreamAppenderTest.java | 13 ++- .../core/issue/lbcore258/Logback1362.java | 105 ++++++++++++++++++ 5 files changed, 129 insertions(+), 15 deletions(-) create mode 100644 logback-core/src/test/java/ch/qos/logback/core/issue/lbcore258/Logback1362.java diff --git a/logback-core/src/main/java/ch/qos/logback/core/AppenderBase.java b/logback-core/src/main/java/ch/qos/logback/core/AppenderBase.java index 1061407894..9c8b1e3e0c 100644 --- a/logback-core/src/main/java/ch/qos/logback/core/AppenderBase.java +++ b/logback-core/src/main/java/ch/qos/logback/core/AppenderBase.java @@ -22,7 +22,9 @@ import ch.qos.logback.core.status.WarnStatus; /** - * Sets a skeleton implementation for appenders. + *

Sets a skeleton implementation for appenders.

+ * + *

It is coarsely synchronized in its {@link #doAppend(E)} method.

* *

* For more information about this appender, please refer to the online manual diff --git a/logback-core/src/main/java/ch/qos/logback/core/OutputStreamAppender.java b/logback-core/src/main/java/ch/qos/logback/core/OutputStreamAppender.java index 3e20daf4f1..ccd6f56d62 100755 --- a/logback-core/src/main/java/ch/qos/logback/core/OutputStreamAppender.java +++ b/logback-core/src/main/java/ch/qos/logback/core/OutputStreamAppender.java @@ -77,9 +77,17 @@ public void start() { addStatus(new ErrorStatus("No output stream set for the appender named \"" + name + "\".", this)); errors++; } + + if (encoder == null) { + addWarn("Encoder has not been set. Cannot invoke its init method."); + errors++; + } + + // only error free appenders should be activated if (errors == 0) { super.start(); + encoderInit(); } } @@ -164,12 +172,7 @@ public void setOutputStream(OutputStream outputStream) { // close any previously opened output stream closeOutputStream(); this.outputStream = outputStream; - if (encoder == null) { - addWarn("Encoder has not been set. Cannot invoke its init method."); - return; - } - encoderInit(); } finally { streamWriteLock.unlock(); } @@ -198,8 +201,11 @@ private void writeBytes(byte[] byteArray) throws IOException { return; streamWriteLock.lock(); + try { - writeByteArrayToOutputStreamWithPossibleFlush(byteArray); + if(isStarted()) { + writeByteArrayToOutputStreamWithPossibleFlush(byteArray); + } } finally { streamWriteLock.unlock(); } diff --git a/logback-core/src/main/java/ch/qos/logback/core/UnsynchronizedAppenderBase.java b/logback-core/src/main/java/ch/qos/logback/core/UnsynchronizedAppenderBase.java index aa8ff2409b..79ea30e18b 100644 --- a/logback-core/src/main/java/ch/qos/logback/core/UnsynchronizedAppenderBase.java +++ b/logback-core/src/main/java/ch/qos/logback/core/UnsynchronizedAppenderBase.java @@ -22,7 +22,7 @@ import ch.qos.logback.core.status.WarnStatus; /** - * Similar to AppenderBase except that derived appenders need to handle thread + * Similar to {@link AppenderBase} except that derived appenders need to handle thread * synchronization on their own. * * @author Ceki Gülcü @@ -30,7 +30,7 @@ */ abstract public class UnsynchronizedAppenderBase extends ContextAwareBase implements Appender { - protected boolean started = false; + protected volatile boolean started = false; // using a ThreadLocal instead of a boolean add 75 nanoseconds per // doAppend invocation. This is tolerable as doAppend takes at least a few diff --git a/logback-core/src/test/java/ch/qos/logback/core/OutputStreamAppenderTest.java b/logback-core/src/test/java/ch/qos/logback/core/OutputStreamAppenderTest.java index 99281b6729..73f9d95f43 100644 --- a/logback-core/src/test/java/ch/qos/logback/core/OutputStreamAppenderTest.java +++ b/logback-core/src/test/java/ch/qos/logback/core/OutputStreamAppenderTest.java @@ -82,8 +82,8 @@ public void nullFileFooter() { public void headerFooterCheck(String fileHeader, String presentationHeader, String presentationFooter, String fileFooter) { - OutputStreamAppender wa = new OutputStreamAppender(); - wa.setContext(context); + OutputStreamAppender osa = new OutputStreamAppender(); + osa.setContext(context); ByteArrayOutputStream baos = new ByteArrayOutputStream(); SamplePatternLayout spl = new SamplePatternLayout(); @@ -99,16 +99,17 @@ public void headerFooterCheck(String fileHeader, String presentationHeader, Stri encoder.setLayout(spl); encoder.setContext(context); - wa.setEncoder(encoder); - wa.setOutputStream(baos); - wa.start(); + osa.setEncoder(encoder); + osa.setOutputStream(baos); + osa.start(); - wa.stop(); + osa.stop(); String result = baos.toString(); String expectedHeader = emtptyIfNull(fileHeader) + emtptyIfNull(presentationHeader); System.out.println(result); + Assertions.assertTrue(result.startsWith(expectedHeader), result); String expectedFooter = emtptyIfNull(presentationFooter) + emtptyIfNull(fileFooter); diff --git a/logback-core/src/test/java/ch/qos/logback/core/issue/lbcore258/Logback1362.java b/logback-core/src/test/java/ch/qos/logback/core/issue/lbcore258/Logback1362.java new file mode 100644 index 0000000000..b14a8faa28 --- /dev/null +++ b/logback-core/src/test/java/ch/qos/logback/core/issue/lbcore258/Logback1362.java @@ -0,0 +1,105 @@ +/* + * Logback: the reliable, generic, fast and flexible logging framework. + * Copyright (C) 1999-2024, QOS.ch. All rights reserved. + * + * This program and the accompanying materials are dual-licensed under + * either the terms of the Eclipse Public License v1.0 as published by + * the Eclipse Foundation + * + * or (per the licensee's choosing) + * + * under the terms of the GNU Lesser General Public License version 2.1 + * as published by the Free Software Foundation. + */ + +package ch.qos.logback.core.issue.lbcore258; + + +import java.io.IOException; +import java.io.OutputStream; + + +import ch.qos.logback.core.OutputStreamAppender; +import ch.qos.logback.core.encoder.EncoderBase; +import org.junit.jupiter.api.Test; + +/** + * Provided by Alexander Kudrevatykh in LOGBACK-1362 + */ +public class Logback1362 { + + long startNanos = System.nanoTime(); + long DELAY = 20; + long getNanos() { + return (System.nanoTime() - startNanos); + } + + @Test + public void testAppender() throws InterruptedException { + + OutputStreamAppender appender = new OutputStreamAppender() { + @Override + public void addError(String msg, Throwable ex) { + throw new RuntimeException(getNanos()+" "+msg, ex); + } + }; + + appender.setEncoder(new EncoderBase() { + + @Override + public byte[] headerBytes() { + return null; + } + + @Override + public byte[] encode(Object event) { + delay(DELAY*2); + return new byte[]{'A'}; + } + + @Override + public byte[] footerBytes() { + // TODO Auto-generated method stub + return null; + } + }); + appender.setOutputStream(new OutputStream() { + + @Override + public void write(int b) throws IOException { + throw new RuntimeException("not closed appender"); + } + }); + + System.out.println(getNanos() + " About to call appender.start()"); + appender.start(); + System.out.println(getNanos() + " After call to appender.start()"); + + Thread t = new Thread(new Runnable() { + @Override + public void run() { + delay(DELAY); + System.out.println(getNanos() + " About to call appender.stop()"); + appender.stop(); + System.out.println(getNanos() + " After call to appender.stop()"); + } + }); + t.start(); + System.out.println(getNanos() + " Calling appender.doAppend(new Object());"); + appender.doAppend(new Object()); + System.out.println("xxxxxxxxxxxxxxxxxxxxxx"); + System.out.println(getNanos()+ " After call to appender.doAppender(new Object())"); + t.join(); + } + + private void delay(long delayInMillis) { + try { + Thread.sleep(delayInMillis); + } catch (InterruptedException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + } + +} +