Monday, November 30, 2009

Groovy: Java Enum Generation from XML

Besides the obvious use of Groovy to write applications, Groovy is also very useful for performing Java development related tasks such building applications, deploying applications, and managing/monitoring applications. In this post, I look at an example of generating a Java enum from XML source using Groovy.

I have previously blogged on slurping XML with Groovy. That technique is used again here. The Groovy code that follows uses this technique to read the source XML and then writes out a 100% compilable Java enum called AlbumsEnum.java. The Groovy script is shown next.

  1. #!/usr/bin/env groovy  
  2. // generateAlbumsEnumFromXml.groovy  
  3. //  
  4. // Demonstrates use of Groovy's XML slurping to generate  
  5. // a Java enum from source XML.  
  6. //  
  7.   
  8. // Set up enum attributes' names and data types  
  9. attributes = ['albumTitle' : 'String''artistName' : 'String''year' : 'int']  
  10.   
  11. // Base package name off command-line parameter if provided  
  12. packageName = args.length > 0 ? args[0] : "albums"  
  13.   
  14. NEW_LINE = System.getProperty("line.separator")  
  15. SINGLE_INDENT = '   '  
  16. DOUBLE_INDENT = SINGLE_INDENT.multiply(2)  
  17. outputFile = new File("AlbumsEnum.java")  
  18. outputFile.write "package ${packageName};${NEW_LINE.multiply(2)}"  
  19. outputFile << "public enum AlbumsEnum${NEW_LINE}"  
  20. outputFile << "{${NEW_LINE}"  
  21.   
  22. outputFile << generateEnumConstants()  
  23.   
  24. // Build enum attributes  
  25. attributesSection = new StringBuilder();  
  26. attributesAccessors = new StringBuilder();  
  27. attributesCtorSetters = new StringBuilder();  
  28. attributes.each  
  29. {  
  30.    attributesSection << generateAttributeDeclaration(it.key, it.value)  
  31.    attributesAccessors << buildAccessor(it.key, it.value) << NEW_LINE  
  32.    attributesCtorSetters << buildConstructorAssignments(it.key)  
  33. }  
  34. outputFile << attributesSection  
  35. outputFile << NEW_LINE  
  36.   
  37. outputFile << generateParameterizedConstructor(attributes)  
  38. outputFile << NEW_LINE  
  39.   
  40. outputFile << attributesAccessors  
  41.   
  42. outputFile << '}'  
  43.   
  44. def String generateEnumConstants()  
  45. {  
  46.    // Get input from XML source  
  47.    albums = new XmlSlurper().parse("albums.xml")  
  48.    def enumConstants = new StringBuilder()  
  49.    albums.Album.each  
  50.    {  
  51.       enumConstants << SINGLE_INDENT  
  52.       enumConstants << it.@artist.toString().replace(' ''_').toUpperCase() << '_'  
  53.       enumConstants << it.@title.toString().replace(' ''_').toUpperCase()  
  54.       enumConstants << "(\"${it.@title}\", \"${it.@artist}\", ${it.@year.toInteger()}),"  
  55.       enumConstants << NEW_LINE  
  56.    }  
  57.    // Subtract three off end of substring: one for new line, one for extra comma,  
  58.    // and one for zero-based indexing.  
  59.    returnStr = new StringBuilder(enumConstants.toString().substring(0, enumConstants.size()-3))  
  60.    returnStr << ';' << NEW_LINE.multiply(2)  
  61.    return returnStr  
  62. }  
  63.   
  64. def String generateAttributeDeclaration(String attrName, String attrType)  
  65. {  
  66.    return "${SINGLE_INDENT}private ${attrType} ${attrName};${NEW_LINE}"  
  67. }  
  68.   
  69. def String buildAccessor(String attrName, String attrType)  
  70. {  
  71.    returnStr = new StringBuilder()  
  72.    returnStr << SINGLE_INDENT << "public ${attrType} get${capitalizeFirstLetter(attrName)}()" << NEW_LINE  
  73.    returnStr << SINGLE_INDENT << '{' << NEW_LINE  
  74.    returnStr << DOUBLE_INDENT << "return this.${attrName};" << NEW_LINE  
  75.    returnStr << SINGLE_INDENT << '}' << NEW_LINE  
  76.    return returnStr  
  77. }  
  78.   
  79. def String generateParameterizedConstructor(Map<String,String> attributesMap)  
  80. {  
  81.    constructorInit = new StringBuilder()  
  82.    constructorInit << SINGLE_INDENT << 'AlbumsEnum('  
  83.    attributesMap.each  
  84.    {  
  85.       constructorInit << "final ${it.value} new${capitalizeFirstLetter(it.key)}, "  
  86.    }  
  87.    constructorFinal = new StringBuilder(constructorInit.substring(0, constructorInit.size()-2))  
  88.    constructorFinal << ')'  
  89.    constructorFinal << NEW_LINE << SINGLE_INDENT << '{' << NEW_LINE  
  90.    constructorFinal << attributesCtorSetters  
  91.    constructorFinal << SINGLE_INDENT << "}${NEW_LINE}"  
  92.    return constructorFinal  
  93. }  
  94.   
  95. def String buildConstructorAssignments(String attrName)  
  96. {  
  97.    return "${DOUBLE_INDENT}this.${attrName} = new${capitalizeFirstLetter(attrName)};${NEW_LINE}"  
  98. }  
  99.   
  100. def String capitalizeFirstLetter(String word)  
  101. {  
  102.    firstLetter = word.getAt(0)  
  103.    uppercaseLetter = firstLetter.toUpperCase()  
  104.    return word.replaceFirst(firstLetter, uppercaseLetter)  
  105. }  


