All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.spf4j.base.BasicExecutionContext Maven / Gradle / Ivy

Go to download

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...

There is a newer version: 8.10.0
Show newest version
/*
 * 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