Listing All Classes In A Jar File Using ColdFusion

Posted October 2, 2008 at 8:05 PM by Ben Nadel

Tags: ColdFusion

You'll probably never need to do this, but today, I needed to find out what Java classes were available in a given JAR file using ColdFusion. The reason - I was doing some work on my POI Utility ColdFusion custom tags and I needed to see if the version of the POI JAR that ships with ColdFusion supports certain classes. Turns out, this is a fairly straightforward task. I wrapped the functionality up in a ColdFusion user defined function that takes the expanded path to the target JAR file:

  • <cffunction
  • name="GetJarClasses"
  • access="public"
  • returntype="array"
  • output="false"
  • hint="I return an array of classes in the given JAR file (expanded path).">
  •  
  • <!--- Define arguments. --->
  • <cfargument
  • name="JarFilePath"
  • type="string"
  • required="true"
  • hint="I am the expanded path of the JAR file."
  • />
  •  
  • <!--- Define the local scope. --->
  • <cfset var LOCAL = {} />
  •  
  • <!--- Create a default array of classes. --->
  • <cfset LOCAL.Classes = [] />
  •  
  •  
  • <!---
  • Create a JAR input stream to read in the line items
  • from our target JAR file.
  • --->
  • <cfset LOCAL.JarFile = CreateObject(
  • "java",
  • "java.util.jar.JarInputStream"
  • ).Init(
  •  
  • CreateObject(
  • "java",
  • "java.io.FileInputStream"
  • ).Init(
  •  
  • JavaCast(
  • "string",
  • ARGUMENTS.JarFilePath
  • )
  • )
  • )
  • />
  •  
  •  
  • <!---
  • Now that we have our JAR file input stream, let's loop
  • over all the entries looking for CLASS files.
  • --->
  • <cfloop condition="true">
  •  
  • <!---
  • Get the next entry. This might return NULL if the
  • JAR file has no more classes.
  • --->
  • <cfset LOCAL.JarEntry = LOCAL.JarFile.GetNextJarEntry() />
  •  
  • <!---
  • Check to see if the entry variable exists. If it
  • does not, then it means the JAR file return NULL
  • and we are done finding classes.
  • --->
  • <cfif StructKeyExists( LOCAL, "JarEntry" )>
  •  
  • <!---
  • Check to make sure that this entry is not a
  • directory, but is, in fact a class.
  • --->
  • <cfif REFindNoCase( "\.class$", LOCAL.JarEntry.GetName() )>
  •  
  • <!---
  • Add this class to the array. Since the JAR
  • file really has a directory structure, let's
  • replace the path separators with dots.
  • --->
  • <cfset LOCAL.ClassName = REReplace(
  • LOCAL.JarEntry.GetName(),
  • "[\\/]",
  • ".",
  • "all"
  • ) />
  •  
  • <!--- Strip off the ".class" path item. --->
  • <cfset LOCAL.ClassName = REReplaceNoCase(
  • LOCAL.ClassName,
  • "\.class$",
  • "",
  • "one"
  • ) />
  •  
  • <!--- Add the formatted class name. --->
  • <cfset ArrayAppend(
  • LOCAL.Classes,
  • LOCAL.ClassName
  • ) />
  •  
  • </cfif>
  •  
  • <cfelse>
  •  
  • <cfbreak />
  •  
  • </cfif>
  •  
  • </cfloop>
  •  
  •  
  • <!--- Return the array of classes. --->
  • <cfreturn LOCAL.Classes />
  • </cffunction>

To call it, you just do this (I had a copy of the installed POI JAR file in the same directory as my test file):

  • <!--- Get file path to POI installed JAR file. --->
  • <cfset strJarFilePath = ExpandPath(
  • "./poi-2.5.1-final-20040804.jar"
  • ) />
  •  
  • <!--- Output the classes. --->
  • <cfdump
  • var="#GetJarClasses( strJarFilePath )#"
  • label="POI 2.5.1 Final Classes"
  • />

Running this, we get an array with the following values (abbreviated):

