com.thoughtworks.qdox.builder.impl.ModelBuilder Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of qdox Show documentation
Show all versions of qdox Show documentation
QDox is a high speed, small footprint parser for extracting class/interface/method definitions from source files
complete with JavaDoc @tags. It is designed to be used by active code generators or documentation tools.
The newest version!
package com.thoughtworks.qdox.builder.impl;
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.
*/
import java.net.URL;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;
import com.thoughtworks.qdox.builder.Builder;
import com.thoughtworks.qdox.builder.TypeAssembler;
import com.thoughtworks.qdox.library.ClassLibrary;
import com.thoughtworks.qdox.model.DocletTag;
import com.thoughtworks.qdox.model.DocletTagFactory;
import com.thoughtworks.qdox.model.JavaAnnotatedElement;
import com.thoughtworks.qdox.model.JavaAnnotation;
import com.thoughtworks.qdox.model.JavaClass;
import com.thoughtworks.qdox.model.JavaConstructor;
import com.thoughtworks.qdox.model.JavaExecutable;
import com.thoughtworks.qdox.model.JavaGenericDeclaration;
import com.thoughtworks.qdox.model.JavaMethod;
import com.thoughtworks.qdox.model.JavaModule;
import com.thoughtworks.qdox.model.JavaParameter;
import com.thoughtworks.qdox.model.JavaSource;
import com.thoughtworks.qdox.model.JavaType;
import com.thoughtworks.qdox.model.JavaTypeVariable;
import com.thoughtworks.qdox.model.expression.Expression;
import com.thoughtworks.qdox.model.impl.AbstractBaseJavaEntity;
import com.thoughtworks.qdox.model.impl.DefaultJavaClass;
import com.thoughtworks.qdox.model.impl.DefaultJavaConstructor;
import com.thoughtworks.qdox.model.impl.DefaultJavaField;
import com.thoughtworks.qdox.model.impl.DefaultJavaInitializer;
import com.thoughtworks.qdox.model.impl.DefaultJavaMethod;
import com.thoughtworks.qdox.model.impl.DefaultJavaModule;
import com.thoughtworks.qdox.model.impl.DefaultJavaModuleDescriptor;
import com.thoughtworks.qdox.model.impl.DefaultJavaModuleDescriptor.DefaultJavaExports;
import com.thoughtworks.qdox.model.impl.DefaultJavaModuleDescriptor.DefaultJavaOpens;
import com.thoughtworks.qdox.model.impl.DefaultJavaModuleDescriptor.DefaultJavaProvides;
import com.thoughtworks.qdox.model.impl.DefaultJavaModuleDescriptor.DefaultJavaRequires;
import com.thoughtworks.qdox.model.impl.DefaultJavaModuleDescriptor.DefaultJavaUses;
import com.thoughtworks.qdox.model.impl.DefaultJavaPackage;
import com.thoughtworks.qdox.model.impl.DefaultJavaParameter;
import com.thoughtworks.qdox.model.impl.DefaultJavaSource;
import com.thoughtworks.qdox.model.impl.DefaultJavaType;
import com.thoughtworks.qdox.model.impl.DefaultJavaTypeVariable;
import com.thoughtworks.qdox.parser.expression.ExpressionDef;
import com.thoughtworks.qdox.parser.structs.AnnoDef;
import com.thoughtworks.qdox.parser.structs.ClassDef;
import com.thoughtworks.qdox.parser.structs.CompactConstructorDef;
import com.thoughtworks.qdox.parser.structs.FieldDef;
import com.thoughtworks.qdox.parser.structs.InitDef;
import com.thoughtworks.qdox.parser.structs.MethodDef;
import com.thoughtworks.qdox.parser.structs.ModuleDef;
import com.thoughtworks.qdox.parser.structs.ModuleDef.ExportsDef;
import com.thoughtworks.qdox.parser.structs.ModuleDef.OpensDef;
import com.thoughtworks.qdox.parser.structs.ModuleDef.ProvidesDef;
import com.thoughtworks.qdox.parser.structs.ModuleDef.RequiresDef;
import com.thoughtworks.qdox.parser.structs.ModuleDef.UsesDef;
import com.thoughtworks.qdox.parser.structs.PackageDef;
import com.thoughtworks.qdox.parser.structs.RecordFieldsDef;
import com.thoughtworks.qdox.parser.structs.TagDef;
import com.thoughtworks.qdox.parser.structs.TypeDef;
import com.thoughtworks.qdox.parser.structs.TypeVariableDef;
import com.thoughtworks.qdox.type.TypeResolver;
import com.thoughtworks.qdox.writer.ModelWriterFactory;
/**
* @author Joe Walnes
* @author Robert Scholte
*/
public class ModelBuilder implements Builder {
private final DefaultJavaSource source;
private DefaultJavaModule module;
private DefaultJavaModuleDescriptor moduleDescriptor;
private LinkedList classStack = new LinkedList();
private LinkedList recordHeaderStack = new LinkedList();
private List parameterList = new LinkedList();
private DefaultJavaConstructor currentConstructor;
private DefaultJavaMethod currentMethod;
private DefaultJavaField currentField;
private List currentAnnoDefs;
private List currentArguments;
private String lastComment;
private List lastTagSet = new LinkedList();
private ClassLibrary classLibrary;
private DocletTagFactory docletTagFactory;
private ModelWriterFactory modelWriterFactory;
public ModelBuilder( ClassLibrary classLibrary, DocletTagFactory docletTagFactory )
{
this.classLibrary = classLibrary;
this.docletTagFactory = docletTagFactory;
this.source = new DefaultJavaSource( classLibrary );
this.currentAnnoDefs = new LinkedList();
this.currentArguments = new LinkedList();
}
/** {@inheritDoc} */
public void setModelWriterFactory( ModelWriterFactory modelWriterFactory )
{
this.modelWriterFactory = modelWriterFactory;
source.setModelWriterFactory( modelWriterFactory );
}
/** {@inheritDoc} */
public void setModule( final ModuleDef moduleDef )
{
this.moduleDescriptor = new DefaultJavaModuleDescriptor(moduleDef.getName());
this.module = new DefaultJavaModule(moduleDef.getName(), moduleDescriptor);
}
/** {@inheritDoc} */
public void addExports( ExportsDef exportsDef )
{
// for now use anonymous modules
List targets = new ArrayList(exportsDef.getTargets().size());
for ( String moduleName : exportsDef.getTargets() )
{
targets.add( new DefaultJavaModule( moduleName, null ) );
}
DefaultJavaExports exports =
new DefaultJavaExports( new DefaultJavaPackage(exportsDef.getSource()), targets );
exports.setLineNumber( exportsDef.getLineNumber() );
exports.setModelWriterFactory( modelWriterFactory );
moduleDescriptor.addExports( exports );
}
/** {@inheritDoc} */
public void addOpens( OpensDef opensDef )
{
// for now use anonymous modules
List targets = new ArrayList(opensDef.getTargets().size());
for ( String moduleName : opensDef.getTargets() )
{
targets.add( new DefaultJavaModule( moduleName, null ) );
}
DefaultJavaOpens exports =
new DefaultJavaOpens( new DefaultJavaPackage(opensDef.getSource()), targets );
exports.setLineNumber( opensDef.getLineNumber() );
exports.setModelWriterFactory( modelWriterFactory );
moduleDescriptor.addOpens( exports );
}
/** {@inheritDoc} */
public void addProvides( ProvidesDef providesDef )
{
JavaClass service = createType( providesDef.getService(), 0 );
List implementations = new LinkedList();
for ( TypeDef implementType : providesDef.getImplementations() )
{
implementations.add( createType( implementType, 0 ) );
}
DefaultJavaProvides provides = new DefaultJavaProvides( service, implementations );
provides.setLineNumber( providesDef.getLineNumber() );
provides.setModelWriterFactory( modelWriterFactory );
moduleDescriptor.addProvides( provides );
}
/** {@inheritDoc} */
public void addRequires( RequiresDef requiresDef )
{
JavaModule module = new DefaultJavaModule( requiresDef.getName(), null );
DefaultJavaRequires requires = new DefaultJavaRequires( module, requiresDef.getModifiers() );
requires.setLineNumber( requiresDef.getLineNumber() );
requires.setModelWriterFactory( modelWriterFactory );
moduleDescriptor.addRequires( requires );
}
/** {@inheritDoc} */
public void addUses( UsesDef usesDef )
{
DefaultJavaUses uses = new DefaultJavaUses( createType( usesDef.getService(), 0 ) );
uses.setLineNumber( usesDef.getLineNumber() );
uses.setModelWriterFactory( modelWriterFactory );
moduleDescriptor.addUses( uses );
}
/** {@inheritDoc} */
public void addPackage( PackageDef packageDef )
{
DefaultJavaPackage jPackage = new DefaultJavaPackage( packageDef.getName() );
jPackage.setClassLibrary( classLibrary );
jPackage.setLineNumber( packageDef.getLineNumber() );
jPackage.setModelWriterFactory( modelWriterFactory );
addJavaDoc( jPackage );
setAnnotations( jPackage );
source.setPackage( jPackage );
}
/** {@inheritDoc} */
public void addImport( String importName )
{
source.addImport( importName );
}
public void addImplements( Set implementSet )
{
List implementz = new LinkedList();
for ( TypeDef implementType : implementSet )
{
implementz.add( createType( implementType, 0 ) );
}
classStack.getFirst().setImplementz( implementz );
}
/** {@inheritDoc} */
public void addJavaDoc( String text )
{
lastComment = text;
}
/** {@inheritDoc} */
public void addJavaDocTag( TagDef tagDef )
{
lastTagSet.add( tagDef );
}
/** {@inheritDoc} */
public void beginClass(ClassDef def)
{
DefaultJavaClass newClass = new DefaultJavaClass( source );
newClass.setLineNumber( def.getLineNumber() );
newClass.setModelWriterFactory( modelWriterFactory );
// basic details
newClass.setName( def.getName() );
newClass.setInterface( ClassDef.INTERFACE.equals( def.getType() ) );
newClass.setEnum( ClassDef.ENUM.equals( def.getType() ) );
newClass.setRecord( ClassDef.RECORD.equals( def.getType() ) );
newClass.setAnnotation( ClassDef.ANNOTATION_TYPE.equals( def.getType() ) );
// superclass
if ( newClass.isInterface() )
{
newClass.setSuperClass( null );
}
else if ( !newClass.isEnum() )
{
newClass.setSuperClass( def.getExtends().size() > 0 ? createType( def.getExtends().iterator().next(), 0 )
: null );
}
// implements
Set implementSet = newClass.isInterface() ? def.getExtends() : def.getImplements();
List implementz = new LinkedList();
for ( TypeDef implementType : implementSet )
{
implementz.add( createType( implementType, 0 ) );
}
newClass.setImplementz( implementz );
// modifiers
newClass.setModifiers( new LinkedList( def.getModifiers() ) );
// typeParameters
if ( def.getTypeParameters() != null )
{
List> typeParams = new LinkedList>();
for ( TypeVariableDef typeVariableDef : def.getTypeParameters() )
{
typeParams.add( createTypeVariable( typeVariableDef, (JavaClass) newClass ) );
}
newClass.setTypeParameters( typeParams );
}
// javadoc
addJavaDoc( newClass );
// // ignore annotation types (for now)
// if (ClassDef.ANNOTATION_TYPE.equals(def.type)) {
// System.out.println( currentClass.getFullyQualifiedName() );
// return;
// }
// annotations
setAnnotations( newClass );
classStack.addFirst( bindClass( newClass ) );
}
protected DefaultJavaClass bindClass( DefaultJavaClass newClass )
{
if ( currentField != null )
{
classStack.getFirst().addClass( newClass );
currentField.setEnumConstantClass( newClass );
}
else if ( !classStack.isEmpty() )
{
classStack.getFirst().addClass( newClass );
newClass.setDeclaringClass( classStack.getFirst() );
}
else
{
source.addClass( newClass );
}
return newClass;
}
/** {@inheritDoc} */
public void endClass()
{
classStack.removeFirst();
}
/** {@inheritDoc} */
public void endRecord( RecordFieldsDef def )
{
DefaultJavaClass cls = classStack.getFirst();
for ( FieldDef param : def.getFields() )
{
int dimensions =
param.isVarArgs()
? param.getDimensions() + 1
: param.getDimensions();
FieldDef field = new FieldDef();
field.setName(param.getName());
field.setType(param.getType());
field.setDimensions(dimensions);
field.setEnumConstant(false);
field.getModifiers().addAll(param.getModifiers());
field.getModifiers().add("private");
field.getModifiers().add("final");
field.setLineNumber(param.getLineNumber());
beginField(field);
endField();
if( cls.getMethod( param.getName(), new LinkedList(), false ) == null )
{
MethodDef mth = new MethodDef();
mth.setName(param.getName());
mth.setLineNumber(param.getLineNumber());
mth.setReturnType(param.getType());
mth.getModifiers().add("public");
mth.setDimensions(dimensions);
mth.setTypeParams(new LinkedList());
mth.setLineNumber(param.getLineNumber());
beginMethod();
endMethod(mth);
}
}
recordHeaderStack.removeFirst();
classStack.removeFirst();
}
/**
* this one is specific for those cases where dimensions can be part of both the type and identifier
* i.e. private String[] matrix[]; //field
* public abstract String[] getMatrix[](); //method
*
* @param typeDef
* @param dimensions
* @return the Type
*/
private DefaultJavaType createType( TypeDef typeDef, int dimensions )
{
if ( typeDef == null )
{
return null;
}
TypeResolver typeResolver;
if(classStack.isEmpty())
{
typeResolver = TypeResolver.byPackageName( source.getPackageName(), classLibrary, source.getImports() );
}
else
{
typeResolver = TypeResolver.byClassName( classStack.getFirst().getBinaryName(), classLibrary, source.getImports() );
}
return TypeAssembler.createUnresolved( typeDef, dimensions, typeResolver );
}
private void addJavaDoc( AbstractBaseJavaEntity entity )
{
entity.setComment( lastComment );
List tagList = new LinkedList();
for ( TagDef tagDef : lastTagSet )
{
tagList.add( docletTagFactory.createDocletTag( tagDef.getName(), tagDef.getText(),
(JavaAnnotatedElement) entity, tagDef.getLineNumber() ) );
}
entity.setTags( tagList );
lastTagSet.clear();
lastComment = null;
}
/** {@inheritDoc} */
public void addInitializer( InitDef def )
{
DefaultJavaInitializer initializer = new DefaultJavaInitializer();
initializer.setLineNumber( def.getLineNumber() );
initializer.setBlock( def.getBlockContent() );
initializer.setStatic( def.isStatic() );
classStack.getFirst().addInitializer( initializer );
}
/** {@inheritDoc} */
public void beginConstructor()
{
currentConstructor = new DefaultJavaConstructor();
currentConstructor.setDeclaringClass( classStack.getFirst() );
currentConstructor.setModelWriterFactory( modelWriterFactory );
addJavaDoc( currentConstructor );
setAnnotations( currentConstructor );
DefaultJavaClass cls = classStack.getFirst();
if( cls.isRecord() && cls.getConstructors().isEmpty() )
{
recordHeaderStack.addFirst( currentConstructor );
}
cls.addConstructor( currentConstructor );
}
/** {@inheritDoc} */
public void endConstructor( MethodDef def )
{
currentConstructor.setLineNumber( def.getLineNumber() );
// basic details
currentConstructor.setName( def.getName() );
// typeParameters
if ( def.getTypeParams() != null )
{
List> typeParams =
new LinkedList>();
for ( TypeVariableDef typeVariableDef : def.getTypeParams() )
{
typeParams.add( createTypeVariable( typeVariableDef, (JavaConstructor) currentConstructor ) );
}
currentConstructor.setTypeParameters( typeParams );
}
// exceptions
List exceptions = new LinkedList();
for ( TypeDef type : def.getExceptions() )
{
exceptions.add( createType( type, 0 ) );
}
currentConstructor.setExceptions( exceptions );
// modifiers
currentConstructor.setModifiers( new LinkedList( def.getModifiers() ) );
if ( !parameterList.isEmpty() )
{
currentConstructor.setParameters( new ArrayList( parameterList ) );
parameterList.clear();
}
currentConstructor.setSourceCode( def.getBody() );
}
/** {@inheritDoc} */
public void addCompactConstructor( CompactConstructorDef def )
{
DefaultJavaConstructor javaConstructor = recordHeaderStack.getFirst();
javaConstructor.setModifiers( new LinkedList( def.getModifiers() ) );
javaConstructor.setSourceCode( def.getBody() );
javaConstructor.setLineNumber( def.getLineNumber() );
}
/** {@inheritDoc} */
public void beginMethod()
{
currentMethod = new DefaultJavaMethod();
if ( currentField == null )
{
currentMethod.setDeclaringClass( classStack.getFirst() );
classStack.getFirst().addMethod( currentMethod );
}
currentMethod.setModelWriterFactory( modelWriterFactory );
addJavaDoc( currentMethod );
setAnnotations( currentMethod );
}
/** {@inheritDoc} */
public void endMethod( MethodDef def )
{
currentMethod.setLineNumber( def.getLineNumber() );
// basic details
currentMethod.setName( def.getName() );
currentMethod.setReturns( createType( def.getReturnType(), def.getDimensions() ) );
// typeParameters
if ( def.getTypeParams() != null )
{
List> typeParams =
new LinkedList>();
for ( TypeVariableDef typeVariableDef : def.getTypeParams() )
{
typeParams.add( createTypeVariable( typeVariableDef, (JavaMethod) currentMethod ) );
}
currentMethod.setTypeParameters( typeParams );
}
// exceptions
List exceptions = new LinkedList();
for ( TypeDef type : def.getExceptions() )
{
exceptions.add( createType( type, 0 ) );
}
currentMethod.setExceptions( exceptions );
// modifiers
currentMethod.setDefault( def.getModifiers().remove( "default" ) );
currentMethod.setModifiers( new LinkedList( def.getModifiers() ) );
if ( !parameterList.isEmpty() )
{
currentMethod.setParameters( new ArrayList( parameterList ) );
parameterList.clear();
}
currentMethod.setSourceCode( def.getBody() );
}
private DefaultJavaTypeVariable createTypeVariable( TypeVariableDef typeVariableDef, G genericDeclaration)
{
if ( typeVariableDef == null )
{
return null;
}
JavaClass declaringClass = getContext( genericDeclaration );
// can't select a declaring class based on the genericDeclaration
// likely method specifies its own genericDecleration
if ( declaringClass == null )
{
return null;
}
TypeResolver typeResolver = TypeResolver.byClassName( declaringClass.getBinaryName(), classLibrary, source.getImports() );
DefaultJavaTypeVariable result = new DefaultJavaTypeVariable( typeVariableDef.getName(), typeResolver );
if ( typeVariableDef.getBounds() != null && !typeVariableDef.getBounds().isEmpty() )
{
List bounds = new LinkedList();
for ( TypeDef typeDef : typeVariableDef.getBounds() )
{
bounds.add( createType( typeDef, 0 ) );
}
result.setBounds( bounds );
}
return result;
}
private static JavaClass getContext( JavaGenericDeclaration genericDeclaration )
{
JavaClass result;
if ( genericDeclaration instanceof JavaClass )
{
result = (JavaClass) genericDeclaration;
}
else if ( genericDeclaration instanceof JavaExecutable )
{
result = ( (JavaExecutable) genericDeclaration ).getDeclaringClass();
}
else
{
throw new IllegalArgumentException( "Unknown JavaGenericDeclaration implementation" );
}
return result;
}
/** {@inheritDoc} */
public void beginField( FieldDef def )
{
currentField = new DefaultJavaField( def.getName() );
currentField.setDeclaringClass( classStack.getFirst() );
currentField.setLineNumber( def.getLineNumber() );
currentField.setModelWriterFactory( modelWriterFactory );
currentField.setType( createType( def.getType(), def.getDimensions() ) );
currentField.setEnumConstant( def.isEnumConstant() );
// modifiers
{
currentField.setModifiers( new LinkedList( def.getModifiers() ) );
}
// code body
currentField.setInitializationExpression( def.getBody() );
// javadoc
addJavaDoc( currentField );
// annotations
setAnnotations( currentField );
}
/** {@inheritDoc} */
public void endField()
{
if ( currentArguments != null && !currentArguments.isEmpty() )
{
TypeResolver typeResolver;
if( classStack.isEmpty() )
{
typeResolver = TypeResolver.byPackageName( source.getPackageName(), classLibrary, source.getImports() );
}
else
{
typeResolver = TypeResolver.byClassName( classStack.getFirst().getBinaryName(), classLibrary, source.getImports() );
}
//DefaultExpressionTransformer??
DefaultJavaAnnotationAssembler assembler = new DefaultJavaAnnotationAssembler( currentField.getDeclaringClass(), classLibrary, typeResolver );
List arguments = new LinkedList();
for ( ExpressionDef annoDef : currentArguments )
{
arguments.add( assembler.assemble( annoDef ) );
}
currentField.setEnumConstantArguments( arguments );
currentArguments.clear();
}
classStack.getFirst().addField(currentField);
currentField = null;
}
/** {@inheritDoc} */
public void addParameter( FieldDef fieldDef )
{
DefaultJavaParameter jParam =
new DefaultJavaParameter( createType( fieldDef.getType(), fieldDef.getDimensions() ), fieldDef.getName(),
fieldDef.isVarArgs() );
if( currentMethod != null )
{
jParam.setExecutable( currentMethod );
}
else
{
jParam.setExecutable( currentConstructor );
}
jParam.setModelWriterFactory( modelWriterFactory );
addJavaDoc( jParam );
setAnnotations( jParam );
parameterList.add( jParam );
}
private void setAnnotations( final AbstractBaseJavaEntity entity )
{
if ( !currentAnnoDefs.isEmpty() )
{
TypeResolver typeResolver;
if( classStack.isEmpty() )
{
typeResolver = TypeResolver.byPackageName( source.getPackageName(), classLibrary, source.getImports() );
}
else
{
typeResolver = TypeResolver.byClassName( classStack.getFirst().getBinaryName(), classLibrary, source.getImports() );
}
DefaultJavaAnnotationAssembler assembler = new DefaultJavaAnnotationAssembler( entity.getDeclaringClass(), classLibrary, typeResolver );
List annotations = new LinkedList();
for ( AnnoDef annoDef : currentAnnoDefs )
{
annotations.add( assembler.assemble( annoDef ) );
}
entity.setAnnotations( annotations );
currentAnnoDefs.clear();
}
}
// Don't resolve until we need it... class hasn't been defined yet.
/** {@inheritDoc} */
public void addAnnotation( AnnoDef annotation )
{
currentAnnoDefs.add( annotation );
}
/** {@inheritDoc} */
public void addArgument( ExpressionDef argument )
{
currentArguments.add( argument );
}
/** {@inheritDoc} */
public JavaSource getSource()
{
return source;
}
/** {@inheritDoc} */
public JavaModule getModuleInfo()
{
return module;
}
/** {@inheritDoc} */
public void setUrl( URL url )
{
source.setURL( url );
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy