Package com.jitlogic.zorka.core

Source Code of com.jitlogic.zorka.core.ZorkaLib

/**
* Copyright 2012-2014 Rafal Lewczuk <rafal.lewczuk@jitlogic.com>
*
* ZORKA is free software. You can redistribute it and/or modify it under the
* terms of the GNU General Public License as published by the Free Software
* Foundation, either version 3 of the License, or (at your option) any later
* version.
*
* ZORKA 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 General Public License along with
* ZORKA. If not, see <http://www.gnu.org/licenses/>.
*/

package com.jitlogic.zorka.core;

import java.util.*;
import java.util.concurrent.ConcurrentHashMap;

import javax.management.*;

import com.jitlogic.zorka.common.ZorkaService;
import com.jitlogic.zorka.common.stats.AgentDiagnostics;
import com.jitlogic.zorka.common.util.*;
import com.jitlogic.zorka.common.util.FileTrapper;
import com.jitlogic.zorka.core.integ.QueryTranslator;
import com.jitlogic.zorka.core.spy.SpyClassTransformer;
import com.jitlogic.zorka.core.spy.SpyDefinition;
import com.jitlogic.zorka.core.spy.SpyMatcherSet;
import com.jitlogic.zorka.core.util.*;
import com.jitlogic.zorka.core.mbeans.MBeanServerRegistry;
import com.jitlogic.zorka.core.perfmon.*;
import com.jitlogic.zorka.core.mbeans.AttrGetter;
import com.jitlogic.zorka.common.stats.ValGetter;
import com.jitlogic.zorka.core.mbeans.ZorkaMappedMBean;


/**
* Standard library for zorka-agent. All public methods implemented in this module will be available for
* zorka configuration scripts in 'zorka' namespace.
*
* @author rafal.lewczuk@jitlogic.com
*/
public class ZorkaLib implements ZorkaService {

    private static final ZorkaLog log = ZorkaLogger.getLog(ZorkaLib.class);

    public static final ZorkaLogLevel TRACE = ZorkaLogLevel.TRACE;
    public static final ZorkaLogLevel DEBUG = ZorkaLogLevel.DEBUG;
    public static final ZorkaLogLevel INFO = ZorkaLogLevel.INFO;
    public static final ZorkaLogLevel WARN = ZorkaLogLevel.WARN;
    public static final ZorkaLogLevel ERROR = ZorkaLogLevel.ERROR;
    public static final ZorkaLogLevel FATAL = ZorkaLogLevel.FATAL;

    private static final int MINUTE = 60000;
    private static final int SECOND = 1000;

    private ZorkaLogger logger = ZorkaLogger.getLogger();

    private ZorkaBshAgent agent;
    private Set<JmxObject> registeredObjects = new HashSet<JmxObject>();

    private MBeanServerRegistry mbsRegistry;

    private String hostname;

    private AvgRateCounter rateCounter = new AvgRateCounter(this);
    private Map<String, FileTrapper> fileTrappers = new ConcurrentHashMap<String, FileTrapper>();

    private TaskScheduler scheduler = TaskScheduler.instance();

    private String version;

    private AgentConfig config;

    private QueryTranslator translator;

    private AgentInstance instance;

    /**
     * Standard constructor
     */
    public ZorkaLib(AgentInstance instance, QueryTranslator translator) {
        this.agent = instance.getZorkaAgent();
        this.mbsRegistry = instance.getMBeanServerRegistry();
        this.config = instance.getConfig();
        this.translator = translator;
        this.instance = instance;

        this.hostname = config.getProperties().getProperty("zorka.hostname").trim();
        this.version = config.getProperties().getProperty("zorka.version").trim();
    }


    /**
     * Returns agent version.
     *
     * @return string containing agent version
     */
    public String version() {
        return version;
    }


    /**
     * Returns hostname (name agent will present to monitoring systems communicating with agent).
     * Note that this does have to be host name of server running application server hosting this agent.
     * Administrator can configure many JVMs on a single server and each can present different name.
     * Agent hostname can be configured in zorka.properties file as 'zorka.hostname' property.
     * <p/>
     * By convention 'app-name.os-hostname.domain' should be used.
     *
     * @return hostname string
     */
    public String getHostname() {
        return "" + hostname;
    }


