| /* |
| * Copyright 2007-2011 Rickard Öberg. |
| * Copyright 2007-2010 Niclas Hedhman. |
| * |
| * 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 |
| * ied. |
| * |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| package org.qi4j.api.query.grammar; |
| |
| import java.lang.reflect.AccessibleObject; |
| import java.lang.reflect.Member; |
| import java.lang.reflect.ParameterizedType; |
| import java.lang.reflect.Proxy; |
| import java.lang.reflect.Type; |
| import org.qi4j.api.association.Association; |
| import org.qi4j.api.composite.Composite; |
| import org.qi4j.api.composite.CompositeInstance; |
| import org.qi4j.api.property.GenericPropertyInfo; |
| import org.qi4j.api.property.Property; |
| import org.qi4j.api.query.NotQueryableException; |
| import org.qi4j.api.query.QueryExpressionException; |
| import org.qi4j.api.util.Classes; |
| import org.qi4j.functional.Function; |
| |
| import static org.qi4j.api.util.Classes.typeOf; |
| |
| /** |
| * Function to get Entity Properties. |
| */ |
| public class PropertyFunction<T> |
| implements Function<Composite, Property<T>> |
| { |
| private final PropertyFunction<?> traversedProperty; |
| private final AssociationFunction<?> traversedAssociation; |
| private final ManyAssociationFunction<?> traversedManyAssociation; |
| private final NamedAssociationFunction<?> traversedNamedAssociation; |
| private final AccessibleObject accessor; |
| |
| public PropertyFunction( PropertyFunction<?> traversedProperty, |
| AssociationFunction<?> traversedAssociation, |
| ManyAssociationFunction<?> traversedManyAssociation, |
| NamedAssociationFunction<?> traversedNamedAssociation, |
| AccessibleObject accessor |
| ) |
| { |
| this.traversedProperty = traversedProperty; |
| this.traversedAssociation = traversedAssociation; |
| this.traversedManyAssociation = traversedManyAssociation; |
| this.traversedNamedAssociation = traversedNamedAssociation; |
| this.accessor = accessor; |
| |
| // Verify that the property accessor is not marked as non queryable |
| NotQueryableException.throwIfNotQueryable( accessor ); |
| // Verify that the property type itself (value composites) is not marked as non queryable |
| |
| Type returnType = typeOf( accessor ); |
| if( !Property.class.isAssignableFrom( Classes.RAW_CLASS.map( returnType ) ) ) |
| { |
| throw new QueryExpressionException( "Not a property type:" + returnType ); |
| } |
| Type propertyTypeAsType = GenericPropertyInfo.toPropertyType( returnType ); |
| if( propertyTypeAsType instanceof ParameterizedType ) |
| { |
| propertyTypeAsType = ( (ParameterizedType) propertyTypeAsType ).getRawType(); |
| } |
| |
| if( !( propertyTypeAsType instanceof Class ) ) |
| { |
| throw new QueryExpressionException( "Unsupported property type:" + propertyTypeAsType ); |
| } |
| @SuppressWarnings( "unchecked" ) |
| Class<T> type = (Class<T>) propertyTypeAsType; |
| NotQueryableException.throwIfNotQueryable( type ); |
| } |
| |
| public PropertyFunction<?> traversedProperty() |
| { |
| return traversedProperty; |
| } |
| |
| public AssociationFunction<?> traversedAssociation() |
| { |
| return traversedAssociation; |
| } |
| |
| public ManyAssociationFunction<?> traversedManyAssociation() |
| { |
| return traversedManyAssociation; |
| } |
| |
| public NamedAssociationFunction<?> traversedNamedAssociation() |
| { |
| return traversedNamedAssociation; |
| } |
| |
| public AccessibleObject accessor() |
| { |
| return accessor; |
| } |
| |
| @Override |
| public Property<T> map( Composite entity ) |
| { |
| try |
| { |
| Object target = entity; |
| if( traversedProperty != null ) |
| { |
| Property<?> property = traversedProperty.map( entity ); |
| if( property == null ) |
| { |
| return null; |
| } |
| target = property.get(); |
| } |
| else if( traversedAssociation != null ) |
| { |
| Association<?> association = traversedAssociation.map( entity ); |
| if( association == null ) |
| { |
| return null; |
| } |
| target = association.get(); |
| } |
| else if( traversedManyAssociation != null ) |
| { |
| throw new IllegalArgumentException( "Cannot evaluate a ManyAssociation" ); |
| } |
| else if( traversedNamedAssociation != null ) |
| { |
| throw new IllegalArgumentException( "Cannot evaluate a NamedAssociation" ); |
| } |
| |
| if( target == null ) |
| { |
| return null; |
| } |
| |
| CompositeInstance handler = (CompositeInstance) Proxy.getInvocationHandler( target ); |
| return handler.state().propertyFor( accessor ); |
| } |
| catch( IllegalArgumentException e ) |
| { |
| throw e; |
| } |
| catch( Throwable e ) |
| { |
| throw new IllegalArgumentException( e ); |
| } |
| } |
| |
| @Override |
| public String toString() |
| { |
| if( traversedProperty != null ) |
| { |
| return traversedProperty.toString() + "." + ( (Member) accessor ).getName(); |
| } |
| else if( traversedAssociation != null ) |
| { |
| return traversedAssociation.toString() + "." + ( (Member) accessor ).getName(); |
| } |
| else |
| { |
| return ( (Member) accessor ).getName(); |
| } |
| } |
| } |