METAGEN-36

This commit is contained in:
Hardy Ferentschik 2010-10-04 13:45:28 +00:00 committed by Strong Liu
parent ba43b863f5
commit d01ae40533
7 changed files with 130 additions and 15 deletions

View File

@ -40,7 +40,7 @@ import org.hibernate.jpamodelgen.util.Constants;
* @author Hardy Ferentschik * @author Hardy Ferentschik
* @author Emmanuel Bernard * @author Emmanuel Bernard
*/ */
public class Context { public final class Context {
private static final String DEFAULT_PERSISTENCE_XML_LOCATION = "/META-INF/persistence.xml"; private static final String DEFAULT_PERSISTENCE_XML_LOCATION = "/META-INF/persistence.xml";
/** /**

View File

@ -19,6 +19,7 @@
package org.hibernate.jpamodelgen; package org.hibernate.jpamodelgen;
import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
import java.util.HashSet; import java.util.HashSet;
import java.util.List; import java.util.List;
@ -139,19 +140,28 @@ public class JPAMetaModelEntityProcessor extends AbstractProcessor {
} }
private void createMetaModelClasses() { private void createMetaModelClasses() {
// keep track of all classes for which model have been generated
Collection<String> generatedModelClasses = new ArrayList<String>();
for ( MetaEntity entity : context.getMetaEntities() ) { for ( MetaEntity entity : context.getMetaEntities() ) {
context.logMessage( Diagnostic.Kind.OTHER, "Writing meta model for entity " + entity ); context.logMessage( Diagnostic.Kind.OTHER, "Writing meta model for entity " + entity );
ClassWriter.writeFile( entity, context ); ClassWriter.writeFile( entity, context );
generatedModelClasses.add( entity.getQualifiedName() );
} }
// we cannot process the delayed entities in any order. There might be dependencies between them. // we cannot process the delayed entities in any order. There might be dependencies between them.
// we need to process the top level entities first // we need to process the top level entities first
// TODO make sure that we don't run into circular dependencies here
Collection<MetaEntity> toProcessEntities = context.getMetaEmbeddables(); Collection<MetaEntity> toProcessEntities = context.getMetaEmbeddables();
while ( !toProcessEntities.isEmpty() ) { while ( !toProcessEntities.isEmpty() ) {
Set<MetaEntity> processedEntities = new HashSet<MetaEntity>(); Set<MetaEntity> processedEntities = new HashSet<MetaEntity>();
int toProcessCountBeforeLoop = toProcessEntities.size();
for ( MetaEntity entity : toProcessEntities ) { for ( MetaEntity entity : toProcessEntities ) {
if ( containedInEntity( toProcessEntities, entity ) ) { // see METAGEN-36
if ( generatedModelClasses.contains( entity.getQualifiedName() ) ) {
toProcessEntities.remove( entity );
continue;
}
if ( modelGenerationNeedsToBeDeferred( toProcessEntities, entity ) ) {
continue; continue;
} }
context.logMessage( context.logMessage(
@ -161,10 +171,15 @@ public class JPAMetaModelEntityProcessor extends AbstractProcessor {
processedEntities.add( entity ); processedEntities.add( entity );
} }
toProcessEntities.removeAll( processedEntities ); toProcessEntities.removeAll( processedEntities );
if ( toProcessEntities.size() >= toProcessCountBeforeLoop ) {
context.logMessage(
Diagnostic.Kind.ERROR, "Potential endless loop in generation of entities."
);
}
} }
} }
private boolean containedInEntity(Collection<MetaEntity> entities, MetaEntity containedEntity) { private boolean modelGenerationNeedsToBeDeferred(Collection<MetaEntity> entities, MetaEntity containedEntity) {
ContainsAttributeTypeVisitor visitor = new ContainsAttributeTypeVisitor( ContainsAttributeTypeVisitor visitor = new ContainsAttributeTypeVisitor(
containedEntity.getTypeElement(), context containedEntity.getTypeElement(), context
); );
@ -207,7 +222,7 @@ public class JPAMetaModelEntityProcessor extends AbstractProcessor {
continue; continue;
} }
String fqn = ( ( TypeElement ) element ).getQualifiedName().toString(); String fqn = ( (TypeElement) element ).getQualifiedName().toString();
MetaEntity alreadyExistingMetaEntity = tryGettingExistingEntityFromContext( mirror, fqn ); MetaEntity alreadyExistingMetaEntity = tryGettingExistingEntityFromContext( mirror, fqn );
if ( alreadyExistingMetaEntity != null && alreadyExistingMetaEntity.isMetaComplete() ) { if ( alreadyExistingMetaEntity != null && alreadyExistingMetaEntity.isMetaComplete() ) {
String msg = "Skipping processing of annotations for " + fqn + " since xml configuration is metadata complete."; String msg = "Skipping processing of annotations for " + fqn + " since xml configuration is metadata complete.";
@ -217,10 +232,10 @@ public class JPAMetaModelEntityProcessor extends AbstractProcessor {
AnnotationMetaEntity metaEntity; AnnotationMetaEntity metaEntity;
if ( TypeUtils.containsAnnotation( element, Embeddable.class ) ) { if ( TypeUtils.containsAnnotation( element, Embeddable.class ) ) {
metaEntity = new AnnotationEmbeddable( ( TypeElement ) element, context ); metaEntity = new AnnotationEmbeddable( (TypeElement) element, context );
} }
else { else {
metaEntity = new AnnotationMetaEntity( ( TypeElement ) element, context ); metaEntity = new AnnotationMetaEntity( (TypeElement) element, context );
} }
if ( alreadyExistingMetaEntity != null ) { if ( alreadyExistingMetaEntity != null ) {
@ -267,7 +282,7 @@ public class JPAMetaModelEntityProcessor extends AbstractProcessor {
@Override @Override
public Boolean visitDeclared(DeclaredType declaredType, Element element) { public Boolean visitDeclared(DeclaredType declaredType, Element element) {
TypeElement returnedElement = ( TypeElement ) context.getTypeUtils().asElement( declaredType ); TypeElement returnedElement = (TypeElement) context.getTypeUtils().asElement( declaredType );
String fqNameOfReturnType = returnedElement.getQualifiedName().toString(); String fqNameOfReturnType = returnedElement.getQualifiedName().toString();
String collection = Constants.COLLECTIONS.get( fqNameOfReturnType ); String collection = Constants.COLLECTIONS.get( fqNameOfReturnType );
@ -275,7 +290,7 @@ public class JPAMetaModelEntityProcessor extends AbstractProcessor {
TypeMirror collectionElementType = TypeUtils.getCollectionElementType( TypeMirror collectionElementType = TypeUtils.getCollectionElementType(
declaredType, fqNameOfReturnType, null, context declaredType, fqNameOfReturnType, null, context
); );
returnedElement = ( TypeElement ) context.getTypeUtils().asElement( collectionElementType ); returnedElement = (TypeElement) context.getTypeUtils().asElement( collectionElementType );
} }
if ( type.getQualifiedName().toString().equals( returnedElement.getQualifiedName().toString() ) ) { if ( type.getQualifiedName().toString().equals( returnedElement.getQualifiedName().toString() ) ) {

View File

@ -0,0 +1,33 @@
/*
* JBoss, Home of Professional Open Source
* Copyright 2010, Red Hat Middleware LLC, and individual contributors
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* 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 implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
// $Id:$
package org.hibernate.jpamodelgen.test.embeddablemappedsuperclass;
import javax.persistence.Embeddable;
import javax.persistence.MappedSuperclass;
/**
* @author Hardy Ferentschik
*/
@Embeddable
@MappedSuperclass
public class EmbeddableAndMappedSuperClass {
}

View File

@ -0,0 +1,46 @@
/*
* JBoss, Home of Professional Open Source
* Copyright 2010, Red Hat Middleware LLC, and individual contributors
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* 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 implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
// $Id: BlobTest.java 20721 2010-09-27 12:40:10Z hardy.ferentschik $
package org.hibernate.jpamodelgen.test.embeddablemappedsuperclass;
import org.testng.annotations.Test;
import org.hibernate.jpamodelgen.test.util.CompilationTest;
import static org.hibernate.jpamodelgen.test.util.TestUtil.assertMetamodelClassGeneratedFor;
import static org.hibernate.jpamodelgen.test.util.TestUtil.assertNoCompilationError;
/**
* @author Hardy Ferentschik
*/
public class EmbeddableMappedSuperClassTest extends CompilationTest {
/**
* METAGEN-36
*/
@Test
public void testMetaModelsGenerated() {
assertMetamodelClassGeneratedFor( EmbeddableAndMappedSuperClass.class );
assertNoCompilationError(getCompilationDiagnostics());
}
@Override
protected String getPackageNameOfTestSources() {
return EmbeddableMappedSuperClassTest.class.getPackage().getName();
}
}

View File

@ -43,17 +43,22 @@ import org.hibernate.jpamodelgen.JPAMetaModelEntityProcessor;
import static org.testng.FileAssert.fail; import static org.testng.FileAssert.fail;
/** /**
* Base class for annotation processor tests.
*
* @author Hardy Ferentschik * @author Hardy Ferentschik
*/ */
public abstract class CompilationTest { public abstract class CompilationTest {
private static final Logger log = LoggerFactory.getLogger( CompilationTest.class ); private static final Logger log = LoggerFactory.getLogger( CompilationTest.class );
private static final String PATH_SEPARATOR = System.getProperty( "file.separator" ); private static final String PATH_SEPARATOR = System.getProperty( "file.separator" );
private static final String ANNOTATION_PROCESSOR_OPTION_PREFIX = "-A"; private static final String ANNOTATION_PROCESSOR_OPTION_PREFIX = "-A";
private static final String PROC_NONE = "-proc:none";
private static final String SOURCE_BASE_DIR_PROPERTY = "sourceBaseDir"; private static final String SOURCE_BASE_DIR_PROPERTY = "sourceBaseDir";
private static final String OUT_BASE_DIR_PROPERTY = "outBaseDir"; private static final String OUT_BASE_DIR_PROPERTY = "outBaseDir";
private static final String sourceBaseDir; private static final String sourceBaseDir;
private static final String outBaseDir; private static final String outBaseDir;
private List<Diagnostic> compilationDiagnostics;
static { static {
String tmp = System.getProperty( SOURCE_BASE_DIR_PROPERTY ); String tmp = System.getProperty( SOURCE_BASE_DIR_PROPERTY );
if ( tmp == null ) { if ( tmp == null ) {
@ -69,6 +74,11 @@ public abstract class CompilationTest {
} }
public CompilationTest() { public CompilationTest() {
compilationDiagnostics = new ArrayList<Diagnostic>();
}
public final List<Diagnostic> getCompilationDiagnostics() {
return compilationDiagnostics;
} }
@BeforeClass @BeforeClass
@ -91,8 +101,9 @@ public abstract class CompilationTest {
compilationUnits = fileManager.getJavaFileObjectsFromFiles( compilationUnits = fileManager.getJavaFileObjectsFromFiles(
getCompilationUnits( outBaseDir ) getCompilationUnits( outBaseDir )
); );
options.add( "-proc:none" ); // for the second compile skip the processor options.add( PROC_NONE ); // for the second compile skip the processor
compileSources( options, compiler, diagnostics, fileManager, compilationUnits ); compileSources( options, compiler, diagnostics, fileManager, compilationUnits );
compilationDiagnostics.addAll( diagnostics.getDiagnostics() );
fileManager.close(); fileManager.close();
} }

View File

@ -28,6 +28,8 @@ import java.lang.reflect.Field;
import java.lang.reflect.GenericArrayType; import java.lang.reflect.GenericArrayType;
import java.lang.reflect.ParameterizedType; import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type; import java.lang.reflect.Type;
import java.util.List;
import javax.tools.Diagnostic;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -209,10 +211,6 @@ public class TestUtil {
} }
} }
private static boolean hasFieldInMetamodelFor(Class<?> clazz, String fieldName) {
return getFieldFromMetamodelFor( clazz, fieldName ) != null;
}
public static Field getFieldFromMetamodelFor(Class<?> entityClass, String fieldName) { public static Field getFieldFromMetamodelFor(Class<?> entityClass, String fieldName) {
Class<?> metaModelClass = getMetamodelClassFor( entityClass ); Class<?> metaModelClass = getMetamodelClassFor( entityClass );
Field field; Field field;
@ -229,6 +227,18 @@ public class TestUtil {
return fcn.replace( PACKAGE_SEPARATOR, PATH_SEPARATOR ); return fcn.replace( PACKAGE_SEPARATOR, PATH_SEPARATOR );
} }
public static void assertNoCompilationError(List<Diagnostic> diagnostics) {
for ( Diagnostic diagnostic : diagnostics ) {
if ( diagnostic.getKind().equals( Diagnostic.Kind.ERROR ) ) {
fail( "There was a compilation error. " + diagnostic.getMessage( null ) );
}
}
}
private static boolean hasFieldInMetamodelFor(Class<?> clazz, String fieldName) {
return getFieldFromMetamodelFor( clazz, fieldName ) != null;
}
private static class MetaModelFilenameFilter implements FileFilter { private static class MetaModelFilenameFilter implements FileFilter {
@Override @Override
public boolean accept(File pathName) { public boolean accept(File pathName) {