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

com.devebot.opflow.OpflowLogTracer Maven / Gradle / Ivy

package com.devebot.opflow;

import com.devebot.opflow.supports.OpflowTextFormat;
import com.google.gson.Gson;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.jar.Attributes;
import java.util.jar.Manifest;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 *
 * @author drupalex
 */
public class OpflowLogTracer {
    private final static Gson GSON = new Gson();
    private final static Logger LOG = LoggerFactory.getLogger(OpflowLogTracer.class);
    private final static String OPFLOW_VERSION = "0.1.x";
    private final static String OPFLOW_INSTANCE_ID = OpflowUtil.getLogID();
    private final static String DEFAULT_PARENT_ID_NAME = "_parentId_";
    private final static String DEFAULT_NODE_ID_NAME = "_nodeId_";
    private final static String DEFAULT_NODE_TYPE_NAME = "_nodeType_";
    
    private final static Set ALWAYS_ENABLED;
    private final static int TRACKING_DEPTH;
    private final static boolean KEEP_ORDER;
    private final static String TAGS_FIELD_NAME;
    private final static String TEXT_FIELD_NAME;
    private final static boolean IS_INTERCEPTOR_ENABLED;
    private final static boolean IS_TRACING_ID_PREDEFINED;
    private final static boolean IS_STRINGIFY_ENABLED;
    private final static boolean IS_TAGS_EMBEDDABLE;
    private final static boolean IS_TEXT_EMBEDDABLE;
    private final static boolean IS_TEMPLATE_APPLIED;

    static {
        ALWAYS_ENABLED = new HashSet();
        String[] _levels = OpflowUtil.getSystemProperty("OPFLOW_ALWAYS_ENABLED", "").split(",");
        for(String _level: _levels) {
            ALWAYS_ENABLED.add(_level.trim());
        }
        
        String treepath = OpflowUtil.getSystemProperty("OPFLOW_TRACKING_DEPTH", null);
        if ("none".equals(treepath)) TRACKING_DEPTH = 0;
        else if ("parent".equals(treepath)) TRACKING_DEPTH = 1;
        else if ("full".equals(treepath)) TRACKING_DEPTH = 2;
        else TRACKING_DEPTH = 2;
        
        KEEP_ORDER = (OpflowUtil.getSystemProperty("OPFLOW_LOGKEEPORDER", null) == null);
        
        TAGS_FIELD_NAME = OpflowUtil.getSystemProperty("OPFLOW_TAGS_FIELD_NAME", "_tags_");
        TEXT_FIELD_NAME = OpflowUtil.getSystemProperty("OPFLOW_TEXT_FIELD_NAME", "_text_");
        
        IS_TRACING_ID_PREDEFINED = "true".equals(OpflowUtil.getSystemProperty("OPFLOW_TRACING_ID_PREDEFINED", null));
        IS_TAGS_EMBEDDABLE = !"false".equals(OpflowUtil.getSystemProperty("OPFLOW_TAGS_EMBEDDABLE", null));
        IS_TEXT_EMBEDDABLE = "false".equals(OpflowUtil.getSystemProperty("OPFLOW_TEXT_EMBEDDABLE", null));
        IS_TEMPLATE_APPLIED = !"false".equals(OpflowUtil.getSystemProperty("OPFLOW_TEMPLATE_APPLIED", null));
        IS_INTERCEPTOR_ENABLED = !"false".equals(OpflowUtil.getSystemProperty("OPFLOW_DEBUGLOG", null));
        IS_STRINGIFY_ENABLED = true;
    }
    
    private final OpflowLogTracer parent;
    private final String key;
    private final Object value;
    private final Map fields;
    private final Set frozen;
    private final Set tags;
    private String template;
    
    public final static OpflowLogTracer ROOT = new OpflowLogTracer();
    
    public OpflowLogTracer() {
        this(null, "instanceId", OpflowUtil.getSystemProperty("OPFLOW_INSTANCE_ID", OPFLOW_INSTANCE_ID));
    }
    
    private OpflowLogTracer(OpflowLogTracer ref, String key, Object value) {
        this.parent = ref;
        this.key = key;
        this.value = value;
        this.fields = KEEP_ORDER ? new LinkedHashMap() : new HashMap();
        this.frozen = new HashSet();
        this.tags = new HashSet();
        this.reset();
    }
    
    public OpflowLogTracer copy() {
        return new OpflowLogTracer(this.parent, this.key, this.value);
    }
    
    public OpflowLogTracer branch(String key, Object value) {
        return new OpflowLogTracer(this, key, value);
    }
    
    public final OpflowLogTracer clear() {
        if (this.fields.size() > this.frozen.size()) {
            this.fields.keySet().retainAll(this.frozen);
        }
        return this;
    }
    
    public final OpflowLogTracer reset() {
        this.fields.clear();
        this.frozen.clear();
        if (IS_TRACING_ID_PREDEFINED) {
            if (this.parent != null) {
                this.fields.put(DEFAULT_PARENT_ID_NAME, this.parent.value);
            }
            this.fields.put(DEFAULT_NODE_ID_NAME, value);
            this.fields.put(DEFAULT_NODE_TYPE_NAME, key);
        } else {
            if (TRACKING_DEPTH > 0) {
                if (TRACKING_DEPTH == 1) {
                    if (this.parent != null) {
                        this.fields.put(this.parent.key, this.parent.value);
                    }
                } else {
                    OpflowLogTracer ref = this.parent;
                    while(ref != null) {
                        this.fields.put(ref.key, ref.value);
                        ref = ref.parent;
                    }
                }
            }
            this.fields.put(key, value);
        }
        this.frozen.addAll(this.fields.keySet());
        return this;
    }
    
