All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.persistence.internal.jpa.jpql.EclipseLinkSemanticValidatorHelper Maven / Gradle / Ivy

There is a newer version: 4.0.3
Show newest version
/*******************************************************************************
 * Copyright (c) 2012, 2013, 2013 Oracle and/or its affiliates. All rights reserved.
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v1.0 and Eclipse Distribution License v. 1.0
 * which accompanies this distribution.
 * The Eclipse Public License is available at http://www.eclipse.org/legal/epl-v10.html
 * and the Eclipse Distribution License is available at
 * http://www.eclipse.org/org/documents/edl-v10.php.
 *
 * Contributors:
 *     Oracle - initial API and implementation
 *
 ******************************************************************************/
package org.eclipse.persistence.internal.jpa.jpql;

import java.lang.reflect.Constructor;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import org.eclipse.persistence.descriptors.ClassDescriptor;
import org.eclipse.persistence.jpa.jpql.ExpressionTools;
import org.eclipse.persistence.jpa.jpql.ITypeHelper;
import org.eclipse.persistence.jpa.jpql.JPQLQueryDeclaration;
import org.eclipse.persistence.jpa.jpql.SemanticValidatorHelper;
import org.eclipse.persistence.jpa.jpql.parser.Expression;
import org.eclipse.persistence.jpa.jpql.parser.IdentificationVariable;
import org.eclipse.persistence.jpa.jpql.parser.JPQLGrammar;
import org.eclipse.persistence.jpa.jpql.parser.SimpleSelectStatement;
import org.eclipse.persistence.mappings.DatabaseMapping;
import org.eclipse.persistence.mappings.querykeys.QueryKey;

/**
 * The EclipseLink implementation of {@link SemanticValidatorHelper}, which directly accesses
 * EclipseLink objects without using Hermes SPI.
 *
 * @version 2.5
 * @since 2.4
 * @author Pascal Filion
 */
final class EclipseLinkSemanticValidatorHelper implements SemanticValidatorHelper {

	/**
	 * The context used to query information about the JPQL query.
	 */
	private final JPQLQueryContext queryContext;

	/**
	 * Creates a new EclipseLinkSemanticValidatorHelper.
	 *
	 * @param queryContext The context used to query information about the JPQL query
	 */
	EclipseLinkSemanticValidatorHelper(JPQLQueryContext queryContext) {
		super();
		this.queryContext = queryContext;
	}

