blob: 4318a17d4c4a300974a86b0ba9233ea3f65a7ddd [file] [log] [blame]
/*
* 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.
*/
package org.apache.johnzon.mapper.access;
import org.apache.johnzon.mapper.JohnzonProperty;
import org.apache.johnzon.mapper.MapperException;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.lang.reflect.Type;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
public class MethodAccessMode implements AccessMode {
private final boolean supportGetterAsWritter;
public MethodAccessMode(final boolean supportGetterAsWritter) {
this.supportGetterAsWritter = supportGetterAsWritter;
}
@Override
public Map<String, Reader> findReaders(final Class<?> clazz) {
final Map<String, Reader> readers = new HashMap<String, Reader>();
final PropertyDescriptor[] propertyDescriptors = getPropertyDescriptors(clazz);
for (final PropertyDescriptor descriptor : propertyDescriptors) {
final Method readMethod = descriptor.getReadMethod();
if (readMethod != null && readMethod.getDeclaringClass() != Object.class) {
if (isIgnored(descriptor.getName())) {
continue;
}
readers.put(extractKey(descriptor), new MethodReader(readMethod));
}
}
return readers;
}
@Override
public Map<String, Writer> findWriters(final Class<?> clazz) {
final Map<String, Writer> writers = new HashMap<String, Writer>();
final PropertyDescriptor[] propertyDescriptors = getPropertyDescriptors(clazz);
for (final PropertyDescriptor descriptor : propertyDescriptors) {
if (descriptor.getPropertyType() == Class.class || isIgnored(descriptor.getName())) {
continue;
}
final Method writeMethod = descriptor.getWriteMethod();
if (writeMethod != null) {
writers.put(extractKey(descriptor), new MethodWriter(writeMethod));
} else if (supportGetterAsWritter
&& Collection.class.isAssignableFrom(descriptor.getPropertyType())
&& descriptor.getReadMethod() != null) {
writers.put(extractKey(descriptor), new MethodGetterAsWriter(descriptor.getReadMethod()));
}
}
return writers;
}
private String extractKey(final PropertyDescriptor f) {
final JohnzonProperty property = f.getReadMethod() == null ? null : f.getReadMethod().getAnnotation(JohnzonProperty.class);
return property != null ? property.value() : f.getName();
}
protected boolean isIgnored(final String name) {
return name.equals("metaClass") || name.contains("$");
}
private PropertyDescriptor[] getPropertyDescriptors(final Class<?> clazz) {
final PropertyDescriptor[] propertyDescriptors;
try {
propertyDescriptors = Introspector.getBeanInfo(clazz).getPropertyDescriptors();
} catch (final IntrospectionException e) {
throw new IllegalStateException(e);
}
return propertyDescriptors;
}
protected static abstract class MethodDecoratedType implements DecoratedType {
protected final Method method;
public MethodDecoratedType(final Method method) {
this.method = method;
if (!method.isAccessible()) {
method.setAccessible(true);
}
}
@Override
public <T extends Annotation> T getAnnotation(final Class<T> clazz) {
return method.getAnnotation(clazz);
}
}
public static class MethodWriter extends MethodDecoratedType implements Writer {
public MethodWriter(final Method method) {
super(method);
}
@Override
public Type getType() {
return method.getGenericParameterTypes()[0];
}
@Override
public void write(final Object instance, final Object value) {
try {
method.invoke(instance, value);
} catch (final Exception e) {
throw new MapperException(e);
}
}
}
public static class MethodReader extends MethodDecoratedType implements Reader {
public MethodReader(final Method method) {
super(method);
}
@Override
public Type getType() {
return method.getGenericReturnType();
}
@Override
public Object read(final Object instance) {
try {
return method.invoke(instance);
} catch (final Exception e) {
throw new MapperException(e);
}
}
}
private class MethodGetterAsWriter extends MethodReader implements Writer {
public MethodGetterAsWriter(final Method readMethod) {
super(readMethod);
}
@Override
public void write(final Object instance, final Object value) {
if (value != null) {
try {
final Collection<?> collection = Collection.class.cast(method.invoke(instance));
if (collection != null) {
collection.addAll(Collection.class.cast(value));
}
} catch (final Exception e) {
throw new MapperException(e);
}
}
}
}
}