org.spf4j.base.BasicExecutionContext Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of spf4j-core Show documentation
Show all versions of spf4j-core Show documentation
A continuously growing collection of utilities to measure performance, get better diagnostics,
improve performance, or do things more reliably, faster that other open source libraries...
/*
* Copyright (c) 2001-2017, Zoltan Farkas All Rights Reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* Additionally licensed with:
*
* 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.spf4j.base;
import com.fasterxml.jackson.core.JsonGenerator;
import com.google.common.annotations.Beta;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
import java.io.IOException;
import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.function.BiFunction;
import java.util.function.Consumer;
import java.util.logging.Logger;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import javax.annotation.ParametersAreNonnullByDefault;
import javax.annotation.concurrent.ThreadSafe;
import org.spf4j.io.AppendableWriter;
import org.spf4j.log.Level;
import org.spf4j.log.Slf4jLogRecord;
import org.spf4j.base.ThreadLocalContextAttacher.Attached;
import org.spf4j.log.LogUtils;
/**
* The simplest execution context possible.
*
* @author Zoltan Farkas
*/
@ThreadSafe
@ParametersAreNonnullByDefault
public class BasicExecutionContext implements ExecutionContext {
private static final int MX_NR_LOGS_PER_CTXT = Integer.getInteger("spf4j.execContext.maxNrLogsPerContext", 100);
private static final Level MIN_LOG_LEVEL
= Level.valueOf(System.getProperty("spf4j.execContext.minLogLevel", "TRACE"));
private final String name;
private final CharSequence id;
private final ExecutionContext source;
private final Relation relation;
private final long startTimeNanos;
private final long deadlineNanos;
private ArrayDeque logs;
private List closeables;
private Map baggage;
private long childCount;
private boolean isClosed = false;
private Level minBackendLogLevel;
private Attached attached;
@SuppressWarnings("unchecked")
@SuppressFBWarnings({"EI_EXPOSE_REP2", "STT_TOSTRING_STORED_IN_FIELD"})
public BasicExecutionContext(final String name, @Nullable final CharSequence id,
@Nullable final ExecutionContext source, final Relation relation,
final long startTimeNanos, final long deadlineNanos) {
this.isClosed = false;
this.relation = relation;
this.name = name;
this.startTimeNanos = startTimeNanos;
if (source != null) {
long parentDeadline = source.getDeadlineNanos();
if (parentDeadline < deadlineNanos) {
this.deadlineNanos = parentDeadline;
} else {
this.deadlineNanos = deadlineNanos;
}
if (id == null) {
CharSequence pId = source.getId();
StringBuilder sb = new StringBuilder(pId.length() + 2).append(pId).append('/');
AppendableUtils.appendUnsignedString(sb, source.nextChildId(), 5);
this.id = sb;
} else {
this.id = id;
}
this.minBackendLogLevel = source.getBackendMinLogLevel();
} else {
this.deadlineNanos = deadlineNanos;
this.id = id == null ? ExecutionContexts.genId() : id;
this.minBackendLogLevel = null;
}
this.source = source;
this.baggage = Collections.EMPTY_MAP;
this.logs = null;
this.closeables = Collections.EMPTY_LIST;
}
@Override
public final String getName() {
return name;
}
@Override
public final long getDeadlineNanos() {
return deadlineNanos;
}
public final long getStartTimeNanos() {
return startTimeNanos;
}
@Nullable
@Beta
@Override
public final synchronized T put(@Nonnull final Tag key, @Nonnull final T data) {
if (baggage == Collections.EMPTY_MAP) {
baggage = new HashMap<>(4);
}
return (T) baggage.put(key, data);
}
@Nullable
@Beta
@Override
public final synchronized T get(@Nonnull final Tag key) {
Object res = baggage.get(key);
if (res == null && source != null && key.isInherited(relation)) {
ExecutionContext src = source;
Relation rel;
do {
res = src.getLocal(key);
rel = src.getRelationToSource();
src = src.getSource();
} while (res == null && src != null && key.isInherited(rel));
}
return (T) res;
}
@Override
@SuppressWarnings("unchecked")
public final synchronized ContextValue getContextAndValue(final Tag key) {
Object res = baggage.get(key);
ExecutionContext ctx = this;
if (res == null && source != null && key.isInherited(relation)) {
ExecutionContext src = source;
Relation rel;
do {
res = src.getLocal(key);
rel = src.getRelationToSource();
ctx = src;
src = src.getSource();
} while (res == null && src != null && key.isInherited(rel));
}
return res == null ? null : new ContextValue(ctx, (T) res);
}
@Nullable
@Beta
@Override
public final synchronized T getLocal(@Nonnull final Tag key) {
return (T) baggage.get(key);
}
@Override
@Nullable
public final synchronized V compute(@Nonnull final Tag key, final BiFunction, V, V> compute) {
if (baggage == Collections.EMPTY_MAP) {
baggage = new HashMap(4);
}
return (V) baggage.compute(key, (BiFunction) compute);
}
@Override
@SuppressFBWarnings(value = "EI_EXPOSE_REP",
justification = "THis is intentional,"
+ " and as such these objects need to be handled with care since they are mutable, but thread safe")
public final ExecutionContext getSource() {
return source;
}
/**
* Close might be overridable to close any additional stuff added in the extended class.
*/
@Override
public synchronized void close() {
if (!isClosed) {
if (attached != null) {
detach();
}
Exception ex = null;
for (int i = closeables.size() - 1; i >= 0; i--) {
try {
closeables.get(i).close();
} catch (Exception e) {
if (ex != null) {
Throwables.suppressLimited(e, ex);
}
ex = e;
}
}
ExecutionContext parent = getNotClosedParent();
if (parent != null) {
if (logs != null) {
parent.addLogs(logs);
}
for (Map.Entry be : baggage.entrySet()) {
Tag key = be.getKey();
if (key.pushOnClose()) {
parent.accumulate(key, be.getValue());
}
}
} else if (this.source != null && this.relation == relation.CHILD_OF) {
// In theory a child context should finish before parent,
// if not, we will log the baggage that was not returned to parent.
StackTraceElement[] stackTrace = null;
Logger orphaned = Logger.getLogger("ORPHAN_CTX_ENTITIES");
if (logs != null) {
for (Slf4jLogRecord lr : logs) {
if (stackTrace == null) {
stackTrace = Thread.currentThread().getStackTrace();
}
LogUtils.logUpgrade(orphaned, Level.INFO, "Orphaned log", lr.toLogRecord("", ""), stackTrace);
}
}
for (Map.Entry be : baggage.entrySet()) {
Tag key = be.getKey();
if (key.pushOnClose()) {
if (stackTrace == null) {
stackTrace = Thread.currentThread().getStackTrace();
}
LogUtils.logUpgrade(orphaned, Level.INFO, "Orphaned baggage", be.getKey().toString(),
be.getValue(), stackTrace);
}
}
}
isClosed = true;
if (ex != null) {
if (ex instanceof RuntimeException) {
throw (RuntimeException) ex;
} else {
throw new RuntimeException(ex);
}
}
}
}
@Override
public final synchronized void detach() {
attached.detach();
attached = null;
}
public final synchronized boolean isAttached() {
return attached != null;
}
@Override
public final synchronized void attach() {
if (attached != null) {
throw new IllegalStateException("Context already attached, can only be attached to one thread at a time: "
+ attached);
}
attached = ExecutionContexts.threadLocalAttacher().attach(this);
}
/**
* Overwrite as needed for debug string.
*/
@Override
public String toString() {
StringBuilder sb = new StringBuilder(64);
writeTo(sb);
return sb.toString();
}
/**
* Overwrite this method to change default json format.
* @param appendable
*/
@Override
public synchronized void writeJsonTo(final Appendable appendable) throws IOException {
JsonGenerator gen = Json.FACTORY.createGenerator(new AppendableWriter(appendable));
gen.setCodec(Json.MAPPER);
gen.writeStartObject();
gen.writeFieldName("name");
gen.writeString(name);
gen.writeFieldName("startTs");
Timing currentTiming = Timing.getCurrentTiming();
gen.writeString(currentTiming.fromNanoTimeToInstant(startTimeNanos).toString());
gen.writeFieldName("deadlineTs");
gen.writeString(currentTiming.fromNanoTimeToInstant(deadlineNanos).toString());
gen.writeEndObject();
gen.flush();
}
@Override
public final synchronized void addLog(final Slf4jLogRecord log) {
if (isClosed) {
if (source == null) {
return;
} else {
source.addLog(log);
return;
}
}
if (logs == null) {
logs = new ArrayDeque<>(4);
}
if (logs.size() >= MX_NR_LOGS_PER_CTXT) {
logs.removeFirst();
}
logs.addLast(log);
}
@Beta
@Override
public final synchronized void addCloseable(final AutoCloseable closeable) {
if (this.closeables.isEmpty()) {
this.closeables = new ArrayList<>(4);
}
this.closeables.add(closeable);
}
@Override
public final synchronized void addLogs(final Collection pLogs) {
if (isClosed) {
if (source == null) {
return;
} else {
source.addLogs(pLogs);
return;
}
}
if (logs == null) {
logs = new ArrayDeque<>(pLogs);
return;
}
int xNrLogs = logs.size();
int toRemove = xNrLogs + pLogs.size() - MX_NR_LOGS_PER_CTXT;
if (toRemove >= xNrLogs) {
logs.clear();
} else {
for (int i = 0; i < toRemove; i++) { // TODO
logs.removeFirst();
}
}
logs.addAll(pLogs);
}
@Override
public final synchronized void streamLogs(final Consumer to) {
if (logs != null) {
for (Slf4jLogRecord log : logs) {
to.accept(log);
}
}
}
@Override
public final synchronized void streamLogs(final Consumer to, final int maxNr) {
if (logs != null) {
if (maxNr >= logs.size()) {
for (Slf4jLogRecord log : logs) {
to.accept(log);
}
} else {
int i = 0;
int toSkip = logs.size() - maxNr;
for (Slf4jLogRecord log : logs) {
if (i >= toSkip) {
to.accept(log);
} else {
i++;
}
}
}
}
}
/**
* Overwrite for more configurable implementation.
* @param loggerName
* @return
*/
@Override
public Level getContextMinLogLevel(final String loggerName) {
return MIN_LOG_LEVEL;
}
/**
* Overwrite for more configurable implementation.
* @param loggerName
* @return
*/
@Override
public synchronized Level getBackendMinLogLevel(final String loggerName) {
return minBackendLogLevel;
}
/**
* Overwrite for more configurable implementation.
* @param loggerName
* @return
*/
@Override
public synchronized Level setBackendMinLogLevel(final String loggerName, final Level level) {
Level result = minBackendLogLevel;
minBackendLogLevel = level;
return result;
}
@Override
public final CharSequence getId() {
return id;
}
@Override
public final synchronized long nextChildId() {
return childCount++;
}
/**
* Add a sample to this execution context.
* this implementation does not retain them.
* @param sample
*/
@Override
public void add(final StackTraceElement[] sample) {
// Basic Context will not staore samples
}
/**
* get the collected stack samples.
* This implementation will always return null.
* overwrite in a context that supports this functionality.
* @return
*/
@Override
@Nullable
public StackSamples getAndClearStackSamples() {
return null;
}
/**
* overwrite in a context that supports this functionality.
* @return a copy of the collected samples.
*/
@Override
@Nullable
public StackSamples getStackSamples() {
return null;
}
@Override
public final synchronized boolean isClosed() {
return isClosed;
}
@Override
public final Relation getRelationToSource() {
return relation;
}
/**
* add a bunch of samples to this context.
* @param samples
*/
@Override
public void add(final StackSamples samples) {
//Basic Context will not staore samples
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy