| /* |
| * 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.felix.dm.impl; |
| |
| import java.lang.reflect.Field; |
| import java.lang.reflect.ParameterizedType; |
| import java.lang.reflect.Proxy; |
| import java.lang.reflect.Type; |
| import java.util.Collection; |
| import java.util.Dictionary; |
| import java.util.Map; |
| import java.util.concurrent.ConcurrentHashMap; |
| import java.util.concurrent.ConcurrentLinkedQueue; |
| |
| import org.apache.felix.dm.Logger; |
| import org.apache.felix.dm.context.DependencyContext; |
| import org.apache.felix.dm.context.Event; |
| |
| /** |
| * Reflection Helper methods, used to inject autoconfig fields in component instances. |
| * |
| * @author <a href="mailto:dev@felix.apache.org">Felix Project Team</a> |
| */ |
| public class FieldUtil { |
| /** |
| * Callbacks for fields to be injected |
| */ |
| private interface FieldFunction { |
| // Inject an updated service in the given field for the the given target. |
| void injectField(Field f, Object target); |
| |
| // Inject an Iterable Field in the given target |
| void injectIterableField(Field f, Object target); |
| |
| // Inject a Map field in the given target (key = dependency service, value = Dictionary with dependency service properties). |
| void injectMapField(Field f, Object target); |
| } |
| |
| /** |
| * Injects some component instances (on a given field, if provided), with an object of a given class. |
| * @param targets the component instances to fill in |
| * @param fieldName the fieldname, or null. If null, the field must exaclty match the injected service classname. |
| * @param clazz the injected service class |
| * @param service the injected service |
| * @param logger the component logger. |
| */ |
| public static boolean injectField(Object[] targets, String fieldName, Class<?> clazz, final Object service, |
| final Logger logger) |
| { |
| if (service == null) { |
| return true; // TODO why service can be null ? |
| } |
| return mapField(true, clazz, targets, fieldName, logger, new FieldFunction() { |
| public void injectField(Field f, Object target) { |
| try { |
| f.setAccessible(true); |
| f.set(target, service); |
| } catch (Throwable e) { |
| logger.log(Logger.LOG_ERROR, "Could not set field " + f + " in class " |
| + target.getClass().getName(), e); |
| } |
| } |
| |
| public void injectIterableField(Field f, Object target) { // never called |
| } |
| |
| public void injectMapField(Field f, Object target) { // never called |
| } |
| }); |
| } |
| |
| /** |
| * Injects a dependency service in some component instances. |
| * Here, we'll inject the dependency services in the component if the field is of the same type of the injected services, |
| * or if the field is a Collection of the injected service, or if the field is a Map<Injected Service class, Dictionary). |
| * @param targets the component instances to fill in |
| * @param fieldName the fieldname, or null. If null, the field must exaclty match the injected service classname. |
| * @param clazz the injected service class |
| * @param service the injected service |
| * @param logger the component logger. |
| */ |
| public static boolean injectDependencyField(Object[] targets, String fieldName, Class<?> clazz, |
| final DependencyContext dc, final Logger logger) |
| { |
| final Event event = dc.getService(); |
| if (event == null) { |
| return true; // TODO check why event can be null |
| } |
| return mapField(false, clazz, targets, fieldName, logger, new FieldFunction() { |
| public void injectField(Field f, Object target) { |
| try { |
| f.setAccessible(true); |
| f.set(target, event.getEvent()); |
| } catch (Throwable e) { |
| logger.log(Logger.LOG_ERROR, "Could not set field " + f + " in class " |
| + target.getClass().getName(), e); |
| } |
| } |
| |
| @SuppressWarnings("unchecked") |
| public void injectIterableField(Field f, Object target) { |
| f.setAccessible(true); |
| |
| try { |
| Iterable<Object> iter = (Iterable<Object>) f.get(target); |
| if (iter == null) { |
| iter = new ConcurrentLinkedQueue<Object>(); |
| f.set(target, iter); |
| } |
| dc.copyToCollection((Collection<Object>) iter); |
| } catch (Throwable e) { |
| logger.log(Logger.LOG_ERROR, "Could not set field " + f + " in class " |
| + target.getClass().getName(), e); |
| } |
| } |
| |
| @SuppressWarnings({ "unchecked", "rawtypes" }) |
| @Override |
| public void injectMapField(Field f, Object target) { |
| f.setAccessible(true); |
| try { |
| Map<Object, Dictionary<?, ?>> map = (Map) f.get(target); |
| if (map == null) { |
| map = new ConcurrentHashMap<>(); |
| f.set(target, map); |
| } |
| dc.copyToMap(map); |
| } catch (Throwable e) { |
| logger.log(Logger.LOG_ERROR, "Could not set field " + f + " in class " |
| + target.getClass().getName(), e); |
| } |
| } |
| }); |
| } |
| |
| /** |
| * Adds, or removes, or update some component instances with an updated dependency service |
| * @param targets the component instances to fill in with the updated service |
| * @param fieldName the component instance fieldname |
| * @param update true if it's a dependency service update, false if the dependency service is added or removed |
| * @param add true if the dependency service has been added, false the dependency service has been removed. |
| * This flag is ignored if the "update" parameter is "true". |
| * @param clazz the clazz of the dependency service |
| * @param event the event holding the dependency service |
| * @param dc the dependency service context |
| * @param logger the logger used when problems occure. |
| */ |
| public static void updateDependencyField(Object[] targets, String fieldName, final boolean update, |
| final boolean add, Class<?> clazz, final Event event, final DependencyContext dc, final Logger logger) |
| { |
| mapField(false, clazz, targets, fieldName, logger, new FieldFunction() { |
| public void injectField(Field f, Object target) { |
| try { |
| f.setAccessible(true); |
| f.set(target, dc.getService().getEvent()); |
| } catch (Throwable e) { |
| logger.log(Logger.LOG_ERROR, "Could not set field " + f + " in class " |
| + target.getClass().getName(), e); |
| } |
| } |
| |
| @SuppressWarnings("unchecked") |
| public void injectIterableField(Field f, Object target) { |
| if (update) { |
| return; |
| } |
| |
| f.setAccessible(true); |
| |
| try { |
| Collection<Object> coll = (Collection<Object>) f.get(target); |
| if (add) { |
| coll.add(event.getEvent()); |
| } else { |
| coll.remove(event.getEvent()); |
| } |
| } catch (Throwable e) { |
| logger.log(Logger.LOG_ERROR, "Could not set field " + f + " in class " |
| + target.getClass().getName(), e); |
| } |
| } |
| |
| @SuppressWarnings({ "rawtypes", "unchecked" }) |
| @Override |
| public void injectMapField(Field f, Object target) { |
| f.setAccessible(true); |
| |
| try { |
| Map<Object, Dictionary<?, ?>> map = (Map) f.get(target); |
| if (add || update) { |
| map.put(event.getEvent(), event.getProperties()); |
| } else { |
| map.remove(event.getEvent()); |
| } |
| } catch (Throwable e) { |
| logger.log(Logger.LOG_ERROR, "Could not set field " + f + " in class " |
| + target.getClass().getName(), e); |
| } |
| } |
| }); |
| } |
| |
| /** |
| * Scans component instances for fields having either the same dependency service type, or being a |
| * Collection of the dependency service, or being a Map<Dependency Service class, Dictionary> (the Dictionary |
| * corresponds to the dependency service properties). |
| * @param strict true if we are only looking for fields having exactly the same type as the dependency service. |
| * In other words, if strict = true, we don't lookup for fields of "Collection" or "Map" types. |
| * @param clazz the dependency service class |
| * @param targets the component instances |
| * @param fieldName the component instances field name or null |
| * @param logger a logger used when exceptions are occuring |
| * @param func the callback used to notify when we find either a field with the same dependency service type, or |
| * with a "Collection" type, or with a "Map" type. |
| */ |
| private static boolean mapField(boolean strict, Class<?> clazz, Object[] targets, String fieldName, Logger logger, |
| FieldFunction func) |
| { |
| boolean injected = false; |
| if (targets != null && clazz != null) { |
| for (int i = 0; i < targets.length; i++) { |
| Object target = targets[i]; |
| Class<?> targetClass = target.getClass(); |
| if (Proxy.isProxyClass(targetClass)) { |
| target = Proxy.getInvocationHandler(target); |
| targetClass = target.getClass(); |
| } |
| while (targetClass != null) { |
| Field[] fields = targetClass.getDeclaredFields(); |
| for (int j = 0; j < fields.length; j++) { |
| Field field = fields[j]; |
| Class<?> fieldType = field.getType(); |
| |
| if (fieldName == null) { |
| // Field type class must match injected service type |
| if (fieldType.equals(clazz)) { |
| injected = true; |
| func.injectField(field, target); |
| } else if (!strict && mayInjectToIterable(clazz, field, true)) { |
| injected = true; |
| func.injectIterableField(field, target); |
| } else if (!strict && mayInjectToMap(clazz, field, true)) { |
| injected = true; |
| func.injectMapField(field, target); |
| } |
| } else if (field.getName().equals(fieldName)) { |
| // Field type may be a superclass of the service type |
| if (fieldType.isAssignableFrom(clazz)) { |
| injected = true; |
| func.injectField(field, target); |
| } else if (!strict && mayInjectToIterable(clazz, field, false)) { |
| injected = true; |
| func.injectIterableField(field, target); |
| } else if (!strict && mayInjectToMap(clazz, field, false)) { |
| injected = true; |
| func.injectMapField(field, target); |
| } else { |
| logger.log( |
| Logger.LOG_ERROR, |
| "Could not set field " + field + " in class " + target.getClass().getName() |
| + ": the type of the field type should be either assignable from " |
| + clazz.getName() + " or Collection, or Map"); |
| } |
| } |
| } |
| targetClass = targetClass.getSuperclass(); |
| } |
| } |
| } |
| return injected; |
| } |
| |
| private static boolean mayInjectToIterable(Class<?> clazz, Field field, boolean strictClassEquality) { |
| Class<?> fieldType = field.getType(); |
| if (Iterable.class.isAssignableFrom(fieldType)) { |
| Type type = field.getGenericType(); |
| |
| // The field must be a parameterized map (generics). |
| if (! (type instanceof ParameterizedType)) { |
| return false; |
| } |
| ParameterizedType parameterType = (ParameterizedType) type; |
| Type[] types = parameterType.getActualTypeArguments(); |
| if (types == null || types.length != 1) { |
| return false; |
| } |
| if (types[0] instanceof Class<?>) { |
| Class<?> parameterizedTypeClass = (Class<?>) types[0]; |
| return strictClassEquality ? parameterizedTypeClass.equals(clazz) |
| : parameterizedTypeClass.isAssignableFrom(clazz); |
| } |
| } |
| return false; |
| } |
| |
| private static boolean mayInjectToMap(Class<?> clazz, Field field, boolean strictClassEquality) { |
| Class<?> fieldType = field.getType(); |
| if (Map.class.isAssignableFrom(fieldType)) { |
| Type type = field.getGenericType(); |
| |
| // The field must be a parameterized map (generics). |
| if (! (type instanceof ParameterizedType)) { |
| return false; |
| } |
| ParameterizedType parameterType = (ParameterizedType) type; |
| Type[] types = parameterType.getActualTypeArguments(); |
| if (types == null || types.length != 2) { |
| return false; |
| } |
| |
| // The map field generic key parameter must be "Class". |
| if (! (types[0] instanceof Class<?>)) { |
| return false; |
| } |
| |
| // The map generic value parameter must be Dictionary, or Dictionary<String, ...> |
| if (types[1] instanceof Class<?>) { |
| // The map field is in the form "Map m_field<Class, Dictionary>" |
| Class<?> mapValueGenericType = (Class<?>) types[1]; |
| if (! mapValueGenericType.equals(Dictionary.class)) { |
| return false; |
| } |
| } else if (types[1] instanceof ParameterizedType) { |
| // The map field is in the form "Map m_field<Class, Dictionary<String, ...>" |
| ParameterizedType mapValueGenericType = (ParameterizedType) types[1]; |
| if (! mapValueGenericType.getRawType().equals(Dictionary.class)) { |
| return false; |
| } |
| } |
| |
| Class<?> K = (Class<?>) types[0]; |
| return strictClassEquality ? K.equals(clazz) : K.isAssignableFrom(clazz); |
| } |
| return false; |
| } |
| } |