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

org.jumpmind.db.util.CallbackClosure Maven / Gradle / Ivy

Go to download

This library is a fork of Apache DdlUtils. DdlUtils is a small, easy-to-use component for working with Database Definition (DDL) files.

There is a newer version: 3.5.19
Show newest version
package org.jumpmind.db.util;

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.
 */

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.Map;

import org.apache.commons.collections.Closure;
import org.jumpmind.db.platform.DdlException;

/**
 * A closure that determines a callback for the type of the object and calls it.
 * Note that inheritance is also taken into account. I.e. if the object is of
 * type B which is a subtype of A, and there is only a callback for type A,
 * then this one will be invoked. If there is however also a callback for type B,
 * then only this callback for type B will be invoked and not the one for type A. 
 * 
 * @version $Revision: $
 */
public class CallbackClosure implements Closure
{
    /** The object on which the callbacks will be invoked. */
    private Object _callee;
    /** The parameter types. */
    private Class[] _parameterTypes;
    /** The parameters. */
    private Object[] _parameters;
    /** The position of the callback parameter type. */
    private int _callbackTypePos = -1;
    /** The cached callbacks. */
    private Map _callbacks = new HashMap();

    /**
     * Creates a new closure object.
     * 
     * @param callee         The object on which the callbacks will be invoked
     * @param callbackName   The name of the callback method
     * @param parameterTypes The parameter types. This array has to contain one null
     *                       for the type of the object for which the callback is invoked.
     *                       null or an empty array is regarded to be the
     *                       same as an array containing a single null
     * @param parameters     The actual arguments. The value at the placeholder position
     *                       will be ignored. Can be null if no parameter types
     *                       where given
     */
    public CallbackClosure(Object callee, String callbackName, Class[] parameterTypes, Object[] parameters)
    {
        _callee = callee;

        if ((parameterTypes == null) || (parameterTypes.length == 0))
        {
            _parameterTypes  = new Class[] { null };
            _parameters      = new Object[] { null };
            _callbackTypePos = 0;
        }
        else
        {
            _parameterTypes = new Class[parameterTypes.length];
            _parameters     = new Object[parameterTypes.length];

            for (int idx = 0; idx < parameterTypes.length; idx++)
            {
                if (parameterTypes[idx] == null)
                {
                    if (_callbackTypePos >= 0)
                    {
                        throw new IllegalArgumentException("The parameter types may contain null only once");
                    }
                    _callbackTypePos = idx;
                }
                else
                {
                    _parameterTypes[idx] = parameterTypes[idx];
                    _parameters[idx]     = parameters[idx];
                }
            }
            if (_callbackTypePos < 0)
            {
                throw new IllegalArgumentException("The parameter types need to a null placeholder");
            }
        }
        
        Class type = callee.getClass();

        // we're caching the callbacks
        do
        {
            Method[] methods = type.getDeclaredMethods();

            if (methods != null)
            {
                for (int idx = 0; idx < methods.length; idx++)
                {
                    Method  method     = methods[idx];
                    Class[] paramTypes = methods[idx].getParameterTypes();

                    method.setAccessible(true);
                    if (method.getName().equals(callbackName) && typesMatch(paramTypes))
                    {
                        if (_callbacks.get(paramTypes[_callbackTypePos]) == null)
                        {
                            _callbacks.put(paramTypes[_callbackTypePos], methods[idx]);
                        }
                    }
                }
            }
            type = type.getSuperclass();
        }
        while ((type != null) && !type.equals(Object.class));
    }

    /**
     * Checks whether the given method parameter types match the expected ones.
     * 
     * @param methodParamTypes The method parameter types
     * @return true if the parameter types match
     */
    private boolean typesMatch(Class[] methodParamTypes)
    {
        if ((methodParamTypes == null) || (_parameterTypes.length != methodParamTypes.length))
        {
            return false;
        }
        for (int idx = 0; idx < _parameterTypes.length; idx++)
        {
            if ((idx != _callbackTypePos) && !_parameterTypes[idx].equals(methodParamTypes[idx]))
            {
                return false;
            }
        }
        return true;
    }
    
    /**
     * {@inheritDoc}
     */
    public void execute(Object obj) throws DdlException
    {
        LinkedList queue = new LinkedList();

        queue.add(obj.getClass());
        while (!queue.isEmpty())
        {
            Class  type     = (Class)queue.removeFirst();
            Method callback = (Method)_callbacks.get(type);

            if (callback != null)
            {
                try
                {
                    _parameters[_callbackTypePos] = obj;
                    callback.invoke(_callee, _parameters);
                    return;
                }
                catch (InvocationTargetException ex)
                {
                    throw new DdlException(ex.getTargetException());
                }
                catch (IllegalAccessException ex)
                {
                    throw new DdlException(ex);
                }
            }
            if ((type.getSuperclass() != null) && !type.getSuperclass().equals(Object.class))
            {
                queue.add(type.getSuperclass());
            }

            Class[] baseInterfaces = type.getInterfaces();

            if (baseInterfaces != null)
            {
                for (int idx = 0; idx < baseInterfaces.length; idx++)
                {
                    queue.add(baseInterfaces[idx]);
                }
            }
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy