Norman Walsh resolver code from xml-commons. Submitted by: Craeg K Strong <cstrong at arielpartners.com> git-svn-id: https://svn.apache.org/repos/asf/ant/core/trunk@273486 13f79535-47bb-0310-9956-ffa450edef68master
@@ -67,6 +67,10 @@ Other changes: | |||||
define ids or paths and use Ant's location magic for filename resolutions | define ids or paths and use Ant's location magic for filename resolutions | ||||
in the XML file. | in the XML file. | ||||
* <xmlcatalog> will now support external catalogs according to the | |||||
OASIS "Open Catalog" standard - if resolver.jar from Apache's | |||||
xml-commons is in your CLASSPATH. | |||||
Changes from Ant 1.5.1Beta1 to 1.5.1 | Changes from Ant 1.5.1Beta1 to 1.5.1 | ||||
==================================== | ==================================== | ||||
@@ -51,6 +51,7 @@ | |||||
<property name="ant.package" value="org/apache/tools/ant"/> | <property name="ant.package" value="org/apache/tools/ant"/> | ||||
<property name="optional.package" value="${ant.package}/taskdefs/optional"/> | <property name="optional.package" value="${ant.package}/taskdefs/optional"/> | ||||
<property name="optional.type.package" value="${ant.package}/types/optional"/> | <property name="optional.type.package" value="${ant.package}/types/optional"/> | ||||
<property name="apache.resolver.type.package" value="${ant.package}/types/resolver"/> | |||||
<property name="util.package" value="${ant.package}/util"/> | <property name="util.package" value="${ant.package}/util"/> | ||||
<property name="regexp.package" value="${util.package}/regexp"/> | <property name="regexp.package" value="${util.package}/regexp"/> | ||||
@@ -195,6 +196,9 @@ | |||||
<selector id="needs.xslp"> | <selector id="needs.xslp"> | ||||
<filename name="${optional.package}/XslpLiaison*"/> | <filename name="${optional.package}/XslpLiaison*"/> | ||||
</selector> | </selector> | ||||
<selector id="needs.apache.resolver"> | |||||
<filename name="${apache.resolver.type.package}/**"/> | |||||
</selector> | |||||
<selector id="needs.junit"> | <selector id="needs.junit"> | ||||
<filename name="${optional.package}/junit/**"/> | <filename name="${optional.package}/junit/**"/> | ||||
</selector> | </selector> | ||||
@@ -341,6 +345,9 @@ | |||||
<available property="xslp.present" | <available property="xslp.present" | ||||
classname="com.kvisco.xsl.XSLProcessor" | classname="com.kvisco.xsl.XSLProcessor" | ||||
classpathref="classpath"/> | classpathref="classpath"/> | ||||
<available property="apache.resolver.present" | |||||
classname="org.apache.xml.resolver.tools.CatalogResolver" | |||||
classpathref="classpath" /> | |||||
<available property="xalan.present" | <available property="xalan.present" | ||||
classname="org.apache.xalan.xslt.XSLTProcessorFactory" | classname="org.apache.xalan.xslt.XSLTProcessorFactory" | ||||
classpathref="classpath"/> | classpathref="classpath"/> | ||||
@@ -570,6 +577,7 @@ | |||||
<selector refid="needs.xalan1" unless="xalan.present"/> | <selector refid="needs.xalan1" unless="xalan.present"/> | ||||
<selector refid="needs.xalan2" unless="xalan2.present"/> | <selector refid="needs.xalan2" unless="xalan2.present"/> | ||||
<selector refid="needs.xslp" unless="xslp.present"/> | <selector refid="needs.xslp" unless="xslp.present"/> | ||||
<selector refid="needs.apache.resolver" unless="apache.resolver.present"/> | |||||
<selector refid="needs.junit" unless="junit.present"/> | <selector refid="needs.junit" unless="junit.present"/> | ||||
<selector refid="needs.jakarta.regexp" | <selector refid="needs.jakarta.regexp" | ||||
unless="jakarta.regexp.present"/> | unless="jakarta.regexp.present"/> | ||||
@@ -739,6 +747,7 @@ | |||||
<selector refid="needs.xalan1"/> | <selector refid="needs.xalan1"/> | ||||
<selector refid="needs.xalan2"/> | <selector refid="needs.xalan2"/> | ||||
<selector refid="needs.xslp"/> | <selector refid="needs.xslp"/> | ||||
<selector refid="needs.apache.resolver"/> | |||||
<selector refid="needs.junit"/> | <selector refid="needs.junit"/> | ||||
<selector refid="needs.jakarta.regexp"/> | <selector refid="needs.jakarta.regexp"/> | ||||
<selector refid="needs.jakarta.oro"/> | <selector refid="needs.jakarta.oro"/> | ||||
@@ -792,6 +801,12 @@ | |||||
<selector refid="needs.xslp"/> | <selector refid="needs.xslp"/> | ||||
</jar> | </jar> | ||||
<jar destfile="${build.lib}/${optional.jars.prefix}-apache-resolver.jar" | |||||
basedir="${build.classes}" | |||||
manifest="${manifest.tmp}"> | |||||
<selector refid="needs.apache.resolver"/> | |||||
</jar> | |||||
<jar destfile="${build.lib}/${optional.jars.prefix}-junit.jar" | <jar destfile="${build.lib}/${optional.jars.prefix}-junit.jar" | ||||
basedir="${build.classes}" | basedir="${build.classes}" | ||||
manifest="${manifest.tmp}"> | manifest="${manifest.tmp}"> | ||||
@@ -19,6 +19,10 @@ documents | |||||
to efficiently allow a local substitution for a resource available on the | to efficiently allow a local substitution for a resource available on the | ||||
web. | web. | ||||
</p> | </p> | ||||
<p><b>Note:</b> This task <em>uses, but does not depend on</em> external | |||||
libraries not included in the Ant distribution. See <a | |||||
href="../install.html#librarydependencies">Library Dependencies</a> for more | |||||
information.</p> | |||||
<p>This data type provides a catalog of resource locations based | <p>This data type provides a catalog of resource locations based | ||||
on the <a | on the <a | ||||
href="http://oasis-open.org/committees/entity/spec-2001-08-06.html"> | href="http://oasis-open.org/committees/entity/spec-2001-08-06.html"> | ||||
@@ -59,17 +63,34 @@ task uses XMLCatalogs for both entity and URI resolution.</p> | |||||
<p>XMLCatalogs are specified as either a reference to another XMLCatalog, | <p>XMLCatalogs are specified as either a reference to another XMLCatalog, | ||||
defined | defined | ||||
previously in a build file, or as a list of <code>dtd</code> or | previously in a build file, or as a list of <code>dtd</code> or | ||||
<code>entity</code> locations. A separate classpath for entity resolution | |||||
<code>entity</code> locations. In addition, external catalog files | |||||
may be specified in <code>catalogfiles</code> filesets, but they will | |||||
be ignored unless the resolver library from xml-commons is available | |||||
in the system classpath. A separate classpath for entity resolution | |||||
may be specified inline via nested <code>classpath</code> elements; | may be specified inline via nested <code>classpath</code> elements; | ||||
otherwise the system classpath is used for this as well.</p> | otherwise the system classpath is used for this as well.</p> | ||||
<p>XMLCatalogs can also be nested inside other XMLCatalogs. For | <p>XMLCatalogs can also be nested inside other XMLCatalogs. For | ||||
example, a "superset" XMLCatalog could be made by including several | example, a "superset" XMLCatalog could be made by including several | ||||
nested XMLCatalogs that referred to other, previously defined | nested XMLCatalogs that referred to other, previously defined | ||||
XMLCatalogs.</p> | XMLCatalogs.</p> | ||||
<p>Resource locations can be specified either in-line or in | |||||
external catalog file(s), or both. In order to use an external | |||||
catalog file, the xml-commons resolver library ("resolver.jar") | |||||
must be in your path. External catalog files may be either <a | |||||
href="http://oasis-open.org/committees/entity/background/9401.html"> | |||||
plain text format</a> or <a | |||||
href="http://www.oasis-open.org/committees/entity/spec-2001-08-06.html"> | |||||
XML format</a>. If the xml-commons resolver library is not found in the | |||||
classpath, external catalog files, specified in <code>catalogfiles</code> | |||||
filesets, will be ignored and a warning will be logged. In this case, however, | |||||
processing of inline entries will proceed normally.</p> | |||||
<p>Currently, only <code><dtd></code> and | <p>Currently, only <code><dtd></code> and | ||||
<code><entity></code> elements may be specified inline; these | <code><entity></code> elements may be specified inline; these | ||||
roughly correspond to OASIS catalog entry types <code>PUBLIC</code> and | roughly correspond to OASIS catalog entry types <code>PUBLIC</code> and | ||||
<code>URI</code> respectively.</p> | |||||
<code>URI</code> respectively. By contrast, external catalog files | |||||
may use any of the entry types defined in the | |||||
<a href="http://oasis-open.org/committees/entity/spec-2001-08-06.html"> | |||||
+OASIS specification</a>.</p> | |||||
<h3><a name="ResolverAlgorithm">Entity/DTD/URI Resolution Algorithm</a></h3> | <h3><a name="ResolverAlgorithm">Entity/DTD/URI Resolution Algorithm</a></h3> | ||||
When an entity, DTD, or URI is looked up by the XML processor, the | When an entity, DTD, or URI is looked up by the XML processor, the | ||||
@@ -101,6 +122,18 @@ will <em>not</em> resolve an entity whose <code>location</code> is | |||||
<code>blat.dtd</code>. | <code>blat.dtd</code>. | ||||
<h4>3a. Apache xml-commons resolver lookup</h4> | |||||
<p>What happens next depends on whether the resolver library from | |||||
xml-commons is available on the classpath. If so, we defer all | |||||
further attempts at resolving to it. The resolver library supports | |||||
extremely sophisticated functionality like URL rewriting and so on, | |||||
which can be accessed by making the appropriate entries in external | |||||
catalog files (XMLCatalog does not yet provide inline support for all | |||||
of the entries defined in the <a | |||||
href="http://oasis-open.org/committees/entity/spec-2001-08-06.html">OASIS | |||||
standard</a>).</p> | |||||
<h4>3. URL-space lookup</h4> | <h4>3. URL-space lookup</h4> | ||||
<p>Finally, we attempt to make a URL out of the <code>location</code>. | <p>Finally, we attempt to make a URL out of the <code>location</code>. | ||||
@@ -171,6 +204,22 @@ basedir. | |||||
<p>The classpath to use for <a href="#ResolverAlgorithm">entity | <p>The classpath to use for <a href="#ResolverAlgorithm">entity | ||||
resolution</a>. The nested <code><classpath></code> is a | resolution</a>. The nested <code><classpath></code> is a | ||||
<a href="../using.html#path">path</a>-like structure.</p> | <a href="../using.html#path">path</a>-like structure.</p> | ||||
<h4>catalogfiles</h4> | |||||
<p> | |||||
The nested <code>catalogfiles</code> element specifies a <a | |||||
href="../CoreTypes/fileset.html">FileSet</a>. All files included in | |||||
this fileset are assumed to be OASIS catalog files, in either | |||||
<a href="http://oasis-open.org/committees/entity/background/9401.html"> | |||||
plain text format</a> or <a | |||||
href="http://www.oasis-open.org/committees/entity/spec-2001-08-06.html"> | |||||
XML format</a>. Multiple <code>catalogfiles</code> filesets may be | |||||
specified. Of course, if you use wildcards in your fileset, you will | |||||
want to use some sort of naming convention to ensure that you don't | |||||
accidentally match non-catalog files. If the resolver library from | |||||
xml-commons is not available in the classpath, all | |||||
<code>catalogfiles</code> will be ignored and a warning will be | |||||
logged. | |||||
</p> | |||||
<h3>Examples</h3> | <h3>Examples</h3> | ||||
<p>Set up an XMLCatalog with a single dtd referenced locally in a user's | <p>Set up an XMLCatalog with a single dtd referenced locally in a user's | ||||
home | home | ||||
@@ -197,7 +246,8 @@ filesystem (relative to the Ant project basedir) or in the classpath: | |||||
</pre></blockquote> | </pre></blockquote> | ||||
<p>Set up an XMLCatalog with a combination of DTDs and entities as | <p>Set up an XMLCatalog with a combination of DTDs and entities as | ||||
well as a nested XMLCatalog:</p> | |||||
well as a nested XMLCatalog and external catalog files in both | |||||
formats:</p> | |||||
<blockquote><pre> | <blockquote><pre> | ||||
<xmlcatalog id="allcatalogs"> | <xmlcatalog id="allcatalogs"> | ||||
@@ -207,7 +257,13 @@ well as a nested XMLCatalog:</p> | |||||
<entity | <entity | ||||
publicId="LargeLogo" | publicId="LargeLogo" | ||||
location="com/arielpartners/images/ariel-logo-large.gif"/> | location="com/arielpartners/images/ariel-logo-large.gif"/> | ||||
<xmlcatalog refid="commonDTDs"/> | |||||
<xmlcatalog refid="commonDTDs"/> | |||||
<catalogfiles | |||||
dir="/anetwork/drive" | |||||
includes="**/catalog"/> | |||||
<catalogfiles | |||||
dir="/my/catalogs" | |||||
includes="**/catalog.xml"/> | |||||
</xmlcatalog> | </xmlcatalog> | ||||
</pre></blockquote> | </pre></blockquote> | ||||
<p>To reference the above XMLCatalog in an <code>xslt</code> task:<p> | <p>To reference the above XMLCatalog in an <code>xslt</code> task:<p> | ||||
@@ -381,6 +381,12 @@ Installing Ant / Optional Tasks</a> section above.</p> | |||||
<td><a href="http://www.clarkware.com/software/JDepend.html" | <td><a href="http://www.clarkware.com/software/JDepend.html" | ||||
target="_top">http://www.clarkware.com/software/JDepend.html</a></td> | target="_top">http://www.clarkware.com/software/JDepend.html</a></td> | ||||
</tr> | </tr> | ||||
<tr> | |||||
<td>resolver.jar</td> | |||||
<td>xmlcatalog datatype <em>only if support for external catalog files is desired</em></td> | |||||
<td><a href="http://xml.apache.org/dist/commons" | |||||
target="_top">http://xml.apache.org/dist/commons</a></td> | |||||
</tr> | |||||
</table> | </table> | ||||
<br> | <br> | ||||
<hr> | <hr> | ||||
@@ -38,8 +38,18 @@ | |||||
</xmlcatalog> | </xmlcatalog> | ||||
</xmlvalidate> | </xmlvalidate> | ||||
</target> | </target> | ||||
<target name="xmlcatalogfiles"> | |||||
<xmlvalidate warn="false"> | |||||
<fileset dir="xml" includes="**/about.xml"/> | |||||
<xmlcatalog classpath="xml"> | |||||
<catalogfiles dir="xml" includes="catalog"/> | |||||
<dtd publicID="-//stevo//DTD doc 1.0//EN" | |||||
location="doc.dtd"/> | |||||
</xmlcatalog> | |||||
</xmlvalidate> | |||||
</target> | |||||
<target name="testSchemaGood"> | <target name="testSchemaGood"> | ||||
<xmlvalidate warn="false" lenient="no" > | <xmlvalidate warn="false" lenient="no" > | ||||
<fileset dir="xml" includes="endpiece.xml"/> | <fileset dir="xml" includes="endpiece.xml"/> | ||||
@@ -67,6 +67,7 @@ import org.apache.tools.ant.types.Path; | |||||
import org.apache.tools.ant.types.Reference; | import org.apache.tools.ant.types.Reference; | ||||
import org.apache.tools.ant.types.XMLCatalog; | import org.apache.tools.ant.types.XMLCatalog; | ||||
import org.apache.tools.ant.util.FileUtils; | import org.apache.tools.ant.util.FileUtils; | ||||
import javax.xml.transform.URIResolver; | |||||
/** | /** | ||||
* Processes a set of XML documents via XSLT. This is | * Processes a set of XML documents via XSLT. This is | ||||
@@ -54,44 +54,25 @@ | |||||
package org.apache.tools.ant.types; | package org.apache.tools.ant.types; | ||||
/** | /** | ||||
* Helper class to handle the DTD and Entity nested elements. | |||||
* <p>Helper class to handle the DTD nested element. Instances of | |||||
* this class correspond to the <code>PUBLIC</code> catalog entry type | |||||
* of the <a | |||||
* href="http://oasis-open.org/committees/entity/spec-2001-08-06.html"> | |||||
* OASIS "Open Catalog" standard</a>.</p> | |||||
* | * | ||||
* <p>Possible Future Enhancement: Bring the Ant element name into | |||||
* conformance with the OASIS standard.</p> | |||||
* | |||||
* @see org.apache.xml.resolver.Catalog | |||||
* @author Conor MacNeill | * @author Conor MacNeill | ||||
* @author dIon Gillard | * @author dIon Gillard | ||||
* @author <a href="mailto:cstrong@arielpartners.com">Craeg Strong</a> | |||||
* @version $Id$ | |||||
*/ | */ | ||||
public class DTDLocation { | |||||
/** publicId of the dtd/entity */ | |||||
private String publicId = null; | |||||
/** location of the dtd/entity - a file/resource/URL */ | |||||
private String location = null; | |||||
/** | |||||
* @param publicId uniquely identifies the resource | |||||
*/ | |||||
public void setPublicId(String publicId) { | |||||
this.publicId = publicId; | |||||
} | |||||
public class DTDLocation extends ResourceLocation { | |||||
/** | |||||
* @param location the location of the resource associated with the | |||||
* publicId | |||||
*/ | |||||
public void setLocation(String location) { | |||||
this.location = location; | |||||
public DTDLocation() { | |||||
super("PUBLIC"); | |||||
} | } | ||||
/** | |||||
* @return the publicId | |||||
*/ | |||||
public String getPublicId() { | |||||
return publicId; | |||||
} | |||||
/** | |||||
* @return the location of the resource identified by the publicId | |||||
*/ | |||||
public String getLocation() { | |||||
return location; | |||||
} | |||||
} | } | ||||
@@ -0,0 +1,77 @@ | |||||
/* | |||||
* The Apache Software License, Version 1.1 | |||||
* | |||||
* Copyright (c) 2002 The Apache Software Foundation. All rights | |||||
* reserved. | |||||
* | |||||
* Redistribution and use in source and binary forms, with or without | |||||
* modification, are permitted provided that the following conditions | |||||
* are met: | |||||
* | |||||
* 1. Redistributions of source code must retain the above copyright | |||||
* notice, this list of conditions and the following disclaimer. | |||||
* | |||||
* 2. Redistributions in binary form must reproduce the above copyright | |||||
* notice, this list of conditions and the following disclaimer in | |||||
* the documentation and/or other materials provided with the | |||||
* distribution. | |||||
* | |||||
* 3. The end-user documentation included with the redistribution, if | |||||
* any, must include the following acknowlegement: | |||||
* "This product includes software developed by the | |||||
* Apache Software Foundation (http://www.apache.org/)." | |||||
* Alternately, this acknowlegement may appear in the software itself, | |||||
* if and wherever such third-party acknowlegements normally appear. | |||||
* | |||||
* 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||||
* Foundation" must not be used to endorse or promote products derived | |||||
* from this software without prior written permission. For written | |||||
* permission, please contact apache@apache.org. | |||||
* | |||||
* 5. Products derived from this software may not be called "Apache" | |||||
* nor may "Apache" appear in their names without prior written | |||||
* permission of the Apache Group. | |||||
* | |||||
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||||
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||||
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||||
* DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||||
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||||
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||||
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||||
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||||
* SUCH DAMAGE. | |||||
* ==================================================================== | |||||
* | |||||
* This software consists of voluntary contributions made by many | |||||
* individuals on behalf of the Apache Software Foundation. For more | |||||
* information on the Apache Software Foundation, please see | |||||
* <http://www.apache.org/>. | |||||
*/ | |||||
package org.apache.tools.ant.types; | |||||
/** | |||||
* Helper class to handle the Entity nested element. Instances of | |||||
* this class correspond to the <code>URI</code> catalog entry type of | |||||
* the <a | |||||
* href="http://oasis-open.org/committees/entity/spec-2001-08-06.html"> | |||||
* OASIS "Open Catalog" standard</a>.</p> | |||||
* | |||||
* <p>Possible Future Enhancement: Bring the Ant element name into | |||||
* conformance with the OASIS standard.</p> | |||||
* | |||||
* @see org.apache.xml.resolver.Catalog | |||||
* @author Conor MacNeill | |||||
* @author dIon Gillard | |||||
* @author <a href="mailto:cstrong@arielpartners.com">Craeg Strong</a> | |||||
*/ | |||||
public class EntityLocation extends ResourceLocation { | |||||
public EntityLocation() { | |||||
super("URI"); | |||||
} | |||||
} |
@@ -0,0 +1,165 @@ | |||||
/* | |||||
* The Apache Software License, Version 1.1 | |||||
* | |||||
* Copyright (c) 2002 The Apache Software Foundation. All rights | |||||
* reserved. | |||||
* | |||||
* Redistribution and use in source and binary forms, with or without | |||||
* modification, are permitted provided that the following conditions | |||||
* are met: | |||||
* | |||||
* 1. Redistributions of source code must retain the above copyright | |||||
* notice, this list of conditions and the following disclaimer. | |||||
* | |||||
* 2. Redistributions in binary form must reproduce the above copyright | |||||
* notice, this list of conditions and the following disclaimer in | |||||
* the documentation and/or other materials provided with the | |||||
* distribution. | |||||
* | |||||
* 3. The end-user documentation included with the redistribution, if | |||||
* any, must include the following acknowlegement: | |||||
* "This product includes software developed by the | |||||
* Apache Software Foundation (http://www.apache.org/)." | |||||
* Alternately, this acknowlegement may appear in the software itself, | |||||
* if and wherever such third-party acknowlegements normally appear. | |||||
* | |||||
* 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||||
* Foundation" must not be used to endorse or promote products derived | |||||
* from this software without prior written permission. For written | |||||
* permission, please contact apache@apache.org. | |||||
* | |||||
* 5. Products derived from this software may not be called "Apache" | |||||
* nor may "Apache" appear in their names without prior written | |||||
* permission of the Apache Group. | |||||
* | |||||
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||||
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||||
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||||
* DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||||
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||||
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||||
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||||
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||||
* SUCH DAMAGE. | |||||
* ==================================================================== | |||||
* | |||||
* This software consists of voluntary contributions made by many | |||||
* individuals on behalf of the Apache Software Foundation. For more | |||||
* information on the Apache Software Foundation, please see | |||||
* <http://www.apache.org/>. | |||||
*/ | |||||
package org.apache.tools.ant.types; | |||||
import java.net.URL; | |||||
/** | |||||
* <p>Helper class to handle the <code><dtd></code> and | |||||
* <code><entity></code> nested elements. These correspond to | |||||
* the <code>PUBLIC</code> and <code>URI</code> catalog entry types, | |||||
* respectively, as defined in the <a | |||||
* href="http://oasis-open.org/committees/entity/spec-2001-08-06.html"> | |||||
* OASIS "Open Catalog" standard</a>.</p> | |||||
* | |||||
* <p>Possible Future Enhancements: | |||||
* <ul> | |||||
* <li>Bring the Ant element names into conformance with the OASIS standard</li> | |||||
* <li>Add support for additional OASIS catalog entry types</li> | |||||
* </ul> | |||||
* </p> | |||||
* | |||||
* @see org.apache.xml.resolver.Catalog | |||||
* @author Conor MacNeill | |||||
* @author dIon Gillard | |||||
* @author <a href="mailto:cstrong@arielpartners.com">Craeg Strong</a> | |||||
* @version $Id$ | |||||
*/ | |||||
public class ResourceLocation { | |||||
//-- Fields ---------------------------------------------------------------- | |||||
/** | |||||
* name of the catalog entry type, as per OASIS spec. | |||||
*/ | |||||
protected String name = null; | |||||
/** publicId of the dtd/entity. */ | |||||
private String publicId = null; | |||||
/** location of the dtd/entity - a file/resource/URL. */ | |||||
private String location = null; | |||||
/** | |||||
* base URL of the dtd/entity, or null. If null, the Ant project | |||||
* basedir is assumed. If the location specifies a relative | |||||
* URL/pathname, it is resolved using the base. The default base | |||||
* for an external catalog file is the directory in which it is | |||||
* located. | |||||
*/ | |||||
private String base = null; | |||||
//-- Methods --------------------------------------------------------------- | |||||
protected ResourceLocation(String name) { | |||||
this.name = name; | |||||
} | |||||
/** | |||||
* @param publicId uniquely identifies the resource. | |||||
*/ | |||||
public void setPublicId(String publicId) { | |||||
this.publicId = publicId; | |||||
} | |||||
/** | |||||
* @param location the location of the resource associated with the | |||||
* publicId. | |||||
*/ | |||||
public void setLocation(String location) { | |||||
this.location = location; | |||||
} | |||||
/** | |||||
* @param base the base URL of the resource associated with the | |||||
* publicId. If the location specifies a relative URL/pathname, | |||||
* it is resolved using the base. The default base for an | |||||
* external catalog file is the directory in which it is located. | |||||
*/ | |||||
public void setBase(String base) { | |||||
this.base = base; | |||||
} | |||||
/** | |||||
* @return the publicId of the resource. | |||||
*/ | |||||
public String getPublicId() { | |||||
return publicId; | |||||
} | |||||
/** | |||||
* @return the location of the resource identified by the publicId. | |||||
*/ | |||||
public String getLocation() { | |||||
return location; | |||||
} | |||||
/** | |||||
* @return the base of the resource identified by the publicId. | |||||
*/ | |||||
public String getBase() { | |||||
return base; | |||||
} | |||||
/** | |||||
* @return the name of the catalog entry type. Currently this is | |||||
* one of <code>PUBLIC</code> or <code>URI</code>. | |||||
* | |||||
* @see org.apache.xml.resolver.Catalog | |||||
*/ | |||||
public String getName() { | |||||
return name; | |||||
} | |||||
} //-- ResourceLocation |
@@ -54,6 +54,8 @@ | |||||
package org.apache.tools.ant.types; | package org.apache.tools.ant.types; | ||||
import java.lang.reflect.Method; | |||||
import java.io.File; | import java.io.File; | ||||
import java.io.FileInputStream; | import java.io.FileInputStream; | ||||
import java.io.FileNotFoundException; | import java.io.FileNotFoundException; | ||||
@@ -72,6 +74,7 @@ import javax.xml.transform.URIResolver; | |||||
import javax.xml.transform.sax.SAXSource; | import javax.xml.transform.sax.SAXSource; | ||||
import org.apache.tools.ant.AntClassLoader; | import org.apache.tools.ant.AntClassLoader; | ||||
import org.apache.tools.ant.BuildException; | import org.apache.tools.ant.BuildException; | ||||
import org.apache.tools.ant.DirectoryScanner; | |||||
import org.apache.tools.ant.Project; | import org.apache.tools.ant.Project; | ||||
import org.apache.tools.ant.util.FileUtils; | import org.apache.tools.ant.util.FileUtils; | ||||
import org.xml.sax.EntityResolver; | import org.xml.sax.EntityResolver; | ||||
@@ -138,13 +141,14 @@ import org.xml.sax.XMLReader; | |||||
* @author <a href="mailto:cstrong@arielpartners.com">Craeg Strong</a> | * @author <a href="mailto:cstrong@arielpartners.com">Craeg Strong</a> | ||||
* @version $Id$ | * @version $Id$ | ||||
*/ | */ | ||||
public class XMLCatalog extends DataType implements Cloneable, EntityResolver, URIResolver { | |||||
public class XMLCatalog extends DataType | |||||
implements Cloneable, EntityResolver, URIResolver { | |||||
/** File utilities instance */ | /** File utilities instance */ | ||||
private FileUtils fileUtils = FileUtils.newFileUtils(); | private FileUtils fileUtils = FileUtils.newFileUtils(); | ||||
//-- Fields ---------------------------------------------------------------- | //-- Fields ---------------------------------------------------------------- | ||||
/** holds dtd/entity objects until needed */ | |||||
/** Holds dtd/entity objects and catalog filesets until needed. */ | |||||
private Vector elements = new Vector(); | private Vector elements = new Vector(); | ||||
/** | /** | ||||
@@ -152,6 +156,14 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
*/ | */ | ||||
private Path classpath; | private Path classpath; | ||||
/** | |||||
* The name of the bridge to the Apache xml-commons resolver | |||||
* class, used to determine whether resolver.jar is present in the | |||||
* classpath. | |||||
*/ | |||||
public static final String APACHE_RESOLVER | |||||
= "org.apache.tools.ant.types.resolver.ApacheCatalogResolver"; | |||||
//-- Methods --------------------------------------------------------------- | //-- Methods --------------------------------------------------------------- | ||||
public XMLCatalog() { | public XMLCatalog() { | ||||
@@ -159,9 +171,11 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
} | } | ||||
/** | /** | ||||
* Returns the elements of the catalog - DTDLocation objects. | |||||
* Returns the elements of the catalog - ResolverLocation and FileSet | |||||
* objects. | |||||
* | * | ||||
* @return the elements of the catalog - DTDLocation objects | |||||
* @return the elements of the catalog - ResolverLocation and FileSet | |||||
* objects | |||||
*/ | */ | ||||
private Vector getElements() { | private Vector getElements() { | ||||
return elements; | return elements; | ||||
@@ -177,14 +191,18 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
} | } | ||||
/** | /** | ||||
* Set the list of DTDLocation objects in the catalog. Not | |||||
* allowed if this catalog is itself a reference to another | |||||
* catalog -- that is, a catalog cannot both refer to another | |||||
* <em>and</em> contain elements or other attributes. | |||||
* Set the list of ResourceLocation objects and FileSets in the catalog. | |||||
* Not allowed if this catalog is itself a reference to another catalog -- | |||||
* that is, a catalog cannot both refer to another <em>and</em> contain | |||||
* elements or other attributes. | |||||
* | * | ||||
* @param aVector the new list of DTD Locations to use in the catalog. | |||||
* @param aVector the new list of ResourceLocations and FileSets | |||||
* to use in the catalog. | |||||
*/ | */ | ||||
private void setElements(Vector aVector) { | private void setElements(Vector aVector) { | ||||
if (isReference()) { | |||||
throw noChildrenAllowed(); | |||||
} | |||||
elements = aVector; | elements = aVector; | ||||
} | } | ||||
@@ -237,6 +255,22 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
setChecked( false ); | setChecked( false ); | ||||
} | } | ||||
/** Creates the nested <code><catalogfiles></code> element. Not | |||||
* allowed if this catalog is itself a reference to another catalog -- that | |||||
* is, a catalog cannot both refer to another <em>and</em> contain elements | |||||
* or other attributes. | |||||
* | |||||
* @param fs the fileset of external catalogs. | |||||
* @exception BuildException | |||||
* if this is a reference and no nested elements are allowed. | |||||
*/ | |||||
public void addCatalogfiles(FileSet fs) throws BuildException { | |||||
if (isReference()) { | |||||
throw noChildrenAllowed(); | |||||
} | |||||
getElements().addElement(fs); | |||||
} | |||||
/** | /** | ||||
* Creates the nested <code><dtd></code> element. Not | * Creates the nested <code><dtd></code> element. Not | ||||
* allowed if this catalog is itself a reference to another | * allowed if this catalog is itself a reference to another | ||||
@@ -248,7 +282,7 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
* @exception BuildException if this is a reference and no nested | * @exception BuildException if this is a reference and no nested | ||||
* elements are allowed. | * elements are allowed. | ||||
*/ | */ | ||||
public void addDTD(DTDLocation dtd) throws BuildException { | |||||
public void addDTD(ResourceLocation dtd) throws BuildException { | |||||
if (isReference()) { | if (isReference()) { | ||||
throw noChildrenAllowed(); | throw noChildrenAllowed(); | ||||
} | } | ||||
@@ -256,6 +290,9 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
getElements().addElement(dtd); | getElements().addElement(dtd); | ||||
setChecked( false ); | setChecked( false ); | ||||
} | } | ||||
public void addDTD(DTDLocation dtd) throws BuildException { | |||||
addDTD((ResourceLocation)dtd); | |||||
} | |||||
/** | /** | ||||
* Creates the nested <code><entity></code> element. Not | * Creates the nested <code><entity></code> element. Not | ||||
@@ -263,13 +300,34 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
* catalog -- that is, a catalog cannot both refer to another | * catalog -- that is, a catalog cannot both refer to another | ||||
* <em>and</em> contain elements or other attributes. | * <em>and</em> contain elements or other attributes. | ||||
* | * | ||||
* @param dtd the information about the URI resource mapping to be | |||||
* added to the catalog | |||||
* @param entity the information about the URI resource mapping to | |||||
* be added to the catalog. | |||||
* @exception BuildException if this is a reference and no nested | * @exception BuildException if this is a reference and no nested | ||||
* elements are allowed. | * elements are allowed. | ||||
*/ | */ | ||||
public void addEntity(DTDLocation dtd) throws BuildException { | |||||
addDTD(dtd); | |||||
public void addEntity(EntityLocation entity) throws BuildException { | |||||
if (isReference()) { | |||||
throw noChildrenAllowed(); | |||||
} | |||||
getElements().addElement(entity); | |||||
} | |||||
/** | |||||
* Creates the nested <code><entity></code> element. Not | |||||
* allowed if this catalog is itself a reference to another | |||||
* catalog -- that is, a catalog cannot both refer to another | |||||
* <em>and</em> contain elements or other attributes. | |||||
* | |||||
* @param entity the information about the URI resource mapping to | |||||
* be added to the catalog. | |||||
* @exception BuildException if this is a reference and no nested | |||||
* elements are allowed. | |||||
*/ | |||||
public void addEntity(DTDLocation entity) throws BuildException { | |||||
if (isReference()) { | |||||
throw noChildrenAllowed(); | |||||
} | |||||
getElements().addElement(entity); | |||||
} | } | ||||
/** | /** | ||||
@@ -337,17 +395,18 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
public InputSource resolveEntity(String publicId, String systemId) | public InputSource resolveEntity(String publicId, String systemId) | ||||
throws SAXException, IOException { | throws SAXException, IOException { | ||||
if (!isChecked()) { | |||||
// make sure we don't have a circular reference here | |||||
Stack stk = new Stack(); | |||||
stk.push(this); | |||||
dieOnCircularReference(stk, getProject()); | |||||
} | |||||
if (!isChecked()) { | |||||
// make sure we don't have a circular reference here | |||||
Stack stk = new Stack(); | |||||
stk.push(this); | |||||
dieOnCircularReference(stk, getProject()); | |||||
} | |||||
log("resolveEntity: '" + publicId + "': '" + systemId + "'", | log("resolveEntity: '" + publicId + "': '" + systemId + "'", | ||||
Project.MSG_DEBUG); | Project.MSG_DEBUG); | ||||
InputSource inputSource = resolveEntityImpl(publicId ); | |||||
InputSource inputSource = | |||||
getCatalogResolver().resolveEntity(publicId, systemId); | |||||
if (inputSource == null) { | if (inputSource == null) { | ||||
log("No matching catalog entry found, parser will use: '" + | log("No matching catalog entry found, parser will use: '" + | ||||
@@ -365,12 +424,12 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
public Source resolve(String href, String base) | public Source resolve(String href, String base) | ||||
throws TransformerException { | throws TransformerException { | ||||
if (!isChecked()) { | |||||
// make sure we don't have a circular reference here | |||||
Stack stk = new Stack(); | |||||
stk.push(this); | |||||
dieOnCircularReference(stk, getProject()); | |||||
} | |||||
if (!isChecked()) { | |||||
// make sure we don't have a circular reference here | |||||
Stack stk = new Stack(); | |||||
stk.push(this); | |||||
dieOnCircularReference(stk, getProject()); | |||||
} | |||||
SAXSource source = null; | SAXSource source = null; | ||||
@@ -389,11 +448,11 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
// | // | ||||
source = new SAXSource(); | source = new SAXSource(); | ||||
try | try | ||||
{ | |||||
URL baseURL = new URL(base); | |||||
URL url = (uri.length() == 0 ? baseURL : new URL(baseURL, uri)); | |||||
source.setInputSource(new InputSource(url.toString())); | |||||
} | |||||
{ | |||||
URL baseURL = new URL(base); | |||||
URL url = (uri.length() == 0 ? baseURL : new URL(baseURL, uri)); | |||||
source.setInputSource(new InputSource(url.toString())); | |||||
} | |||||
catch (MalformedURLException ex) { | catch (MalformedURLException ex) { | ||||
// At this point we are probably in failure mode, but | // At this point we are probably in failure mode, but | ||||
// try to use the bare URI as a last gasp | // try to use the bare URI as a last gasp | ||||
@@ -406,25 +465,67 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
} | } | ||||
/** | /** | ||||
* Find a DTDLocation instance for the given publicId. | |||||
* The instance of the CatalogResolver strategy to use. | |||||
*/ | |||||
private static CatalogResolver catalogResolver = null; | |||||
/** | |||||
* Factory method for creating the appropriate CatalogResolver | |||||
* strategy implementation. | |||||
* <p> Until we query the classpath, we don't know whether the Apache | |||||
* resolver (Norm Walsh's library from xml-commons) is available or not. | |||||
* This method determines whether the library is available and creates the | |||||
* appropriate implementation of CatalogResolver based on the answer.</p> | |||||
* <p>This is an application of the Gang of Four Strategy Pattern | |||||
* combined with Template Method.</p> | |||||
* | * | ||||
* @param publicId the publicId of the Resource for which local information | * @param publicId the publicId of the Resource for which local information | ||||
* is required | * is required | ||||
* @return a DTDLocation instance with information on the local location | |||||
* @return a ResourceLocation instance with information on the local location | |||||
* of the Resource or null if no such information is available | * of the Resource or null if no such information is available | ||||
*/ | */ | ||||
private DTDLocation findMatchingEntry(String publicId) { | |||||
Enumeration elements = getElements().elements(); | |||||
DTDLocation element = null; | |||||
while (elements.hasMoreElements()) { | |||||
element = (DTDLocation) elements.nextElement(); | |||||
if (element.getPublicId().equals(publicId)) { | |||||
return element; | |||||
private CatalogResolver getCatalogResolver() { | |||||
if (catalogResolver == null) { | |||||
AntClassLoader loader = null; | |||||
loader = new AntClassLoader(project, Path.systemClasspath); | |||||
try { | |||||
Class clazz = loader.forceLoadSystemClass(APACHE_RESOLVER); | |||||
Object obj = clazz.newInstance(); | |||||
// | |||||
// Success! The xml-commons resolver library is | |||||
// available, so use it. | |||||
// | |||||
catalogResolver = new ApacheResolver(clazz, obj); | |||||
} | |||||
catch (Throwable ex) { | |||||
// | |||||
// The xml-commons resolver library is not | |||||
// available, so we can't use it. | |||||
// | |||||
catalogResolver = new InternalResolver(); | |||||
// | |||||
// If any <catalogfiles> are specified, warn that they | |||||
// will be ignored. | |||||
// | |||||
Enumeration enum = getElements().elements(); | |||||
while (enum.hasMoreElements()) { | |||||
Object o = enum.nextElement(); | |||||
if (o instanceof FileSet) { | |||||
log("Warning: External catalogfiles will be ignored", | |||||
Project.MSG_WARN); | |||||
break; | |||||
} | |||||
} | |||||
} | } | ||||
} | } | ||||
return null; | |||||
return catalogResolver; | |||||
} | } | ||||
/** | /** | ||||
* <p>This is called from the URIResolver to set an EntityResolver | * <p>This is called from the URIResolver to set an EntityResolver | ||||
* on the SAX parser to be used for new XML documents that are | * on the SAX parser to be used for new XML documents that are | ||||
@@ -464,6 +565,29 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
source.setXMLReader(reader); | source.setXMLReader(reader); | ||||
} | } | ||||
/** | |||||
* Find a ResourceLocation instance for the given publicId. | |||||
* | |||||
* @param publicId the publicId of the Resource for which local information is | |||||
* required. | |||||
* @return a ResourceLocation instance with information on the local location | |||||
* of the Resource or null if no such information is available. | |||||
*/ | |||||
private ResourceLocation findMatchingEntry(String publicId) { | |||||
Enumeration enum = getElements().elements(); | |||||
ResourceLocation element = null; | |||||
while (enum.hasMoreElements()) { | |||||
Object o = enum.nextElement(); | |||||
if (o instanceof ResourceLocation) { | |||||
element = (ResourceLocation)o; | |||||
if (element.getPublicId().equals(publicId)) { | |||||
return element; | |||||
} | |||||
} | |||||
} | |||||
return null; | |||||
} | |||||
/** | /** | ||||
* Utility method to remove trailing fragment from a URI. | * Utility method to remove trailing fragment from a URI. | ||||
* For example, | * For example, | ||||
@@ -484,17 +608,17 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
} | } | ||||
/** | /** | ||||
* Utility method to lookup a DTDLocation in the filesystem. | |||||
* Utility method to lookup a ResourceLocation in the filesystem. | |||||
* | * | ||||
* @return An InputSource for reading the file, or <code>null</code> | * @return An InputSource for reading the file, or <code>null</code> | ||||
* if the file does not exist or is not readable. | * if the file does not exist or is not readable. | ||||
*/ | */ | ||||
private InputSource filesystemLookup(DTDLocation matchingEntry) { | |||||
private InputSource filesystemLookup(ResourceLocation matchingEntry) { | |||||
String uri = matchingEntry.getLocation(); | String uri = matchingEntry.getLocation(); | ||||
// | // | ||||
// The DTDLocation may specify a relative path for its | |||||
// The ResourceLocation may specify a relative path for its | |||||
// location attribute. This is resolved using the appropriate | // location attribute. This is resolved using the appropriate | ||||
// base. | // base. | ||||
// | // | ||||
@@ -522,12 +646,12 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
} | } | ||||
/** | /** | ||||
* Utility method to lookup a DTDLocation in the classpath. | |||||
* Utility method to lookup a ResourceLocation in the classpath. | |||||
* | * | ||||
* @return An InputSource for reading the resource, or <code>null</code> | * @return An InputSource for reading the resource, or <code>null</code> | ||||
* if the resource does not exist in the classpath or is not readable. | * if the resource does not exist in the classpath or is not readable. | ||||
*/ | */ | ||||
private InputSource classpathLookup(DTDLocation matchingEntry) { | |||||
private InputSource classpathLookup(ResourceLocation matchingEntry) { | |||||
InputSource source = null; | InputSource source = null; | ||||
@@ -557,7 +681,7 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
} | } | ||||
/** | /** | ||||
* Utility method to lookup a DTDLocation in URL-space. | |||||
* Utility method to lookup a ResourceLocation in URL-space. | |||||
* | * | ||||
* @return An InputSource for reading the resource, or <code>null</code> | * @return An InputSource for reading the resource, or <code>null</code> | ||||
* if the resource does not identify a valid URL or is not readable. | * if the resource does not identify a valid URL or is not readable. | ||||
@@ -602,31 +726,33 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
/** | /** | ||||
* Implements the guts of the resolveEntity() lookup strategy. | * Implements the guts of the resolveEntity() lookup strategy. | ||||
*/ | */ | ||||
private InputSource resolveEntityImpl(String publicId) { | |||||
/* | |||||
private InputSource resolveEntityImpl(String publicId) { | |||||
InputSource result = null; | |||||
InputSource result = null; | |||||
DTDLocation matchingEntry = findMatchingEntry(publicId); | |||||
ResourceLocation matchingEntry = findMatchingEntry(publicId); | |||||
if (matchingEntry != null) { | |||||
if (matchingEntry != null) { | |||||
log("Matching catalog entry found for publicId: '" + | |||||
matchingEntry.getPublicId() + "' location: '" + | |||||
matchingEntry.getLocation() + "'", | |||||
Project.MSG_DEBUG); | |||||
log("Matching catalog entry found for publicId: '" + | |||||
matchingEntry.getPublicId() + "' location: '" + | |||||
matchingEntry.getLocation() + "'", | |||||
Project.MSG_DEBUG); | |||||
result = filesystemLookup(matchingEntry); | |||||
result = filesystemLookup(matchingEntry); | |||||
if (result == null) { | |||||
result = classpathLookup(matchingEntry); | |||||
} | |||||
if (result == null) { | |||||
result = classpathLookup(matchingEntry); | |||||
} | |||||
if (result == null) { | |||||
result = urlLookup(matchingEntry.getLocation(), null); | |||||
} | |||||
} | |||||
return result; | |||||
} | |||||
if (result == null) { | |||||
result = urlLookup(matchingEntry.getLocation(), null); | |||||
} | |||||
} | |||||
return result; | |||||
} | |||||
*/ | |||||
/** | /** | ||||
* Implements the guts of the resolve() lookup strategy. | * Implements the guts of the resolve() lookup strategy. | ||||
@@ -636,7 +762,7 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
SAXSource result = null; | SAXSource result = null; | ||||
InputSource source = null; | InputSource source = null; | ||||
DTDLocation matchingEntry = findMatchingEntry(href); | |||||
ResourceLocation matchingEntry = findMatchingEntry(href); | |||||
if (matchingEntry != null) { | if (matchingEntry != null) { | ||||
@@ -661,4 +787,312 @@ public class XMLCatalog extends DataType implements Cloneable, EntityResolver, U | |||||
} | } | ||||
return result; | return result; | ||||
} | } | ||||
/** | |||||
* Interface implemented by both the InternalResolver strategy and | |||||
* the ApacheResolver strategy. | |||||
*/ | |||||
private interface CatalogResolver extends URIResolver, EntityResolver { | |||||
InputSource resolveEntity(String publicId, String systemId); | |||||
Source resolve(String href, String base) throws TransformerException; | |||||
} | |||||
/** | |||||
* The InternalResolver strategy is used if the Apache resolver | |||||
* library (Norm Walsh's library from xml-commons) is not | |||||
* available. In this case, external catalog files will be | |||||
* ignored. | |||||
* | |||||
*/ | |||||
private class InternalResolver implements CatalogResolver { | |||||
public InternalResolver() { | |||||
log("Apache resolver library not found, internal resolver will be used", | |||||
Project.MSG_INFO); | |||||
} | |||||
public InputSource resolveEntity(String publicId, | |||||
String systemId) { | |||||
InputSource result = null; | |||||
ResourceLocation matchingEntry = findMatchingEntry(publicId); | |||||
if (matchingEntry != null) { | |||||
log("Matching catalog entry found for publicId: '" + | |||||
matchingEntry.getPublicId() + "' location: '" + | |||||
matchingEntry.getLocation() + "'", | |||||
Project.MSG_DEBUG); | |||||
result = filesystemLookup(matchingEntry); | |||||
if (result == null) { | |||||
result = classpathLookup(matchingEntry); | |||||
} | |||||
if (result == null) { | |||||
result = urlLookup(matchingEntry.getLocation(), null); | |||||
} | |||||
} | |||||
return result; | |||||
} | |||||
public Source resolve(String href, String base) | |||||
throws TransformerException { | |||||
SAXSource result = null; | |||||
InputSource source = null; | |||||
ResourceLocation matchingEntry = findMatchingEntry(href); | |||||
if (matchingEntry != null) { | |||||
log("Matching catalog entry found for uri: '" + | |||||
matchingEntry.getPublicId() + "' location: '" + | |||||
matchingEntry.getLocation() + "'", | |||||
Project.MSG_DEBUG); | |||||
source = filesystemLookup(matchingEntry); | |||||
if (source == null) { | |||||
source = classpathLookup(matchingEntry); | |||||
} | |||||
if (source == null) { | |||||
source = urlLookup(matchingEntry.getLocation(), base); | |||||
} | |||||
if (source != null) { | |||||
result = new SAXSource(source); | |||||
} | |||||
} | |||||
return result; | |||||
} | |||||
} | |||||
/** | |||||
* The ApacheResolver strategy is used if the Apache resolver | |||||
* library (Norm Walsh's library from xml-commons) is available in | |||||
* the classpath. The ApacheResolver is a essentially a superset | |||||
* of the InternalResolver. | |||||
* | |||||
*/ | |||||
private class ApacheResolver implements CatalogResolver { | |||||
private Method setXMLCatalog = null; | |||||
private Method parseCatalog = null; | |||||
private Method resolveEntity = null; | |||||
private Method resolve = null; | |||||
/** The instance of the ApacheCatalogResolver bridge class */ | |||||
private Object resolverImpl = null; | |||||
private boolean externalCatalogsProcessed = false; | |||||
public ApacheResolver(Class resolverImplClass, | |||||
Object resolverImpl) { | |||||
this.resolverImpl = resolverImpl; | |||||
// | |||||
// Get Method instances for each of the methods we need to | |||||
// call on the resolverImpl using reflection. We can't | |||||
// call them directly, because they require on the | |||||
// xml-commons resolver library which may not be available | |||||
// in the classpath. | |||||
// | |||||
try { | |||||
setXMLCatalog = | |||||
resolverImplClass.getMethod("setXMLCatalog", | |||||
new Class[] | |||||
{ XMLCatalog.class }); | |||||
parseCatalog = | |||||
resolverImplClass.getMethod("parseCatalog", | |||||
new Class[] | |||||
{ String.class }); | |||||
resolveEntity = | |||||
resolverImplClass.getMethod("resolveEntity", | |||||
new Class[] | |||||
{ String.class, String.class }); | |||||
resolve = | |||||
resolverImplClass.getMethod("resolve", | |||||
new Class[] | |||||
{ String.class, String.class }); | |||||
} | |||||
catch (NoSuchMethodException ex) { | |||||
throw new BuildException(ex); | |||||
} | |||||
log("Apache resolver library found, xml-commons resolver will be used", | |||||
Project.MSG_INFO); | |||||
} | |||||
public InputSource resolveEntity(String publicId, | |||||
String systemId) { | |||||
InputSource result = null; | |||||
processExternalCatalogs(); | |||||
ResourceLocation matchingEntry = findMatchingEntry(publicId); | |||||
if (matchingEntry != null) { | |||||
log("Matching catalog entry found for publicId: '" + | |||||
matchingEntry.getPublicId() + "' location: '" + | |||||
matchingEntry.getLocation() + "'", | |||||
Project.MSG_DEBUG); | |||||
result = filesystemLookup(matchingEntry); | |||||
if (result == null) { | |||||
result = classpathLookup(matchingEntry); | |||||
} | |||||
if (result == null) { | |||||
try { | |||||
result = | |||||
(InputSource)resolveEntity.invoke(resolverImpl, | |||||
new Object[] | |||||
{ publicId, systemId }); | |||||
} | |||||
catch (Exception ex) { | |||||
throw new BuildException(ex); | |||||
} | |||||
} | |||||
} | |||||
else { | |||||
// | |||||
// We didn't match a ResourceLocation, but since we | |||||
// only support PUBLIC and URI entry types, it is | |||||
// still possible that there is another entry in an | |||||
// external catalog that will match. We call Apache | |||||
// resolver's resolveEntity method to cover this | |||||
// possibility. | |||||
// | |||||
try { | |||||
result = | |||||
(InputSource)resolveEntity.invoke(resolverImpl, | |||||
new Object[] | |||||
{ publicId, systemId }); | |||||
} | |||||
catch (Exception ex) { | |||||
throw new BuildException(ex); | |||||
} | |||||
} | |||||
return result; | |||||
} | |||||
public Source resolve(String href, String base) | |||||
throws TransformerException { | |||||
SAXSource result = null; | |||||
InputSource source = null; | |||||
processExternalCatalogs(); | |||||
ResourceLocation matchingEntry = findMatchingEntry(href); | |||||
if (matchingEntry != null) { | |||||
log("Matching catalog entry found for uri: '" + | |||||
matchingEntry.getPublicId() + "' location: '" + | |||||
matchingEntry.getLocation() + "'", | |||||
Project.MSG_DEBUG); | |||||
source = filesystemLookup(matchingEntry); | |||||
if (source == null) { | |||||
source = classpathLookup(matchingEntry); | |||||
} | |||||
if (source != null) { | |||||
result = new SAXSource(source); | |||||
} else { | |||||
try { | |||||
result = | |||||
(SAXSource)resolve.invoke(resolverImpl, | |||||
new Object[] | |||||
{ href, base }); | |||||
} | |||||
catch (Exception ex) { | |||||
throw new BuildException(ex); | |||||
} | |||||
} | |||||
} | |||||
else { | |||||
// | |||||
// We didn't match a ResourceLocation, but since we | |||||
// only support PUBLIC and URI entry types, it is | |||||
// still possible that there is another entry in an | |||||
// external catalog that will match. We call Apache | |||||
// resolver's resolveEntity method to cover this | |||||
// possibility. | |||||
// | |||||
try { | |||||
result = | |||||
(SAXSource)resolve.invoke(resolverImpl, | |||||
new Object[] | |||||
{ href, base }); | |||||
} | |||||
catch (Exception ex) { | |||||
throw new BuildException(ex); | |||||
} | |||||
} | |||||
return result; | |||||
} | |||||
/** | |||||
* Process each external catalog file specified in a | |||||
* <code><catalogfiles></code> FileSet. It will be | |||||
* parsed by the resolver library, and the individual elements | |||||
* will be added back to us (that is, the controlling | |||||
* XMLCatalog instance) via a callback mechanism. | |||||
*/ | |||||
private void processExternalCatalogs() { | |||||
if (externalCatalogsProcessed == false) { | |||||
try { | |||||
setXMLCatalog.invoke(resolverImpl, | |||||
new Object[] | |||||
{ XMLCatalog.this }); | |||||
} | |||||
catch (Exception ex) { | |||||
throw new BuildException(ex); | |||||
} | |||||
Enumeration enum = getElements().elements(); | |||||
while (enum.hasMoreElements()) { | |||||
Object o = enum.nextElement(); | |||||
if (o instanceof FileSet) { | |||||
FileSet fs = (FileSet)o; | |||||
DirectoryScanner ds = | |||||
fs.getDirectoryScanner(getProject()); | |||||
String[] files = ds.getIncludedFiles(); | |||||
for (int i = 0; i < files.length; i++) { | |||||
File catFile = new File(ds.getBasedir(), files[i]); | |||||
try { | |||||
parseCatalog.invoke(resolverImpl, | |||||
new Object[] | |||||
{ catFile.getPath() }); | |||||
} | |||||
catch (Exception ex) { | |||||
throw new BuildException(ex); | |||||
} | |||||
} | |||||
} | |||||
} | |||||
} | |||||
externalCatalogsProcessed = true; | |||||
} | |||||
} | |||||
} | } |
@@ -0,0 +1,155 @@ | |||||
/* | |||||
* The Apache Software License, Version 1.1 | |||||
* | |||||
* Copyright (c) 2002 The Apache Software Foundation. All rights | |||||
* reserved. | |||||
* | |||||
* Redistribution and use in source and binary forms, with or without | |||||
* modification, are permitted provided that the following conditions | |||||
* are met: | |||||
* | |||||
* 1. Redistributions of source code must retain the above copyright | |||||
* notice, this list of conditions and the following disclaimer. | |||||
* | |||||
* 2. Redistributions in binary form must reproduce the above copyright | |||||
* notice, this list of conditions and the following disclaimer in | |||||
* the documentation and/or other materials provided with the | |||||
* distribution. | |||||
* | |||||
* 3. The end-user documentation included with the redistribution, if | |||||
* any, must include the following acknowlegement: | |||||
* "This product includes software developed by the | |||||
* Apache Software Foundation (http://www.apache.org/)." | |||||
* Alternately, this acknowlegement may appear in the software itself, | |||||
* if and wherever such third-party acknowlegements normally appear. | |||||
* | |||||
* 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||||
* Foundation" must not be used to endorse or promote products derived | |||||
* from this software without prior written permission. For written | |||||
* permission, please contact apache@apache.org. | |||||
* | |||||
* 5. Products derived from this software may not be called "Apache" | |||||
* nor may "Apache" appear in their names without prior written | |||||
* permission of the Apache Group. | |||||
* | |||||
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||||
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||||
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||||
* DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||||
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||||
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||||
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||||
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||||
* SUCH DAMAGE. | |||||
* ==================================================================== | |||||
* | |||||
* This software consists of voluntary contributions made by many | |||||
* individuals on behalf of the Apache Software Foundation. For more | |||||
* information on the Apache Software Foundation, please see | |||||
* <http://www.apache.org/>. | |||||
*/ | |||||
package org.apache.tools.ant.types.resolver; | |||||
import org.apache.xml.resolver.Catalog; | |||||
import org.apache.xml.resolver.CatalogEntry; | |||||
import org.apache.xml.resolver.helpers.Debug; | |||||
import org.apache.xml.resolver.helpers.PublicId; | |||||
/** | |||||
* This class extends the Catalog class provided by Norman Walsh's | |||||
* resolver library in xml-commons in order to add classpath entity | |||||
* and URI resolution. Since XMLCatalog already does classpath | |||||
* resolution, we simply add all CatalogEntry instances back to the | |||||
* controlling XMLCatalog instance. This is done via a callback | |||||
* mechanism. ApacheCatalog is <em>only</em> used for external | |||||
* catalog files. Inline entries (currently <code><dtd></code> | |||||
* and <code><entity></code>) are not added to ApacheCatalog. | |||||
* See XMLCatalog.java for the details of the entity and URI | |||||
* resolution algorithms. | |||||
* | |||||
* @see org.apache.tools.ant.types.XMLCatalog.CatalogResolver | |||||
* @author <a href="mailto:cstrong@arielpartners.com">Craeg Strong</a> | |||||
* @version $Id$ | |||||
*/ | |||||
public class ApacheCatalog extends Catalog { | |||||
/** The resolver object to callback. */ | |||||
private ApacheCatalogResolver resolver = null; | |||||
/** | |||||
* <p>Create a new ApacheCatalog instance.</p> | |||||
* | |||||
* <p>This method overrides the superclass method of the same name | |||||
* in order to set the resolver object for callbacks. The reason | |||||
* we have to do this is that internally Catalog creates a new | |||||
* instance of itself for each external catalog file processed. | |||||
* That is, if two external catalog files are processed, there | |||||
* will be a total of two ApacheCatalog instances, and so on.</p> | |||||
*/ | |||||
protected Catalog newCatalog() { | |||||
ApacheCatalog cat = (ApacheCatalog)super.newCatalog(); | |||||
cat.setResolver(resolver); | |||||
return cat; | |||||
} | |||||
/** Set the resolver object to callback. */ | |||||
public void setResolver(ApacheCatalogResolver resolver) { | |||||
this.resolver = resolver; | |||||
} | |||||
/** | |||||
* <p>This method overrides the superclass method of the same name | |||||
* in order to add catalog entries back to the controlling | |||||
* XMLCatalog instance. In this way, we can add classpath lookup | |||||
* for these entries.</p> | |||||
* | |||||
* <p>When we add an external catalog file, the entries inside it | |||||
* get parsed by this method. Therefore, we override it to add | |||||
* each of them back to the controlling XMLCatalog instance. This | |||||
* is done by performing a callback to the ApacheCatalogResolver, | |||||
* which in turn calls the XMLCatalog.</p> | |||||
* | |||||
* <p>XMLCatalog currently only understands <code>PUBLIC</code> | |||||
* and <code>URI</code> entry types, so we ignore the other types.</p> | |||||
* | |||||
* @param entry The CatalogEntry to process. | |||||
*/ | |||||
public void addEntry(CatalogEntry entry) { | |||||
int type = entry.getEntryType(); | |||||
if (type == PUBLIC) { | |||||
String publicid = PublicId.normalize(entry.getEntryArg(0)); | |||||
String systemid = normalizeURI(entry.getEntryArg(1)); | |||||
if (resolver == null) { | |||||
Debug.message(1, "Internal Error: null ApacheCatalogResolver"); | |||||
} | |||||
else { | |||||
resolver.addPublicEntry(publicid, systemid, base.toExternalForm()); | |||||
} | |||||
} else if (type == URI) { | |||||
String uri = normalizeURI(entry.getEntryArg(0)); | |||||
String altURI = normalizeURI(entry.getEntryArg(1)); | |||||
if (resolver == null) { | |||||
Debug.message(1, "Internal Error: null ApacheCatalogResolver"); | |||||
} | |||||
else { | |||||
resolver.addURIEntry(uri, altURI, base.toExternalForm()); | |||||
} | |||||
} | |||||
super.addEntry(entry); | |||||
} | |||||
} //- ApacheCatalog |
@@ -0,0 +1,207 @@ | |||||
/* | |||||
* The Apache Software License, Version 1.1 | |||||
* | |||||
* Copyright (c) 2002 The Apache Software Foundation. All rights | |||||
* reserved. | |||||
* | |||||
* Redistribution and use in source and binary forms, with or without | |||||
* modification, are permitted provided that the following conditions | |||||
* are met: | |||||
* | |||||
* 1. Redistributions of source code must retain the above copyright | |||||
* notice, this list of conditions and the following disclaimer. | |||||
* | |||||
* 2. Redistributions in binary form must reproduce the above copyright | |||||
* notice, this list of conditions and the following disclaimer in | |||||
* the documentation and/or other materials provided with the | |||||
* distribution. | |||||
* | |||||
* 3. The end-user documentation included with the redistribution, if | |||||
* any, must include the following acknowlegement: | |||||
* "This product includes software developed by the | |||||
* Apache Software Foundation (http://www.apache.org/)." | |||||
* Alternately, this acknowlegement may appear in the software itself, | |||||
* if and wherever such third-party acknowlegements normally appear. | |||||
* | |||||
* 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||||
* Foundation" must not be used to endorse or promote products derived | |||||
* from this software without prior written permission. For written | |||||
* permission, please contact apache@apache.org. | |||||
* | |||||
* 5. Products derived from this software may not be called "Apache" | |||||
* nor may "Apache" appear in their names without prior written | |||||
* permission of the Apache Group. | |||||
* | |||||
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||||
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||||
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||||
* DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||||
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||||
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||||
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||||
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||||
* SUCH DAMAGE. | |||||
* ==================================================================== | |||||
* | |||||
* This software consists of voluntary contributions made by many | |||||
* individuals on behalf of the Apache Software Foundation. For more | |||||
* information on the Apache Software Foundation, please see | |||||
* <http://www.apache.org/>. | |||||
*/ | |||||
package org.apache.tools.ant.types.resolver; | |||||
import java.io.IOException; | |||||
import java.net.MalformedURLException; | |||||
import org.apache.tools.ant.BuildException; | |||||
import org.apache.tools.ant.types.XMLCatalog; | |||||
import org.apache.tools.ant.types.DTDLocation; | |||||
import org.apache.tools.ant.types.EntityLocation; | |||||
import org.apache.xml.resolver.Catalog; | |||||
import org.apache.xml.resolver.CatalogManager; | |||||
import org.apache.xml.resolver.tools.CatalogResolver; | |||||
/** | |||||
* <p>This class extends the CatalogResolver class provided by Norman | |||||
* Walsh's resolver library in xml-commons. It provides the bridge | |||||
* between the Ant XMLCatalog datatype and the xml-commons Catalog | |||||
* class. XMLCatalog calls methods in this class using Reflection in | |||||
* order to avoid requiring the xml-commons resolver library in the | |||||
* path.</p> | |||||
* | |||||
* <p>The {@link org.apache.tools.ant.types.resolver.ApacheCatalog | |||||
* ApacheCatalog} class is used to parse external catalog files, which | |||||
* can be in either <a | |||||
* href="http://oasis-open.org/committees/entity/background/9401.html"> | |||||
* plain text format</a> or <a | |||||
* href="http://www.oasis-open.org/committees/entity/spec-2001-08-06.html"> | |||||
* XML format</a>.</p> | |||||
* | |||||
* <p>For each entry found in an external catalog file, if any, an | |||||
* instance of {@link org.apache.tools.ant.types.ResourceLocation | |||||
* ResourceLocation} is created and added to the controlling | |||||
* XMLCatalog datatype. In this way, these entries will be included | |||||
* in XMLCatalog's lookup algorithm. See XMLCatalog.java for more | |||||
* details.</p> | |||||
* | |||||
* @see org.apache.tools.ant.types.XMLCatalog.CatalogResolver | |||||
* @see org.apache.xml.resolver.CatalogManager | |||||
* @author <a href="mailto:cstrong@arielpartners.com">Craeg Strong</a> | |||||
* @version $Id$ | |||||
*/ | |||||
public class ApacheCatalogResolver extends CatalogResolver { | |||||
/** The XMLCatalog object to callback. */ | |||||
private XMLCatalog xmlCatalog = null; | |||||
static | |||||
{ | |||||
// | |||||
// If you don't do this, you get all sorts of annoying | |||||
// warnings about a missing properties file. However, it | |||||
// seems to work just fine with default values. Ultimately, | |||||
// we should probably include a "CatalogManager.properties" | |||||
// file in the ant jarfile with some default property | |||||
// settings. See CatalogManager.java for more details. | |||||
// | |||||
CatalogManager.ignoreMissingProperties(true); | |||||
// | |||||
// Make sure CatalogResolver instantiates ApacheCatalog, | |||||
// rather than a plain Catalog | |||||
// | |||||
System.setProperty("xml.catalog.className", | |||||
ApacheCatalog.class.getName()); | |||||
// debug | |||||
// System.setProperty("xml.catalog.verbosity", "4"); | |||||
} | |||||
/** Set the XMLCatalog object to callback. */ | |||||
public void setXMLCatalog(XMLCatalog xmlCatalog) { | |||||
this.xmlCatalog = xmlCatalog; | |||||
} | |||||
/** | |||||
* XMLCatalog calls this to add an external catalog file for each | |||||
* file within a <code><catalogfiles></code> fileset. | |||||
*/ | |||||
public void parseCatalog(String file) { | |||||
ApacheCatalog catalog = (ApacheCatalog)getCatalog(); | |||||
// Pass in reference to ourselves so we can be called back. | |||||
catalog.setResolver(this); | |||||
try { | |||||
catalog.parseCatalog(file); | |||||
} | |||||
catch(MalformedURLException ex) { | |||||
throw new BuildException(ex); | |||||
} | |||||
catch(IOException ex) { | |||||
throw new BuildException(ex); | |||||
} | |||||
} | |||||
/** | |||||
* <p>Add a PUBLIC catalog entry to the controlling XMLCatalog instance. | |||||
* ApacheCatalog calls this for each PUBLIC entry found in an external | |||||
* catalog file.</p> | |||||
* | |||||
* @param publicid The public ID of the resource | |||||
* @param systemid The system ID (aka location) of the resource | |||||
* @param base The base URL of the resource. If the systemid | |||||
* specifies a relative URL/pathname, it is resolved using the | |||||
* base. The default base for an external catalog file is the | |||||
* directory in which the catalog is located. | |||||
* | |||||
*/ | |||||
public void addPublicEntry(String publicid, | |||||
String systemid, | |||||
String base) { | |||||
DTDLocation dtd = new DTDLocation(); | |||||
dtd.setBase(base); | |||||
dtd.setPublicId(publicid); | |||||
dtd.setLocation(systemid); | |||||
xmlCatalog.addDTD(dtd); | |||||
} | |||||
/** | |||||
* <p>Add a URI catalog entry to the controlling XMLCatalog instance. | |||||
* ApacheCatalog calls this for each URI entry found in an external | |||||
* catalog file.</p> | |||||
* | |||||
* @param URI The URI of the resource | |||||
* @param altURI The URI to which the resource should be mapped | |||||
* (aka the location) | |||||
* @param base The base URL of the resource. If the altURI | |||||
* specifies a relative URL/pathname, it is resolved using the | |||||
* base. The default base for an external catalog file is the | |||||
* directory in which the catalog is located. | |||||
* | |||||
*/ | |||||
public void addURIEntry(String uri, | |||||
String altURI, | |||||
String base) { | |||||
EntityLocation entity = new EntityLocation(); | |||||
entity.setBase(base); | |||||
entity.setPublicId(uri); | |||||
entity.setLocation(altURI); | |||||
xmlCatalog.addEntity(entity); | |||||
} | |||||
} //-- ApacheCatalogResolver |
@@ -0,0 +1,23 @@ | |||||
<body> | |||||
Ant integration with xml-commons resolver. | |||||
<p>These classes enhance the <code><xmlcatalog></code> datatype | |||||
to support external catalog files using the xml-commons resolver, in | |||||
accordance with the | |||||
<a href="http://oasis-open.org/committees/entity/spec-2001-08-06.html"> | |||||
OASIS "Open Catalog" standard</a>. They will be used if and only if | |||||
the xml-commons resolver library is available on the classpath.</p> | |||||
@see <A HREF="http://xml.apache.org/commons">Apache xml-commons Project</A> | |||||
@see org.apache.tools.ant.types.XMLCatalog | |||||
@see org.apache.tools.ant.types.resolver.ApacheCatalogResolver | |||||
@see org.apache.tools.ant.types.resolver.ApacheCatalog | |||||
@author <A HREF="mailto:cstrong@arielpartners.com">Craeg Strong</A> | |||||
<hr> | |||||
<p align="center">Copyright © 2002 Apache Software Foundation. All rights | |||||
Reserved.</p> | |||||
</body> |
@@ -121,6 +121,17 @@ public class XmlValidateTest extends BuildFileTest { | |||||
executeTarget("xmlcatalog"); | executeTarget("xmlcatalog"); | ||||
} | } | ||||
/** | |||||
* catalogfiles fileset should be ignored | |||||
* if resolver.jar is not present, but will | |||||
* be used if it is. either way, test should | |||||
* work b/c we have a nested dtd with the same | |||||
* entity | |||||
*/ | |||||
public void testXmlCatalogFiles() { | |||||
executeTarget("xmlcatalogfiles"); | |||||
} | |||||
/** | /** | ||||
* Test nested xmlcatalog definitions | * Test nested xmlcatalog definitions | ||||
*/ | */ | ||||