/*
* Distributed as part of c3p0 0.9.5-pre10
*
* Copyright (C) 2014 Machinery For Change, Inc.
*
* Author: Steve Waldman <swaldman@mchange.com>
*
* This library is free software; you can redistribute it and/or modify
* it under the terms of EITHER:
*
* 1) The GNU Lesser General Public License (LGPL), version 2.1, as
* published by the Free Software Foundation
*
* OR
*
* 2) The Eclipse Public License (EPL), version 1.0
*
* You may choose which license to accept if you wish to redistribute
* or modify this work. You may offer derivatives of this work
* under the license you have chosen, or you may provide the same
* choice of license which you have been offered here.
*
* This software 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.
*
* You should have received copies of both LGPL v2.1 and EPL v1.0
* along with this software; see the files LICENSE-EPL and LICENSE-LGPL.
* If not, the text of these licenses are currently available at
*
* LGPL v2.1: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html
* EPL v1.0: http://www.eclipse.org/org/documents/epl-v10.php
*
*/
package com.mchange.v2.c3p0.management;
import java.beans.BeanInfo;
import java.beans.Introspector;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.util.*;
import javax.management.Attribute;
import javax.management.AttributeList;
import javax.management.AttributeNotFoundException;
import javax.management.DynamicMBean;
import javax.management.InvalidAttributeValueException;
import javax.management.MBeanAttributeInfo;
import javax.management.MBeanConstructorInfo;
import javax.management.MBeanException;
import javax.management.MBeanInfo;
import javax.management.MBeanOperationInfo;
import javax.management.MBeanParameterInfo;
import javax.management.MBeanServer;
import javax.management.ObjectName;
import javax.management.ObjectInstance;
import javax.management.ReflectionException;
import javax.sql.ConnectionPoolDataSource;
import javax.sql.DataSource;
import com.mchange.v1.lang.ClassUtils;
import com.mchange.v2.lang.ObjectUtils;
import com.mchange.v2.c3p0.AbstractComboPooledDataSource;
import com.mchange.v2.c3p0.DriverManagerDataSource;
import com.mchange.v2.c3p0.PooledDataSource;
import com.mchange.v2.c3p0.PoolBackedDataSource;
import com.mchange.v2.c3p0.WrapperConnectionPoolDataSource;
import com.mchange.v2.c3p0.impl.AbstractPoolBackedDataSource;
import com.mchange.v2.log.MLog;
import com.mchange.v2.log.MLogger;
import com.mchange.v2.log.MLevel;
import com.mchange.v2.management.ManagementUtils;
public class DynamicPooledDataSourceManagerMBean implements DynamicMBean
{
final static MLogger logger = MLog.getLogger( DynamicPooledDataSourceManagerMBean.class );
final static Set HIDE_PROPS;
final static Set HIDE_OPS;
final static Set FORCE_OPS;
final static Set FORCE_READ_ONLY_PROPS;
static
{
Set hpTmp = new HashSet();
hpTmp.add("connectionPoolDataSource");
hpTmp.add("nestedDataSource");
hpTmp.add("reference");
hpTmp.add("connection");
hpTmp.add("password");
hpTmp.add("pooledConnection");
hpTmp.add("properties");
hpTmp.add("logWriter");
hpTmp.add("lastAcquisitionFailureDefaultUser");
hpTmp.add("lastCheckoutFailureDefaultUser");
hpTmp.add("lastCheckinFailureDefaultUser");
hpTmp.add("lastIdleTestFailureDefaultUser");
hpTmp.add("lastConnectionTestFailureDefaultUser");
HIDE_PROPS = Collections.unmodifiableSet( hpTmp );
Class[] userPassArgs = new Class[] { String.class, String.class };
Set hoTmp = new HashSet();
try
{
hoTmp.add(PooledDataSource.class.getMethod("close", new Class[] { boolean.class }) );
hoTmp.add(PooledDataSource.class.getMethod("getConnection", userPassArgs ) );
hoTmp.add(PooledDataSource.class.getMethod("getLastAcquisitionFailure", userPassArgs ) );
hoTmp.add(PooledDataSource.class.getMethod("getLastCheckinFailure", userPassArgs ) );
hoTmp.add(PooledDataSource.class.getMethod("getLastCheckoutFailure", userPassArgs ) );
hoTmp.add(PooledDataSource.class.getMethod("getLastIdleTestFailure", userPassArgs ) );
hoTmp.add(PooledDataSource.class.getMethod("getLastConnectionTestFailure", userPassArgs ) );
}
catch (Exception e)
{
logger.log(MLevel.WARNING, "Tried to hide an operation from being exposed by mbean, but failed to find the operation!", e);
}
HIDE_OPS = Collections.unmodifiableSet(hoTmp);
Set fropTmp = new HashSet();
fropTmp.add("identityToken");
FORCE_READ_ONLY_PROPS = Collections.unmodifiableSet(fropTmp);
Set foTmp = new HashSet();
FORCE_OPS = Collections.unmodifiableSet(foTmp);
}
final static MBeanOperationInfo[] OP_INFS = extractOpInfs();
MBeanInfo info = null;
PooledDataSource pds;
String mbeanName;
MBeanServer mbs;
ConnectionPoolDataSource cpds;
DataSource unpooledDataSource;
//attr names to attr infos
Map pdsAttrInfos;
Map cpdsAttrInfos;
Map unpooledDataSourceAttrInfos;
PropertyChangeListener pcl = new PropertyChangeListener()
{
public void propertyChange(PropertyChangeEvent evt)
{
String propName = evt.getPropertyName();
Object val = evt.getNewValue();
if ("nestedDataSource".equals(propName) || "connectionPoolDataSource".equals(propName) || "dataSourceName".equals(propName))
reinitialize();
}
};
public DynamicPooledDataSourceManagerMBean(PooledDataSource pds, String mbeanName, MBeanServer mbs)
throws Exception
{
this.pds = pds;
this.mbeanName = mbeanName;
this.mbs = mbs;
if (pds instanceof AbstractComboPooledDataSource)
((AbstractComboPooledDataSource) pds).addPropertyChangeListener(pcl);
else if (pds instanceof AbstractPoolBackedDataSource)
((AbstractPoolBackedDataSource) pds).addPropertyChangeListener(pcl);
else
logger.warning(this + "managing an unexpected PooledDataSource. Only top-level attributes will be available. PooledDataSource: " + pds);
Exception e = reinitialize();
if (e != null)
throw e;
}
private synchronized Exception reinitialize()
{
try
{
// for AbstractComboPooledDataSource, everything we care about is exposed via the PooledDataSource
// for other implementations, we have to pay attention to nested DataSources
if (!(pds instanceof AbstractComboPooledDataSource) && pds instanceof AbstractPoolBackedDataSource)
{
if (this.cpds instanceof WrapperConnectionPoolDataSource) //implies non-null, this is a reinit
((WrapperConnectionPoolDataSource) this.cpds).removePropertyChangeListener(pcl);
// yeah, we reassign instantly, but for my comfort...
this.cpds = null;
this.unpooledDataSource = null;
this.cpds = ((AbstractPoolBackedDataSource) pds).getConnectionPoolDataSource();
if (cpds instanceof WrapperConnectionPoolDataSource)
{
this.unpooledDataSource = ((WrapperConnectionPoolDataSource) cpds).getNestedDataSource();
((WrapperConnectionPoolDataSource) this.cpds).addPropertyChangeListener(pcl);
}
}
pdsAttrInfos = extractAttributeInfos( pds );
cpdsAttrInfos = extractAttributeInfos( cpds );
unpooledDataSourceAttrInfos = extractAttributeInfos( unpooledDataSource );
Set allAttrNames = new HashSet();
allAttrNames.addAll(pdsAttrInfos.keySet());
allAttrNames.addAll(cpdsAttrInfos.keySet());
allAttrNames.addAll(unpooledDataSourceAttrInfos.keySet());
Set allAttrs = new HashSet();
for(Iterator ii = allAttrNames.iterator(); ii.hasNext();)
{
String name = (String) ii.next();
Object attrInfo;
attrInfo = pdsAttrInfos.get(name);
if (attrInfo == null)
attrInfo = cpdsAttrInfos.get(name);
if (attrInfo == null)
attrInfo = unpooledDataSourceAttrInfos.get(name);
allAttrs.add(attrInfo);
}
String className = this.getClass().getName();
MBeanAttributeInfo[] attrInfos = (MBeanAttributeInfo[]) allAttrs.toArray(new MBeanAttributeInfo[ allAttrs.size() ]);
Class[] ctorArgClasses = {PooledDataSource.class, String.class, MBeanServer.class};
MBeanConstructorInfo[] constrInfos
= new MBeanConstructorInfo[] { new MBeanConstructorInfo("Constructor from PooledDataSource", this.getClass().getConstructor(ctorArgClasses)) };
this.info = new MBeanInfo( this.getClass().getName(),
"An MBean to monitor and manage a PooledDataSource",
attrInfos,
constrInfos,
OP_INFS,
null);
// we need to reregister when the attributes we support may have changed, to be sure
// that the MBeanInfo is reread.
try
{
ObjectName oname = ObjectName.getInstance( mbeanName );
if (mbs.isRegistered( oname ))
{
mbs.unregisterMBean( oname );
if (logger.isLoggable(MLevel.FINE))
logger.log(MLevel.FINE, "MBean: " + oname.toString() + " unregistered, in order to be reregistered after update.");
}
// the name may have changed because dataSourceName may have changed.
this.mbeanName = ActiveManagementCoordinator.getPdsObjectNameStr(pds);
oname = ObjectName.getInstance(mbeanName);
mbs.registerMBean( this, oname );
if (logger.isLoggable(MLevel.FINE))
logger.log(MLevel.FINE, "MBean: " + oname.toString() + " registered.");
return null;
}
catch (Exception e)
{
if ( logger.isLoggable(MLevel.WARNING) )
logger.log(MLevel.WARNING,
"An Exception occurred while registering/reregistering mbean " + mbeanName +
". MBean may not be registered, or may not work properly.",
e );
return e;
}
}
catch (NoSuchMethodException e)
{
if (logger.isLoggable(MLevel.SEVERE))
logger.log( MLevel.SEVERE,
"Huh? We can't find our own constructor?? The one we're in?",
e);
return e;
}
}
// this method is fragile, makes assumptions that may have to change with
// the PooledDataSource interface. It presumes that methods that look like
// JavaBean properties should be skipped as they are attributes, that methods with
// two string arguments are always username and password, that methods with
// a return value are simple getters, while void methods are modifiers. At the
// time of this writing, these assumptions all hold for PooledDataSource.
// But beware the future.
private static MBeanOperationInfo[] extractOpInfs()
{
MBeanParameterInfo user = new MBeanParameterInfo("user", "java.lang.String", "The database username of a pool-owner.");
MBeanParameterInfo pwd = new MBeanParameterInfo("password", "java.lang.String", "The database password of a pool-owner.");
MBeanParameterInfo[] userPass = {user, pwd};
MBeanParameterInfo[] empty = {};
Method[] meths = PooledDataSource.class.getMethods();
Set attrInfos = new TreeSet(ManagementUtils.OP_INFO_COMPARATOR);
for (int i = 0; i < meths.length; ++i)
{
Method meth = meths[i];
if (HIDE_OPS.contains(meth))
continue;
String mname = meth.getName();
Class[] params = meth.getParameterTypes();
if (! FORCE_OPS.contains(mname))
{
//get rid of things we'd have picked up as attributes
if (mname.startsWith("set") && params.length == 1)
continue;
if ((mname.startsWith("get") || mname.startsWith("is")) && params.length == 0)
continue;
}
Class retType = meth.getReturnType();
int impact = (retType == void.class ? MBeanOperationInfo.ACTION : MBeanOperationInfo.INFO);
MBeanParameterInfo[] pi;
if (params.length == 2 && params[0] == String.class && params[1] == String.class)
pi = userPass;
else if (params.length == 0)
pi = empty;
else
pi = null;
MBeanOperationInfo opi;
if (pi != null)
opi = new MBeanOperationInfo( mname, // name
null, // desc
pi,
retType.getName(),
impact );
else
{
//System.err.println("autobuilding opi from meth " + meth);
opi = new MBeanOperationInfo(meth.toString(), meth);
}
//System.err.println("Created MBeanOperationInfo: " + opi + " [" + opi.getName() + ']');
attrInfos.add( opi );
}
return (MBeanOperationInfo[]) attrInfos.toArray( new MBeanOperationInfo[ attrInfos.size() ] );
}
public synchronized Object getAttribute(String attr) throws AttributeNotFoundException, MBeanException, ReflectionException
{
try
{
AttrRec rec = attrRecForAttribute(attr);
if (rec == null)
throw new AttributeNotFoundException(attr);
else
{
MBeanAttributeInfo ai = rec.attrInfo;
if (! ai.isReadable() )
throw new IllegalArgumentException(attr + " not readable.");
else
{
String name = ai.getName();
String pfx = ai.isIs() ? "is" : "get";
String mname = pfx + Character.toUpperCase(name.charAt(0)) + name.substring(1);
Object target = rec.target;
Method m = target.getClass().getMethod(mname, (Class[]) null); // cast to suppress inexact type warning
return m.invoke(target, (Object[]) null); // cast to suppress inexact type warning
}
}
}
catch (Exception e)
{
if (logger.isLoggable(MLevel.WARNING))
logger.log(MLevel.WARNING, "Failed to get requested attribute: " + attr, e);
throw new MBeanException(e);
}
}
public synchronized AttributeList getAttributes(String[] attrs)
{
AttributeList al = new AttributeList();
for (int i = 0, len = attrs.length; i < len; ++i)
{
String attr = attrs[i];
try
{
Object val = getAttribute(attr);
al.add(new Attribute(attr, val));
}
catch (Exception e)
{
if (logger.isLoggable(MLevel.WARNING))
logger.log(MLevel.WARNING, "Failed to get requested attribute (for list): " + attr, e);
}
}
return al;
}
private AttrRec attrRecForAttribute(String attr)
{
assert (Thread.holdsLock(this));
if (pdsAttrInfos.containsKey(attr))
return new AttrRec(pds, (MBeanAttributeInfo) pdsAttrInfos.get(attr));
else if (cpdsAttrInfos.containsKey(attr))
return new AttrRec(cpds, (MBeanAttributeInfo) cpdsAttrInfos.get(attr));
else if (unpooledDataSourceAttrInfos.containsKey(attr))
return new AttrRec(unpooledDataSource, (MBeanAttributeInfo) unpooledDataSourceAttrInfos.get(attr));
else
return null;
}
public synchronized MBeanInfo getMBeanInfo()
{
if (info == null)
reinitialize();
return info;
}
public synchronized Object invoke(String operation, Object[] paramVals, String[] signature) throws MBeanException, ReflectionException
{
try
{
int slen = signature.length;
Class[] paramTypes = new Class[ slen ];
for (int i = 0; i < slen; ++i)
paramTypes[i] = ClassUtils.forName( signature[i] );
//all operations should be on pds
Method m = pds.getClass().getMethod(operation, paramTypes);
return m.invoke(pds, paramVals);
}
catch (NoSuchMethodException e)
{
// although not generally legal as of the latest JMX spec,
// someone could be trying to work with attributes through
// invoke. If so, we try to deal
try
{
boolean two = false;
if (signature.length == 0 && ( operation.startsWith("get") || (two = operation.startsWith("is")) ))
{
int i = two ? 2 : 3;
String attr = Character.toLowerCase(operation.charAt(i)) + operation.substring(i + 1);
return getAttribute( attr );
}
else if (signature.length == 1 && operation.startsWith("set"))
{
setAttribute(new Attribute(Character.toLowerCase(operation.charAt(3)) + operation.substring(4), paramVals[0]));
return null;
}
else
throw new MBeanException(e);
}
catch (Exception e2)
{ throw new MBeanException(e2); }
}
catch (Exception e)
{ throw new MBeanException(e); }
}
public synchronized void setAttribute(Attribute attrObj) throws AttributeNotFoundException, InvalidAttributeValueException, MBeanException, ReflectionException
{
try
{
String attr = attrObj.getName();
Object curVal = getAttribute( attr );
Object newVal = attrObj.getValue();
if ( !ObjectUtils.eqOrBothNull( curVal, newVal ) )
{
if (attr == "factoryClassLocation") // special case
{
if (pds instanceof AbstractComboPooledDataSource)
{
((AbstractComboPooledDataSource) pds).setFactoryClassLocation((String) newVal);
return;
}
else if (pds instanceof AbstractPoolBackedDataSource)
{
String strval = (String) newVal;
AbstractPoolBackedDataSource apbds = ((AbstractPoolBackedDataSource) pds);
apbds.setFactoryClassLocation( strval );
ConnectionPoolDataSource checkDs1 = apbds.getConnectionPoolDataSource();
if (checkDs1 instanceof WrapperConnectionPoolDataSource)
{
WrapperConnectionPoolDataSource wcheckDs1 = (WrapperConnectionPoolDataSource) checkDs1;
wcheckDs1.setFactoryClassLocation( strval );
DataSource checkDs2 = wcheckDs1.getNestedDataSource();
if (checkDs2 instanceof DriverManagerDataSource)
((DriverManagerDataSource) checkDs2).setFactoryClassLocation( strval );
}
return;
}
// else try treating factoryClassLocation like any other attribute
// on the presumption that some future, unexpected DataSource that
// exposes this property will not require the property to be set at
// multiple levels, as PoolBackedDataSource does...
}
AttrRec rec = attrRecForAttribute(attr);
if (rec == null)
throw new AttributeNotFoundException(attr);
else
{
MBeanAttributeInfo ai = rec.attrInfo;
if (! ai.isWritable() )
throw new IllegalArgumentException(attr + " not writable.");
else
{
Class attrType = ClassUtils.forName( rec.attrInfo.getType() );
String name = ai.getName();
String pfx = "set";
String mname = pfx + Character.toUpperCase(name.charAt(0)) + name.substring(1);
Object target = rec.target;
Method m = target.getClass().getMethod(mname, new Class[] {attrType});
m.invoke(target, new Object[] { newVal });
// if we were unable to set this attribute directly in the PooledDataSource,
// we are updating a property of a nested DataSource, and we should reset
// the pool manager of the PooledDataSource implementation so that these
// properties are reread and the changes take effect.
if (target != pds)
{
if (pds instanceof AbstractPoolBackedDataSource)
((AbstractPoolBackedDataSource) pds).resetPoolManager(false);
else if (logger.isLoggable(MLevel.WARNING))
logger.warning("MBean set a nested ConnectionPoolDataSource or DataSource parameter on an unknown PooledDataSource type. " +
"Could not reset the pool manager, so the changes may not take effect. " + "" +
"c3p0 may need to be updated for PooledDataSource type " + pds.getClass() + ".");
}
}
}
}
}
catch (Exception e)
{
if (logger.isLoggable(MLevel.WARNING))
logger.log(MLevel.WARNING, "Failed to set requested attribute: " + attrObj, e);
throw new MBeanException(e);
}
}
public synchronized AttributeList setAttributes(AttributeList al)
{
AttributeList out = new AttributeList();
for (int i = 0, len = al.size(); i < len; ++i)
{
Attribute attrObj = (Attribute) al.get(i);
try
{
this.setAttribute( attrObj );
out.add(attrObj);
}
catch (Exception e)
{
if (logger.isLoggable(MLevel.WARNING))
logger.log(MLevel.WARNING, "Failed to set requested attribute (from list): " + attrObj, e);
}
}
return out;
}
private static Map extractAttributeInfos(Object bean)
{
if ( bean != null)
{
try
{
Map out = new HashMap();
BeanInfo beanInfo = Introspector.getBeanInfo( bean.getClass(), Object.class ); //so we don't see getClass() as a property
PropertyDescriptor[] pds = beanInfo.getPropertyDescriptors();
//System.err.println("ignoreProps: " + ignoreProps );
for( int i = 0, len = pds.length; i < len; ++i)
{
PropertyDescriptor pd = pds[i];
String name;
String desc;
Method getter;
Method setter;
name = pd.getName();
if (HIDE_PROPS.contains( name ))
continue;
desc = getDescription( name );
getter = pd.getReadMethod();
setter = pd.getWriteMethod();
if (FORCE_READ_ONLY_PROPS.contains(name))
setter = null;
/*
* Note that it's not a problem that these
* getters and setters are not against this class
* the MBeanAttributInfo just uses the method
* names and attribute type to construct itself,
* and does not hold the methods themselves for
* future invocation.
*/
try
{
out.put( name, new MBeanAttributeInfo(name, desc, getter, setter) );
}
catch (javax.management.IntrospectionException e)
{
if (logger.isLoggable( MLevel.WARNING ))
logger.log( MLevel.WARNING, "IntrospectionException while setting up MBean attribute '" + name + "'", e);
}
}
return Collections.synchronizedMap(out);
}
catch (java.beans.IntrospectionException e)
{
if (logger.isLoggable( MLevel.WARNING ))
logger.log( MLevel.WARNING, "IntrospectionException while setting up MBean attributes for " + bean, e);
return Collections.EMPTY_MAP;
}
}
else
return Collections.EMPTY_MAP;
}
// TODO: use a ResourceBundle for attribute descriptions.
// (Extra credit -- build from xml file, build docs same way)
private static String getDescription(String attrName)
{ return null; }
private static class AttrRec
{
Object target;
MBeanAttributeInfo attrInfo;
AttrRec(Object target, MBeanAttributeInfo attrInfo)
{
this.target = target;
this.attrInfo = attrInfo;
}
}
}