    /**
     * Returns list of objects from given mbean server.
     *
     * @param args attribute chain (as in ObjectInspector.get() function)
     * @return list of objects
     *         <p/>
     *         TODO fix parameters of this method
     */
    public List<Object> jmxList(List<Object> args) {
        List<Object> objs = new ArrayList<Object>();
        if (args.size() < 2) {
            log.error(ZorkaLogger.ZAG_ERRORS, "Zorka JMX function takes at least 2 arguments.");
            return objs;
        }
        String conname = args.get(0).toString();
        MBeanServerConnection conn = mbsRegistry.lookup(conname);
        if (conn == null) {
            log.error(ZorkaLogger.ZAG_ERRORS, "MBean server named '" + args.get(0) + "' is not registered.");
            return objs;
        }
        ClassLoader cl0 = Thread.currentThread().getContextClassLoader(), cl1 = mbsRegistry.getClassLoader(conname);

        Set<ObjectName> names = ObjectInspector.queryNames(conn, args.get(1).toString());
        if (args.size() == 2) {
            for (ObjectName name : names) {
                objs.add(new JmxObject(name, conn, cl1));
            }
        } else {
            if (cl1 != null) {
                Thread.currentThread().setContextClassLoader(cl1);
            }
            for (ObjectName name : names) {
                Object obj = null;
                try {
                    obj = conn.getAttribute(name, args.get(2).toString());
                } catch (AttributeNotFoundException e) {
                    log.error(ZorkaLogger.ZAG_ERRORS, "Object '" + conname + "|" + name + "' has no attribute '" + args.get(2) + "'.", e);
                } catch (Exception e) {
                    log.error(ZorkaLogger.ZAG_ERRORS, "Error getting attribute '" + args.get(2) + "' from '" + conname + "|" + name + "'", e);
                }

                if (args.size() > 3) {
                    obj = ObjectInspector.get(obj, args.subList(3, args.size()).toArray(new Object[0]));
                }
                objs.add(obj);
            }
            if (cl1 != null) {
                Thread.currentThread().setContextClassLoader(cl0);
            }
        }
        return objs;
    } // jmxList()


    /**
     * Return text dump of selected JMX objects
     *
     * @param args attribute chain (as in ObjectInspector.get() function)
     * @return dump of selected JMX objects
     *         TODO fix parameters of this method
     */
    public String dump(Object... args) {

        StringBuffer sb = new StringBuffer();

        for (Object obj : jmxList(Arrays.asList(args))) {
            sb.append(ObjectDumper.objectDump(obj));
        }

        return sb.toString();
    } // dump()


    public Object jmxv(Object defval, Object... args) {
        Object v = jmx(args);
        return v != null ? v : defval;
    }


    /**
     * Retrieves object from JMX
     *
     * @param args attribute chain (as in ObjectInspector.get() function)
     * @return retrieved obejct
     */
    public Object jmx(Object... args) {

        List<Object> argList = Arrays.asList(args);

        if (argList.size() < 2) {
            log.error(ZorkaLogger.ZAG_ERRORS, "zorka.jmx() function requires at least 2 arguments");
            return null;
        }

        String conname = argList.get(0).toString();

        MBeanServerConnection conn = mbsRegistry.lookup(conname);

        if (conn == null) {
            log.error(ZorkaLogger.ZAG_ERRORS, "MBean server named '" + argList.get(0) + "' is not registered.");
            return null;
        }

        Set<ObjectName> names = ObjectInspector.queryNames(conn, argList.get(1).toString());

        if (names.isEmpty()) {
            return null;
        }

        ObjectName name = names.iterator().next();

        ClassLoader cl0 = Thread.currentThread().getContextClassLoader(), cl1 = mbsRegistry.getClassLoader(conname);

        if (argList.size() == 2) {
            return new JmxObject(name, conn, cl1);
        }

        Object obj = null;
        try {
            if (cl1 != null) {
                Thread.currentThread().setContextClassLoader(cl1);
            }
            obj = conn.getAttribute(name, argList.get(2).toString());
        } catch (AttributeNotFoundException e) {
            log.error(ZorkaLogger.ZAG_ERRORS, "Object '" + conname + "|" + name + "' has no attribute '" + argList.get(2) + "'.", e);
            return null;
        } catch (Exception e) {
            log.error(ZorkaLogger.ZAG_ERRORS, "Error getting attribute '" + argList.get(2) + "' from '" + conname + "|" + name + "'", e);
        } finally {
            if (cl1 != null) {
                Thread.currentThread().setContextClassLoader(cl0);
            }
        }

        if (argList.size() > 3 && obj != null) {
            obj = ObjectInspector.get(obj, argList.subList(3, argList.size()).toArray(new Object[0]));
        }

        return obj;
    } // jmx()


