Reputation: 5168
I am trying to update files inside an existing WAR file using the ANT WAR task. I need to replace a set of xml files in a folder in WAR with new ones from my HDD.
<war destfile="myApp.war" update="true" >
<zipfileset dir="<PathToStubsFolderOnHDD>" includes="**/*.xml" prefix="<PathToStubsFolderInWAR>"/>
</war>
This works fine if the original WAR does not have xmls with same name. However if the original WAR contains xmls with same name; WAR task does not update them with files from HDD.
The ANT WAR task documentation reads:
update | indicates whether to update or overwrite the destination file if it already exists. Default is "false".
duplicate | behavior when a duplicate file is found. Valid values are "add", "preserve", and "fail". The default value is "add".
if i use update="false"; all other files in the original WAR are deleted and only the new xmls stored.
using duplicate="add" didnot have any effect either.
Any suggestions on how this can be achieved??
Upvotes: 16
Views: 27326
Reputation: 5509
Neither touch option worked for me. The original war file being created by a maven artifact from within the ant script might have something to do with this. Ended up repackaging the war file into a temp file and overwriting the original war file like this:
<target name="update-properties">
<war destfile="target/${war.name}temp" needxmlfile='false'>
<zipfileset src="target/${war.name}" includes="**/*"
excludes="${path.to.properties.in.war}" />
<zipfileset file="${path.to.new.properties}"
fullpath="${path.to.properties.in.war}" />
</war>
<move file="target/${local.war.name}temp" tofile="target/${local.war.name}"/>
<delete file="target/${local.war.name}temp"/>
</target>
where ${path.to.properties.in.war
} can be something like "WEB-INF/classes/META-INF/spring/database.properties"
Upvotes: 0
Reputation: 10400
This is my solution to replace existing files in zip(.war) file. Initial state is I have build.xml to compile and package mywebapp.war for Tomcat6 server. Minor configuration changes within war file are required for Tomcat7 server.
Reason I copy webT7 content to a temp build folder is content management systems. I don't want to change timestamps of the original repository files for no reason. Everything else compile,jar,war targets are always same no matter what Tomcat target I use.
Zip update="true" attribute as was said earlier do not replace files, it updates only if zip had older file than the one we give. This may introduce problems if I have web/config.jsp(2013-01-21 14:01:01) and webT7/config.jsp(2012-12-21 15:02:03) files. File was not replaced.
Snippet from build.xml file
<target name="war" depends="compile,jar" description="Create a .war file">
<delete file="${name}.war.zip" />
<zip destfile="${name}.war.zip"
basedir="./web/"
excludes="
**/CVS*
"
/>
</target>
<target name="warT7" depends="war" description="Create a .war file for Tomcat7">
<delete dir="${build}" />
<mkdir dir="${build}" />
<delete file="${name}.war_T7.zip" />
<copy file="${name}.war.zip"
tofile="${name}.war_T7.zip" overwrite="true" preservelastmodified="true"
/>
<copy todir="${build}" overwrite="true">
<fileset dir="./webT7" />
</copy>
<touch datetime="01/31/1981 01:00:00 AM" file="${name}.war_T7.zip" />
<zip destfile="${name}.war_T7.zip" update="true">
<zipfileset dir="${build}" prefix="" />
</zip>
<delete dir="${build}" />
</target>
Upvotes: 1
Reputation: 6872
Seems that with update=true
option, the war file is newer than the files that you want to update. Someone suggest to apply the touch
task with '0' to workaround the problem.
The zip task checks only if the source files you whish to add to an existing zip are more recent than the zip. A workaround would be to
<touch millis="0" />
the zip file before adding.
Or you could do the reverse stuff:
Performing a
touch
before thewar
task on XML file(s) makes it work.
Upvotes: 12
Reputation: 5168
Thanks Aito!
Here is the complete ANT script:
<target name = "UpdateWARWithStubs"
description="Updates WAR file with files from Stub folder">
<!-- Use touch to set modification time of all stubs to current time. This will force war task to update stub files inside the war -->
<tstamp> <format property="touch.time" pattern="MM/dd/yyyy hh:mm aa"/> </tstamp>
<touch datetime="${touch.time}">
<fileset dir="${path.to.stubs.on.hdd}" />
</touch>
<war destfile="myApp.war" update="true">
<zipfileset dir="${path.to.stubs.on.hdd}" prefix="${path.to.stubs.in.war}"/>
</war>
</target>
Upvotes: 9
Reputation: 272237
Perhaps the easiest way is to explode the war to a temporary directory, make your changes and then rebuild the war. Not nice, admittedly.
Upvotes: 0