/** * * Copyright * 2009-2015 Jayway Products AB * 2016-2017 Föreningen Sambruk * * Licensed under AGPL, Version 3.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.gnu.org/licenses/agpl.txt * * 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 se.streamsource.streamflow.web.domain.generic; import java.beans.Introspector; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import org.qi4j.api.common.AppliesTo; import org.qi4j.api.common.AppliesToFilter; import org.qi4j.api.entity.EntityComposite; import org.qi4j.api.entity.association.EntityStateHolder; import org.qi4j.api.entity.association.ManyAssociation; import org.qi4j.api.injection.scope.State; import org.qi4j.api.injection.scope.This; import se.streamsource.streamflow.infrastructure.event.domain.DomainEvent; /** * Generic mixin for simple event methods that remove an entity from a collection. They have to follow this pattern: * void removedFoo(DomainEvent event, SomeType entity) * This will remove the entity from the ManyAssociation called "foos" */ @AppliesTo(EventEntityRemovedMixin.EventEntityRemovedAppliesTo.class) public class EventEntityRemovedMixin implements InvocationHandler { private static Map<Method, Method> methodMappings = new ConcurrentHashMap(); @State EntityStateHolder state; @This EntityComposite composite; public Object invoke( Object proxy, Method method, Object[] args ) throws Throwable { Method manyAssociationMethod = methodMappings.get( method ); if (manyAssociationMethod == null) { // Find ManyAssociation method String removedName = Introspector.decapitalize( method.getName().substring( "removed".length() ) ) + "s"; manyAssociationMethod = composite.getClass().getInterfaces()[0].getMethod( removedName ); methodMappings.put( method, manyAssociationMethod ); } // Lookup the ManyAssociation ManyAssociation<Object> manyAssociation = state.getManyAssociation( manyAssociationMethod ); // Remove entity from ManyAssociation manyAssociation.remove( args[1] ); return null; } public static class EventEntityRemovedAppliesTo implements AppliesToFilter { public boolean appliesTo( Method method, Class<?> mixin, Class<?> compositeType, Class<?> fragmentClass ) { return method.getParameterTypes().length == 2 && method.getParameterTypes()[0].equals( DomainEvent.class ) && method.getName().startsWith( "removed" ) && method.getReturnType().equals( Void.TYPE ); } } }