    /**
     * Lists attributes of given object(s)
     *
     * @param mbsName    mbean server name
     * @param objectName object name (mask)
     * @param args       attribute chain (as in ObjectInspector.get())
     * @return string listing attributes and their values
     */
    public String ls(String mbsName, String objectName, Object... args) {

        MBeanServerConnection conn = mbsRegistry.lookup(mbsName);

        if (conn == null) {
            log.error(ZorkaLogger.ZAG_ERRORS, "MBean server named '" + mbsName + "' is not registered.");
            return null;
        }

        QueryDef qdef = new QueryDef(mbsName, objectName, "*");

        for (int i = 0; i < args.length; i++) {
            qdef = qdef.listAs(args[i].toString(), "ARG" + i);
        }

        ClassLoader cl0 = Thread.currentThread().getContextClassLoader(), cl1 = mbsRegistry.getClassLoader(mbsName);

        List<String> lst = new ArrayList<String>();

        try {

            if (cl1 != null) {
                Thread.currentThread().setContextClassLoader(cl1);
                log.debug(ZorkaLogger.ZAG_DEBUG, "Switching to MBS class loader ...");
            }

            List<QueryResult> results = new QueryLister(mbsRegistry, qdef).list();


            for (QueryResult result : results) {
                StringBuilder sb = new StringBuilder();
                int n = 0;
                for (Map.Entry<String, Object> e : result.attrSet()) {
                    if (n > 0) {
                        sb.append(n == 1 ? ": " : ".");
                    }
                    n++;
                    sb.append(e.getValue());
                }
                sb.append(" -> ");
                sb.append(result.getValue());
                lst.add(sb.toString());
            }

        } finally {
            if (cl1 != null) {
                Thread.currentThread().setContextClassLoader(cl0);
                log.debug(ZorkaLogger.ZAG_DEBUG, "Switching back from class loader ...");
            }
        }

        Collections.sort(lst);
        return ZorkaUtil.join("\n", lst);
    }


    /**
     * Creates zorka dynamic MBean. Such mbean object can be populated with attributes
     * using put() or setAttr() methods.
     *
     * @param mbs  mbean server at which bean will register
     * @param name object name
     * @return mbean object
     */
    public ZorkaMappedMBean mbean(String mbs, String name) {
        return mbean(mbs, name, "");
    }


    /**
     * Creates zorka dynamic MBean. Such mbean object can be populated with attributes
     *
     * @param mbs  mbean server at which bean will register
     * @param name object name
     * @param desc mbean description
     * @return mbean object
     */
    public ZorkaMappedMBean mbean(String mbs, String name, String desc) {
        // TODO wyrugować rÄ™czne tworzenie mbeanów, zdefiniować jednÄ… (wspólnÄ…) metodÄ™ do tego i używać jej
        try {
            ZorkaMappedMBean mbean = new ZorkaMappedMBean(desc);
            MBeanServer conn = (MBeanServer) mbsRegistry.lookup(mbs);
            if (conn == null) {
                throw new IllegalArgumentException("There is no mbean server named '" + mbs + "'");
            }

            ObjectName on = new ObjectName(name);
            conn.registerMBean(mbean, on);
            registeredObjects.add(new JmxObject(on, conn, Thread.currentThread().getContextClassLoader()));
            return mbean;
        } catch (Exception e) {
            // TODO zalogowac problem
            return null;
        }
    }


    /**
     * Creates Attribute getter object.
     *
     * @param obj
     * @param attrs
     * @return
     */
    public ValGetter getter(Object obj, Object... attrs) {
        return new AttrGetter(obj, attrs);
    }


    /**
     * Calculates windowed rate of two associated parameters (eg. execution time and number of calls).
     * This is useful for statistics that only provide total values (and no averages). It requires
     * caller to periodically call this function in order to maintain collected deltas, so this method
     * will be deprecated as soon as viable alternative appears.
     *
     * @param args first two arguments are mbean server name and object name, last three arguments are
     *             nominator field, divider field and time horizon, all remaining middle arguemnts are
     *             optional and are part of attribute chain needed to reach from mbean to proper object
     *             containing statistic
     * @return calculated rate
     */
    public Double rate(Object... args) {

        if (args.length < 5) {
            log.error(ZorkaLogger.ZAG_ERRORS, "Too little arguments for zorka.rate(). At least 5 args are required");
            return null;
        }

        Object oh = args[args.length - 1];
        long horizon = 0;

        if (oh instanceof String && ((String) oh).matches("^AVG[0-9]+$")) {
            horizon = Long.parseLong(oh.toString().substring(3)) * MINUTE;
        } else {
            horizon = rateCounter.coerce(args[args.length - 1]) * SECOND;
        }

        if (horizon == 0) {
            log.error(ZorkaLogger.ZAG_ERRORS, "Invalid time horizon in zorka.rate()");
            return null;
        }

        String div = (String) args[args.length - 2];
        String nom = (String) args[args.length - 3];

        return rateCounter.get(Arrays.asList(ZorkaUtil.clipArray(args, args.length - 3)), nom, div, horizon);
    }


    /**
     * Sends DEBUG message to zorka log.
     *
     * @param message message text
     * @param args    optional arguments (if message contains format string markers)
     */
    public void logDebug(String message, Object... args) {
        log(ZorkaLogLevel.DEBUG, message, args);
    }

    /**
     * Sends INFO message to zorka log.
     *
     * @param message message text
     * @param args    optional arguments (if message contains format string markers)
     */
    public void logInfo(String message, Object... args) {
        log(ZorkaLogLevel.INFO, message, args);
    }

    /**
     * Sends WARNING message to zorka log.
     *
     * @param message message text
     * @param args    optional arguments (if message contains format string markers)
     */
    public void logWarning(String message, Object... args) {
        log(ZorkaLogLevel.WARN, message, args);
    }

    /**
     * Sends ERROR message to zorka log.
     *
     * @param message message text
     * @param args    optional arguments (if message contains format string markers)
     */
    public void logError(String message, Object... args) {
        log(ZorkaLogLevel.ERROR, message, args);
    }

    /**
     * Sends FATAL message to zorka log.
     *
     * @param message message text
     * @param args    optional arguments (if message contains format string markers)
     */
    public void logFatal(String message, Object... args) {
        log(ZorkaLogLevel.ERROR, message, args);
    }

    /**
     * Logs arbitrary message to zorka log.
     *
     * @param level   log level
     * @param message message text
     * @param argv    optional arguments (if message is a format string)
     */
    private void log(ZorkaLogLevel level, String message, Object... argv) {
        Throwable ex = null;
        Object[] args = argv;
        if (args.length > 0 && args[args.length - 1] instanceof Throwable) {
            ex = (Throwable) args[args.length - 1];
            args = args.length > 1 ? ZorkaUtil.clipArray(args, -1) : new Object[0];
        }
        logger.trap(level, "bsh", message, ex, args);
    }


    /**
     * Reloads agent configuration and scripts.
     *
     * @return
     */
    public String reload() {
        instance.reload();
        long l = AgentDiagnostics.get(AgentDiagnostics.CONFIG_ERRORS);
        return l == 0 ? "OK" : "ERRORS(" + l + ") see agent log.";
    }

    public String loadScript(String script) {
        return agent.loadScript(script);
    }


    public String require(String... names) {
        String s = "";
        for (String name : names) {
            s += agent.require(name) + "; ";
        }
        return s;
    }


    /**
     * Returns true if agent has been initialized (i.e. executing BSH code is
     * executing after initial execution of configuration scripts
     *
     * @return true if agent has been initialized
     */
    public boolean isInitialized() {
        return agent.isInitialized();
    }


    /**
     * Registers mbean server in agent mbean server registry. It is useful for some
     * application servers that have non-standard (additional) mbean servers or when
     * agent cannot access platform mbean server at start up time (notably JBoss 4/5/6/7).
     *
     * @param name name at which mbean server will be registered
     * @param mbs  reference to mbean server connection
     */
    public void registerMbs(String name, MBeanServerConnection mbs) {
        mbsRegistry.register(name, mbs, mbs.getClass().getClassLoader());
    }


    /**
     * Registers mbean server in agent mbean server registry. It is useful for some
     * application servers that have non-standard (additional) mbean servers or when
     * agent cannot access platform mbean server at start up time (notably JBoss 4/5/6/7).
     *
     * @param name        name at which mbean server will be registered
     * @param mbs         reference to mbean server connection
     * @param classLoader context class loader
     */
    public void registerMbs(String name, MBeanServerConnection mbs, ClassLoader classLoader) {
        mbsRegistry.register(name, mbs, classLoader);
    }


    public boolean isMbsRegistered(String name) {
        return mbsRegistry.lookup(name) != null;
    }


    /**
     * Presents object as an attribute of an mbean. If there is already such attribute, its value
     * will be returned instead. If mbean exists, it must be ZorkaMappedBean.
     *
     * @param mbsName  mbean server name
     * @param beanName object name
     * @param attrName attribute name
     * @param obj      object to be presented
     * @param <T>      type of object to be presented
     * @return obj if there was no such attribute or value of existing attribute
     */
    public <T> T registerAttr(String mbsName, String beanName, String attrName, T obj) {
        return mbsRegistry.getOrRegister(mbsName, beanName, attrName, obj);
    }

    /**
     * Presents object as an attribute of an mbean. If there is already such attribute, its value
     * will be returned instead. If mbean exists, it must be ZorkaMappedBean.
     *
     * @param mbsName  mbean server name
     * @param beanName object name
     * @param attrName attribute name
     * @param obj      object to be presented
     * @param <T>      type of object to be presented
     * @param desc     attribute description
     * @return obj if there was no such attribute or value of existing attribute
     */
    public <T> T registerAttr(String mbsName, String beanName, String attrName, T obj, String desc) {
        return mbsRegistry.getOrRegister(mbsName, beanName, attrName, obj, desc);
    }


    /**
     * Creates JMX lister object that can be used to create rankings
     *
     * @param mbsName mbean server name
     * @param onMask  object name (or mask)
     * @param <T>     wrapped object type (if any)
     * @return JMX rank lister object
     */
    public <T extends Rankable<?>> RankLister<T> jmxLister(String mbsName, String onMask) {
        JmxAggregatingLister<T> lister = new JmxAggregatingLister<T>(mbsRegistry, mbsName, onMask);
        return lister;
    }


    /**
     * Thread rank lister (if it has been created)
     */
    private ThreadRankLister threadRankLister;


    /**
     * Returns thread rank lister. Creates and starts a new one if none has been creater (yet).
     * As thread lister causes some stress on JVM, it is a singleton object and it is created in lazy manner.
     *
     * @return thread rank lister object
     */
    public synchronized ThreadRankLister threadRankLister() {
        if (threadRankLister == null) {
            threadRankLister = new ThreadRankLister(mbsRegistry);
            scheduler.schedule(threadRankLister, 15000, 0);
        }

        return threadRankLister;
    }

    /**
     * Creates EJB rank lister object that can be used to create rankings.
     * It will list EJB statistics from selected mbeans.
     *
     * @param mbsName  mbean server name
     * @param objNames object names
     * @param attr     attribute name
     * @return EJB rank lister object
     */
    public EjbRankLister ejbRankLister(String mbsName, String objNames, String attr) {
        EjbRankLister lister = new EjbRankLister(mbsRegistry, mbsName, objNames, attr);
        scheduler.schedule(lister, 15000, 0);
        return lister;
    }


    /**
     * Looks for file trapper and returns if trapper exists.
     *
     * @param id trapper ID
     * @return file trapper or null
     */
    public FileTrapper fileTrapper(String id) {
        return fileTrappers.get(id);
    }


    /**
     * Loooks for file trapper registered as 'id' or creates and registers rolling file trapper.
     *
     * @param id            trapper ID
     * @param logLevel      log level (only messages with such or higher log level will be logged)
     * @param path          path to log file (excluding numbered suffixes)
     * @param count         number of archived files (excluding current one)
     * @param maxSize       maximum file size
     * @param logExceptions if true, stack traces of passed exceptions will be logged
     * @return file trapper
     */
    public FileTrapper rollingFileTrapper(String id, String logLevel, String path, int count, long maxSize, boolean logExceptions) {
        FileTrapper trapper = fileTrappers.get(id);

        if (trapper == null) {
            trapper = FileTrapper.rolling(ZorkaLogLevel.valueOf(logLevel), formatCfg(path), count, maxSize, logExceptions);
            trapper.start();
            fileTrappers.put(id, trapper);
        }

        return trapper;
    }


    /**
     * Looks for file trapper registered as 'id' or creates and registers daily file trapper.
     *
     * @param id            trapper ID
     * @param logLevel      trapper log level (only messages with such or higher log level will be logged)
     * @param path          path to log file (excluding suffix indicating log date)
     * @param logExceptions if true, trapper will log stack traces of passed exceptions
     * @return file trapper
     */
    public FileTrapper dailyFileTrapper(String id, ZorkaLogLevel logLevel, String path, boolean logExceptions) {
        FileTrapper trapper = fileTrappers.get(id);

        if (trapper == null) {
            trapper = FileTrapper.daily(logLevel, formatCfg(path), logExceptions);
            trapper.start();
            fileTrappers.put(id, trapper);
        }

        return trapper;
    }


    /**
     * Stops and unregisters file trapper
     *
     * @param id trapper ID
     */
    public void removeFileTrapper(String id) {
        FileTrapper trapper = fileTrappers.get(id);

        if (trapper != null) {
            trapper.shutdown();
        }
    }

    @Override
    public void shutdown() {
        for (FileTrapper trapper : fileTrappers.values()) {
            trapper.shutdown();
        }
        fileTrappers.clear();
    }


    /**
     * Formats string containing references to zorka properties.
     *
     * @param input zorka properties
     * @return
     */
    public String formatCfg(String input) {
        return config.formatCfg(input);
    }


    /**
     * Returns true if given config property is set in zorka.properties file and has non-zero length
     *
     * @param key property key
     * @return true if entry exists and is non-empty
     */
    public boolean hasCfg(String key) {
        return config.hasCfg(key);
    }


    public Boolean boolCfg(String key) {
        return boolCfg(key, null);
    }


    public Boolean boolCfg(String key, Boolean defval) {
        return config.boolCfg(key, defval);
    }


    public Integer intCfg(String key) {
        return intCfg(key, null);
    }


    public Integer intCfg(String key, Integer defval) {
        return config.intCfg(key, defval);
    }


    public Long longCfg(String key) {
        return longCfg(key, null);
    }


    public Long longCfg(String key, Long defval) {
        return config.longCfg(key, defval);
    }


    public Long kiloCfg(String key) {
        return kiloCfg(key, null);
    }


    public Long kiloCfg(String key, Long defval) {
        return config.kiloCfg(key, defval);
    }


    public String stringCfg(String key) {
        return stringCfg(key, null);
    }


    public String stringCfg(String key, String defval) {
        return config.stringCfg(key, defval);
    }


    public void defCfg(String key, String defVal) {
        if (!hasCfg(key)) {
            config.setCfg(key, defVal);
        }
    }

    public void defCfg(String key, Object defVal) {
        String strVal = defVal != null ? defVal.toString() : null;
        defCfg(key, strVal);
    }

    public void setCfg(String key, String val) {
        config.setCfg(key, val);
    }

    public Set<String> setCfg(String key) {
        Set<String> set = new HashSet<String>();
        set.addAll(listCfg(key));
        return set;
    }


    public Object ifCfg(String key, boolean defVal, Object thenVal, Object elseVal) {
        return boolCfg(key, defVal) ? thenVal : elseVal;
    }


    public Object ifCfg(String key, boolean defVal, Object thenVal) {
        return ifCfg(key, defVal, thenVal, null);
    }


    public Properties loadCfg(String fname) {
        String path = ZorkaUtil.path(config.getHomeDir(), fname);
        Properties props = config.loadCfg(config.getProperties(), path, false);
        if (props != null) {
            log.info(ZorkaLogger.ZAG_INFO, "Loaded property file: " + path);
        } else {
            log.info(ZorkaLogger.ZAG_INFO, "Property file not found: " + path);
        }
        return props;
    }


    public Properties loadCfg(Properties properties, String fname, boolean verbose) {
        String path = ZorkaUtil.path(config.getHomeDir(), fname);
        Properties props = config.loadCfg(properties, path, verbose);
        if (props != null) {
            log.info(ZorkaLogger.ZAG_INFO, "Loaded property file: " + path);
        } else {
            log.info(ZorkaLogger.ZAG_INFO, "Property file not found: " + path);
        }
        return props;
    }


    /**
     * Parses comma-separated configuration setting and returns it as list of strings.
     *
     * @param key     key in zorka.properties file
     * @param defVals set of default values (if key is missing)
     * @return parsed list
     */
    public List<String> listCfg(String key, String... defVals) {
        return config.listCfg(key, defVals);
    }


    /**
     * Schedules a task.
     *
     * @param task     task (must be Runnable)
     * @param interval run interval (in milliseconds)
     */
    public void schedule(Runnable task, long interval, long delay) {
        scheduler.schedule(task, interval, delay);
    }


    public QueryDef query(String mbsName, String query, String... attrs) {
        return new QueryDef(mbsName, query, attrs);
    }


    public void allow(String... funcs) {
        for (String func : funcs) {
            translator.allow(func);
        }
    }

}
TOP

Related Classes of com.jitlogic.zorka.core.ZorkaLib

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.