    public OpflowLogTracer put(String key, Object value) {
        this.fields.put(key, value);
        return this;
    }
    
    public Object get(String key) {
        return this.fields.get(key);
    }
    
    public OpflowLogTracer tags(String t) {
        this.tags.add(t);
        return this;
    }
    
    public OpflowLogTracer tags(String[] ts) {
        this.tags.addAll(Arrays.asList(ts));
        return this;
    }
    
    public OpflowLogTracer text(String s) {
        this.template = s;
        return this;
    }
    
    @Override
    public String toString() {
        return stringify();
    }
    
    public String stringify() {
        return stringify(true);
    }
    
    public String stringify(String template) {
        return stringify(template, true);
    }
    
    public String stringify(boolean clear) {
        return stringify(null, clear);
    }
    
    public String stringify(String template, boolean clear) {
        String output = null, text = null;
        Set tagz = null;
        
        if (template == null) {
            template = this.template;
        }
        
        if (IS_TAGS_EMBEDDABLE || IS_INTERCEPTOR_ENABLED) {
            tagz = this.tags;
        }
        
        if (IS_TEXT_EMBEDDABLE || (IS_TEMPLATE_APPLIED && template != null)) {
            text = OpflowTextFormat.format(template, this.fields);
        }
        
        if (IS_STRINGIFY_ENABLED) {
            if (IS_TEMPLATE_APPLIED && text != null) {
                output = text;
            } else {
                if (IS_TAGS_EMBEDDABLE && tagz != null && tagz.size() > 0) {
                    this.fields.put(TAGS_FIELD_NAME, tagz);
                }
                if (IS_TEXT_EMBEDDABLE && text != null) {
                    this.fields.put(TEXT_FIELD_NAME, text);
                }
                output = GSON.toJson(fields);
            }
        }
        
        if (IS_INTERCEPTOR_ENABLED && !interceptors.isEmpty()) {
            Map cloned = new HashMap();
            cloned.putAll(fields);
            tagz = new HashSet(this.tags);
            for(StringifyInterceptor interceptor:interceptors) {
                interceptor.intercept(cloned, tagz);
            }
        }
        this.tags.clear();
        this.template = null;
        if (clear) this.clear();
        return output;
    }
    
    public interface StringifyInterceptor {
        public void intercept(Map logdata, Set tags);
    }
    
    private static List interceptors = new ArrayList();
    
    public static boolean addInterceptor(StringifyInterceptor interceptor) {
        if (interceptor != null && interceptors.indexOf(interceptor) < 0) {
            interceptors.add(interceptor);
            return true;
        }
        return false;
    }
    
    @Deprecated
    public static boolean addStringifyInterceptor(StringifyInterceptor interceptor) {
        return addInterceptor(interceptor);
    }
    
    public static boolean removeInterceptor(StringifyInterceptor interceptor) {
        return interceptors.remove(interceptor);
    }
    
    @Deprecated
    public static boolean removeStringifyInterceptor(StringifyInterceptor interceptor) {
        return removeInterceptor(interceptor);
    }
    
    public static void clearInterceptors() {
        interceptors.clear();
    }
    
    @Deprecated
    public static void clearStringifyInterceptors() {
        clearInterceptors();
    }
    
    private static String libraryInfo = null;;
    
    public static String getLibraryInfo() {
        if (libraryInfo == null) {
            libraryInfo = new OpflowLogTracer()
                    .put("message", "Opflow Library Information")
                    .put("lib_name", "opflow-java")
                    .put("lib_version", getVersionNameFromManifest())
                    .put("os_name", System.getProperty("os.name"))
                    .put("os_version", System.getProperty("os.version"))
                    .put("os_arch", System.getProperty("os.arch"))
                    .text("[${instanceId}] Library: ${lib_name}@${lib_version} - ${message}")
                    .stringify();
        }
        return libraryInfo;
    }
    
    private static String getVersionNameFromPOM() {
        try {
            Properties props = new Properties();
            String POM_PROPSFILE = "META-INF/maven/com.devebot.opflow/opflow-core/pom.properties";
            props.load(OpflowLogTracer.class.getClassLoader().getResourceAsStream(POM_PROPSFILE));
            return props.getProperty("version");
        } catch (Exception ioe) {}
        return OPFLOW_VERSION;
    }
    
    private static String getVersionNameFromManifest() {
        try {
            InputStream manifestStream = OpflowLogTracer.class.getClassLoader().getResourceAsStream("META-INF/MANIFEST.MF");
            if (manifestStream != null) {
                Manifest manifest = new Manifest(manifestStream);
                Attributes attributes = manifest.getMainAttributes();
                return attributes.getValue("Implementation-Version");
            }
        } catch (Exception ioe) {}
        return OPFLOW_VERSION;
    }
    
    public static boolean has(Logger logger, String level) {
        if (logger == null) return false;
        if (ALWAYS_ENABLED.contains("all")) return true;
        if (ALWAYS_ENABLED.contains(level)) return true;
        if ("debug".equalsIgnoreCase(level)) return logger.isDebugEnabled();
        if ("trace".equalsIgnoreCase(level)) return logger.isTraceEnabled();
        if ("info".equalsIgnoreCase(level)) return logger.isInfoEnabled();
        if ("warn".equalsIgnoreCase(level)) return logger.isWarnEnabled();
        if ("error".equalsIgnoreCase(level)) return logger.isErrorEnabled();
        return false;
    }
    
    static {
        if (has(LOG, "info")) LOG.info(getLibraryInfo());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy