/*
* Licensed 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.
*/
package org.jctools.channels.mapping;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import static java.lang.reflect.Modifier.isAbstract;
import static java.util.Arrays.asList;
public class TypeInspector {
private final Class<?> flyweightClass;
final List<Method> getters;
final Map<String, Method> setters;
public TypeInspector(Class<?> flyweightClass) {
this.flyweightClass = flyweightClass;
if(!flyweightClass.isInterface())
throw new InvalidInterfaceException("Your flyweight class must be an interface");
getters = findGetters();
setters = findSetters();
checkRemainingMethods(flyweightClass);
}
private void checkRemainingMethods(Class<?> klass) {
List<Method> methods = new ArrayList<Method>(asList(klass.getDeclaredMethods()));
methods.removeAll(getters);
methods.removeAll(setters.values());
for (Method method : methods)
if (isAbstract(method.getModifiers()))
throw new InvalidInterfaceException(klass.getName() + " has abstract methods that are neither getters nor setters");
}
private List<Method> findGetters() {
List<Method> methods = new ArrayList<Method>();
for (Method method : flyweightClass.getDeclaredMethods()) {
String name = method.getName();
if (!name.startsWith("get"))
continue;
ensureAbstract(method);
returnsPrimitive(method);
hasNoParameters(method);
methods.add(method);
}
return methods;
}
private void ensureAbstract(Method method) {
if (!isAbstract(method.getModifiers()))
throw new InvalidInterfaceException(method + " must be abstract, since its a getter or setter");
}
private void hasNoParameters(Method method) {
if (method.getParameterTypes().length != 0)
throw new InvalidInterfaceException(method.getName() + " is a getter with one or more parameters");
}
private void returnsPrimitive(Method method) {
if (!method.getReturnType().isPrimitive())
throw new InvalidInterfaceException(method.getName() + " is a getter that doesn't return a primitive");
}
Primitive getReturn(Method method) {
return Primitive.of(method.getReturnType());
}
private Map<String, Method> findSetters() {
Map<String, Method> methods = new HashMap<String, Method>();
for (Method method : flyweightClass.getDeclaredMethods()) {
if (!method.getName().startsWith("set"))
continue;
ensureAbstract(method);
returnsVoid(method);
hasOnePrimitiveParameter(method);
methods.put(method.getName(), method);
}
return methods;
}
private void hasOnePrimitiveParameter(Method method) {
Class<?>[] parameters = method.getParameterTypes();
if (parameters.length != 1)
throw new InvalidInterfaceException(method.getName() + " is a setter with more than one parameter");
if (!parameters[0].isPrimitive())
throw new InvalidInterfaceException(method.getName() + " is a setter with a non-primitive parameter");
}
private void returnsVoid(Method method) {
if (method.getReturnType() != Void.TYPE)
throw new InvalidInterfaceException(method.getName() + " is a setter that doesn't return void");
}
public int getSizeInBytes() {
int total = 0;
for (Method getter : getters) {
total += getReturn(getter).sizeInBytes;
}
return total;
}
public Method setterFor(Method getter) {
String name = getter.getName().replaceFirst("get", "set");
Method method = setters.get(name);
if (method == null)
throw new InvalidInterfaceException("Unable to find setter with name: " + name);
return method;
}
}