[Git][java-team/jboss-logmanager][upstream] New upstream version 2.1.18
Markus Koschany (@apo)
gitlab at salsa.debian.org
Wed Sep 8 12:11:48 BST 2021
Markus Koschany pushed to branch upstream at Debian Java Maintainers / jboss-logmanager
Commits:
5887c688 by Markus Koschany at 2021-09-08T13:00:50+02:00
New upstream version 2.1.18
- - - - -
17 changed files:
- pom.xml
- src/main/java/org/jboss/logmanager/config/AbstractPropertyConfiguration.java
- src/main/java/org/jboss/logmanager/config/HandlerConfigurationImpl.java
- src/main/java/org/jboss/logmanager/config/LoggerConfigurationImpl.java
- + src/main/java/org/jboss/logmanager/config/WrappedAction.java
- src/main/java/org/jboss/logmanager/formatters/StackTraceFormatter.java
- src/main/java/org/jboss/logmanager/handlers/PeriodicRotatingFileHandler.java
- src/main/java/org/jboss/logmanager/handlers/PeriodicSizeRotatingFileHandler.java
- + src/main/java/org/jboss/logmanager/handlers/SecurityActions.java
- src/main/java/org/jboss/logmanager/handlers/SizeRotatingFileHandler.java
- src/main/java/org/jboss/logmanager/handlers/SuffixRotator.java
- src/main/java9/org/jboss/logmanager/formatters/StackTraceFormatter.java
- src/test/java/org/jboss/logmanager/formatters/PatternFormatterTests.java
- src/test/resources/client-keystore.jks
- src/test/resources/client.cer
- src/test/resources/server-keystore.jks
- src/test/resources/server.cer
Changes:
=====================================
pom.xml
=====================================
@@ -28,7 +28,7 @@
<groupId>org.jboss.logmanager</groupId>
<artifactId>jboss-logmanager</artifactId>
<packaging>jar</packaging>
- <version>2.1.15.Final</version>
+ <version>2.1.18.Final</version>
<parent>
<groupId>org.jboss</groupId>
@@ -58,8 +58,8 @@
<org.jboss.test.port>4560</org.jboss.test.port>
<org.jboss.test.alt.port>14560</org.jboss.test.alt.port>
- <maven.compiler.source>1.8</maven.compiler.source>
- <maven.compiler.target>1.8</maven.compiler.target>
+ <maven.compiler.source>8</maven.compiler.source>
+ <maven.compiler.target>8</maven.compiler.target>
<jdk.min.version>9</jdk.min.version>
</properties>
@@ -220,6 +220,7 @@
<artifactId>maven-javadoc-plugin</artifactId>
<configuration>
<doclint>none</doclint>
+ <javadocExecutable>${java.home}${file.separator}bin</javadocExecutable>
</configuration>
</plugin>
<plugin>
=====================================
src/main/java/org/jboss/logmanager/config/AbstractPropertyConfiguration.java
=====================================
@@ -44,6 +44,7 @@ import org.jboss.modules.ModuleLoader;
*/
abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfiguration<T, C>> extends AbstractBasicConfiguration<T, C> implements ObjectConfigurable<T>, PropertyConfigurable {
private final Class<? extends T> actualClass;
+ private final ClassLoader classLoader;
private final String moduleName;
private final String className;
private final String[] constructorProperties;
@@ -61,11 +62,13 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
final ClassLoader classLoader;
if (moduleName != null) try {
classLoader = ModuleFinder.getClassLoader(moduleName);
+ this.classLoader = classLoader;
} catch (Throwable e) {
throw new IllegalArgumentException(String.format("Failed to load module \"%s\" for %s \"%s\"", moduleName, getDescription(), name), e);
}
else {
classLoader = getClass().getClassLoader();
+ this.classLoader = null;
}
final Class<? extends T> actualClass;
try {
@@ -89,18 +92,19 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
final Class<?>[] paramTypes = new Class<?>[length];
for (int i = 0; i < length; i++) {
final String property = constructorProperties[i];
- final Class<?> type = getConstructorPropertyType(actualClass, property);
+ final Class<?> type = WrappedAction.execute(() -> getConstructorPropertyType(actualClass, property), classLoader);
if (type == null) {
throw new IllegalArgumentException(String.format("No property named \"%s\" for %s \"%s\"", property, getDescription(), getName()));
}
paramTypes[i] = type;
}
- final Constructor<? extends T> constructor;
- try {
- constructor = actualClass.getConstructor(paramTypes);
- } catch (Exception e) {
- throw new IllegalArgumentException(String.format("Failed to locate constructor in class \"%s\" for %s \"%s\"", className, getDescription(), getName()), e);
- }
+ final Constructor<? extends T> constructor = WrappedAction.execute(() -> {
+ try {
+ return actualClass.getConstructor(paramTypes);
+ } catch (Exception e) {
+ throw new IllegalArgumentException(String.format("Failed to locate constructor in class \"%s\" for %s \"%s\"", className, getDescription(), getName()), e);
+ }
+ }, classLoader);
final Object[] params = new Object[length];
for (int i = 0; i < length; i++) {
final String property = constructorProperties[i];
@@ -111,11 +115,13 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
final Object value = getConfiguration().getValue(actualClass, property, paramTypes[i], valueExpression, true).getObject();
params[i] = value;
}
- try {
- return constructor.newInstance(params);
- } catch (Exception e) {
- throw new IllegalArgumentException(String.format("Failed to instantiate class \"%s\" for %s \"%s\"", className, getDescription(), getName()), e);
- }
+ return WrappedAction.execute(() -> {
+ try {
+ return constructor.newInstance(params);
+ } catch (Exception e) {
+ throw new IllegalArgumentException(String.format("Failed to instantiate class \"%s\" for %s \"%s\"", className, getDescription(), getName()), e);
+ }
+ }, classLoader);
}
public void applyPreCreate(final T param) {
@@ -189,7 +195,7 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
private void setPropertyValueExpression(final String propertyName, final ValueExpression<String> expression) {
final boolean replacement = properties.containsKey(propertyName);
final boolean constructorProp = contains(constructorProperties, propertyName);
- final Method setter = getPropertySetter(actualClass, propertyName);
+ final Method setter = WrappedAction.execute(() -> getPropertySetter(actualClass, propertyName), classLoader);
if (setter == null && ! constructorProp) {
throw new IllegalArgumentException(String.format("No property \"%s\" setter found for %s \"%s\"", propertyName, getDescription(), getName()));
}
@@ -199,11 +205,13 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
if (setter == null) {
return ObjectProducer.NULL_PRODUCER;
}
- final Class<?> propertyType = getPropertyType(actualClass, propertyName);
- if (propertyType == null) {
- throw new IllegalArgumentException(String.format("No property \"%s\" type could be determined for %s \"%s\"", propertyName, getDescription(), getName()));
- }
- return getConfiguration().getValue(actualClass, propertyName, propertyType, expression, false);
+ return WrappedAction.execute(() -> {
+ final Class<?> propertyType = getPropertyType(actualClass, propertyName);
+ if (propertyType == null) {
+ throw new IllegalArgumentException(String.format("No property \"%s\" type could be determined for %s \"%s\"", propertyName, getDescription(), getName()));
+ }
+ return getConfiguration().getValue(actualClass, propertyName, propertyType, expression, false);
+ }, classLoader);
}
public void applyPreCreate(final ObjectProducer param) {
@@ -212,44 +220,48 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
public void applyPostCreate(final ObjectProducer param) {
if (setter != null) {
- final T instance = getRefs().get(getName());
- try {
- setter.invoke(instance, param.getObject());
- } catch (Throwable e) {
- StandardOutputStreams.printError(e, "Failed to invoke setter %s with value %s%n.", setter.getName(), param.getObject());
- }
+ WrappedAction.execute(() -> {
+ final T instance = getRefs().get(getName());
+ try {
+ setter.invoke(instance, param.getObject());
+ } catch (Throwable e) {
+ StandardOutputStreams.printError(e, "Failed to invoke setter %s with value %s%n.", setter.getName(), param.getObject());
+ }
+ }, classLoader);
}
}
public void rollback() {
- final Class<?> propertyType = getPropertyType(actualClass, propertyName);
- if (propertyType == null) {
- // We don't want the rest of the rollback to fail so we'll just log a message
- StandardOutputStreams.printError("No property \"%s\" type could be determined for %s \"%s\"", propertyName, getDescription(), getName());
- return;
- }
- final ObjectProducer producer;
- if (replacement) {
- properties.put(propertyName, oldValue);
- producer = getConfiguration().getValue(actualClass, propertyName, propertyType, oldValue, true);
- } else {
- properties.remove(propertyName);
- producer = getDefaultValue(propertyType);
- }
- if (setter != null) {
- // Get the reference instance, the old value and reset to the old value
- final T instance = getRefs().get(getName());
- if (instance != null) {
- try {
- setter.invoke(instance, producer.getObject());
- } catch (Throwable e) {
- StandardOutputStreams.printError(e, "Failed to invoke setter %s with value %s%n.", setter.getName(), producer.getObject());
- }
+ WrappedAction.execute(() -> {
+ final Class<?> propertyType = getPropertyType(actualClass, propertyName);
+ if (propertyType == null) {
+ // We don't want the rest of the rollback to fail so we'll just log a message
+ StandardOutputStreams.printError("No property \"%s\" type could be determined for %s \"%s\"", propertyName, getDescription(), getName());
+ return;
+ }
+ final ObjectProducer producer;
+ if (replacement) {
+ properties.put(propertyName, oldValue);
+ producer = getConfiguration().getValue(actualClass, propertyName, propertyType, oldValue, true);
} else {
- // If the instance is not available don't keep the property
properties.remove(propertyName);
+ producer = getDefaultValue(propertyType);
}
- }
+ if (setter != null) {
+ // Get the reference instance, the old value and reset to the old value
+ final T instance = getRefs().get(getName());
+ if (instance != null) {
+ try {
+ setter.invoke(instance, producer.getObject());
+ } catch (Throwable e) {
+ StandardOutputStreams.printError(e, "Failed to invoke setter %s with value %s%n.", setter.getName(), producer.getObject());
+ }
+ } else {
+ // If the instance is not available don't keep the property
+ properties.remove(propertyName);
+ }
+ }
+ }, classLoader);
}
});
}
@@ -262,7 +274,7 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
if (isRemoved()) {
throw new IllegalArgumentException(String.format("Cannot remove property \"%s\" on %s \"%s\" (removed)", propertyName, getDescription(), getName()));
}
- final Method setter = getPropertySetter(actualClass, propertyName);
+ final Method setter = WrappedAction.execute(() -> getPropertySetter(actualClass, propertyName), classLoader);
if (setter == null) {
throw new IllegalArgumentException(String.format("No property \"%s\" setter found for %s \"%s\"", propertyName, getDescription(), getName()));
}
@@ -270,11 +282,13 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
if (oldValue != null) {
getConfiguration().addAction(new ConfigAction<ObjectProducer>() {
public ObjectProducer validate() throws IllegalArgumentException {
- final Class<?> propertyType = getPropertyType(actualClass, propertyName);
- if (propertyType == null) {
- throw new IllegalArgumentException(String.format("No property \"%s\" type could be determined for %s \"%s\"", propertyName, getDescription(), getName()));
- }
- return getDefaultValue(propertyType);
+ return WrappedAction.execute(() -> {
+ final Class<?> propertyType = getPropertyType(actualClass, propertyName);
+ if (propertyType == null) {
+ throw new IllegalArgumentException(String.format("No property \"%s\" type could be determined for %s \"%s\"", propertyName, getDescription(), getName()));
+ }
+ return getDefaultValue(propertyType);
+ }, classLoader);
}
public void applyPreCreate(final ObjectProducer param) {
@@ -282,30 +296,34 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
}
public void applyPostCreate(final ObjectProducer param) {
- final T instance = getRefs().get(getName());
- try {
- setter.invoke(instance, param.getObject());
- } catch (Throwable e) {
- StandardOutputStreams.printError(e, "Failed to invoke setter %s with value %s%n.", setter.getName(), param.getObject());
- }
+ WrappedAction.execute(() -> {
+ final T instance = getRefs().get(getName());
+ try {
+ setter.invoke(instance, param.getObject());
+ } catch (Throwable e) {
+ StandardOutputStreams.printError(e, "Failed to invoke setter %s with value %s%n.", setter.getName(), param.getObject());
+ }
+ }, classLoader);
}
public void rollback() {
- // We need to once again determine the property type
- final Class<?> propertyType = getPropertyType(actualClass, propertyName);
- if (propertyType == null) {
- // We don't want the rest of the rollback to fail so we'll just log a message
- StandardOutputStreams.printError("No property \"%s\" type could be determined for %s \"%s\"", propertyName, getDescription(), getName());
- return;
- }
- // Get the reference instance, the old value and reset to the old value
- final T instance = getRefs().get(getName());
- final ObjectProducer producer = getConfiguration().getValue(actualClass, propertyName, propertyType, oldValue, true);
- try {
- setter.invoke(instance, producer.getObject());
- } catch (Throwable e) {
- StandardOutputStreams.printError(e, "Failed to invoke setter %s with value %s%n.", setter.getName(), producer.getObject());
- }
+ WrappedAction.execute(() -> {
+ // We need to once again determine the property type
+ final Class<?> propertyType = getPropertyType(actualClass, propertyName);
+ if (propertyType == null) {
+ // We don't want the rest of the rollback to fail so we'll just log a message
+ StandardOutputStreams.printError("No property \"%s\" type could be determined for %s \"%s\"", propertyName, getDescription(), getName());
+ return;
+ }
+ // Get the reference instance, the old value and reset to the old value
+ final T instance = getRefs().get(getName());
+ final ObjectProducer producer = getConfiguration().getValue(actualClass, propertyName, propertyType, oldValue, true);
+ try {
+ setter.invoke(instance, producer.getObject());
+ } catch (Throwable e) {
+ StandardOutputStreams.printError(e, "Failed to invoke setter %s with value %s%n.", setter.getName(), producer.getObject());
+ }
+ }, classLoader);
}
});
return true;
@@ -339,11 +357,13 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
}
configuration.addAction(new ConfigAction<Method>() {
public Method validate() throws IllegalArgumentException {
- try {
- return actualClass.getMethod(methodName);
- } catch (NoSuchMethodException e) {
- throw new IllegalArgumentException(String.format("Method '%s' not found on '%s'", methodName, actualClass.getName()));
- }
+ return WrappedAction.execute(() -> {
+ try {
+ return actualClass.getMethod(methodName);
+ } catch (NoSuchMethodException e) {
+ throw new IllegalArgumentException(String.format("Method '%s' not found on '%s'", methodName, actualClass.getName()));
+ }
+ }, classLoader);
}
public void applyPreCreate(final Method param) {
@@ -383,12 +403,13 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
public Map<String, Method> validate() throws IllegalArgumentException {
final Map<String, Method> result = new LinkedHashMap<String, Method>();
for (String methodName : names) {
- try {
- result.put(methodName, actualClass.getMethod(methodName));
- } catch (NoSuchMethodException e) {
- throw new IllegalArgumentException(String.format("Method '%s' not found on '%s'", methodName, actualClass.getName()));
- }
-
+ WrappedAction.execute(() -> {
+ try {
+ result.put(methodName, actualClass.getMethod(methodName));
+ } catch (NoSuchMethodException e) {
+ throw new IllegalArgumentException(String.format("Method '%s' not found on '%s'", methodName, actualClass.getName()));
+ }
+ }, classLoader);
}
return result;
}
@@ -471,12 +492,14 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
}
public void applyPostCreate(final Method param) {
- final T instance = getRefs().get(getName());
- try {
- param.invoke(instance);
- } catch (Throwable e) {
- StandardOutputStreams.printError(e, "Failed to invoke method %s%n.", param.getName());
- }
+ WrappedAction.execute(() -> {
+ final T instance = getRefs().get(getName());
+ try {
+ param.invoke(instance);
+ } catch (Throwable e) {
+ StandardOutputStreams.printError(e, "Failed to invoke method %s%n.", param.getName());
+ }
+ }, classLoader);
}
public void rollback() {
@@ -493,17 +516,17 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
return getConfiguration().removePostConfigurationActions(name);
}
- static Class<?> getPropertyType(Class<?> clazz, String propertyName) {
+ private static Class<?> getPropertyType(Class<?> clazz, String propertyName) {
final Method setter = getPropertySetter(clazz, propertyName);
return setter != null ? setter.getParameterTypes()[0] : null;
}
- static Class<?> getConstructorPropertyType(Class<?> clazz, String propertyName) {
+ private static Class<?> getConstructorPropertyType(Class<?> clazz, String propertyName) {
final Method getter = getPropertyGetter(clazz, propertyName);
return getter != null ? getter.getReturnType() : getPropertyType(clazz, propertyName);
}
- static Method getPropertySetter(Class<?> clazz, String propertyName) {
+ private static Method getPropertySetter(Class<?> clazz, String propertyName) {
final String upperPropertyName = Character.toUpperCase(propertyName.charAt(0)) + propertyName.substring(1);
final String set = "set" + upperPropertyName;
for (Method method : clazz.getMethods()) {
@@ -514,7 +537,7 @@ abstract class AbstractPropertyConfiguration<T, C extends AbstractPropertyConfig
return null;
}
- static Method getPropertyGetter(Class<?> clazz, String propertyName) {
+ private static Method getPropertyGetter(Class<?> clazz, String propertyName) {
final String upperPropertyName = Character.toUpperCase(propertyName.charAt(0)) + propertyName.substring(1);
final Pattern pattern = Pattern.compile("(get|has|is)(" + Pattern.quote(upperPropertyName) + ")");
for (Method method : clazz.getMethods()) {
=====================================
src/main/java/org/jboss/logmanager/config/HandlerConfigurationImpl.java
=====================================
@@ -375,19 +375,20 @@ final class HandlerConfigurationImpl extends AbstractPropertyConfiguration<Handl
}
final int index = handlerNames.indexOf(name);
handlerNames.remove(index);
- configuration.addAction(new ConfigAction<Void>() {
- public Void validate() throws IllegalArgumentException {
- return null;
+ configuration.addAction(new ConfigAction<Handler>() {
+ public Handler validate() throws IllegalArgumentException {
+ final Map<String, Handler> handlerRefs = configuration.getHandlerRefs();
+ return handlerRefs.get(name);
}
- public void applyPreCreate(final Void param) {
+ public void applyPreCreate(final Handler param) {
addPostConfigurationActions();
}
- public void applyPostCreate(final Void param) {
+ public void applyPostCreate(final Handler param) {
final Map<String, Handler> handlerRefs = configuration.getHandlerRefs();
final ExtHandler handler = (ExtHandler) handlerRefs.get(getName());
- handler.removeHandler(handlerRefs.get(name));
+ handler.removeHandler(param);
}
public void rollback() {
=====================================
src/main/java/org/jboss/logmanager/config/LoggerConfigurationImpl.java
=====================================
@@ -310,19 +310,19 @@ final class LoggerConfigurationImpl extends AbstractBasicConfiguration<Logger, L
}
final int index = handlerNames.indexOf(name);
handlerNames.remove(index);
- configuration.addAction(new ConfigAction<Void>() {
- public Void validate() throws IllegalArgumentException {
- return null;
+ configuration.addAction(new ConfigAction<Handler>() {
+ public Handler validate() throws IllegalArgumentException {
+ final Map<String, Handler> handlerRefs = configuration.getHandlerRefs();
+ return handlerRefs.get(name);
}
- public void applyPreCreate(final Void param) {
+ public void applyPreCreate(final Handler param) {
}
- public void applyPostCreate(final Void param) {
- final Map<String, Handler> handlerRefs = configuration.getHandlerRefs();
+ public void applyPostCreate(final Handler param) {
final Map<String, Logger> loggerRefs = configuration.getLoggerRefs();
- final Logger logger = (Logger) loggerRefs.get(getName());
- logger.removeHandler(handlerRefs.get(name));
+ final Logger logger = loggerRefs.get(getName());
+ logger.removeHandler(param);
}
public void rollback() {
=====================================
src/main/java/org/jboss/logmanager/config/WrappedAction.java
=====================================
@@ -0,0 +1,90 @@
+/*
+ * JBoss, Home of Professional Open Source.
+ *
+ * Copyright 2021 Red Hat, Inc., and individual contributors
+ * as indicated by the @author tags.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.jboss.logmanager.config;
+
+import java.security.AccessController;
+import java.security.PrivilegedAction;
+import java.util.function.Supplier;
+
+/**
+ * @author <a href="mailto:jperkins at redhat.com">James R. Perkins</a>
+ */
+class WrappedAction {
+
+ /**
+ * Executes the action with the {@linkplain Thread#getContextClassLoader() TCCL} set to the class loader of the type.
+ *
+ * @param action the action to run
+ * @param cl the class loader to use for the TCCL
+ * @param <R> the return type
+ *
+ * @return the value from the action
+ */
+ static <R> R execute(final Supplier<R> action, final ClassLoader cl) {
+ if (cl == null) {
+ return action.get();
+ }
+ final ClassLoader current = getTccl();
+ try {
+ setTccl(cl);
+ return action.get();
+ } finally {
+ setTccl(current);
+ }
+ }
+
+ /**
+ * Executes the action with the {@linkplain Thread#getContextClassLoader() TCCL} set to the class loader of the type.
+ *
+ * @param action the action to run
+ * @param cl the class loader to use for the TCCL
+ */
+ static void execute(final Runnable action, final ClassLoader cl) {
+ if (cl == null) {
+ action.run();
+ } else {
+ final ClassLoader current = getTccl();
+ try {
+ setTccl(cl);
+ action.run();
+ } finally {
+ setTccl(current);
+ }
+ }
+ }
+
+ private static ClassLoader getTccl() {
+ if (System.getSecurityManager() == null) {
+ return Thread.currentThread().getContextClassLoader();
+ }
+ return AccessController.doPrivileged((PrivilegedAction<ClassLoader>) () -> Thread.currentThread().getContextClassLoader());
+ }
+
+ private static void setTccl(final ClassLoader cl) {
+ if (System.getSecurityManager() == null) {
+ Thread.currentThread().setContextClassLoader(cl);
+ } else {
+ AccessController.doPrivileged((PrivilegedAction<Object>) () -> {
+ Thread.currentThread().setContextClassLoader(cl);
+ return null;
+ });
+ }
+ }
+}
=====================================
src/main/java/org/jboss/logmanager/formatters/StackTraceFormatter.java
=====================================
@@ -106,7 +106,9 @@ class StackTraceFormatter {
private void renderStackTrace(final StackTraceElement[] parentStack, final Throwable child, final String caption, final String prefix) {
if (seen.contains(child)) {
- builder.append("\t[CIRCULAR REFERENCE:")
+ builder.append(prefix)
+ .append(caption)
+ .append("[CIRCULAR REFERENCE: ")
.append(child)
.append(']');
newLine();
=====================================
src/main/java/org/jboss/logmanager/handlers/PeriodicRotatingFileHandler.java
=====================================
@@ -22,6 +22,10 @@ package org.jboss.logmanager.handlers;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
+import java.io.UncheckedIOException;
+import java.security.AccessControlContext;
+import java.security.AccessController;
+import java.security.PrivilegedAction;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
@@ -36,6 +40,7 @@ import org.jboss.logmanager.ExtLogRecord;
*/
public class PeriodicRotatingFileHandler extends FileHandler {
+ private final AccessControlContext acc = AccessController.getContext();
private SimpleDateFormat format;
private String nextSuffix;
private Period period = Period.NEVER;
@@ -128,7 +133,7 @@ public class PeriodicRotatingFileHandler extends FileHandler {
* @throws IllegalArgumentException if the suffix is not valid
*/
public void setSuffix(String suffix) throws IllegalArgumentException {
- final SuffixRotator suffixRotator = SuffixRotator.parse(suffix);
+ final SuffixRotator suffixRotator = SuffixRotator.parse(acc, suffix);
final String dateSuffix = suffixRotator.getDatePattern();
final SimpleDateFormat format = new SimpleDateFormat(dateSuffix);
format.setTimeZone(timeZone);
@@ -191,12 +196,16 @@ public class PeriodicRotatingFileHandler extends FileHandler {
private void rollOver() {
try {
final File file = getFile();
+ if (file == null) {
+ // no file is set; a direct output stream or writer was specified
+ return;
+ }
// first, close the original file (some OSes won't let you move/rename a file that is open)
- setFile(null);
+ setFileInternal(null);
// next, rotate it
- suffixRotator.rotate(getErrorManager(), file.toPath(), nextSuffix);
+ suffixRotator.rotate(SecurityActions.getErrorManager(acc, this), file.toPath(), nextSuffix);
// start new file
- setFile(file);
+ setFileInternal(file);
} catch (IOException e) {
reportError("Unable to rotate log file", e, ErrorManager.OPEN_FAILURE);
}
@@ -292,6 +301,22 @@ public class PeriodicRotatingFileHandler extends FileHandler {
this.timeZone = timeZone;
}
+ private void setFileInternal(final File file) throws FileNotFoundException {
+ // At this point we should have already checked the security required
+ if (System.getSecurityManager() == null) {
+ super.setFile(file);
+ } else {
+ AccessController.doPrivileged((PrivilegedAction<Void>) () -> {
+ try {
+ super.setFile(file);
+ } catch (FileNotFoundException e) {
+ throw new UncheckedIOException(e);
+ }
+ return null;
+ }, acc);
+ }
+ }
+
private static <T extends Comparable<? super T>> T min(T a, T b) {
return a.compareTo(b) <= 0 ? a : b;
}
=====================================
src/main/java/org/jboss/logmanager/handlers/PeriodicSizeRotatingFileHandler.java
=====================================
@@ -23,6 +23,10 @@ import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.OutputStream;
+import java.io.UncheckedIOException;
+import java.security.AccessControlContext;
+import java.security.AccessController;
+import java.security.PrivilegedAction;
import java.util.logging.ErrorManager;
import org.jboss.logmanager.ExtLogRecord;
@@ -38,6 +42,7 @@ import org.jboss.logmanager.ExtLogRecord;
* @author <a href="mailto:jperkins at redhat.com">James R. Perkins</a>
*/
public class PeriodicSizeRotatingFileHandler extends PeriodicRotatingFileHandler {
+ private final AccessControlContext acc = AccessController.getContext();
// by default, rotate at 10MB
private long rotateSize = 0xa0000L;
private int maxBackupIndex = 1;
@@ -148,6 +153,7 @@ public class PeriodicSizeRotatingFileHandler extends PeriodicRotatingFileHandler
*/
@Override
public void setFile(final File file) throws FileNotFoundException {
+ checkAccess(this);
synchronized (outputLock) {
// Check for a rotate
if (rotateOnBoot && maxBackupIndex > 0 && file != null && file.exists() && file.length() > 0L) {
@@ -155,11 +161,12 @@ public class PeriodicSizeRotatingFileHandler extends PeriodicRotatingFileHandler
final SuffixRotator suffixRotator = getSuffixRotator();
if (suffixRotator != SuffixRotator.EMPTY && suffix != null) {
// Make sure any previous files are closed before we attempt to rotate
- setFileInternal(null);
+ setFileInternal(null, false);
+ // Previous actions have already performed
suffixRotator.rotate(getErrorManager(), file.toPath(), suffix, maxBackupIndex);
}
}
- setFileInternal(file);
+ setFileInternal(file, false);
}
}
@@ -224,19 +231,34 @@ public class PeriodicSizeRotatingFileHandler extends PeriodicRotatingFileHandler
return;
}
// close the old file.
- setFileInternal(null);
- getSuffixRotator().rotate(getErrorManager(), file.toPath(), getNextSuffix(), maxBackupIndex);
+ setFileInternal(null, true);
+ getSuffixRotator().rotate(SecurityActions.getErrorManager(acc, this), file.toPath(), getNextSuffix(), maxBackupIndex);
// start with new file.
- setFileInternal(file);
+ setFileInternal(file, true);
} catch (IOException e) {
reportError("Unable to rotate log file", e, ErrorManager.OPEN_FAILURE);
}
}
}
- private void setFileInternal(final File file) throws FileNotFoundException {
- super.setFile(file);
- if (outputStream != null)
- outputStream.currentSize = file == null ? 0L : file.length();
+ private void setFileInternal(final File file, final boolean doPrivileged) throws FileNotFoundException {
+ if (System.getSecurityManager() == null || !doPrivileged) {
+ super.setFile(file);
+ if (outputStream != null) {
+ outputStream.currentSize = file == null ? 0L : file.length();
+ }
+ } else {
+ AccessController.doPrivileged((PrivilegedAction<Void>) () -> {
+ try {
+ super.setFile(file);
+ if (outputStream != null) {
+ outputStream.currentSize = file == null ? 0L : file.length();
+ }
+ } catch (FileNotFoundException e) {
+ throw new UncheckedIOException(e);
+ }
+ return null;
+ }, acc);
+ }
}
}
=====================================
src/main/java/org/jboss/logmanager/handlers/SecurityActions.java
=====================================
@@ -0,0 +1,68 @@
+/*
+ * JBoss, Home of Professional Open Source.
+ *
+ * Copyright 2020 Red Hat, Inc., and individual contributors
+ * as indicated by the @author tags.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.jboss.logmanager.handlers;
+
+import java.security.AccessControlContext;
+import java.security.AccessController;
+import java.security.PrivilegedAction;
+import java.util.function.Supplier;
+import java.util.logging.ErrorManager;
+import java.util.logging.Handler;
+
+/**
+ * @author <a href="mailto:jperkins at redhat.com">James R. Perkins</a>
+ */
+class SecurityActions {
+
+ static ErrorManager getErrorManager(final AccessControlContext acc, final Handler handler) {
+ Supplier<ErrorManager> supplier = () -> {
+ if (System.getSecurityManager() == null) {
+ return handler.getErrorManager();
+ }
+ return AccessController.doPrivileged((PrivilegedAction<ErrorManager>) handler::getErrorManager, acc);
+ };
+ return new LazyErrorManager(supplier);
+ }
+
+ private static class LazyErrorManager extends ErrorManager {
+ private final Supplier<ErrorManager> supplier;
+ private volatile ErrorManager delegate;
+
+ private LazyErrorManager(final Supplier<ErrorManager> supplier) {
+ this.supplier = supplier;
+ }
+
+ @Override
+ public synchronized void error(final String msg, final Exception ex, final int code) {
+ getDelegate().error(msg, ex, code);
+ }
+
+ private ErrorManager getDelegate() {
+ if (delegate == null) {
+ synchronized (this) {
+ if (delegate == null) {
+ delegate = supplier.get();
+ }
+ }
+ }
+ return delegate;
+ }
+ }
+}
=====================================
src/main/java/org/jboss/logmanager/handlers/SizeRotatingFileHandler.java
=====================================
@@ -25,9 +25,14 @@ import java.io.File;
import java.io.FileNotFoundException;
import org.jboss.logmanager.ExtLogRecord;
+import java.io.UncheckedIOException;
+import java.security.AccessControlContext;
+import java.security.AccessController;
+import java.security.PrivilegedAction;
import java.util.logging.ErrorManager;
public class SizeRotatingFileHandler extends FileHandler {
+ private final AccessControlContext acc = AccessController.getContext();
// by default, rotate at 10MB
private long rotateSize = 0xa0000L;
private int maxBackupIndex = 1;
@@ -136,14 +141,15 @@ public class SizeRotatingFileHandler extends FileHandler {
* @throws RuntimeException if there is an attempt to rotate file and the rotation fails
*/
public void setFile(final File file) throws FileNotFoundException {
+ checkAccess(this);
synchronized (outputLock) {
// Check for a rotate
if (rotateOnBoot && maxBackupIndex > 0 && file != null && file.exists() && file.length() > 0L) {
// Make sure any previous files are closed before we attempt to rotate
- setFileInternal(null);
+ setFileInternal(null, false);
suffixRotator.rotate(getErrorManager(), file.toPath(), maxBackupIndex);
}
- setFileInternal(file);
+ setFileInternal(file, false);
}
}
@@ -228,7 +234,7 @@ public class SizeRotatingFileHandler extends FileHandler {
public void setSuffix(final String suffix) {
checkAccess(this);
synchronized (outputLock) {
- this.suffixRotator = SuffixRotator.parse(suffix);
+ this.suffixRotator = SuffixRotator.parse(acc, suffix);
}
}
@@ -244,19 +250,34 @@ public class SizeRotatingFileHandler extends FileHandler {
return;
}
// close the old file.
- setFileInternal(null);
- suffixRotator.rotate(getErrorManager(), file.toPath(), maxBackupIndex);
+ setFileInternal(null, true);
+ suffixRotator.rotate(SecurityActions.getErrorManager(acc, this), file.toPath(), maxBackupIndex);
// start with new file.
- setFileInternal(file);
+ setFileInternal(file, true);
} catch (IOException e) {
reportError("Unable to rotate log file", e, ErrorManager.OPEN_FAILURE);
}
}
}
- private void setFileInternal(final File file) throws FileNotFoundException {
- super.setFile(file);
- if (outputStream != null)
- outputStream.currentSize = file == null ? 0L : file.length();
+ private void setFileInternal(final File file, final boolean doPrivileged) throws FileNotFoundException {
+ if (System.getSecurityManager() == null || !doPrivileged) {
+ super.setFile(file);
+ if (outputStream != null) {
+ outputStream.currentSize = file == null ? 0L : file.length();
+ }
+ } else {
+ AccessController.doPrivileged((PrivilegedAction<Void>) () -> {
+ try {
+ super.setFile(file);
+ if (outputStream != null) {
+ outputStream.currentSize = file == null ? 0L : file.length();
+ }
+ } catch (FileNotFoundException e) {
+ throw new UncheckedIOException(e);
+ }
+ return null;
+ }, acc);
+ }
}
}
=====================================
src/main/java/org/jboss/logmanager/handlers/SuffixRotator.java
=====================================
@@ -21,11 +21,16 @@ package org.jboss.logmanager.handlers;
import java.io.IOException;
import java.io.InputStream;
+import java.io.OutputStream;
+import java.io.UncheckedIOException;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardCopyOption;
+import java.security.AccessControlContext;
+import java.security.AccessController;
+import java.security.PrivilegedAction;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;
@@ -53,15 +58,18 @@ class SuffixRotator {
/**
* An empty rotation suffix.
*/
- static final SuffixRotator EMPTY = new SuffixRotator("", "", "", CompressionType.NONE);
+ static final SuffixRotator EMPTY = new SuffixRotator(AccessController.getContext(), "", "", "", CompressionType.NONE);
+ private final AccessControlContext acc;
private final String originalSuffix;
private final String datePattern;
private final SimpleDateFormat formatter;
private final String compressionSuffix;
private final CompressionType compressionType;
- private SuffixRotator(final String originalSuffix, final String datePattern, final String compressionSuffix, final CompressionType compressionType) {
+ private SuffixRotator(final AccessControlContext acc, final String originalSuffix, final String datePattern,
+ final String compressionSuffix, final CompressionType compressionType) {
+ this.acc = acc;
this.originalSuffix = originalSuffix;
this.datePattern = datePattern;
this.compressionSuffix = compressionSuffix;
@@ -80,7 +88,7 @@ class SuffixRotator {
*
* @return the file rotator used to determine the suffix parts and rotate the file.
*/
- static SuffixRotator parse(final String suffix) {
+ static SuffixRotator parse(final AccessControlContext acc, final String suffix) {
if (suffix == null || suffix.isEmpty()) {
return EMPTY;
}
@@ -103,9 +111,9 @@ class SuffixRotator {
}
}
if (compressionSuffix.isEmpty() && datePattern.isEmpty()) {
- return new SuffixRotator(suffix, suffix, "", CompressionType.NONE);
+ return new SuffixRotator(acc, suffix, suffix, "", CompressionType.NONE);
}
- return new SuffixRotator(suffix, datePattern, compressionSuffix, compressionType);
+ return new SuffixRotator(acc, suffix, datePattern, compressionSuffix, compressionType);
}
/**
@@ -156,7 +164,7 @@ class SuffixRotator {
try {
archiveGzip(source, target);
// Delete the file after it's archived to behave like a file move or rename
- Files.delete(source);
+ deleteFile(source);
} catch (Exception e) {
errorManager.error(String.format("Failed to compress %s to %s. Compressed file may be left on the " +
"filesystem corrupted.", source, target), e, ErrorManager.WRITE_FAILURE);
@@ -165,7 +173,7 @@ class SuffixRotator {
try {
archiveZip(source, target);
// Delete the file after it's archived to behave like a file move or rename
- Files.delete(source);
+ deleteFile(source);
} catch (Exception e) {
errorManager.error(String.format("Failed to compress %s to %s. Compressed file may be left on the " +
"filesystem corrupted.", source, target), e, ErrorManager.WRITE_FAILURE);
@@ -219,13 +227,13 @@ class SuffixRotator {
final String fileWithSuffix = source.toAbsolutePath() + rotationSuffix;
final Path lastFile = Paths.get(fileWithSuffix + "." + maxBackupIndex + compressionSuffix);
try {
- Files.deleteIfExists(lastFile);
+ deleteFile(lastFile);
} catch (Exception e) {
errorManager.error(String.format("Failed to delete file %s", lastFile), e, ErrorManager.GENERIC_FAILURE);
}
for (int i = maxBackupIndex - 1; i >= 1; i--) {
final Path src = Paths.get(fileWithSuffix + "." + i + compressionSuffix);
- if (Files.exists(src)) {
+ if (fileExists(src)) {
final Path target = Paths.get(fileWithSuffix + "." + (i + 1) + compressionSuffix);
move(errorManager, src, target);
}
@@ -242,19 +250,23 @@ class SuffixRotator {
}
private void move(final ErrorManager errorManager, final Path src, final Path target) {
- try {
- Files.move(src, target, StandardCopyOption.REPLACE_EXISTING);
- } catch (Exception e) {
- // Report the error, but allow the rotation to continue
- errorManager.error(String.format("Failed to move file %s to %s.", src, target), e, ErrorManager.GENERIC_FAILURE);
+ if (System.getSecurityManager() == null) {
+ try {
+ Files.move(src, target, StandardCopyOption.REPLACE_EXISTING);
+ } catch (Exception e) {
+ // Report the error, but allow the rotation to continue
+ errorManager.error(String.format("Failed to move file %s to %s.", src, target), e, ErrorManager.GENERIC_FAILURE);
+ }
+ } else {
+ AccessController.doPrivileged(new MoveFileAction(errorManager, src, target), acc);
}
}
- private static void archiveGzip(final Path source, final Path target) throws IOException {
+ private void archiveGzip(final Path source, final Path target) throws IOException {
final byte[] buff = new byte[512];
- try (final GZIPOutputStream out = new GZIPOutputStream(Files.newOutputStream(target), true)) {
- try (final InputStream in = Files.newInputStream(source)) {
+ try (final GZIPOutputStream out = new GZIPOutputStream(newOutputStream(target), true)) {
+ try (final InputStream in = newInputStream(source)) {
int len;
while ((len = in.read(buff)) != -1) {
out.write(buff, 0, len);
@@ -264,12 +276,12 @@ class SuffixRotator {
}
}
- private static void archiveZip(final Path source, final Path target) throws IOException {
+ private void archiveZip(final Path source, final Path target) throws IOException {
final byte[] buff = new byte[512];
- try (final ZipOutputStream out = new ZipOutputStream(Files.newOutputStream(target), StandardCharsets.UTF_8)) {
+ try (final ZipOutputStream out = new ZipOutputStream(newOutputStream(target), StandardCharsets.UTF_8)) {
final ZipEntry entry = new ZipEntry(source.getFileName().toString());
out.putNextEntry(entry);
- try (final InputStream in = Files.newInputStream(source)) {
+ try (final InputStream in = newInputStream(source)) {
int len;
while ((len = in.read(buff)) != -1) {
out.write(buff, 0, len);
@@ -278,4 +290,120 @@ class SuffixRotator {
out.closeEntry();
}
}
+
+ @SuppressWarnings("UnusedReturnValue")
+ private boolean deleteFile(final Path path) throws IOException {
+ if (System.getSecurityManager() == null) {
+ return Files.deleteIfExists(path);
+ }
+ return AccessController.doPrivileged(new DeleteFileAction(path), acc);
+ }
+
+ private boolean fileExists(final Path file) {
+ if (System.getSecurityManager() == null) {
+ return Files.exists(file);
+ }
+ return AccessController.doPrivileged(new FileExistsAction(file), acc);
+ }
+
+ private InputStream newInputStream(final Path file) throws IOException {
+ if (System.getSecurityManager() == null) {
+ return Files.newInputStream(file);
+ }
+ return AccessController.doPrivileged(new InputStreamAction(file), acc);
+ }
+
+ private OutputStream newOutputStream(final Path file) throws IOException {
+ if (System.getSecurityManager() == null) {
+ return Files.newOutputStream(file);
+ }
+ return AccessController.doPrivileged(new OutputStreamAction(file), acc);
+ }
+
+ private static class DeleteFileAction implements PrivilegedAction<Boolean> {
+ private final Path file;
+
+ private DeleteFileAction(final Path file) {
+ this.file = file;
+ }
+
+ @Override
+ public Boolean run() {
+ try {
+ return Files.deleteIfExists(file);
+ } catch (IOException e) {
+ throw new UncheckedIOException(e);
+ }
+ }
+ }
+
+ private static class MoveFileAction implements PrivilegedAction<Path> {
+ private final ErrorManager errorManager;
+ private final Path source;
+ private final Path target;
+
+ private MoveFileAction(final ErrorManager errorManager, final Path source, final Path target) {
+ this.errorManager = errorManager;
+ this.source = source;
+ this.target = target;
+ }
+
+ @Override
+ public Path run() {
+ try {
+ return Files.move(source, target, StandardCopyOption.REPLACE_EXISTING);
+ } catch (Exception e) {
+ // Report the error, but allow the rotation to continue
+ errorManager.error(String.format("Failed to move file %s to %s.", source, target), e, ErrorManager.GENERIC_FAILURE);
+ }
+ return null;
+ }
+ }
+
+ private static class FileExistsAction implements PrivilegedAction<Boolean> {
+ private final Path file;
+
+ private FileExistsAction(final Path file) {
+ this.file = file;
+ }
+
+ @Override
+ public Boolean run() {
+ return Files.exists(file);
+ }
+ }
+
+ private static class InputStreamAction implements PrivilegedAction<InputStream> {
+ private final Path file;
+
+ private InputStreamAction(final Path file) {
+ this.file = file;
+ }
+
+ @Override
+ public InputStream run() {
+ try {
+ return Files.newInputStream(file);
+ } catch (IOException e) {
+ throw new UncheckedIOException(e);
+ }
+ }
+ }
+
+ private static class OutputStreamAction implements PrivilegedAction<OutputStream> {
+ private final Path file;
+
+ private OutputStreamAction(final Path file) {
+ this.file = file;
+ }
+
+ @Override
+ public OutputStream run() {
+ try {
+ return Files.newOutputStream(file);
+ } catch (IOException e) {
+ throw new UncheckedIOException(e);
+ }
+ }
+ }
}
=====================================
src/main/java9/org/jboss/logmanager/formatters/StackTraceFormatter.java
=====================================
@@ -84,7 +84,9 @@ class StackTraceFormatter {
private void renderStackTrace(final StackTraceElement[] parentStack, final Throwable child, final String caption, final String prefix) {
if (seen.contains(child)) {
- builder.append("\t[CIRCULAR REFERENCE:")
+ builder.append(prefix)
+ .append(caption)
+ .append("[CIRCULAR REFERENCE: ")
.append(child)
.append(']');
newLine();
=====================================
src/test/java/org/jboss/logmanager/formatters/PatternFormatterTests.java
=====================================
@@ -245,7 +245,7 @@ public class PatternFormatterTests {
formatter = new PatternFormatter("%e");
cause.addSuppressed(suppressedLevel1);
formatted = formatter.format(record);
- Assert.assertTrue(formatted.contains("CIRCULAR REFERENCE:java.lang.IllegalStateException: suppressedLevel1"));
+ Assert.assertTrue(formatted.contains("CIRCULAR REFERENCE: java.lang.IllegalStateException: suppressedLevel1"));
}
@Test
=====================================
src/test/resources/client-keystore.jks
=====================================
Binary files a/src/test/resources/client-keystore.jks and b/src/test/resources/client-keystore.jks differ
=====================================
src/test/resources/client.cer
=====================================
Binary files a/src/test/resources/client.cer and b/src/test/resources/client.cer differ
=====================================
src/test/resources/server-keystore.jks
=====================================
Binary files a/src/test/resources/server-keystore.jks and b/src/test/resources/server-keystore.jks differ
=====================================
src/test/resources/server.cer
=====================================
Binary files a/src/test/resources/server.cer and b/src/test/resources/server.cer differ
View it on GitLab: https://salsa.debian.org/java-team/jboss-logmanager/-/commit/5887c68889f29d58d871a0ca8376d2ead968528b
--
View it on GitLab: https://salsa.debian.org/java-team/jboss-logmanager/-/commit/5887c68889f29d58d871a0ca8376d2ead968528b
You're receiving this email because of your account on salsa.debian.org.
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://alioth-lists.debian.net/pipermail/pkg-java-commits/attachments/20210908/47d68e98/attachment.htm>
More information about the pkg-java-commits
mailing list