	private void addIdentificationVariable(IdentificationVariable identificationVariable,
	                                       Map> identificationVariables) {

		String variableName = (identificationVariable != null) ? identificationVariable.getVariableName() : null;

		if (ExpressionTools.stringIsNotEmpty(variableName)) {

			// Add the IdentificationVariable to the list
			List variables = identificationVariables.get(variableName);

			if (variables == null) {
				variables = new ArrayList();
				identificationVariables.put(variableName, variables);
			}

			variables.add(identificationVariable);
		}
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void collectAllDeclarationIdentificationVariables(Map> identificationVariables) {

		JPQLQueryContext currentContext = queryContext.getCurrentContext();

		while (currentContext != null) {
			collectLocalDeclarationIdentificationVariables(currentContext, identificationVariables);
			currentContext = currentContext.getActualParent();
		}
	}

	private void collectLocalDeclarationIdentificationVariables(JPQLQueryContext queryContext,
	                                                            Map> identificationVariables) {

		DeclarationResolver declarationResolver = queryContext.getDeclarationResolverImp();

		// Collect the identification variables from the declarations
		for (Declaration declaration : declarationResolver.getDeclarations()) {
			IdentificationVariable identificationVariable = declaration.identificationVariable;
			addIdentificationVariable(identificationVariable, identificationVariables);
		}

		// Collect the result variables
		for (IdentificationVariable identificationVariable : declarationResolver.getResultVariables()) {
			addIdentificationVariable(identificationVariable, identificationVariables);
		}
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void collectLocalDeclarationIdentificationVariables(Map> identificationVariables) {
		collectLocalDeclarationIdentificationVariables(queryContext, identificationVariables);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void disposeSubqueryContext() {
		queryContext.disposeSubqueryContext();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public String[] entityNames() {

		List names = new ArrayList();

		for (ClassDescriptor descriptor : queryContext.getSession().getDescriptors().values()) {
			if (!descriptor.isAggregateDescriptor()) {
				String name = descriptor.getAlias();
				if (ExpressionTools.stringIsEmpty(name)) {
					name = descriptor.getJavaClass().getSimpleName();
				}
				names.add(name);
			}
		}

		return names.toArray(new String[names.size()]);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public List getAllDeclarations() {

		List declarations = new ArrayList();
		JPQLQueryContext context = queryContext.getCurrentContext();

		while (context != null) {
			declarations.addAll(context.getDeclarationResolverImp().getDeclarations());
			context = context.getActualParent();
		}

		return declarations;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Object[] getConstructors(Object type) {
		return (type != null) ? ((Class) type).getDeclaredConstructors() : ExpressionTools.EMPTY_ARRAY;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	@SuppressWarnings({ "rawtypes", "unchecked" })
	public List getDeclarations() {
		return queryContext.getDeclarations();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public ClassDescriptor getEmbeddable(Object type) {

		ClassDescriptor descriptor = queryContext.getDescriptor((Class) type);

		if ((descriptor != null) && descriptor.isAggregateDescriptor()) {
			return descriptor;
		}

		return null;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public ClassDescriptor getEntityNamed(String entityName) {
		return queryContext.getDescriptor(entityName);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	@SuppressWarnings("unchecked")
	public String[] getEnumConstants(Object type) {

		Enum[] constants = ((Class>) type).getEnumConstants();
		String[] names = new String[constants.length];

		for (int index = constants.length; --index >= 0; ) {
			names[index] = constants[index].name();
		}

		return names;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public JPQLGrammar getGrammar() {
		return queryContext.getGrammar();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Object getManagedType(Expression expression) {
		return queryContext.resolveDescriptor(expression);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Object getReferenceManagedType(Object mapping) {

		if (mapping == null) {
			return null;
		}

		return ((DatabaseMapping) mapping).getReferenceDescriptor();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Object getMappingNamed(Object entity, String name) {

		ClassDescriptor descriptor = (ClassDescriptor) entity;
		Object mapping = descriptor.getObjectBuilder().getMappingForAttributeName(name);

		if (mapping == null) {
			mapping = descriptor.getQueryKeyNamed(name);
		}

		return mapping;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Class getMappingType(Object mapping) {

		if (mapping == null) {
			return null;
		}

		try {
			// We do a try/catch, it should be faster than doing an instance of
			// since a QueryKey is used a lot less than a DatabaseMapping
			return queryContext.calculateMappingType((DatabaseMapping) mapping);
		}
		catch (ClassCastException e) {
			return queryContext.calculateQueryKeyType((QueryKey) mapping);
		}
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Type[] getMethodParameterTypeDeclarations(Object constructor) {
		return ((Constructor) constructor).getGenericParameterTypes();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Class getType(Expression expression) {
		return queryContext.getType(expression);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Object getType(Object typeDeclaration) {
		// Not used
		return null;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Class getType(String className) {
		return queryContext.getType(className);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Object getTypeDeclaration(Expression expression) {
		// Not used
		return null;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public ITypeHelper getTypeHelper() {
		// Not used
		return null;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public String getTypeName(Object type) {
		return ((Class) type).getName();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isAssignableTo(Object type1, Object type2) {
		return ((Class) type2).isAssignableFrom((Class) type1) ;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isCollectionIdentificationVariable(String variableName) {
		return queryContext.isCollectionIdentificationVariable(variableName);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isCollectionMapping(Object mapping) {

		if (mapping == null) {
			return false;
		}

		try {
			// We do a try/catch, it should be faster than doing an instance of
			// since a QueryKey is used a lot less than a DatabaseMapping
			return ((DatabaseMapping) mapping).isCollectionMapping();
		}
		catch (ClassCastException e) {
			return ((QueryKey) mapping).isCollectionQueryKey();
		}
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isEnumType(Object type) {
		return (type != null) && ((Class) type).isEnum();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isIdentificationVariableValidInComparison(IdentificationVariable expression) {

		Declaration declaration = queryContext.findDeclaration(expression.getVariableName());

		if (declaration == null) {
			return false;
		}

		DatabaseMapping mapping = declaration.getMapping();

		if (mapping == null) {
			return false;
		}

		// Direct collection is not an object so it's valid
		return mapping.isDirectCollectionMapping() ||
		       mapping.isAbstractCompositeDirectCollectionMapping();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isManagedTypeResolvable(Object managedType) {
		return managedType != null;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isPropertyMapping(Object mapping) {

		if (mapping == null) {
			return false;
		}

		try {
			// We do a try/catch, it should be faster than doing an instance of
			// since a QueryKey is used a lot less than a DatabaseMapping
			return ((DatabaseMapping) mapping).isDirectToFieldMapping();
		}
		catch (ClassCastException e) {
			return ((QueryKey) mapping).isDirectQueryKey();
		}
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isRelationshipMapping(Object mapping) {

		if (mapping == null) {
			return false;
		}

		if (mapping instanceof DatabaseMapping) {

			DatabaseMapping databaseMapping = (DatabaseMapping) mapping;

			return databaseMapping.isForeignReferenceMapping()            ||
			       databaseMapping.isAbstractCompositeCollectionMapping() ||
			       databaseMapping.isAbstractCompositeDirectCollectionMapping();
		}

		return ((QueryKey) mapping).isForeignReferenceQueryKey();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isResultVariable(String variableName) {
		return queryContext.isResultVariable(variableName);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isTransient(Object mapping) {
		return mapping == null;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isTypeDeclarationAssignableTo(Object typeDeclaration1, Object typeDeclaration2) {
		// Not used
		return false;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isTypeResolvable(Object type) {
		return type != null;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void newSubqueryContext(SimpleSelectStatement expression) {
		queryContext.newSubQueryContext(expression, null);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Object resolveMapping(Expression expression) {
		return queryContext.resolveMappingObject(expression);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Object resolveMapping(String variableName, String name) {

		// Find the declaration associated with the identification variable
		Declaration declaration = queryContext.findDeclaration(variableName);

		if (declaration == null) {
			return null;
		}

		// Retrieve the associated descriptor
		ClassDescriptor descriptor = declaration.getDescriptor();

		if (descriptor == null) {
			return null;
		}

		// Now, retrieve the mapping
		Object mapping = descriptor.getObjectBuilder().getMappingForAttributeName(name);

		// No mapping was found, look for a query key
		if (mapping == null) {
			mapping = descriptor.getQueryKeyNamed(name);
		}

		return mapping;
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy