/** * Copyright (C) 2009 Original Authors * * This file is part of Spring ME. * * Spring ME is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation; either version 2, or (at your option) any * later version. * * Spring ME 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. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with Spring ME; see the file COPYING. If not, write to the Free * Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * 02110-1301 USA. * * Linking this library statically or dynamically with other modules is * making a combined work based on this library. Thus, the terms and * conditions of the GNU General Public License cover the whole * combination. * * As a special exception, the copyright holders of this library give you * permission to link this library with independent modules to produce an * executable, regardless of the license terms of these independent * modules, and to copy and distribute the resulting executable under * terms of your choice, provided that you also meet, for each linked * independent module, the terms and conditions of the license of that * module. An independent module is a module which is not derived from or * based on this library. If you modify this library, you may extend this * exception to your version of the library, but you are not obligated to * do so. If you do not wish to do so, delete this exception statement * from your version. */ package me.springframework.di.spring; import java.util.Collection; import java.util.Map; import me.springframework.di.Sink; import me.springframework.di.Source; import me.springframework.di.base.AbstractSink; import me.springframework.di.base.MutableContext; import me.springframework.di.base.MutableInstance; import me.springframework.di.base.MutableInstanceReference; import org.springframework.beans.factory.FactoryBean; /** * Augments sinks with information about any type casts required at that sink. * <p> * This must be done after the {@link QDoxAugmentation} because it relies on knowing * whether a source refers to a {@link FactoryBean}. The QDoxAugmentation cannot * guarantee to process FactoryBean definitions before the bean definitions in which they * are used. The augmentation should also be applied after the {@link AutowiringAugmentation}, * because that augmentation can introduce new sinks. */ public class SinkAugmentation implements Augmentation { public void augment(MutableContext context) { Map<String, MutableInstance> instances = context.getInstances(); for (MutableInstance instance : instances.values()) { attribute(instance, context); } } private void attribute(MutableInstance instance, MutableContext context) { attribute(instance.getConstructorArguments(), context); attribute(instance.getSetters(), context); } /** * Sets the {@link AbstractSink#setCastRequired(boolean)} flag on the given * {@link Sink} instances. */ private void attribute(Collection<? extends AbstractSink> sinks, MutableContext context) { if (sinks == null || sinks.size() == 0) { return; } for (AbstractSink sink : sinks) { attribute(context, sink); Source source = sink.getSource(); if (source instanceof MutableInstance) { attribute((MutableInstance) source, context); } } } protected void attribute(MutableContext context, AbstractSink sink) { Source source = sink.getSource(); if (isFactoryBean(source, context)) { sink.setCastTo(sink.getType().replace('$', '.')); } } /** * Returns true if a given source refers to a {@link FactoryBean}. * * @param source Any type of Source instance. * @param instances All known instances. * @return True if the source refers to a FactoryBean, otherwise false. */ private boolean isFactoryBean(Source source, MutableContext context) { boolean factoryBean = false; if (source instanceof MutableInstanceReference) { MutableInstanceReference r = (MutableInstanceReference) source; factoryBean = context.getByName(r.getName()).isFactoryBean(); } else if (source instanceof MutableInstance) { MutableInstance r = (MutableInstance) source; factoryBean = r.isFactoryBean(); } return factoryBean; } }