Developer Dad
The Blog of Jeff Layton / jeffl8n
The Blog of Jeff Layton / jeffl8n
Oct 23rd
I’ve had a few requests to get an example of the build script I referenced in my post, Ant Foreach Properties, so here it is.
In our ant build.xml file we have a task called buildAllWsdls which gets all of the WSDL files from the web service URL, then feeds each of them into Axis2 to convert them to Java classes.
Here are the basic steps of what this build does:
<propertyselector property="services.list" delimiter="," match="(\w)*Service" />
<foreach list="${services.list}" inheritall="true" target="-getSingleWsdl" delimiter="," param="prettyName" />
<loadfile srcfile="${conf.dir}/NStoPkg.properties" property="ns2p.all">
<filterchain>
<striplinecomments>
<comment value="//" />
</striplinecomments>
<prefixlines prefix="," />
<striplinebreaks />
<tokenfilter>
<trim />
<ignoreblank />
</tokenfilter>
</filterchain>
</loadfile>
<foreach target="-createStubs" param="file" inheritall="true">
<path>
<fileset dir="${wsdldir}" />
</path>
</foreach>
<target name="-createStubs">
<basename property="wsdl" file="${file}" />
<echo message="Generating client code for service: ${wsdl}" />
<java fork="true" classname="org.apache.axis2.wsdl.WSDL2Java" failonerror="true">
<arg line="-ns2p ${ns2p.all} -s -u -uri ${wsdldir}/${wsdl} -o ${stubsdir} --noBuildXML" />
<classpath refid="cp.axis" />
</java>
</target>
If you don’t mind your web services being in a Java package that corresponds with the web service namespace, you can skip step 3 and leave out the -ns2p command line argument in the java ant task in step 4.
Here is the full example Ant build.xml file for doing this:
<project name="Axis2WSDLBuild" default="buildAllWsdls" basedir=".">
<tstamp>
<format property="TODAY" pattern="MM/dd/yyyy HH:mm:ss"/>
</tstamp>
<property name="ws-domain" value="http://services.test.com:8080" />
<property name="webservices.dir" value="${app.dir}/axisStubsProject" />
<property name="wsdldir" value="${webservices.dir}/wsdls" />
<property name="stubsdir" value="${webservices.dir}" />
<!-- patternsets -->
<patternset id="jar.files">
<include name="**/*.jar"/>
</patternset>
<path id="cp.axis">
<fileset dir="${lib.dir}/axis">
<patternset refid="jar.files"/>
</fileset>
</path>
<path id="cp.antcontrib">
<fileset dir="${lib.dir}/antcontrib">
<patternset refid="jar.files"/>
</fileset>
</path>
<!-- Allow usage of ant-contrib defined tasks -->
<taskdef resource="net/sf/antcontrib/antlib.xml">
<classpath refid="cp.antcontrib"/>
</taskdef>
<!-- =================================================================== -->
<!-- Project initialization -->
<!-- =================================================================== -->
<target name="init">
<property name="version" value="build ${TODAY}"/>
<property file="${conf.dir}/soa-addresses.properties"/>
<filter token="version" value="${version}"/>
<filter token="today" value="${TODAY}"/>
</target>
<!--=======================================================-->
<!-- TARGET [buildAllWsdls] -->
<!-- Target which gets and creates the web service stubs necessary for client invocation. -->
<!--=======================================================-->
<target name="buildAllWsdls" depends="init">
<antcall target="wsdlGet" />
<antcall target="buildWsdls" />
</target>
<!--=======================================================-->
<!-- TARGET [buildWsdls] -->
<!-- Target which creates the web service stubs necessary for client invocation. -->
<!--=======================================================-->
<target name="buildWsdls">
<!-- create the download directory -->
<mkdir dir="${stubsdir}" />
<!--
Translate the namespace to package properties file (as typically used in
an Axis 1 implementation) into a comma delimited list (as required by
Axis 2).
-->
<loadfile srcfile="${conf.dir}/NStoPkg.properties" property="ns2p.all">
<filterchain>
<striplinecomments>
<comment value="//" />
</striplinecomments>
<prefixlines prefix="," />
<striplinebreaks />
<tokenfilter>
<trim />
<ignoreblank />
</tokenfilter>
</filterchain>
</loadfile>
<!-- run client generation for each WSDL in the download directory -->
<foreach target="-createStubs" param="file" inheritall="true">
<path>
<fileset dir="${wsdldir}" />
</path>
</foreach>
</target>
<!--=======================================================-->
<!-- TARGET [wsdlGet] -->
<!-- Target retrieving all the WSDL files necessary for generating client code. -->
<!--=======================================================-->
<target name="wsdlGet">
<!-- create the download directory -->
<mkdir dir="${wsdldir}" />
<!-- find the properties related to WSDL locations -->
<propertyselector property="services.list" delimiter="," match="(\w)*Service" />
<!-- for each property found, retrieve the WSDL -->
<foreach list="${services.list}" inheritall="true" target="-getSingleWsdl" delimiter="," param="prettyName" />
</target>
<!--=======================================================-->
<!-- TARGET [-getSingleWsdl] -->
<!-- Hidden Target to retrieve a single WSDL. -->
<!--=======================================================-->
<target name="-getSingleWsdl">
<!-- get the property value -->
<propertycopy property="fullPath" from="${prettyName}" />
<!-- download the WSDL file -->
<get src="${ws-domain}/${fullPath}?wsdl" dest="${wsdldir}/${prettyName}.wsdl" verbose="true" usetimestamp="true" />
</target>
<!--=======================================================-->
<!-- TARGET [-createStubs] -->
<!-- Hidden target to run WSDL2Java on an input file. -->
<!--=======================================================-->
<target name="-createStubs">
<!-- strip the full path from the file -->
<basename property="wsdl" file="${file}" />
<echo message="Generating client code for service: ${wsdl}" />
<java fork="true" classname="org.apache.axis2.wsdl.WSDL2Java" failonerror="true">
<arg line="-ns2p ${ns2p.all} -s -u -uri ${wsdldir}/${wsdl} -o ${stubsdir} --noBuildXML" />
<classpath refid="cp.axis" />
</java>
</target>
</project>
the soa-addresses.properties file, which contains the different web service URLs:
ATestService=services/ATestService BTestService=services/BTestService CTestService=services/CTestService DTestService=services/DTestService ETestService=services/ETestService
and the NStoPkg.properties file which converts the default web service namespace based off the WSDL URL to a specified java package:
http://typens.test.com/common=com.test.axis.common.types http://servicens.test.com/common/ATestService=com.test.axis.common.services.atestservice http://servicens.test.com/common/BTestService=com.test.axis.common.services.btestservice http://servicens.test.com/common/CTestService=com.test.axis.common.services.ctestservice http://servicens.test.com/common/DTestService=com.test.axis.common.services.dtestservice http://servicens.test.com/common/ETestService=com.test.axis.common.services.etestservice
Jul 28th
Facebook has been doing a few sketchy things recently with privacy. First, there was a big terms of service change which most of the Facebook community petitioned against and was later reverted back and the terms were changed with the community’s input. Now Facebook is using user’s profile information in their Ads. Now these ads are Facebook ads (not third party) and only shown to people you have confirmed as a friend, but I still would prefer to not have random ads which basically say I endorse something because I am a “fan” of it. I thought other people might want to know how to as well, so I made this quick how to for making sure your profile won’t show up in Facebook Ads. It’s just three simple steps…


