/*
* Copyright (c) 2012 Fraunhofer IGD
*
* All rights reserved. This program and the accompanying materials are made
* available under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation, either version 3 of the License,
* or (at your option) any later version.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this distribution. If not, see <http://www.gnu.org/licenses/>.
*
* Contributors:
* Fraunhofer IGD
*/
package eu.esdihumboldt.hale.io.xslt;
import java.io.OutputStream;
import javax.xml.namespace.NamespaceContext;
import javax.xml.namespace.QName;
import javax.xml.stream.XMLOutputFactory;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.XMLStreamWriter;
import org.apache.commons.lang.StringEscapeUtils;
import org.apache.velocity.VelocityContext;
import org.apache.velocity.app.event.EventCartridge;
import eu.esdihumboldt.hale.common.align.model.impl.TypeEntityDefinition;
import eu.esdihumboldt.hale.common.schema.model.TypeDefinition;
import eu.esdihumboldt.hale.io.gml.writer.internal.IndentingXMLStreamWriter;
import eu.esdihumboldt.hale.io.xsd.constraint.XmlElements;
import eu.esdihumboldt.hale.io.xsd.model.XmlElement;
import eu.esdihumboldt.hale.io.xslt.internal.FailOnInvalidReference;
import eu.esdihumboldt.hale.io.xslt.xpath.FilterToXPath;
/**
* Utility methods that may be useful in {@link XslTransformation}s.
*
* @author Simon Templer
*/
@SuppressWarnings("restriction")
public abstract class XslTransformationUtil {
/**
* Create a XPath statement to select instances specified by the given type
* entity definition.
*
* @param ted the type entity definition
* @param context the context for the XPath expression, e.g. the empty
* string for the document root or <code>/</code> for anywhere in
* the document
* @param namespaces the namespace context
* @return the XPath expression or <code>null</code> if there are no
* elements that match the type
*/
public static String selectInstances(TypeEntityDefinition ted, String context,
NamespaceContext namespaces) {
TypeDefinition type = ted.getDefinition();
// get the XML elements associated to the type
XmlElements elements = type.getConstraint(XmlElements.class);
if (elements.getElements().isEmpty()) {
/*
* XXX dirty hack
*
* In CityGML 1.0 no element for AppearanceType is defined, only a
* property that is not detected in this way. The source route
* element is not known here, so we also cannot do a search based on
* the type. Thus for now we handle it as a special case.
*/
QName typeName = ted.getDefinition().getName();
if ("http://www.opengis.net/citygml/appearance/1.0".equals(typeName.getNamespaceURI())
&& "AppearanceType".equals(typeName.getLocalPart())) {
// create a dummy XML element
elements = new XmlElements();
elements.addElement(new XmlElement(new QName(
"http://www.opengis.net/citygml/appearance/1.0", "Appearance"), ted
.getDefinition(), null));
}
else
// XXX dirty hack end
return null;
}
// XXX which elements should be used?
// for now use all elements
StringBuilder select = new StringBuilder();
boolean first = true;
for (XmlElement element : elements.getElements()) {
if (first) {
first = false;
}
else {
select.append(" | ");
}
select.append(context);
select.append('/');
String ns = element.getName().getNamespaceURI();
if (ns != null && !ns.isEmpty()) {
String prefix = namespaces.getPrefix(ns);
if (prefix != null && !prefix.isEmpty()) {
select.append(prefix);
select.append(':');
}
}
select.append(element.getName().getLocalPart());
}
// filter
if (ted.getFilter() != null) {
String filterxpath = FilterToXPath.toXPath(ted.getDefinition(), namespaces,
ted.getFilter());
if (filterxpath != null && !filterxpath.isEmpty()) {
select.insert(0, '(');
select.append(")[");
select.append(StringEscapeUtils.escapeXml(filterxpath));
select.append(']');
}
}
return select.toString();
}
/**
* Setup a XML writer configured with the namespace prefixes and UTF-8
* encoding.
*
* @param outStream the output stream to write the XML content to
* @param namespaces the namespace context, e.g. as retrieved from a
* {@link XsltGenerationContext}
* @return the XML stream writer
* @throws XMLStreamException if an error occurs setting up the writer
*/
public static XMLStreamWriter setupXMLWriter(OutputStream outStream, NamespaceContext namespaces)
throws XMLStreamException {
// create and set-up a writer
XMLOutputFactory outputFactory = XMLOutputFactory.newInstance();
// will set namespaces if these not set explicitly
outputFactory.setProperty("javax.xml.stream.isRepairingNamespaces", //$NON-NLS-1$
Boolean.valueOf(true));
// create XML stream writer with UTF-8 encoding
XMLStreamWriter tmpWriter = outputFactory.createXMLStreamWriter(outStream, "UTF-8"); //$NON-NLS-1$
tmpWriter.setNamespaceContext(namespaces);
return new IndentingXMLStreamWriter(tmpWriter);
}
/**
* Create a new {@link VelocityContext} that lets template merging fail if
* an invalid reference is encountered.
*
* @return the velocity context
*/
public static VelocityContext createStrictVelocityContext() {
VelocityContext context = new VelocityContext();
EventCartridge eventCartridge = new EventCartridge();
eventCartridge.addEventHandler(new FailOnInvalidReference());
context.attachEventCartridge(eventCartridge);
return context;
}
}