org.jboss.logmanager.JDKSpecific Maven / Gradle / Ivy
/*
* Copyright 2018 Red Hat, Inc.
*
* 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;
import java.security.AccessController;
import java.security.PrivilegedAction;
import org.jboss.modules.Module;
import org.jboss.modules.Version;
/**
* @author David M. Lloyd
*/
final class JDKSpecific {
private JDKSpecific() {}
private static final Gateway GATEWAY;
private static final boolean JBOSS_MODULES;
static {
GATEWAY = AccessController.doPrivileged(new GatewayPrivilegedAction());
boolean jbossModules = false;
try {
Module.getStartTime();
jbossModules = true;
} catch (Throwable ignored) {}
JBOSS_MODULES = jbossModules;
}
static final class Gateway extends SecurityManager {
protected Class>[] getClassContext() {
return super.getClassContext();
}
}
static void calculateCaller(ExtLogRecord logRecord) {
final String loggerClassName = logRecord.getLoggerClassName();
final StackTraceElement[] stackTrace = new Throwable().getStackTrace();
final Class>[] classes = GATEWAY.getClassContext();
// The stack trace may be missing classes, but the class context is not, so if we find a mismatch, we skip the class context items.
int i = 1, j = 0;
Class> clazz = classes[i++];
StackTraceElement element = stackTrace[j++];
boolean found = false;
for (;;) {
if (clazz.getName().equals(element.getClassName())) {
if (clazz.getName().equals(loggerClassName)) {
// next entry could be the one we want!
found = true;
} else {
if (found) {
logRecord.setSourceClassName(element.getClassName());
logRecord.setSourceMethodName(element.getMethodName());
logRecord.setSourceFileName(element.getFileName());
logRecord.setSourceLineNumber(element.getLineNumber());
if (JBOSS_MODULES) {
calculateModule(logRecord, clazz);
}
return;
}
}
if (j == stackTrace.length) {
logRecord.setUnknownCaller();
return;
}
element = stackTrace[j ++];
}
if (i == classes.length) {
logRecord.setUnknownCaller();
return;
}
clazz = classes[i ++];
}
}
private static void calculateModule(final ExtLogRecord logRecord, final Class> clazz) {
final Module module = Module.forClass(clazz);
if (module != null) {
logRecord.setSourceModuleName(module.getName());
final Version version = module.getVersion();
if (version != null) {
logRecord.setSourceModuleVersion(version.toString());
} else {
logRecord.setSourceModuleVersion(null);
}
}
}
static class GatewayPrivilegedAction implements PrivilegedAction {
GatewayPrivilegedAction() {
}
public Gateway run() {
return new Gateway();
}
}
}