That’s all you have to do!
Jul 8th
Well, it’s been quite a while since I’ve posted. In fact the last time I posted, the date was a significant math date, as was today. This time only happens once a century (though 12:34:56 7/8/90 is probably a more significant math date…). Anyway, we’ve been moving and getting our new house together the past couple months, which is partially the reason for my lack of posting. Now that we’re mostly done (if you’re a home owner, you know a house is never truly finished fixing…), I will be posting a bit more again. In the mean time, hope you had a nice 12:34:56 7/8/9 today!
Mar 13th
Update 10/23/2009: An example of this build has been posted.
I’ve been working with Apache Ant a bit at work the past couple weeks. We’ve been setting up a Hudson continuous integration server to automate some of our builds and deployments. We’re also hoping to setup more automated tests using tools like Selenium, HTTPUnit, JUnit, etc., but those are going to be gradually added with the coming releases.
I’ve worked with Ant a lot before, but hadn’t really used the Ant-Contrib Tasks until modifying this build script. The foreach task is useful for running a list of things through a certain ant target. We have a properties file that has the webservice name and the associated WSDL URL. We feed that list to an Ant target through the foreach task (for each URL, go get the WSDL file/definition), then we generate Axis2 stubs from each WSDL file. To automate the generation/retrieving of these WSDL files I was moving the base URL (ex: http://services.test.com/) to the environment property file, since this is different in each environment (Test, QA, Production), but the relative URL (ex: /services/…) stays the same. So to summarize, the webservices properties file now has ExampleService=/services/ExampleService and the environment properties file has SERVICES-DOMAIN=http://services.test.com (SERVICES-DOMAIN is assigned to the ws-domain after the environment properties file is loaded in the ant build file).
Here’s where I ran into a little issue that I wanted to share so no one else has to spend a half hour debugging it. Well, also so I can find the answer here the next time I run into it. The property for the base URL wasn’t evaluating when I was running the WSDL list through the foreach call to get/download the WSDL file. It kept erroring saying ${ws-domain}/services/ExampleService?wsdl couldn’t be found. I spent a few minutes putting echo’s in the ant script to see where it was failing until I saw ${ws-domain} was set up until the ant target that gets the WSDL file. It was then I saw this foreach task attribute:
Attribute Description
inheritall Iftrue, pass all properties to the called target. Defaults tofalse.
Now I’m not sure why you wouldn’t want properties passed, but I’m sure there is some good reason why this is false by default. Maybe. So of course, adding this attribute set as true to the foreach ant task made everything work without a hitch.