org.apache.poi.ddf.DefaultEscherRecordFactory
org.apache.poi.ddf.EscherArrayProperty
org.apache.poi.ddf.EscherBSERecord
org.apache.poi.ddf.EscherBlipRecord
..... several hundred classes .....
org.apache.poi.util.ShortList
org.apache.poi.util.StringUtil
org.apache.poi.util.SystemOutLogger

Anyway, just thought I would post that in case anyone ever needs this kind of functionality.




Reader Comments

Oct 3, 2008 at 9:09 AM // reply »
48 Comments

How is this different then just listing the JAR out with cfzip?


Oct 3, 2008 at 9:10 AM // reply »
32 Comments

I never really play around with JAR files much, so this comment could be totally off base, but, could you have not just used the cfzip tag? It accepts a JAR file and you could use the list action and just dumped that out, right? or am i missing the point? It's Friday so my brain has shut of ffor the weekend :-)


Oct 3, 2008 at 9:10 AM // reply »
32 Comments

@Todd - beat me to it :-)


Oct 3, 2008 at 9:13 AM // reply »
10,640 Comments

@Todd, @Simon,

Oh snap, I totally forgot that CFZip could read JAR files :)


Oct 3, 2008 at 9:30 AM // reply »
10,640 Comments

Thanks guys; amends have been made:

http://www.bennadel.com/index.cfm?dax=blog:1373.view


Post A Comment

Comment Etiquette: Please do not post spam. Please keep the comments on-topic. Please do not post unrelated questions or large chunks of code. And, above all, please be nice to each other - we're trying to have a good conversation here.

Please review the following issues:

Author Name:


Author Email:

Author Website:

Comment:

Supported HTML tags for formatting: <strong>bold</strong>   <em>italic</em>   <code>code</code>







  • Help Wanted - Find Your Next ColdFusion Job
InVision App - Prototyping Made Beautiful With Prototyping Tools Ben Nadel's Company - Epicenter Consulting Recent Blog Comments
Feb 10, 2012 at 7:21 PM
jQuery AJAX Strips Script Tags And Inserts Them After Parent-Most Elements
Update! Instead of $(eval(options.insertAfter)).after(data['insertData']); I now use: var ajaxNode = document.createElement('span'); var parent = $(eval(options.insertAfter))[0].parentNode; ... read »
Feb 10, 2012 at 6:18 PM
jQuery AJAX Strips Script Tags And Inserts Them After Parent-Most Elements
encountered this same, what I consider, jQuery bug last week. I'm building a site in which I load some content via AJAX. This content contains Linkedin share button placeholders which Linkedin API ne ... read »
Feb 10, 2012 at 11:30 AM
Cross-Origin Resource Sharing (CORS) AJAX Requests Between jQuery And Node.js
After you understand the concepts here, this is an awesome cheatsheet for enabling CORS in just about anything http://enable-cors.org/ ... read »
JM
Feb 10, 2012 at 9:10 AM
My Safari Browser SQLite Database Hello World Example
@Amy, Here is a very good tutorial on how to use JOIN: http://www.sqltutorial.org/sqljoin-innerjoin.aspx ... read »
Feb 10, 2012 at 4:42 AM
Building A Twitter-Inspired RESTful API Architecture In ColdFusion
This is great, very useful Ben. I spotted a small typo in the api.cgm listing: <cfthrow type="Unauthroized" /> Cheers Stefan ... read »
Feb 9, 2012 at 10:35 PM
CFDirectory Filtering Uses Pipe Character For Multiple Filters (Thanks Steve Withington)
I was wondering if there would be a filter you could apply so that you got everything but what you included in the filter. As in show me all docs that are not a .pdf. ... read »
Feb 9, 2012 at 10:29 PM
Learning ColdFusion 9: Application-Specific Data Sources
@Ben, No offence, but if people were really wanting advanced features they would be using a platform like ASP.NET MVC. CFML is so structurally compromised as a tag-based scripting language that ... read »
Feb 9, 2012 at 10:03 PM
Subversion - Cleanup Failed To Process The Following Paths
@Leviaguirre, do you still have problems with this? ... read »