The XML source that this script is run against is shown next:

  1. <?xml version="1.0"?>  
  2. <Albums>  
  3.    <Album title="Frontiers" artist="Journey" year="1983">  
  4.       <Song title="Separate Ways" peak="8" />  
  5.       <Song title="Send Her My Love" peak="23" />  
  6.       <Song title="Faithfully" peak="12" />  
  7.    </Album>  
  8.    <Album title="Hysteria" artist="Def Leppard" year="1987">  
  9.       <Song title="Hysteria" peak="10" />  
  10.       <Song title="Animal" peak="19" />  
  11.       <Song title="Women" />  
  12.       <Song title="Pour Some Sugar On Me" peak="2" />  
  13.       <Song title="Love Bites" peak="1" />  
  14.       <Song title="Armageddon It" peak="3" />  
  15.       <Song title="Rocket" peak="15" />  
  16.    </Album>  
  17.    <Album title="The Joshua Tree" artist="U2" year="1987">  
  18.       <Song title="With or Without You" peak="1" />  
  19.       <Song title="I Still Haven't Found What I'm Looking For" peak="1" />  
  20.       <Song title="Where The Streets Have No Name" peak="13" />  
  21.       <Song title="In God's Country" peak="14" />  
  22.    </Album>  
  23.    <Album title="Songs from the Big Chair" artist="Tears for Fears" year="1985">  
  24.       <Song title="Shout" peak="1" />  
  25.       <Song title="Everybody Wants to Rule the World" peak="1" />  
  26.       <Song title="Head Over Heels" peak="3" />  
  27.       <Song title="Mothers Talk" peak="27" />  
  28.    </Album>  
  29. </Albums>  


When the Groovy script is run against this source XML, the Java enum that it generates is shown next.

  1. package albums;  
  2.   
  3. public enum AlbumsEnum  
  4. {  
  5.    JOURNEY_FRONTIERS("Frontiers""Journey"1983),  
  6.    DEF_LEPPARD_HYSTERIA("Hysteria""Def Leppard"1987),  
  7.    U2_THE_JOSHUA_TREE("The Joshua Tree""U2"1987),  
  8.    TEARS_FOR_FEARS_SONGS_FROM_THE_BIG_CHAIR("Songs from the Big Chair""Tears for Fears"1985);  
  9.   
  10.    private String albumTitle;  
  11.    private String artistName;  
  12.    private int year;  
  13.   
  14.    AlbumsEnum(final String newAlbumTitle, final String newArtistName, final int newYear)  
  15.    {  
  16.       this.albumTitle = newAlbumTitle;  
  17.       this.artistName = newArtistName;  
  18.       this.year = newYear;  
  19.    }  
  20.   
  21.    public String getAlbumTitle()  
  22.    {  
  23.       return this.albumTitle;  
  24.    }  
  25.   
  26.    public String getArtistName()  
  27.    {  
  28.       return this.artistName;  
  29.    }  
  30.   
  31.    public int getYear()  
  32.    {  
  33.       return this.year;  
  34.    }  
  35.   
  36. }  


There are, of course, many ways in which this simple draft script could be improved and many features could be added to it. However, it serves the purpose of illustrating how easy it is to use Groovy to generate Java code from source data such as XML.

No comments: