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

com.nitorcreations.willow.deployer.statistics.CustomJMXStatsSender Maven / Gradle / Ivy

There is a newer version: 2.0.2
Show newest version
package com.nitorcreations.willow.deployer.statistics;

import java.io.IOException;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Properties;
import java.util.concurrent.TimeUnit;
import java.util.logging.Level;

import javax.inject.Named;
import javax.management.AttributeNotFoundException;
import javax.management.InstanceNotFoundException;
import javax.management.IntrospectionException;
import javax.management.MBeanException;
import javax.management.MBeanInfo;
import javax.management.MBeanOperationInfo;
import javax.management.MBeanParameterInfo;
import javax.management.MBeanServerConnection;
import javax.management.MalformedObjectNameException;
import javax.management.ObjectName;
import javax.management.ReflectionException;

import org.apache.commons.beanutils.ConvertUtils;

import com.nitorcreations.willow.messages.HashMessage;
import com.nitorcreations.willow.utils.MergeableProperties;

@Named("customjmx")
public class CustomJMXStatsSender extends AbstractJMXStatisticsSender {
  private long nextJmx;
  private long interval;
  private Map props = new LinkedHashMap<>();
  private Map opDescs = new HashMap<>();
  private static final class JMXAttributeReader implements JMXReader {
    private final ObjectName objectName;
    private final String attribute;
    private JMXAttributeReader(ObjectName objectName, String attribute) {
      this.objectName = objectName;
      this.attribute = attribute;
    }
    @Override
    public Object read(MBeanServerConnection server) throws AttributeNotFoundException, InstanceNotFoundException, MBeanException, ReflectionException, IOException {
      return server.getAttribute(objectName, attribute);
    }
  }
  private interface JMXReader {
    Object read(MBeanServerConnection server) throws AttributeNotFoundException, InstanceNotFoundException, MBeanException, ReflectionException, IOException, IntrospectionException, ClassNotFoundException;
  }
  private static class OperationDesriptor {
    public Object[] params;
    public String[] signature;
  }
  @Override
  public void setProperties(Properties properties) {
    super.setProperties(properties);
    MergeableProperties tmp = new MergeableProperties();
    tmp.putAll(properties);
    List lst = tmp.getArrayProperty("property");
    for (int i=0; i argList = tmp.getArrayProperty("property[" + i + "].argument");
          final int index = i;
          if (operationName != null) {
            props.put(reportname, new JMXReader() {
              @Override
              public Object read(MBeanServerConnection server) throws AttributeNotFoundException, InstanceNotFoundException, MBeanException, ReflectionException, IOException, IntrospectionException, ClassNotFoundException {
                OperationDesriptor desc = getOperationDescriptor(server, objectName, operationName, argList, index);
                if (desc != null) {
                  return server.invoke(objectName, operationName, desc.params, desc.signature);
                } else {
                  return null;
                }
              }
            });
          }
        }
      } catch (MalformedObjectNameException e) {
        logger.info("Malformed ObjectName:" + properties.getProperty("property[" + i + "].objectname"));
      }
    }
    interval = Long.parseLong(properties.getProperty("interval", "5000"));
    nextJmx = System.currentTimeMillis() + interval;
  }
  private OperationDesriptor getOperationDescriptor(MBeanServerConnection server, ObjectName objectName, String operationName, List argList, int i) throws InstanceNotFoundException, ReflectionException, IOException, IntrospectionException, ClassNotFoundException {
    OperationDesriptor desc = opDescs.get(i);
    if (desc != null) {
      return desc;
    }
    desc = new OperationDesriptor();
    MBeanInfo mBeanInfo = server.getMBeanInfo(objectName);
    MBeanOperationInfo[] operations = mBeanInfo.getOperations();
    for (MBeanOperationInfo operation : operations) {
      if (operation.getName().equals(operationName)) {
        MBeanParameterInfo[] parameters = operation.getSignature();
        desc.signature = new String[parameters.length];
        for (int j = 0; j < desc.signature.length; j++) {
          desc.signature[j] = parameters[j].getType();
        }
        desc.params = argList.toArray(new Object[argList.size()]);
        if (desc.params.length != desc.signature.length) {
          continue;
        }
        for (int j = 0; j < desc.signature.length; j++) {
          desc.params[j] = ConvertUtils.convert(desc.params[j], Class.forName(desc.signature[j]));
        }
        opDescs.put(i, desc);
        return desc;
      }
    }
    return null;
  }
  @Override
  public void execute() {
    long now = System.currentTimeMillis();
    if (now > nextJmx) {
      for (String childName : getChildren()) {
        try {
          Map map = new LinkedHashMap<>();
          MBeanServerConnection server = getMBeanServerConnection(childName);
          if (server == null) continue; 
          for (Entry next : props.entrySet()) {
            Object val = next.getValue().read(server);
            if (val != null) {
              map.put(next.getKey(), val.toString());
            }
          }
          if (!map.isEmpty()) {
            HashMessage msg = HashMessage.create(map);
            msg.addTags("category_customjmx_" + childName);
            transmitter.queue(msg);
          }
        } catch (IOException | ReflectionException | AttributeNotFoundException
            | InstanceNotFoundException | IntrospectionException | MBeanException
            | ClassNotFoundException e) {
          logger.log(Level.WARNING, "Failed to get JMX statistics", e);
        }
      }
      nextJmx = nextJmx + interval;
    }
    try {
      TimeUnit.MILLISECONDS.sleep(nextJmx - now);
    } catch (InterruptedException e) {
      logger.info("Process statistics interrupted");
      return;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy