changeset 472:783cc1b6b615

Moved directories to org.dive4elements
author Sascha L. Teichmann <teichmann@intevation.de>
date Thu, 25 Apr 2013 10:53:15 +0200 (2013-04-25)
parents 1a87cb24a446
children d0ac790a6c89
files artifacts-common/src/main/java/de/intevation/artifacts/common/ArtifactNamespaceContext.java artifacts-common/src/main/java/de/intevation/artifacts/common/model/DefaultUser.java artifacts-common/src/main/java/de/intevation/artifacts/common/model/KVP.java artifacts-common/src/main/java/de/intevation/artifacts/common/model/User.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/ClientProtocolUtils.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/Config.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/CreationFilter.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/DateUtils.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/FileTools.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/JSON.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/LRUCache.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/MapXPathVariableResolver.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/StringUtils.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/XMLUtils.java artifacts-common/src/main/java/de/intevation/artifacts/common/utils/XSLTransformer.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/ArtifactNamespaceContext.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/model/DefaultUser.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/model/KVP.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/model/User.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/ClientProtocolUtils.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/Config.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/CreationFilter.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/DateUtils.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/FileTools.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/JSON.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/LRUCache.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/MapXPathVariableResolver.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/StringUtils.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/XMLUtils.java artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/XSLTransformer.java
diffstat 30 files changed, 3437 insertions(+), 3437 deletions(-) [+]
line wrap: on
line diff
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/ArtifactNamespaceContext.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,90 +0,0 @@
-/*
- * Copyright (c) 2010 by Intevation GmbH
- *
- * This program is free software under the LGPL (>=v2.1)
- * Read the file LGPL.txt coming with the software for details
- * or visit http://www.gnu.org/licenses/ if it does not exist.
- */
-
-package de.intevation.artifacts.common;
-
-import java.util.Iterator;
-
-import javax.xml.XMLConstants;
-
-import javax.xml.namespace.NamespaceContext;
-
-/**
- * The namespace used in artifact documents.
- * @author <a href="mailto:sascha.teichmann@intevation">Sascha L. Teichmann</a>
- */
-public class ArtifactNamespaceContext
-implements   NamespaceContext
-{
-    /**
-     * The URI of the namespace of the artifacts.
-     */
-    public final static String NAMESPACE_URI =
-        "http://www.intevation.de/2009/artifacts";
-
-    /**
-     * The XML prefix for the artifacts namespace.
-     */
-    public final static String NAMESPACE_PREFIX = "art";
-
-    /**
-     * Final instance to be easily used to avoid creation
-     * of instances.
-     */
-    public static final ArtifactNamespaceContext INSTANCE =
-        new ArtifactNamespaceContext();
-
-    /**
-     * The default constructor.
-     */
-    public ArtifactNamespaceContext() {
-    }
-
-    /**
-     * @see javax.xml.namespace.NamespaceContext#getNamespaceURI(String)
-     * @param prefix The prefix
-     * @return The corresponing URI
-     */
-    public String getNamespaceURI(String prefix) {
-
-        if (prefix == null) {
-            throw new NullPointerException("Null prefix");
-        }
-
-        if (NAMESPACE_PREFIX.equals(prefix)) {
-            return NAMESPACE_URI;
-        }
-
-        if ("xml".equals(prefix)) {
-            return XMLConstants.XML_NS_URI;
-        }
-
-        return XMLConstants.NULL_NS_URI;
-    }
-
-    /**
-     * @see javax.xml.namespace.NamespaceContext#getPrefix(String)
-     * @param uri The URI
-     * @return nothing.
-     * @throws java.lang.UnsupportedOperationException
-     */
-    public String getPrefix(String uri) {
-        throw new UnsupportedOperationException();
-    }
-
-    /**
-     * @see javax.xml.namespace.NamespaceContext#getPrefixes(java.lang.String)
-     * @param uri The URI
-     * @return nothing
-     * @throws java.lang.UnsupportedOperationException
-     */
-    public Iterator getPrefixes(String uri) {
-        throw new UnsupportedOperationException();
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/model/DefaultUser.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,75 +0,0 @@
-package de.intevation.artifacts.common.model;
-
-/**
- * The default implementation of the {@link User} interface.
- *
- * @author <a href="mailto:ingo.weinzierl@intevation.de">Ingo Weinzierl</a>
- */
-public class DefaultUser implements User {
-
-    /** The user's firstname. */
-    protected String firstName;
-
-    /** The user's lastname. */
-    protected String lastName;
-
-
-    /**
-     * Creates an empty user without name.
-     */
-    public DefaultUser() {
-    }
-
-
-    /**
-     * Creates a user with first and lastname.
-     *
-     * @param firstName The user's firstname.
-     * @param lastName The user's lastname.
-     */
-    public DefaultUser(String firstName, String lastName) {
-        this.firstName = firstName;
-        this.lastName  = lastName;
-    }
-
-
-    /**
-     * This method returns the firstname of the user.
-     *
-     * @return the firstname.
-     */
-    public String getFirstName() {
-        return firstName;
-    }
-
-
-    /**
-     * Sets the user's firstname.
-     *
-     * @param firstName The user's firstname.
-     */
-    public void setFirstName(String firstName) {
-        this.firstName = firstName;
-    }
-
-
-    /**
-     * This method returns the lastname of the user.
-     *
-     * @return the lastname.
-     */
-    public String getLastName() {
-        return lastName;
-    }
-
-
-    /**
-     * Sets the user's lastname.
-     *
-     * @param lastName The user's lastname.
-     */
-    public void setLastName(String lastName) {
-        this.lastName = lastName;
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/model/KVP.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,32 +0,0 @@
-/*
- * Copyright (c) 2010 by Intevation GmbH
- *
- * This program is free software under the LGPL (>=v2.1)
- * Read the file LGPL.txt coming with the software for details
- * or visit http://www.gnu.org/licenses/ if it does not exist.
- */
-package de.intevation.artifacts.common.model;
-
-
-public class KVP<K, V> {
-
-    private K key;
-    private V value;
-
-
-    public KVP(K key, V value) {
-        this.key   = key;
-        this.value = value;
-    }
-
-
-    public K getKey() {
-        return key;
-    }
-
-
-    public V getValue() {
-        return value;
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/model/User.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,43 +0,0 @@
-package de.intevation.artifacts.common.model;
-
-import java.io.Serializable;
-
-/**
- * An interface that describes a user of the system.
- *
- * @author <a href="mailto:ingo.weinzierl@intevation.de">Ingo Weinzierl</a>
- */
-public interface User extends Serializable {
-
-    /**
-     * This method returns the firstname of the user.
-     *
-     * @return the firstname.
-     */
-    public String getFirstName();
-
-
-    /**
-     * Sets the user's firstname.
-     *
-     * @param firstName The user's firstname.
-     */
-    public void setFirstName(String firstName);
-
-
-    /**
-     * This method returns the lastname of the user.
-     *
-     * @return the lastname.
-     */
-    public String getLastName();
-
-
-    /**
-     * Sets the user's lastname.
-     *
-     * @param lastName The user's lastname.
-     */
-    public void setLastName(String lastName);
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/ClientProtocolUtils.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,860 +0,0 @@
-/*
- * Copyright (c) 2011 by Intevation GmbH
- *
- * This program is free software under the LGPL (>=v2.1)
- * Read the file LGPL.txt coming with the software for details
- * or visit http://www.gnu.org/licenses/ if it does not exist.
- */
-package de.intevation.artifacts.common.utils;
-
-import javax.xml.xpath.XPathConstants;
-
-import org.w3c.dom.Document;
-import org.w3c.dom.Element;
-import org.w3c.dom.Node;
-import org.w3c.dom.NodeList;
-
-import de.intevation.artifacts.common.ArtifactNamespaceContext;
-
-
-/**
- * This class provides methods that help creating the artifact protocol
- * documents DESCRIBE, FEED, ADVANCE and OUT.
- *
- * @author <a href="mailto:ingo.weinzierl@intevation.de">Ingo Weinzierl</a>
- */
-public class ClientProtocolUtils {
-
-    /** The XPath to the current state in the DESCRIBE document. */
-    public static final String XPATH_CURRENT_STATE = "/art:result/art:state";
-
-    /** The XPath to the static UI part in the DESCRIBE document. */
-    public static final String XPATH_STATIC  = "/art:result/art:ui/art:static";
-
-    /** The XPath to the dynamic UI part in the DESCRIBE document. */
-    public static final String XPATH_DYNAMIC = "/art:result/art:ui/art:dynamic";
-
-    /** The XPath to the reachable states part in the DESCRIBE document. */
-    public static final String XPATH_STATES  =
-        "/art:result/art:reachable-states";
-
-    /** The XPath to the output modes in the DESCRIBE document. */
-    public static final String XPATH_OUTPUT_MODES  =
-        "/art:result/art:outputmodes/art:output";
-
-    /** The XPath to the select node relative to the dynamic UI node in the
-     * DESCRIBE document. */
-    public static final String XPATH_DATA_SELECT = "art:select";
-
-    /** The XPath to the choices nodes relative to the select node in the
-     * DESCRIBE document. */
-    public static final String XPATH_DATA_ITEMS = "art:choices/art:item";
-
-    /** The XPath that points to the min value of a range.*/
-    public static final String XPATH_MIN_NODE = "art:min/@art:value";
-
-    /** The XPath that points to the max value of a range.*/
-    public static final String XPATH_MAX_NODE = "art:max/@art:value";
-
-    /** The XPath that points to the default min value of a range.*/
-    public static final String XPATH_DEF_MIN = "art:min/@art:default";
-
-    /** The XPath that points to the default max value of a range.*/
-    public static final String XPATH_DEF_MAX = "art:max/@art:default";
-
-    /** The XPath to a label in the artifact's DESCRIBE document. */
-    public static final String XPATH_LABEL = "art:label/text()";
-
-    /** The XPath to a value in the artifact's DESCRIBE document. */
-    public static final String XPATH_VALUE = "art:value/text()";
-
-
-    /**
-     * It should not be necessary to create instances of this class.
-     */
-    private ClientProtocolUtils() {
-    }
-
-
-    /**
-     * This method creates a new CREATE document.
-     *
-     * @return the CREATE document.
-     */
-    public static Document newCreateDocument(String factory) {
-        return newCreateDocument(factory, null);
-    }
-
-
-    /**
-     * This method creates a new CREATE document.
-     *
-     * @return the CREATE document.
-     */
-    public static Document newCreateDocument(String factory, String uuid) {
-        return newCreateDocument(factory, uuid, null);
-    }
-
-    public static Document newCreateDocument(
-        String  factory,
-        String  uuid,
-        String  ids
-    ) {
-        return newCreateDocument(factory, uuid, ids, null);
-    }
-
-    /**
-     * This method creates a new CREATE document.
-     *
-     * @return the CREATE document.
-     */
-    public static Document newCreateDocument(
-        String         factory,
-        String         uuid,
-        String         ids,
-        CreationFilter filter
-    ) {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action = cr.create("action");
-        Element type   = cr.create("type");
-        Element fac    = cr.create("factory");
-
-        type.setAttribute("name", "create");
-        fac.setAttribute("name", factory);
-
-        action.appendChild(type);
-        action.appendChild(fac);
-
-        if (uuid != null) {
-            Element templ = cr.create("template");
-            templ.setAttribute("uuid", uuid);
-            action.appendChild(templ);
-        }
-
-        if (ids != null) {
-            Element id = cr.create("ids");
-            id.setAttribute("value", ids);
-            action.appendChild(id);
-        }
-
-        if (filter != null) {
-            action.appendChild(filter.toXML(cr));
-        }
-
-        doc.appendChild(action);
-
-        return doc;
-    }
-
-
-    /**
-     * This method creates a new FEED document.
-     *
-     * @param theUuid The identifier of the artifact.
-     * @param theHash The hash of the artifact.
-     * @param theData An array that contains key/value pairs that represent the
-     * data that should be included in the FEED document.
-     *
-     * @return the FEED document.
-     */
-    public static Document newFeedDocument(
-        String     theUuid,
-        String     theHash,
-        String[][] theData)
-    {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action = cr.create("action");
-        Element type   = cr.create("type");
-        Element uuid   = cr.create("uuid");
-        Element hash   = cr.create("hash");
-        Element data   = cr.create("data");
-
-        // XXX It is not nice that the type has no attribute namespace, but to
-        // be backward compatible, we don't change this now.
-        cr.addAttr(type, "name", "feed", false);
-        cr.addAttr(uuid, "value", theUuid, true);
-        cr.addAttr(hash, "value", theHash, true);
-
-        for (String[] kvp: theData) {
-            Element input = cr.create("input");
-            cr.addAttr(input, "name", kvp[0], true);
-            cr.addAttr(input, "value", kvp[1], true);
-
-            data.appendChild(input);
-        }
-
-        action.appendChild(type);
-        action.appendChild(uuid);
-        action.appendChild(hash);
-        action.appendChild(data);
-
-        doc.appendChild(action);
-
-        return doc;
-    }
-
-
-    /**
-     * This method creates a new DESCRIBE document.
-     *
-     * @param theUuid The identifier of the artifact.
-     * @param theHash The hash of the artifact.
-     * @param ui If true, the UI part is included.
-     *
-     * @return the DESCRIBE document.
-     */
-    public static Document newDescribeDocument(
-        String  theUuid,
-        String  theHash,
-        boolean incUI)
-    {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action = cr.create("action");
-        Element type   = cr.create("type");
-        Element uuid   = cr.create("uuid");
-        Element hash   = cr.create("hash");
-        Element ui     = cr.create("include-ui");
-
-        // XXX It is not nice that the type has no attribute namespace, but to
-        // be backward compatible, we don't change this now.
-        cr.addAttr(type, "name", "describe", false);
-        cr.addAttr(uuid, "value", theUuid, true);
-        cr.addAttr(hash, "value", theHash, true);
-
-        ui.setTextContent(incUI ? "true" : "false");
-
-        action.appendChild(type);
-        action.appendChild(uuid);
-        action.appendChild(hash);
-        action.appendChild(ui);
-
-        doc.appendChild(action);
-
-        return doc;
-    }
-
-
-    /**
-     * This method creates a new ADVANCE document.
-     *
-     * @param theUuid The identifier of the artifact.
-     * @param theHash The hash of the artifact.
-     * @param theTarget The target state identifier.
-     *
-     * @return the ADVANCE document.
-     */
-    public static Document newAdvanceDocument(
-        String theUuid,
-        String theHash,
-        String theTarget)
-    {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action = cr.create("action");
-        Element type   = cr.create("type");
-        Element uuid   = cr.create("uuid");
-        Element hash   = cr.create("hash");
-        Element target = cr.create("target");
-
-        // XXX It is not nice that the type has no attribute namespace, but to
-        // be backward compatible, we don't change this now.
-        cr.addAttr(type, "name", "advance", false);
-        cr.addAttr(uuid, "value", theUuid, true);
-        cr.addAttr(hash, "value", theHash, true);
-        cr.addAttr(target, "name", theTarget, true);
-
-        action.appendChild(type);
-        action.appendChild(uuid);
-        action.appendChild(hash);
-        action.appendChild(target);
-
-        doc.appendChild(action);
-
-        return doc;
-    }
-
-
-    /**
-     * This method creates a new document that is used to create new artifact
-     * collections in the artifact server.
-     *
-     * @param name <b>Optional</b> name of the collection.
-     *
-     * @return the document to create new collections.
-     */
-    public static Document newCreateCollectionDocument(String name) {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action     = cr.create("action");
-        Element type       = cr.create("type");
-        Element collection = cr.create("collection");
-        Element attribute  = cr.create("attribute");
-
-        cr.addAttr(type, "name", "create");
-        cr.addAttr(collection, "name", name != null ? name : "");
-
-        action.appendChild(type);
-        type.appendChild(collection);
-        collection.appendChild(attribute);
-
-        doc.appendChild(action);
-
-        return doc;
-    }
-
-
-    /**
-     * This method creates a new Document that is used to add an artifact to a
-     * collection in the artifact server.
-     *
-     * @param artId The identifier of the artifact that should be added.
-     * @param attr A document that contains attributes for the artifact's
-     * life in the collection.
-     *
-     * @return the document to add an artifact into a collection.
-     */
-    public static Document newAddArtifactDocument(String artId, Document attr) {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action    = cr.create("action");
-        Element type      = cr.create("type");
-        Element artifact  = cr.create("artifact");
-        Element attribute = cr.create("attribute");
-
-        cr.addAttr(artifact, "uuid", artId);
-        cr.addAttr(type, "name", "addartifact");
-
-        if (attr != null) {
-            attr.appendChild(attr);
-        }
-
-        action.appendChild(type);
-        type.appendChild(artifact);
-        artifact.appendChild(attribute);
-
-        doc.appendChild(action);
-
-        return doc;
-    }
-
-
-    /**
-     * Create a new Document that is used to remove an artifact from a
-     * collection in the artifact server.
-     *
-     * @param artId The identifier of the artifact that should be added.
-     *
-     * @return the document to add an artifact into a collection.
-     */
-    public static Document newRemoveArtifactDocument(String artId) {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action    = cr.create("action");
-        Element type      = cr.create("type");
-        Element artifact  = cr.create("artifact");
-
-        cr.addAttr(artifact, "uuid", artId);
-        cr.addAttr(type, "name", "removeartifact");
-
-        action.appendChild(type);
-        type.appendChild(artifact);
-
-        doc.appendChild(action);
-
-        return doc;
-    }
-
-
-    /**
-     * This method creates a new Document that is used to trigger the DESCRIBE
-     * operation of a collection in the artifact server.
-     *
-     * @param uuid The identifier of the collection that should be described.
-     *
-     * @return the document to describe a collection.
-     */
-    public static Document newDescribeCollectionDocument(String uuid) {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action    = cr.create("action");
-        Element type      = cr.create("type");
-        cr.addAttr(type, "name", "describe");
-
-        action.appendChild(type);
-
-        doc.appendChild(action);
-
-        return doc;
-    }
-
-
-
-    /**
-     * This function builds a document that is used as request document of the
-     * out() operation of Collections.
-     *
-     * @param uuid The identifier of the collection.
-     * @param mode The name of the desired output mode.
-     * @param type The name of the desired output type.
-     *
-     * @return the request document.
-     */
-    public static Document newOutCollectionDocument(
-        String uuid,
-        String mode,
-        String type) {
-        return newOutCollectionDocument(uuid, mode, type, null);
-    }
-
-
-    /**
-     * This function builds a document that is used as request document of the
-     * out() operation of Collections. The document <i>attr</i> might be used to
-     * adjust some settings specific to the output.
-     *
-     * @param uuid The identifier of the collection.
-     * @param mode The name of the desired output mode.
-     * @param type The name of the desired output type.
-     * @param attr A document that contains settings specific to the output.
-     *
-     * @return the request document.
-     */
-    public static Document newOutCollectionDocument(
-        String   uuid,
-        String   mode,
-        String   type,
-        Document attr)
-    {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action = cr.create("action");
-
-        cr.addAttr(action, "name", mode, true);
-        cr.addAttr(action, "type", type, true);
-
-        doc.appendChild(action);
-
-        if (attr != null) {
-            Node root = attr.getFirstChild();
-
-            if (root != null) {
-                action.appendChild(doc.importNode(root, true));
-            }
-        }
-
-        return doc;
-    }
-
-
-    /**
-     * This function creates a document that is used to set the attribute of a
-     * Collection.
-     *
-     * @param uuid The identifier of the Collection.
-     * @param attr The new attribute value for the Collection.
-     *
-     * @return the document that is used to set the attribute.
-     */
-    public static Document newSetAttributeDocument(
-        String uuid,
-        Document attr)
-    {
-        Node root = attr != null ? attr.getFirstChild() : null;
-
-        if (root == null) {
-            return null;
-        }
-
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action     = ec.create("action");
-        Element type       = ec.create("type");
-        Element collection = ec.create("collection");
-
-        ec.addAttr(type, "name", "setattribute", false);
-        ec.addAttr(collection, "uuid", uuid, false);
-
-        doc.appendChild(action);
-        action.appendChild(type);
-        type.appendChild(collection);
-
-        collection.appendChild(doc.importNode(root, true));
-
-        return doc;
-    }
-
-    /**
-     * This function creates a document that is used to set the attribute of a
-     * CollectionItem.
-     *
-     * @param uuid The identifier of the CollectionItem.
-     * @param attr The new attribute value for the CollectionItem.
-     *
-     * @return the document that is used to set the attribute.
-     */
-    public static Document newSetItemAttributeDocument(
-        String uuid,
-        Document attr)
-    {
-        Node root = attr.getFirstChild();
-
-        if (root == null) {
-            return null;
-        }
-
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action   = ec.create("action");
-        Element type     = ec.create("type");
-        Element artifact = ec.create("artifact");
-
-        ec.addAttr(type, "name", "setitemattribute");
-        ec.addAttr(artifact, "uuid", uuid);
-
-        doc.appendChild(action);
-        action.appendChild(type);
-        type.appendChild(artifact);
-
-        artifact.appendChild(doc.importNode(root, true));
-
-        return doc;
-    }
-
-
-    /**
-     * This function creates a document that is used to set the time-to-live
-     * of a collection.
-     *
-     * @param ttl The ttl for the Collection.
-     *
-     * @return the document that is used to set the time-to-live.
-     */
-    public static Document newSetCollectionTTLDocument(String ttl) {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action = ec.create("action");
-        Element type   = ec.create("type");
-        Element ttlEl  = ec.create("ttl");
-
-        ec.addAttr(type, "name", "settimetolive");
-        ec.addAttr(ttlEl, "value", ttl);
-
-        doc.appendChild(action);
-        action.appendChild(type);
-        type.appendChild(ttlEl);
-
-        return doc;
-    }
-
-
-    /**
-     * This function creates a document that is used to set the name of a
-     * collection.
-     *
-     * @param name The name for the Collection.
-     *
-     * @return the document that is used to set the name of a collection.
-     */
-    public static Document newSetCollectionNameDocument(String name) {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action = ec.create("action");
-        Element type   = ec.create("type");
-        Element coll   = ec.create("collection");
-
-        ec.addAttr(type, "name", "setname");
-        ec.addAttr(coll, "name", name);
-
-        doc.appendChild(action);
-        action.appendChild(type);
-        type.appendChild(coll);
-
-        return doc;
-    }
-
-
-    /**
-     * This function creates a document that is used to delete an existing
-     * collection.
-     *
-     * @return the document that is used to delete an existing collection.
-     */
-    public static Document newDeleteCollectionDocument() {
-        Document doc = XMLUtils.newDocument();
-
-        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
-            doc,
-            ArtifactNamespaceContext.NAMESPACE_URI,
-            ArtifactNamespaceContext.NAMESPACE_PREFIX);
-
-        Element action = ec.create("action");
-        Element type   = ec.create("type");
-
-        ec.addAttr(type, "name", "delete");
-
-        doc.appendChild(action);
-        action.appendChild(type);
-
-        return doc;
-    }
-
-
-    /**
-     * Returns string value found by {@link XPATH_LABEL} relative to
-     * <i>node</i>.
-     *
-     * @param node A node.
-     *
-     * @return the string value found by {@link XPATH_LABEL}.
-     */
-    public static String getLabel(Node node) {
-        return (String) XMLUtils.xpath(
-            node,
-            XPATH_LABEL,
-            XPathConstants.STRING,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    /**
-     * Returns string value found by {@link XPATH_VALUE} relative to
-     * <i>node</i>.
-     *
-     * @param node A node.
-     *
-     * @return the string value found by {@link XPATH_VALUE}.
-     */
-    public static String getValue(Node node) {
-        return (String) XMLUtils.xpath(
-            node,
-            XPATH_VALUE,
-            XPathConstants.STRING,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    /**
-     * This method returns the static UI part of the artifact's DESCRIBE
-     * document.
-     *
-     * @param description The document returned by the artifact server's
-     * DESCRIBE operation.
-     *
-     * @return the static UI node.
-     */
-    public static Node getStaticUI(Document description) {
-        return (Node) XMLUtils.xpath(
-            description,
-            XPATH_STATIC,
-            XPathConstants.NODE,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    /**
-     * This method returns the dynamic UI part of the artifact's DESCRIBE
-     * document.
-     *
-     * @param description The document returned by the artifact server's
-     * DESCRIBE operation.
-     *
-     * @return the dynamic UI node.
-     */
-    public static Node getDynamicUI(Document description) {
-        return (Node) XMLUtils.xpath(
-            description,
-            XPATH_DYNAMIC,
-            XPathConstants.NODE,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    /**
-     * This method returns the current state node contained in the DESCRIBE
-     * document.
-     *
-     * @param description The document returned by the artifact server's
-     * DESCRIBE operation.
-     *
-     * @return the node containing information about the current state.
-     */
-    public static Node getCurrentState(Document description) {
-        return (Node) XMLUtils.xpath(
-            description,
-            XPATH_CURRENT_STATE,
-            XPathConstants.NODE,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    /**
-     * This method returns the node that contains information about the
-     * reachable states of the artifact in the artifact's DESCRIBE document.
-     *
-     * @param description The document returned by the artifact server's
-     * DESCRIBE operation.
-     *
-     * @return the node that contains the reachable states.
-     */
-    public static Node getReachableStates(Document description) {
-        return (Node) XMLUtils.xpath(
-            description,
-            XPATH_STATES,
-            XPathConstants.NODE,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    /**
-     * This method returns the output mode nodes of the DESCRIBE document.
-     *
-     * @param description The document returned by the artifact server's
-     * DESCRIBE operation.
-     *
-     * @return the node that contains the output modes.
-     */
-    public static NodeList getOutputModes(Document description) {
-        return (NodeList) XMLUtils.xpath(
-            description,
-            XPATH_OUTPUT_MODES,
-            XPathConstants.NODESET,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    /**
-     * Returns the node found by {@link XPATH_DATA_SELECT}.
-     *
-     * @param dynamicNode The dynamic UI node of the DESCRIBE document.
-     *
-     * @return the select node found in the dynamic UI node.
-     */
-    public static NodeList getSelectNode(Node dynamicNode) {
-        return (NodeList) XMLUtils.xpath(
-            dynamicNode,
-            XPATH_DATA_SELECT,
-            XPathConstants.NODESET,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    /**
-     * Returns the items that could be found in the <i>node</i>.
-     *
-     * @param node A select node.
-     *
-     * @return the choices nodes as node list.
-     */
-    public static NodeList getItemNodes(Node node) {
-        return (NodeList) XMLUtils.xpath(
-            node,
-            XPATH_DATA_ITEMS,
-            XPathConstants.NODESET,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    public static String getMinNode(Node parent) {
-        return (String) XMLUtils.xpath(
-            parent,
-            XPATH_MIN_NODE,
-            XPathConstants.STRING,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    public static String getMaxNode(Node parent) {
-        return (String) XMLUtils.xpath(
-            parent,
-            XPATH_MAX_NODE,
-            XPathConstants.STRING,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    public static String getDefMin(Node parent) {
-        return (String) XMLUtils.xpath(
-            parent,
-            XPATH_DEF_MIN,
-            XPathConstants.STRING,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-
-
-    public static String getDefMax(Node parent) {
-        return (String) XMLUtils.xpath(
-            parent,
-            XPATH_DEF_MAX,
-            XPathConstants.STRING,
-            ArtifactNamespaceContext.INSTANCE);
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/Config.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,276 +0,0 @@
-/*
- * Copyright (c) 2010 by Intevation GmbH
- *
- * This program is free software under the LGPL (>=v2.1)
- * Read the file LGPL.txt coming with the software for details
- * or visit http://www.gnu.org/licenses/ if it does not exist.
- */
-
-package de.intevation.artifacts.common.utils;
-
-import java.io.File;
-import java.io.IOException;
-
-import javax.xml.namespace.QName;
-
-import javax.xml.parsers.DocumentBuilderFactory;
-import javax.xml.parsers.ParserConfigurationException;
-
-import javax.xml.xpath.XPathConstants;
-
-import org.apache.log4j.Logger;
-
-import org.w3c.dom.Document;
-import org.w3c.dom.Node;
-import org.w3c.dom.NodeList;
-
-import org.xml.sax.SAXException;
-
-/**
- * The central access to the configuration of the artifact database.
- * This class provides some static methods to access the central
- * configuration XML file via XPath.
- *
- * @author <a href="mailto:sascha.teichmann@intevation.de">Sascha L. Teichmann</a>
- */
-public final class Config
-{
-    private static Logger logger = Logger.getLogger(Config.class);
-
-    /**
-     * System property name where to find the configuration directory.
-     */
-    public static final String CONFIG_DIR = "artifact.database.dir";
-
-    /**
-     * Default path to the configuration directory if none
-     * was specified by the CONFIG_DIR system property.
-     */
-    public static final File CONFIG_DIR_DEFAULT =
-        new File(new File(System.getProperty("user.home",
-            System.getProperty("user.dir", "."))), ".artifactdb");
-
-    /**
-     * Name of the central configuration XML file.
-     */
-    public static final String CONFIG_FILE  = "conf.xml";
-
-    /**
-     * Name of the configuration filename alias to be use
-     * within the configuration. This alias is replaced by
-     * the real path.
-     */
-    public static final String CONFIG_DIR_PLACEHOLDER =
-        "${artifacts.config.dir}";
-
-    private static Document config;
-
-    private Config() {
-    }
-
-    /**
-     * Singleton access to the central XML configuration document.
-     * @return The central XML configuration document.
-     */
-    public static synchronized final Document getConfig() {
-        if (config == null) {
-            config = loadConfig();
-        }
-        return config;
-    }
-
-    /**
-     * Returns the path to the configuartion directory. If a path
-     * was specified via the CONFIG_DIR system property this one
-     * is used. Else it falls back to default configuration path.
-     * @return The path to the configuartion directory.
-     */
-    public static File getConfigDirectory() {
-        String configDirString = System.getProperty(CONFIG_DIR);
-
-        File configDir = configDirString != null
-            ? new File(configDirString)
-            : CONFIG_DIR_DEFAULT;
-
-        if (!configDir.isDirectory()) {
-            logger.warn("'" + configDir + "' is not a directory.");
-            configDir = CONFIG_DIR_DEFAULT;
-        }
-
-        return configDir;
-    }
-
-    /**
-     * Replaces the CONFIG_DIR_PLACEHOLDER alias with the real path
-     * of the configuration directory.
-     * @param path The path containing the CONFIG_DIR_PLACEHOLDER placeholder.
-     * @return The path where the CONFIG_DIR_PLACEHOLDER placeholders are
-     * replaced by the real path name.
-     */
-    public static String replaceConfigDir(String path) {
-        String configDir = getConfigDirectory().getAbsolutePath();
-        return path.replace(CONFIG_DIR_PLACEHOLDER, configDir);
-    }
-
-    private static Document loadConfig() {
-
-        File configDir = getConfigDirectory();
-
-        File file = new File(configDir, CONFIG_FILE);
-
-        if (!file.canRead() && !file.isFile()) {
-            logger.error("Cannot read config file '"
-                + file + "'.");
-            return null;
-        }
-
-        try {
-            DocumentBuilderFactory factory =
-                DocumentBuilderFactory.newInstance();
-            factory.setValidating(false); // XXX: This may change in future.
-            return factory.newDocumentBuilder().parse(file);
-        }
-        catch (SAXException se) {
-            logger.error(se.getLocalizedMessage(), se);
-        }
-        catch (ParserConfigurationException pce) {
-            logger.error(pce.getLocalizedMessage(), pce);
-        }
-        catch (IOException ioe) {
-            logger.error(ioe.getLocalizedMessage());
-        }
-
-        return null;
-    }
-
-    /**
-     * Convenience method to search within a given document tree via XPath.
-     * See {@link XMLUtils#xpath(Object, String, QName) } for details.
-     * @param root The object which is used as the root of the tree to
-     * be searched in.
-     * @param query The XPath query.
-     * @param returnType The type of the result.
-     * @return The result of type 'returnTyp' or null if something went
-     * wrong during XPath evaluation.
-     */
-    public static final Object getXPath(
-        Object root, String query, QName returnType
-    ) {
-        return XMLUtils.xpath(root, query, returnType);
-    }
-
-    /**
-     * Convenience method to search within the central configuration via XPath.
-     * See {@link XMLUtils#xpath(Object, String, QName) } for details.
-     * @param query The XPath query.
-     * @param returnType The type of the result.
-     * @return The result of type 'returnTyp' or null if something went
-     * wrong during XPath evaluation.
-     */
-    public static final Object getXPath(String query, QName returnType) {
-        return XMLUtils.xpath(getConfig(), query, returnType);
-    }
-
-    /**
-     * Convenience method to search for a node list within the central
-     * configuation document via XPath.
-     * @param query The XPath query.
-     * @return The queried node list or null if something went
-     * wrong during XPath evaluation.
-     */
-    public static final NodeList getNodeSetXPath(String query) {
-        return (NodeList)getXPath(query, XPathConstants.NODESET);
-    }
-
-    /**
-     * Convenience method to search for a node within the central
-     * configuation document via XPath.
-     * @param query The XPath query.
-     * @return The queried node or null if something went
-     * wrong during XPath evaluation.
-     */
-    public static final Node getNodeXPath(String query) {
-        return (Node)getXPath(query, XPathConstants.NODE);
-    }
-
-    /**
-     * Convenience method to search for a string within the central
-     * configuation document via XPath.
-     * @param xpath The XPath query.
-     * @return The queried string or null if something went
-     * wrong during XPath evaluation.
-     */
-    public static final String getStringXPath(String xpath) {
-        return getStringXPath(xpath, null);
-    }
-
-    /**
-     * Convenience method to search for a string within the central
-     * configuation document via XPath.
-     * @param query The XPath query.
-     * @param def The string to be returned if the search has no results.
-     * @return The queried string or the default value if something went
-     * wrong during XPath evaluation.
-     */
-    public static final String getStringXPath(String query, String def) {
-        String s = (String)getXPath(query, XPathConstants.STRING);
-        return s == null || s.length() == 0
-            ? def
-            : s;
-    }
-
-    /**
-     * Convenience method to search for a node list within a given tree
-     * via XPath.
-     * @param root The root of the tree to be searched in.
-     * @param query The XPath query.
-     * @return The queried node list or null if something went
-     * wrong during XPath evaluation.
-     */
-    public static final NodeList getNodeSetXPath(Object root, String query) {
-        return (NodeList)getXPath(root, query, XPathConstants.NODESET);
-    }
-
-    /**
-     * Convenience method to search for a node within a given tree
-     * via XPath.
-     * @param root The root of the tree to be searched in.
-     * @param query The XPath query.
-     * @return The queried node or null if something went
-     * wrong during XPath evaluation.
-     */
-    public static final Node getNodeXPath(Object root, String query) {
-        return (Node)getXPath(root, query, XPathConstants.NODE);
-    }
-
-    /**
-     * Convenience method to search for a string within a given tree
-     * via XPath.
-     * @param root The root of the tree to be searched in.
-     * @param xpath The XPath query.
-     * @return The queried string or null if something went
-     * wrong during XPath evaluation.
-     */
-    public static final String getStringXPath(Object root, String xpath) {
-        return getStringXPath(root, xpath, null);
-    }
-
-    /**
-     * Convenience method to search for a string within a given tree
-     * via XPath.
-     * @param root The root of the tree to be searched in.
-     * @param query xpath The XPath query.
-     * @param def The string to be returned if the search has no results.
-     * @return The queried string or the default value if something went
-     * wrong during XPath evaluation.
-     */
-    public static final String getStringXPath(
-        Object root, String query, String def
-    ) {
-        String s = (String)getXPath(root, query, XPathConstants.STRING);
-        return s == null || s.length() == 0
-            ? def
-            : s;
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/CreationFilter.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,61 +0,0 @@
-package de.intevation.artifacts.common.utils;
-
-import java.util.List;
-import java.util.Map;
-import java.util.HashMap;
-
-import org.w3c.dom.Element;
-
-public class CreationFilter
-{
-    public static class Facet {
-
-        protected String name;
-        protected String index;
-
-        public Facet() {
-        }
-
-        public Facet(String name, String index) {
-            this.name  = name;
-            this.index = index;
-        }
-
-        public String getName() {
-            return name;
-        }
-
-        public String getIndex() {
-            return index;
-        }
-    }
-
-    protected Map<String, List<Facet>> outs;
-
-    public CreationFilter() {
-        outs = new HashMap<String, List<Facet>>();
-    }
-
-    public void add(String out, List<Facet> facets) {
-        outs.put(out, facets);
-    }
-
-    public Element toXML(XMLUtils.ElementCreator ec) {
-        Element filter = ec.create("filter");
-
-        for (Map.Entry<String, List<Facet>> entry: outs.entrySet()) {
-            Element out = ec.create("out");
-            out.setAttribute("name", entry.getKey());
-            for (Facet facet: entry.getValue()) {
-                Element f = ec.create("facet");
-                f.setAttribute("name", facet.getName());
-                f.setAttribute("index", facet.getIndex());
-                out.appendChild(f);
-            }
-            filter.appendChild(out);
-        }
-
-        return filter;
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/DateUtils.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,30 +0,0 @@
-package de.intevation.artifacts.common.utils;
-
-import java.util.Calendar;
-import java.util.Date;
-
-
-public class DateUtils {
-
-    private DateUtils() {
-    }
-
-
-    /**
-     * This function extracts the year as int value from <i>date</i>.
-     *
-     * @param date The source date.
-     *
-     * @return the year as integer or -1 if date is empty.
-     */
-    public static int getYearFromDate(Date date) {
-        if (date == null) {
-            return -1;
-        }
-
-        Calendar cal = Calendar.getInstance();
-        cal.setTime(date);
-
-        return cal.get(Calendar.YEAR);
-    }
-}
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/FileTools.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,613 +0,0 @@
-/*
- * Copyright (c) 2010, 2011 by Intevation GmbH
- *
- * This program is free software under the LGPL (>=v2.1)
- * Read the file LGPL.txt coming with the software for details
- * or visit http://www.gnu.org/licenses/ if it does not exist.
- */
-package de.intevation.artifacts.common.utils;
-
-import java.io.BufferedInputStream;
-import java.io.File;
-import java.io.IOException;
-import java.io.InputStream;
-import java.io.FileInputStream;
-import java.io.FileOutputStream;
-import java.io.OutputStream;
-import java.io.BufferedOutputStream;
-import java.nio.channels.FileChannel;
-
-import java.util.Deque;
-import java.util.ArrayDeque;
-import java.util.List;
-import java.util.Set;
-import java.util.HashSet;
-import java.util.ArrayList;
-import java.util.Enumeration;
-import java.util.zip.ZipFile;
-import java.util.zip.ZipEntry;
-import java.util.zip.ZipOutputStream;
-
-import org.apache.log4j.Logger;
-
-import java.security.MessageDigest;
-import java.security.NoSuchAlgorithmException;
-
-public class FileTools
-{
-    private static Logger log = Logger.getLogger(FileTools.class);
-
-    public static final String DIGEST =
-        System.getProperty("artifacts.common.file.cmp.digest", "MD5");
-
-    private FileTools() {
-    }
-
-
-    /** Remove everything after dot from name. */
-    public static final String removeExtension(String name) {
-        int index = name.lastIndexOf('.');
-        return index == -1
-            ? name
-            : name.substring(0, index);
-    }
-
-
-    public static File getDirectory(String path, String name) {
-        if (path == null || name == null) {
-            return null;
-        }
-
-        File dir = new File(path, name);
-
-        if (!dir.exists()) {
-            log.debug(
-                "Directory '" + dir.getAbsolutePath() + "' doesn't " +
-                "exist. Try to create it.");
-
-            return dir.mkdir() ? dir : null;
-        }
-        else {
-            return dir.isDirectory() ? dir : null;
-        }
-    }
-
-    public static File repair(File file) {
-        file = file.getAbsoluteFile();
-        if (file.exists()) {
-            return file;
-        }
-        Deque<String> parts = new ArrayDeque<String>();
-        File curr = file;
-        while (curr != null) {
-            String name = curr.getName();
-            if (name.length() > 0) {
-                parts.push(curr.getName());
-            }
-            curr = curr.getParentFile();
-        }
-
-        curr = null;
-        OUTER: while (!parts.isEmpty()) {
-            String f = parts.pop();
-            log.debug("fixing: '" + f + "'");
-            if (f.equals(".") || f.equals("..")) {
-                // No need to fix . or ..
-                continue;
-            }
-            if (curr == null) {
-                // XXX: Not totaly correct because there
-                // more than one root on none unix systems.
-                for (File root: File.listRoots()) {
-                    File [] files = root.listFiles();
-                    if (files == null) {
-                        log.warn("cannot list '" + root);
-                        continue;
-                    }
-                    for (File candidate: files) {
-                        if (candidate.getName().equalsIgnoreCase(f)) {
-                            curr = new File(root, candidate.getName());
-                            continue OUTER;
-                        }
-                    }
-                }
-                break;
-            }
-            else {
-                File [] files = curr.listFiles();
-                if (files == null) {
-                    log.warn("cannot list: '" + curr + "'");
-                    return file;
-                }
-                for (File candidate: files) {
-                    if (candidate.getName().equalsIgnoreCase(f)) {
-                        curr = new File(curr, candidate.getName());
-                        continue OUTER;
-                    }
-                }
-                curr = null;
-                break;
-            }
-        }
-
-        if (curr == null) {
-            log.warn("cannot repair path '" + file + "'");
-            return file;
-        }
-
-        return curr;
-    }
-
-    /** Object that can calculate hash of file, compare two hashed files etc. */
-    public static class HashedFile
-    implements Comparable<HashedFile>
-    {
-        protected File    file;
-        protected long    length;
-        protected byte [] hash;
-
-        public HashedFile(File file) {
-            this.file = file;
-            length = file.length();
-        }
-
-        public File getFile() {
-            return file;
-        }
-
-        protected byte [] getHash() {
-            if (hash == null) {
-                InputStream in = null;
-
-                try {
-                    in = new FileInputStream(file);
-
-                    MessageDigest digest = MessageDigest.getInstance(DIGEST);
-
-                    byte [] buf = new byte[40*1024];
-                    int r;
-
-                    while ((r = in.read(buf)) >= 0) {
-                        digest.update(buf, 0, r);
-                    }
-
-                    hash = digest.digest();
-                }
-                catch (IOException ioe) {
-                    log.error(ioe);
-                    hash = new byte[0];
-                }
-                catch (NoSuchAlgorithmException nsae) {
-                    log.error(nsae);
-                    hash = new byte[0];
-                }
-                finally {
-                    if (in != null) {
-                        try {
-                            in.close();
-                        }
-                        catch (IOException ioe) {
-                            log.error(ioe);
-                        }
-                    }
-                }
-            }
-            return hash;
-        }
-
-        @Override
-        public int compareTo(HashedFile other) {
-            if (length < other.length) return -1;
-            if (length > other.length) return +1;
-            return compare(getHash(), other.getHash());
-        }
-
-        private static int compare(byte [] a, byte [] b) {
-            if (a.length < b.length) return -1;
-            if (a.length > b.length) return +1;
-            for (int i = 0; i < a.length; ++i) {
-                int x = a[i] & 0xff;
-                int y = b[i] & 0xff;
-                if (x < y) return -1;
-                if (x > y) return +1;
-            }
-            return 0;
-        }
-
-        @Override
-        public boolean equals(Object other) {
-            return other instanceof HashedFile
-                && ((HashedFile)other).compareTo(this) == 0;
-        }
-
-        @Override
-        public int hashCode() {
-            return (int)(length ^ (length >>> 32));
-        }
-    } // class HashedFile
-
-    public static List<File> uniqueFiles(List<File> files) {
-
-        Set<HashedFile> set = new HashSet<HashedFile>();
-
-        for (File file: files) {
-            if (!set.add(new HashedFile(file))) {
-                log.warn("file '" + file + "' is a duplicate.");
-            }
-        }
-
-        ArrayList<File> out = new ArrayList<File>(set.size());
-        for (HashedFile hf: set) {
-            out.add(hf.file);
-        }
-
-        return out;
-    }
-
-    public interface FileVisitor {
-        boolean visit(File file);
-    } // Visitor
-
-    public static void walkTree(File root, FileVisitor visitor) {
-
-        Deque<File> stack = new ArrayDeque<File>();
-
-        stack.push(root);
-
-        while (!stack.isEmpty()) {
-            File current = stack.pop();
-            if (!visitor.visit(current)) break;
-            if (current.isDirectory()) {
-                File [] subs = current.listFiles();
-                if (subs != null) {
-                    for (File f: subs) {
-                        stack.push(f);
-                    }
-                }
-            }
-        }
-    }
-
-    /**
-     * Deletes everything in a directory.
-     *
-     * @param dir The directory.
-     */
-    public final static void deleteContent(File dir) {
-        if (dir == null || !dir.isDirectory()) {
-            return;
-        }
-
-        File[] files = dir.listFiles();
-        if (files != null) {
-            for (File file: files) {
-                deleteRecursive(file);
-            }
-        }
-
-        return;
-    }
-
-    /**
-     * Delete <i>file</i> and everything in <i>file</i> if it is a directory.
-     *
-     * @param file The file or directory.
-     * @return true, if deletion was successful - otherwise false.
-     */
-    public final static boolean deleteRecursive(File file) {
-
-        if (file == null) {
-            return false;
-        }
-
-        if (file.isDirectory()) {
-            File [] files = file.listFiles();
-            if (files != null) {
-                for (File sub: files) {
-                    if (!deleteRecursive(sub)) {
-                        return false;
-                    }
-                }
-            }
-        }
-
-        return file.delete();
-    }
-
-    /**
-     * Put the given file or directory into a zip archive.
-     *
-     * @param file The file or directory.
-     * @param outputStream The stream to write the archive to.
-     * @throws IOException if an error occured while zip creation or writing to
-     * output stream.
-     */
-    public static void createZipArchive(
-        File         file,
-        OutputStream outputStream
-    )
-    throws IOException
-    {
-        ZipOutputStream out = new ZipOutputStream(outputStream);
-
-        if (file.isFile()) {
-            copyFileToZip("", file, out);
-        }
-        else if (file.isDirectory()) {
-
-            Deque<PrefixDir> stack = new ArrayDeque<PrefixDir>();
-            stack.push(new PrefixDir(file.getName() + "/", file));
-
-            while (!stack.isEmpty()) {
-                PrefixDir pd = stack.pop();
-
-                ZipEntry dirEntry = new ZipEntry(pd.prefix);
-                out.putNextEntry(dirEntry);
-                out.closeEntry();
-
-                File [] files = pd.dir.listFiles();
-                if (files != null) {
-                    for (File sub: files) {
-                        if (sub.isDirectory()) {
-                            stack.push(new PrefixDir(
-                                pd.prefix + sub.getName() + "/",
-                                sub));
-                        }
-                        else if (sub.isFile()) {
-                            copyFileToZip(pd.prefix, sub, out);
-                        }
-                    }
-                }
-            }
-        }
-
-        out.finish();
-    }
-
-
-    public static void extractArchive(File archive, File destDir)
-    throws IOException {
-        if (!destDir.exists()) {
-            destDir.mkdir();
-        }
-
-        ZipFile zipFile = new ZipFile(archive);
-        try {
-            Enumeration<? extends ZipEntry> entries = zipFile.entries();
-
-            byte [] buffer = new byte[16384];
-
-            while (entries.hasMoreElements()) {
-                ZipEntry entry = entries.nextElement();
-
-                String entryFileName = entry.getName();
-
-                File dir = buildDirectoryHierarchyFor(entryFileName, destDir);
-                if (!dir.exists()) {
-                    dir.mkdirs();
-                }
-
-                if (!entry.isDirectory()) {
-                    BufferedInputStream bis = new BufferedInputStream(
-                        zipFile.getInputStream(entry));
-                    try {
-                        BufferedOutputStream bos = new BufferedOutputStream(
-                            new FileOutputStream(new File(destDir, entryFileName)));
-
-                        try {
-                            int len;
-                            while ((len = bis.read(buffer)) > 0) {
-                                bos.write(buffer, 0, len);
-                            }
-                            bos.flush();
-                        }
-                        finally {
-                            bos.close();
-                        }
-                    }
-                    finally {
-                        bis.close();
-                    }
-                } // is file
-            }
-        }
-        finally {
-            zipFile.close();
-        }
-    }
-
-    private static File buildDirectoryHierarchyFor(
-        String entryName,
-        File destDir)
-    {
-        int lastIndex = entryName.lastIndexOf('/');
-        String internalPathToEntry = entryName.substring(0, lastIndex + 1);
-        return new File(destDir, internalPathToEntry);
-    }
-
-    /**
-     * A class representing a directory with a prefix.
-     */
-    private static final class PrefixDir {
-
-        String prefix;
-        File   dir;
-
-        public PrefixDir(String prefix, File dir) {
-            this.prefix = prefix;
-            this.dir    = dir;
-        }
-
-    } // class PrefixDir
-
-    /**
-     * Write a file to zip archive.
-     *
-     * @param prefix A prefix.
-     * @param file The file.
-     * @param out The output stream.
-     * @throws IOException if an error occured while writing to zip output
-     * stream.
-     */
-    private static void copyFileToZip(
-        String          prefix,
-        File            file,
-        ZipOutputStream out
-    )
-    throws IOException
-    {
-        String   entryName = prefix + file.getName();
-        ZipEntry entry     = new ZipEntry(entryName);
-        out.putNextEntry(entry);
-        InputStream in = null;
-        try {
-            in =
-                new BufferedInputStream(
-                new FileInputStream(file), 20*1024);
-
-            byte [] buf = new byte[2048];
-
-            int r;
-            while ((r = in.read(buf)) > 0) {
-                out.write(buf, 0, r);
-            }
-        }
-        finally {
-            if (in != null) {
-                try { in.close(); }
-                catch (IOException ioe) {}
-            }
-        }
-        out.closeEntry();
-    }
-
-
-    /**
-     * Copies a <i>src</i> file to <i>target</i>.
-     *
-     * @param src A file (not a directory) that should be copied.
-     * @param target The destination. This might be a file or a directory.
-     *
-     * @return true, if <i>src</i> has been successfully copied; otherwise
-     * false.
-     */
-    public static boolean copyFile(File src, File target)
-    throws IOException
-    {
-        if (src == null || !src.exists()) {
-            log.warn("Source file does not exist!");
-            return false;
-        }
-
-        if (!src.canRead()) {
-            log.warn("Cannot read Source file!");
-            return false;
-        }
-
-        if (src.isDirectory()) {
-            log.warn("Source is a directory!");
-            return false;
-        }
-
-        if (target.isDirectory()) {
-            target = new File(target, src.getName());
-        }
-
-        FileInputStream  in  = null;
-        FileOutputStream out = null;
-
-        try {
-            in  = new FileInputStream(src);
-            out = new FileOutputStream(target);
-
-            FileChannel inChannel  = in.getChannel();
-            FileChannel outChannel = out.getChannel();
-
-            inChannel.transferTo(0l, inChannel.size(), outChannel);
-
-            return true;
-        }
-        catch (IOException ioe) {
-            log.warn(ioe, ioe);
-        }
-        finally {
-            if (in != null) {
-                try {
-                    in.close();
-                }
-                catch (IOException ioe) { /* do nothing here */ }
-            }
-
-            if (out != null) {
-                try {
-                    out.close();
-                }
-                catch (IOException ioe) { /* do nothing here */ }
-            }
-        }
-
-        return false;
-    }
-
-
-    /**
-     * Copies a directory <i>source</i> to a destination path <i>dest</i>.
-     *
-     * @param source A directory that should be copied.
-     * @param dest A destination directory which is created if it is not
-     * existing yet.
-     *
-     * @return true, if the directory has been successfully copied; otherwise
-     * false.
-     */
-    public static boolean copyDirectory(final File source, final File dest) {
-        if (source == null || !source.exists()) {
-            log.warn("Source directory does not exist!");
-            return false;
-        }
-
-        if (!source.isDirectory()) {
-            log.warn("Source is not a directory!");
-            return false;
-        }
-
-        if (dest == null) {
-            log.warn("Destination directory is null!");
-            return false;
-        }
-
-        if (!dest.exists()) {
-            if (!dest.mkdir()) {
-                log.warn("Cannot create destination directory!");
-                return false;
-            }
-        }
-
-        File[] children = source.listFiles();
-        int    failed   = 0;
-
-        if (children != null && children.length > 0) {
-            for (File child: children) {
-                if (child.isFile()) {
-                    try {
-                        if (!copyFile(child, dest)) {
-                            failed++;
-                        }
-                    }
-                    catch (IOException ioe) {
-                        log.warn(ioe, ioe);
-                        failed++;
-                    }
-                }
-                else if (child.isDirectory()) {
-                    copyDirectory(child, new File(dest, child.getName()));
-                }
-            }
-        }
-
-        log.debug("Failed to copy " + failed + " files.");
-
-        return true;
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/JSON.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,445 +0,0 @@
-package de.intevation.artifacts.common.utils;
-
-import java.util.Map;
-import java.util.LinkedHashMap;
-import java.util.List;
-import java.util.ArrayList;
-import java.util.Iterator;
-
-import java.io.IOException;
-import java.io.PushbackInputStream;
-import java.io.InputStream;
-import java.io.PrintWriter;
-import java.io.ByteArrayInputStream;
-import java.io.StringWriter;
-
-import java.nio.charset.Charset;
-import java.nio.charset.UnsupportedCharsetException;
-
-public final class JSON
-{
-    private JSON() {
-    }
-
-    private static final boolean isDigit(int c) {
-        return c >= '0' && c <= '9';
-    }
-
-    public static final boolean isWhitespace(int c) {
-        return c == ' '  || c == '\n' || c == '\r'
-            || c == '\t' || c == '\f';
-    }
-
-    private static final void match(int c, int x) throws IOException {
-        if (c != x) {
-            throw new IOException(
-                "Expecting '" + (char)c + "' found '" + (char)x + "'");
-        }
-    }
-
-    private static final int eof(InputStream in)
-    throws IOException
-    {
-        int c = in.read();
-        if (c == -1) {
-            throw new IOException("EOF unexpected.");
-        }
-        return c;
-    }
-
-    private static final int whitespace(InputStream in)
-    throws IOException
-    {
-        int c;
-        while (isWhitespace(c = eof(in)));
-        return c;
-    }
-
-    private static final int parseHex(String hex) throws IOException {
-        try {
-            return Integer.parseInt(hex, 16);
-        }
-        catch (NumberFormatException nfe) {
-            throw new IOException("'" + hex + "' is not a hex string.");
-        }
-    }
-
-    public static final String jsonString(String string) {
-        StringBuilder sb = new StringBuilder(string.length()+2);
-
-        sb.append('"');
-
-        for (int i = 0, N = string.length(); i < N; ++i) {
-            char c = string.charAt(i);
-            switch (c) {
-                case '"':  sb.append("\\\""); break;
-                case '\t': sb.append("\\t"); break;
-                case '\r': sb.append("\\r"); break;
-                case '\n': sb.append("\\n"); break;
-                case '\b': sb.append("\\b"); break;
-                case '\f': sb.append("\\f"); break;
-                default:
-                    if (c >= 128) {
-                        sb.append("\\u");
-                        String hex = Integer.toHexString((int)c);
-                        for (int j = 4-hex.length(); j > 0; --j) {
-                            sb.append('0');
-                        }
-                        sb.append(hex);
-                    }
-                    else {
-                        sb.append(c);
-                    }
-            }
-        }
-
-        sb.append('"');
-
-        return sb.toString();
-    }
-
-    public static String toJSONString(Map<String, Object> map) {
-        StringWriter sw = new StringWriter();
-        PrintWriter pw = new PrintWriter(sw);
-        write(pw, map);
-        pw.flush();
-        return sw.toString();
-    }
-
-
-    public static void write(PrintWriter out, Map<String, Object> map) {
-        writeObject(out, map);
-    }
-
-    private static void writeValue(PrintWriter out, Object value) {
-        if (value instanceof Map) {
-            writeObject(out, (Map)value);
-        }
-        else if (value instanceof List) {
-            writeList(out, (List)value);
-        }
-        else if (value instanceof Number) {
-            out.print(value);
-        }
-        else if (value instanceof Boolean) {
-            out.print(((Boolean)value) ? "true" : "false");
-        }
-        else if (value == null) {
-            out.print("null");
-        }
-        else {
-            out.print(jsonString(value.toString()));
-        }
-    }
-
-    private static void writeObject(PrintWriter out, Map map) {
-
-        out.print('{');
-        Iterator iter = map.entrySet().iterator();
-        while (iter.hasNext()) {
-            Map.Entry entry = (Map.Entry)iter.next();
-            out.print(jsonString(entry.getKey().toString()));
-            out.print(':');
-            writeValue(out, entry.getValue());
-            if (iter.hasNext()) {
-                out.print(',');
-            }
-        }
-        out.print('}');
-    }
-
-    private static void writeList(PrintWriter out, List list) {
-        out.print('[');
-        Iterator iter = list.iterator();
-        while (iter.hasNext()) {
-            writeValue(out, iter.next());
-            if (iter.hasNext()) {
-                out.print(',');
-            }
-        }
-        out.print(']');
-    }
-
-    public static Map<String, Object> parse(String in)
-    throws IOException
-    {
-        return parse(asInputStream(in));
-    }
-
-    private static InputStream asInputStream(String in) {
-        byte [] bytes;
-        try {
-            bytes = in.getBytes(Charset.forName("US-ASCII"));
-        }
-        catch (UnsupportedCharsetException uce) {
-            // Should not happen.
-            bytes = in.getBytes();
-        }
-        return new ByteArrayInputStream(bytes);
-    }
-
-    public static Map<String, Object> parse(InputStream in)
-    throws IOException
-    {
-        return parseObject(new PushbackInputStream(in, 1));
-    }
-
-    public static Map<String, Object> parse(PushbackInputStream in)
-    throws IOException
-    {
-        return parseObject(in);
-    }
-
-    private static final String parseString(
-        PushbackInputStream in
-    )
-    throws IOException
-    {
-        StringBuilder sb = new StringBuilder();
-
-        int mode = 0;
-
-        char [] hex = new char[4];
-
-        match('"', eof(in));
-
-        OUT: for (int c = eof(in);; c = eof(in)) {
-
-            switch (mode) {
-                case 0:
-                    if (c == '"') {
-                        break OUT;
-                    }
-                    if (c == '\\') {
-                        mode = 1;
-                    }
-                    else {
-                        sb.append((char)c);
-                    }
-                    break;
-                case 1:
-                    switch (c) {
-                        case 'u':
-                            mode = 2;
-                            continue;
-                        case 'b':
-                            sb.append('\b');
-                            break;
-                        case 'f':
-                            sb.append('\f');
-                            break;
-                        case 'n':
-                            sb.append('\n');
-                            break;
-                        case 'r':
-                            sb.append('\r');
-                            break;
-                        case 't':
-                            sb.append('\t');
-                            break;
-                        default:
-                            sb.append((char)c);
-                    }
-                    mode = 0;
-                    break;
-                case 2:
-                    hex[0] = (char)c;
-                    mode = 3;
-                    break;
-                case 3:
-                    hex[1] = (char)c;
-                    mode = 4;
-                    break;
-                case 4:
-                    hex[2] = (char)c;
-                    mode = 5;
-                    break;
-                case 5:
-                    hex[3] = (char)c;
-                    sb.append((char)parseHex(new String(hex)));
-                    mode = 0;
-                    break;
-            }
-        }
-        return sb.toString();
-    }
-
-    private static final Boolean parseTrue(InputStream in)
-    throws IOException
-    {
-        match('t', eof(in));
-        match('r', eof(in));
-        match('u', eof(in));
-        match('e', eof(in));
-        return Boolean.TRUE;
-    }
-
-    private static final Boolean parseFalse(InputStream in)
-    throws IOException
-    {
-        match('f', eof(in));
-        match('a', eof(in));
-        match('l', eof(in));
-        match('s', eof(in));
-        match('e', eof(in));
-        return Boolean.FALSE;
-    }
-
-    private static final Object parseNull(InputStream in)
-    throws IOException
-    {
-        match('n', eof(in));
-        match('u', eof(in));
-        match('l', eof(in));
-        match('l', eof(in));
-        return null;
-    }
-
-    private static final Number parseNumber(PushbackInputStream in)
-    throws IOException
-    {
-        StringBuilder sb = new StringBuilder();
-
-        boolean isInteger = true;
-
-        int c;
-        OUT: for (;;) {
-            switch (c = eof(in)) {
-                case '0': case '1': case '2': case '3': case '4':
-                case '5': case '6': case '7': case '8': case '9':
-                case '-': case '+':
-                    sb.append((char)c);
-                    break;
-                case '.': case 'e': case 'E':
-                    isInteger = false;
-                    sb.append((char)c);
-                    break;
-                default:
-                    in.unread(c);
-                    break OUT;
-            }
-        }
-
-        try {
-            if (isInteger) {
-                return sb.length() > 9
-                    ? (Number)Long   .valueOf(sb.toString())
-                    : (Number)Integer.valueOf(sb.toString());
-            }
-            return (Number)Double.valueOf(sb.toString());
-        }
-        catch (NumberFormatException nfe) {
-            throw new IOException("Not a number '" + sb + "'");
-        }
-    }
-
-    private static List<Object> parseList(PushbackInputStream in)
-    throws IOException
-    {
-        List<Object> list = new ArrayList<Object>();
-        match('[', whitespace(in));
-        int c = whitespace(in);
-        if (c == ']') {
-            return list;
-        }
-
-        for (;; c = whitespace(in)) {
-            Object value;
-            in.unread(c);
-            switch (c) {
-                case '{':
-                    value = parseObject(in);
-                    break;
-                case '[':
-                    value = parseList(in);
-                    break;
-                case '"':
-                    value = parseString(in);
-                    break;
-                case 't':
-                    value = parseTrue(in);
-                    break;
-                case 'f':
-                    value = parseFalse(in);
-                    break;
-                case 'n':
-                    value = parseNull(in);
-                    break;
-                default:
-                    value = parseNumber(in);
-            }
-            list.add(value);
-
-            if ((c = whitespace(in)) == ']') break;
-            match(',', c);
-        }
-        return list;
-    }
-
-    private static void parsePair(
-        PushbackInputStream in,
-        Map<String, Object> pairs
-    )
-    throws IOException
-    {
-        in.unread(whitespace(in));
-        String string = parseString(in);
-        match(':', whitespace(in));
-
-        Object value;
-
-        int c = whitespace(in);
-        in.unread(c);
-        switch (c) {
-            case '{':
-                value = parseObject(in);
-                break;
-            case '[':
-                value = parseList(in);
-                break;
-            case '"':
-                value = parseString(in);
-                break;
-            case 't':
-                value = parseTrue(in);
-                break;
-            case 'f':
-                value = parseFalse(in);
-                break;
-            case 'n':
-                value = parseNull(in);
-                break;
-            default:
-                value = parseNumber(in);
-        }
-        pairs.put(string, value);
-    }
-
-    private static Map<String, Object> parseObject(PushbackInputStream in)
-    throws IOException
-    {
-        Map<String, Object> pairs = new LinkedHashMap<String, Object>();
-
-        int c = whitespace(in);
-        match('{', c);
-
-        if ((c = whitespace(in)) == '}') {
-            return pairs;
-        }
-
-        in.unread(c);
-
-        for (;;) {
-            parsePair(in, pairs);
-
-            if ((c = whitespace(in)) == '}') {
-                break;
-            }
-
-            if (c == '}') break;
-            match(',', c);
-        }
-
-        return pairs;
-    }
-}
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/LRUCache.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,26 +0,0 @@
-package de.intevation.artifacts.common.utils;
-
-import java.util.Map;
-import java.util.LinkedHashMap;
-
-public class LRUCache<K, V>
-extends      LinkedHashMap<K, V>
-{
-    public static final int DEFAULT_MAX_CAPACITY = 25;
-
-    private int maxCapacity;
-
-    public LRUCache() {
-        this(DEFAULT_MAX_CAPACITY);
-    }
-
-    public LRUCache(int maxCapacity) {
-        this.maxCapacity = maxCapacity;
-    }
-
-    @Override
-    protected boolean removeEldestEntry(Map.Entry<K, V> eldest) {
-        return size() > maxCapacity;
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/MapXPathVariableResolver.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,48 +0,0 @@
-/*
- * Copyright (c) 2011 by Intevation GmbH
- *
- * This program is free software under the LGPL (>=v2.1)
- * Read the file LGPL.txt coming with the software for details
- * or visit http://www.gnu.org/licenses/ if it does not exist.
- */
-package de.intevation.artifacts.common.utils;
-
-import java.util.HashMap;
-import java.util.Map;
-
-import javax.xml.namespace.QName;
-import javax.xml.xpath.XPathVariableResolver;
-
-
-/**
- *  @author <a href="mailto:ingo.weinzierl@intevation.de">Ingo Weinzierl</a>
- */
-public class MapXPathVariableResolver implements XPathVariableResolver {
-
-    protected Map<String, String> variables;
-
-
-    public MapXPathVariableResolver() {
-        this.variables = new HashMap<String, String>();
-    }
-
-
-    public MapXPathVariableResolver(Map<String, String> variables) {
-        this.variables = variables;
-    }
-
-
-    public void addVariable(String name, String value) {
-        if (name != null && value != null) {
-            variables.put(name, value);
-        }
-    }
-
-
-    @Override
-    public Object resolveVariable(QName variableName) {
-        String key = variableName.getLocalPart();
-        return variables.get(key);
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/StringUtils.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,148 +0,0 @@
-/*
- * Copyright (c) 2010 by Intevation GmbH
- *
- * This program is free software under the LGPL (>=v2.1)
- * Read the file LGPL.txt coming with the software for details
- * or visit http://www.gnu.org/licenses/ if it does not exist.
- */
-
-package de.intevation.artifacts.common.utils;
-
-import java.io.UnsupportedEncodingException;
-
-import java.util.UUID;
-
-import org.apache.commons.codec.DecoderException;
-
-import org.apache.commons.codec.binary.Hex;
-
-import org.apache.log4j.Logger;
-
-/**
- * Commonly used string functions.
- *
- * @author <a href="mailto:sascha.teichmann@intevation.de">Sascha L. Teichmann</a>
- */
-public final class StringUtils
-{
-    private static Logger logger = Logger.getLogger(StringUtils.class);
-
-    /**
-     * Generated a random UUIDv4 in form of a string.
-     * @return the UUID
-     */
-    public static final String newUUID() {
-        return UUID.randomUUID().toString();
-    }
-
-    /**
-     * Checks if a given string is a valid UUID.
-     * @param uuid The string to test.
-     * @return true if the string is a valid UUID else false.
-     */
-    public static final boolean checkUUID(String uuid) {
-        try {
-            UUID.fromString(uuid);
-        }
-        catch (IllegalArgumentException iae) {
-            logger.warn(iae.getLocalizedMessage());
-            return false;
-        }
-        return true;
-    }
-
-    /**
-     * Returns the UTF-8 byte array representation of a given string.
-     * @param s The string to be transformed.
-     * @return The byte array representation.
-     */
-    public static final byte [] getUTF8Bytes(String s) {
-        try {
-            return s.getBytes("UTF-8");
-        }
-        catch (UnsupportedEncodingException usee) {
-            logger.error(usee.getLocalizedMessage(), usee);
-            return s.getBytes();
-        }
-    }
-
-    /**
-     * Tries to convert a Base64 encoded string into the
-     * corresponing byte array.
-     * @param s The Base64 encoded string
-     * @return The byte array representation or null if
-     * an decoding error occurs.
-     */
-    public static final byte [] decodeHex(String s) {
-        try {
-            return Hex.decodeHex(s.toCharArray());
-        }
-        catch (DecoderException de) {
-            return null;
-        }
-    }
-
-    public static final String repeat(String s, int count, String sep) {
-        if (count <= 0) {
-            return "";
-        }
-        StringBuilder sb = new StringBuilder(s);
-        for (--count; count > 0; --count) {
-            sb.append(sep).append(s);
-        }
-        return sb.toString();
-    }
-
-    public static final String repeat(char c, int count, char sep) {
-        if (count <= 0) {
-            return "";
-        }
-        StringBuilder sb = new StringBuilder(2*count-1).append(c);
-        for (--count; count > 0; --count) {
-            sb.append(sep).append(c);
-        }
-        return sb.toString();
-    }
-
-    public static final String [] toUpperCase(String [] s) {
-        if (s == null) {
-            return null;
-        }
-        String [] d = new String[s.length];
-        for (int i = 0; i < s.length; ++i) {
-            if (s[i] != null) {
-                d[i] = s[i].toUpperCase();
-            }
-        }
-        return d;
-    }
-
-    public static String join(String sep, String [] strings) {
-        StringBuilder sb = new StringBuilder();
-        for (int i = 0; i < strings.length; ++i) {
-            if (i > 0) sb.append(sep);
-            sb.append(strings[i]);
-        }
-        return sb.toString();
-    }
-
-    public static final String [] join(String [] a, String [] b) {
-        if (a == null && b == null) return null;
-        if (a == null) return b;
-        if (b == null) return a;
-        String [] dst = new String[a.length + b.length];
-        System.arraycopy(a, 0, dst, 0, a.length);
-        System.arraycopy(b, 0, dst, a.length, b.length);
-        return dst;
-    }
-
-    public static final boolean contains(String needle, String [] haystack) {
-        for (String stray: haystack) {
-            if (needle.equals(stray)) {
-                return true;
-            }
-        }
-        return false;
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/XMLUtils.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,617 +0,0 @@
-/*
- * Copyright (c) 2010 by Intevation GmbH
- *
- * This program is free software under the LGPL (>=v2.1)
- * Read the file LGPL.txt coming with the software for details
- * or visit http://www.gnu.org/licenses/ if it does not exist.
- */
-
-package de.intevation.artifacts.common.utils;
-
-import java.util.List;
-import java.util.ArrayList;
-import java.util.Map;
-import java.util.LinkedHashMap;
-import java.util.zip.GZIPInputStream;
-import java.util.zip.GZIPOutputStream;
-
-import java.io.ByteArrayInputStream;
-import java.io.FileInputStream;
-import java.io.BufferedInputStream;
-import java.io.ByteArrayOutputStream;
-import java.io.File;
-import java.io.IOException;
-import java.io.InputStream;
-import java.io.OutputStream;
-import java.io.StringWriter;
-
-import javax.xml.namespace.NamespaceContext;
-import javax.xml.namespace.QName;
-
-import javax.xml.parsers.DocumentBuilderFactory;
-import javax.xml.parsers.ParserConfigurationException;
-
-import javax.xml.transform.Transformer;
-import javax.xml.transform.TransformerConfigurationException;
-import javax.xml.transform.TransformerException;
-import javax.xml.transform.TransformerFactory;
-import javax.xml.transform.TransformerFactoryConfigurationError;
-
-import javax.xml.transform.dom.DOMSource;
-
-import javax.xml.transform.stream.StreamResult;
-
-import javax.xml.xpath.XPath;
-import javax.xml.xpath.XPathConstants;
-import javax.xml.xpath.XPathExpressionException;
-import javax.xml.xpath.XPathFactory;
-import javax.xml.xpath.XPathVariableResolver;
-
-import org.apache.log4j.Logger;
-
-import org.w3c.dom.Attr;
-import org.w3c.dom.Document;
-import org.w3c.dom.Element;
-import org.w3c.dom.Node;
-
-import org.xml.sax.SAXException;
-
-/**
- * Some helper functions to ease work with XML concering namespaces, XPATH
- * and so on.
- *  @author <a href="mailto:sascha.teichmann@intevation.de">Sascha L. Teichmann</a>
- */
-public final class XMLUtils
-{
-    /** W3C URL of XForms. */
-    public static final String XFORM_URL    = "http://www.w3.org/2002/xforms";
-
-    /** W3C prefix of XForms. */
-    public static final String XFORM_PREFIX = "xform";
-
-    /** Logger for this class. */
-    private static Logger logger = Logger.getLogger(XMLUtils.class);
-
-    private XMLUtils() {
-    }
-
-    /**
-     * Helper class to generate elements and attributes with
-     * namespaces.
-     */
-    public static class ElementCreator
-    {
-        /** Owner document of the elements to be created. */
-        protected Document document;
-
-        /** Namespace to be used. */
-        protected String   ns;
-
-        /** Prefix to be used. */
-        protected String   prefix;
-
-        /**
-         * Constructor to create an element/attribute creator
-         * with a given namespace and namespace prefix using a
-         * given owner document.
-         * @param document The owning document
-         * @param ns       The namespace
-         * @param prefix   The namespace prefix
-         */
-        public ElementCreator(Document document, String ns, String prefix) {
-            this.document = document;
-            this.ns       = ns;
-            this.prefix   = prefix;
-        }
-
-        /**
-         * Creates a new element using the owning document with
-         * the this creators namespace and namespace prefix.
-         * @param name The name of the element
-         * @return     The new element
-         */
-        public Element create(String name) {
-            Element element = document.createElementNS(ns, name);
-            element.setPrefix(prefix);
-            return element;
-        }
-
-        /**
-         * Adds a new attribute and its value to a given element.
-         * It does not set the namespace prefix.
-         * @param element The element to add the attribute to
-         * @param name    The name of the attribute
-         * @param value   The value of the attribute
-         */
-        public void addAttr(Element element, String name, String value) {
-            addAttr(element, name, value, false);
-        }
-
-        /**
-         * Adds a new attribute and its value to a given element.
-         * If the namespace prefix is used is decided by the 'addPrefix' flag.
-         * @param element The element to add the attribute to
-         * @param name    The name of the attribute
-         * @param value   The value of the attribute
-         * @param addPrefix If true the creators namespace prefix is
-         * set on the attribute.
-         */
-        public void addAttr(
-            Element element,
-            String  name,
-            String  value,
-            boolean addPrefix
-        ) {
-            if (addPrefix) {
-                Attr attr = document.createAttributeNS(ns, name);
-                attr.setValue(value);
-                attr.setPrefix(prefix);
-
-                element.setAttributeNode(attr);
-            }
-            else {
-                element.setAttribute(name, value);
-            }
-        }
-    } // class ElementCreator
-
-    /**
-     * Creates a new XML document
-     * @return the new XML document ot null if something went wrong during
-     * creation.
-     */
-    public static final Document newDocument() {
-        DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
-        factory.setNamespaceAware(true);
-
-        try {
-            return factory.newDocumentBuilder().newDocument();
-        }
-        catch (ParserConfigurationException pce) {
-            logger.error(pce.getLocalizedMessage(), pce);
-        }
-        return null;
-    }
-
-
-    /**
-     * Create xml/string representation of element (nested in otherwise empty
-     * document).
-     * @param element element to inspect in string.
-     * @return string with xml representation of element.
-     */
-    public final static String toString(Node node) {
-        Document doc = newDocument();
-        doc.appendChild(doc.importNode(node,true));
-        return toString(doc);
-    }
-
-
-    /**
-     * Loads a XML document namespace aware from a file
-     * @param file The file to load.
-     * @return the XML document or null if something went wrong
-     * during loading.
-     */
-    public static final Document parseDocument(File file) {
-        InputStream inputStream = null;
-        try {
-            inputStream = new BufferedInputStream(new FileInputStream(file));
-            return parseDocument(inputStream);
-        }
-        catch (IOException ioe) {
-            logger.error(ioe.getLocalizedMessage(), ioe);
-        }
-        finally {
-            if (inputStream != null) {
-                try { inputStream.close(); }
-                catch (IOException ioe) {}
-            }
-        }
-        return null;
-    }
-
-    /**
-     * Parses a String to a xml document.
-     *
-     * @param string The xml string
-     * @return the XML document or null if something went wrong.
-     */
-    public static final Document parseDocument(String string) {
-        InputStream inputStream = new ByteArrayInputStream(string.getBytes());
-        return parseDocument(inputStream);
-    }
-
-
-    public static final Document parseDocument(InputStream inputStream) {
-        return parseDocument(inputStream, Boolean.TRUE);
-    }
-
-    public static final Document parseDocument(
-        InputStream inputStream,
-        Boolean     namespaceAware
-    ) {
-        DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
-
-        if (namespaceAware != null) {
-            factory.setNamespaceAware(namespaceAware.booleanValue());
-        }
-
-        try {
-            return factory.newDocumentBuilder().parse(inputStream);
-        }
-        catch (ParserConfigurationException pce) {
-            logger.error(pce.getLocalizedMessage(), pce);
-        }
-        catch (SAXException se) {
-            logger.error(se.getLocalizedMessage(), se);
-        }
-        catch (IOException ioe) {
-            logger.error(ioe.getLocalizedMessage(), ioe);
-        }
-        return null;
-    }
-
-    /**
-     * Creates a new XPath without a namespace context.
-     * @return the new XPath.
-     */
-    public static final XPath newXPath() {
-        return newXPath(null, null);
-    }
-
-    /**
-     * Creates a new XPath with a given namespace context.
-     * @param namespaceContext The namespace context to be used or null
-     * if none should be used.
-     * @return The new XPath
-     */
-    public static final XPath newXPath(
-        NamespaceContext      namespaceContext,
-        XPathVariableResolver resolver)
-    {
-        XPathFactory factory = XPathFactory.newInstance();
-        XPath        xpath   = factory.newXPath();
-        if (namespaceContext != null) {
-            xpath.setNamespaceContext(namespaceContext);
-        }
-
-        if (resolver != null) {
-            xpath.setXPathVariableResolver(resolver);
-        }
-        return xpath;
-    }
-
-    /**
-     * Evaluates an XPath query on a given object and returns the result
-     * as a given type. No namespace context is used.
-     * @param root  The object which is used as the root of the tree to
-     * be searched in.
-     * @param query The XPath query
-     * @param returnTyp The type of the result.
-     * @return The result of type 'returnTyp' or null if something
-     * went wrong during XPath evaluation.
-     */
-    public static final Object xpath(
-        Object root,
-        String query,
-        QName  returnTyp
-    ) {
-        return xpath(root, query, returnTyp, null);
-    }
-
-    /**
-     * Evaluates an XPath query on a given object and returns the result
-     * as a string. A given namespace context is used.
-     * @param root  The object which is used as the root of the tree to
-     * be searched in.
-     * @param query The XPath query
-     * @param namespaceContext The namespace context to be used or null
-     * if none should be used.
-     * @return The result of the query or null if something went wrong
-     * during XPath evaluation.
-     */
-    public static final String xpathString(
-        Object root, String query, NamespaceContext namespaceContext
-    ) {
-        return (String)xpath(
-            root, query, XPathConstants.STRING, namespaceContext);
-    }
-
-    /**
-     * Evaluates an XPath query on a given object and returns the result
-     * as a given type. Optionally a namespace context is used.
-     * @param root The object which is used as the root of the tree to
-     * be searched in.
-     * @param query The XPath query
-     * @param returnType The type of the result.
-     * @param namespaceContext The namespace context to be used or null
-     * if none should be used.
-     * @return The result of type 'returnTyp' or null if something
-     * went wrong during XPath evaluation.
-     */
-    public static final Object xpath(
-        Object           root,
-        String           query,
-        QName            returnType,
-        NamespaceContext namespaceContext
-    ) {
-        return xpath(root, query, returnType, namespaceContext, null);
-    }
-
-    public static final Object xpath(
-        Object           root,
-        String           query,
-        QName            returnType,
-        NamespaceContext namespaceContext,
-        Map<String, String> variables)
-    {
-        if (root == null) {
-            return null;
-        }
-
-        XPathVariableResolver resolver = variables != null
-            ? new MapXPathVariableResolver(variables)
-            : null;
-
-        try {
-            XPath xpath = newXPath(namespaceContext, resolver);
-            if (xpath != null) {
-                return xpath.evaluate(query, root, returnType);
-            }
-        }
-        catch (XPathExpressionException xpee) {
-            logger.error(xpee.getLocalizedMessage(), xpee);
-        }
-
-        return null;
-    }
-
-    /**
-     * Streams out an XML document to a given output stream.
-     * @param document The document to be streamed out.
-     * @param out      The output stream to be used.
-     * @return true if operation succeeded else false.
-     */
-    public static boolean toStream(Document document, OutputStream out) {
-        try {
-            Transformer transformer =
-                TransformerFactory.newInstance().newTransformer();
-            DOMSource    source = new DOMSource(document);
-            StreamResult result = new StreamResult(out);
-            transformer.transform(source, result);
-            return true;
-        }
-        catch (TransformerConfigurationException tce) {
-            logger.error(tce.getLocalizedMessage(), tce);
-        }
-        catch (TransformerFactoryConfigurationError tfce) {
-            logger.error(tfce.getLocalizedMessage(), tfce);
-        }
-        catch (TransformerException te) {
-            logger.error(te.getLocalizedMessage(), te);
-        }
-
-        return false;
-    }
-
-    public static String toString(Document document) {
-        try {
-            Transformer transformer =
-                TransformerFactory.newInstance().newTransformer();
-            DOMSource    source = new DOMSource(document);
-            StringWriter out    = new StringWriter();
-            StreamResult result = new StreamResult(out);
-            transformer.transform(source, result);
-            out.flush();
-            return out.toString();
-        }
-        catch (TransformerConfigurationException tce) {
-            logger.error(tce.getLocalizedMessage(), tce);
-        }
-        catch (TransformerFactoryConfigurationError tfce) {
-            logger.error(tfce.getLocalizedMessage(), tfce);
-        }
-        catch (TransformerException te) {
-            logger.error(te.getLocalizedMessage(), te);
-        }
-
-        return null;
-    }
-
-    public static byte [] toByteArray(Document document) {
-        return toByteArray(document, false);
-    }
-
-    /**
-     * Transforms an XML document into a byte array.
-     * @param document The document to be streamed out.
-     * @param compress The document should be compressed, too.
-     * @return the byte array or null if operation failed or
-     * document is null.
-     */
-    public static byte [] toByteArray(Document document, boolean compress) {
-        if (document != null) {
-            try {
-                ByteArrayOutputStream baos = new ByteArrayOutputStream();
-                OutputStream out = compress
-                    ? new GZIPOutputStream(baos)
-                    : baos;
-                boolean success = toStream(document, out);
-                out.flush();
-                out.close();
-                return success
-                    ? baos.toByteArray()
-                    : null;
-            }
-            catch (IOException ioe) {
-                logger.error(ioe);
-            }
-        }
-        return null;
-    }
-
-    public static Document fromByteArray(byte [] data) {
-        return fromByteArray(data, false);
-    }
-
-    public static Document fromByteArray(byte [] data, boolean decompress) {
-        if (data != null) {
-            InputStream in = new ByteArrayInputStream(data);
-            try {
-                if (decompress) {
-                    in = new GZIPInputStream(in);
-                }
-                return parseDocument(in);
-            }
-            catch (IOException ioe) {
-                logger.error(ioe);
-            }
-            finally {
-                try {
-                    in.close();
-                }
-                catch (IOException ioe) {
-                    logger.error(ioe);
-                }
-            }
-        }
-        return null;
-    }
-
-    private static class BuildResult {
-        List<Node>          children;
-        Map<String, String> attributes;
-        BuildResult() {
-            children   = new ArrayList<Node>();
-            attributes = new LinkedHashMap<String, String>();
-        }
-
-        void setAttributes(Element element) {
-            for (Map.Entry<String, String> entry: attributes.entrySet()) {
-                element.setAttribute(entry.getKey(), entry.getValue());
-            }
-        }
-
-        void finish(Element element) {
-            setAttributes(element);
-            for (Node child: children) {
-                element.appendChild(child);
-            }
-        }
-
-        void add(Node node) {
-            children.add(node);
-        }
-
-        void add(String key, Object value) {
-            attributes.put(key, value != null ? value.toString() : "null");
-        }
-
-        int numChildren() {
-            return children.size();
-        }
-
-        Node firstChild() {
-            return children.get(0);
-        }
-    } // class BuildResult
-
-    private static BuildResult recursiveBuild(
-        List     list,
-        Document document
-    ) {
-        BuildResult result = new BuildResult();
-        for (Object entry: list) {
-            if (entry instanceof Map) {
-                BuildResult subResult = recursiveBuild(
-                    (Map<String, Object>)entry, document);
-                if (subResult.numChildren() == 1) {
-                    result.add(subResult.firstChild());
-                }
-                else {
-                    Element element = document.createElement("map");
-                    subResult.finish(element);
-                    result.add(element);
-                }
-            }
-            else if (entry instanceof List) {
-                Element element = document.createElement("list");
-                BuildResult subResult = recursiveBuild((List)entry, document);
-                subResult.finish(element);
-                result.add(element);
-            }
-            else {
-                Element element = document.createElement("entry");
-                element.setAttribute(
-                    "value",
-                    entry != null ? entry.toString() : "null");
-            }
-        }
-        return result;
-    }
-
-    private static BuildResult recursiveBuild(
-        Map<String, Object> map,
-        Document            document
-    ) {
-        BuildResult result = new BuildResult();
-
-        List<Node> nodes = new ArrayList<Node>();
-        for (Map.Entry<String, Object> entry: map.entrySet()) {
-            Object value = entry.getValue();
-            if (value instanceof Map) {
-                Element element = document.createElement(entry.getKey());
-                BuildResult subResult = recursiveBuild(
-                    (Map<String, Object>)value, document);
-                subResult.finish(element);
-                result.add(element);
-            }
-            else if (value instanceof List) {
-                Element element = document.createElement(entry.getKey());
-                BuildResult subResult = recursiveBuild((List)value, document);
-                subResult.finish(element);
-                result.add(element);
-            }
-            else {
-                result.add(entry.getKey(), value);
-            }
-        }
-        return result;
-    }
-
-    public static Document jsonToXML(String input) {
-        Document document = newDocument();
-
-        if (document == null) {
-            return null;
-        }
-
-        Map<String, Object> map;
-        try {
-            map = JSON.parse(input);
-        }
-        catch (IOException ioe) {
-            logger.error(ioe);
-            return null;
-        }
-
-        BuildResult roots = recursiveBuild(map, document);
-
-        int N = roots.children.size();
-
-        if (N == 1) {
-            document.appendChild(roots.children.get(0));
-        }
-        else if (N > 1) {
-            Node root = document.createElement("root");
-            for (int i = 0; i < N; ++i) {
-                root.appendChild(roots.children.get(i));
-            }
-            document.appendChild(root);
-        }
-
-        return document;
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- a/artifacts-common/src/main/java/de/intevation/artifacts/common/utils/XSLTransformer.java	Thu Apr 25 10:50:31 2013 +0200
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,73 +0,0 @@
-package de.intevation.artifacts.common.utils;
-
-import java.io.InputStream;
-import java.io.StringWriter;
-
-import java.util.HashMap;
-import java.util.Map;
-
-import javax.xml.transform.Source;
-import javax.xml.transform.Transformer;
-import javax.xml.transform.TransformerConfigurationException;
-import javax.xml.transform.TransformerException;
-import javax.xml.transform.TransformerFactory;
-
-import javax.xml.transform.dom.DOMSource;
-
-import javax.xml.transform.stream.StreamResult;
-import javax.xml.transform.stream.StreamSource;
-
-import org.apache.log4j.Logger;
-
-import org.w3c.dom.Node;
-
-public class XSLTransformer {
-
-    private static Logger log = Logger.getLogger(XSLTransformer.class);
-
-    protected Map<String, Object> parameters;
-
-    public XSLTransformer() {
-    }
-
-    public String transform(Node source, InputStream transform) {
-
-        try {
-            Source templateSource = new StreamSource(transform);
-            TransformerFactory xformFactory =
-                TransformerFactory.newInstance();
-            Transformer transformer =
-                xformFactory.newTransformer(templateSource);
-
-            if (parameters != null) {
-                for (Map.Entry<String, Object> entry: parameters.entrySet()) {
-                    transformer.setParameter(entry.getKey(), entry.getValue());
-                }
-            }
-
-            StringWriter result = new StringWriter();
-
-            DOMSource    src = new DOMSource(source);
-            StreamResult dst = new StreamResult(result);
-            transformer.transform(src, dst);
-
-            return result.toString();
-        }
-        catch (TransformerConfigurationException tce) {
-            log.error(tce, tce);
-        }
-        catch (TransformerException te) {
-            log.error(te, te);
-        }
-
-        return null;
-    }
-
-    public void addParameter(String key, Object value) {
-        if (parameters == null) {
-            parameters = new HashMap<String, Object>();
-        }
-        parameters.put(key, value);
-    }
-}
-// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/ArtifactNamespaceContext.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,90 @@
+/*
+ * Copyright (c) 2010 by Intevation GmbH
+ *
+ * This program is free software under the LGPL (>=v2.1)
+ * Read the file LGPL.txt coming with the software for details
+ * or visit http://www.gnu.org/licenses/ if it does not exist.
+ */
+
+package de.intevation.artifacts.common;
+
+import java.util.Iterator;
+
+import javax.xml.XMLConstants;
+
+import javax.xml.namespace.NamespaceContext;
+
+/**
+ * The namespace used in artifact documents.
+ * @author <a href="mailto:sascha.teichmann@intevation">Sascha L. Teichmann</a>
+ */
+public class ArtifactNamespaceContext
+implements   NamespaceContext
+{
+    /**
+     * The URI of the namespace of the artifacts.
+     */
+    public final static String NAMESPACE_URI =
+        "http://www.intevation.de/2009/artifacts";
+
+    /**
+     * The XML prefix for the artifacts namespace.
+     */
+    public final static String NAMESPACE_PREFIX = "art";
+
+    /**
+     * Final instance to be easily used to avoid creation
+     * of instances.
+     */
+    public static final ArtifactNamespaceContext INSTANCE =
+        new ArtifactNamespaceContext();
+
+    /**
+     * The default constructor.
+     */
+    public ArtifactNamespaceContext() {
+    }
+
+    /**
+     * @see javax.xml.namespace.NamespaceContext#getNamespaceURI(String)
+     * @param prefix The prefix
+     * @return The corresponing URI
+     */
+    public String getNamespaceURI(String prefix) {
+
+        if (prefix == null) {
+            throw new NullPointerException("Null prefix");
+        }
+
+        if (NAMESPACE_PREFIX.equals(prefix)) {
+            return NAMESPACE_URI;
+        }
+
+        if ("xml".equals(prefix)) {
+            return XMLConstants.XML_NS_URI;
+        }
+
+        return XMLConstants.NULL_NS_URI;
+    }
+
+    /**
+     * @see javax.xml.namespace.NamespaceContext#getPrefix(String)
+     * @param uri The URI
+     * @return nothing.
+     * @throws java.lang.UnsupportedOperationException
+     */
+    public String getPrefix(String uri) {
+        throw new UnsupportedOperationException();
+    }
+
+    /**
+     * @see javax.xml.namespace.NamespaceContext#getPrefixes(java.lang.String)
+     * @param uri The URI
+     * @return nothing
+     * @throws java.lang.UnsupportedOperationException
+     */
+    public Iterator getPrefixes(String uri) {
+        throw new UnsupportedOperationException();
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/model/DefaultUser.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,75 @@
+package de.intevation.artifacts.common.model;
+
+/**
+ * The default implementation of the {@link User} interface.
+ *
+ * @author <a href="mailto:ingo.weinzierl@intevation.de">Ingo Weinzierl</a>
+ */
+public class DefaultUser implements User {
+
+    /** The user's firstname. */
+    protected String firstName;
+
+    /** The user's lastname. */
+    protected String lastName;
+
+
+    /**
+     * Creates an empty user without name.
+     */
+    public DefaultUser() {
+    }
+
+
+    /**
+     * Creates a user with first and lastname.
+     *
+     * @param firstName The user's firstname.
+     * @param lastName The user's lastname.
+     */
+    public DefaultUser(String firstName, String lastName) {
+        this.firstName = firstName;
+        this.lastName  = lastName;
+    }
+
+
+    /**
+     * This method returns the firstname of the user.
+     *
+     * @return the firstname.
+     */
+    public String getFirstName() {
+        return firstName;
+    }
+
+
+    /**
+     * Sets the user's firstname.
+     *
+     * @param firstName The user's firstname.
+     */
+    public void setFirstName(String firstName) {
+        this.firstName = firstName;
+    }
+
+
+    /**
+     * This method returns the lastname of the user.
+     *
+     * @return the lastname.
+     */
+    public String getLastName() {
+        return lastName;
+    }
+
+
+    /**
+     * Sets the user's lastname.
+     *
+     * @param lastName The user's lastname.
+     */
+    public void setLastName(String lastName) {
+        this.lastName = lastName;
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/model/KVP.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,32 @@
+/*
+ * Copyright (c) 2010 by Intevation GmbH
+ *
+ * This program is free software under the LGPL (>=v2.1)
+ * Read the file LGPL.txt coming with the software for details
+ * or visit http://www.gnu.org/licenses/ if it does not exist.
+ */
+package de.intevation.artifacts.common.model;
+
+
+public class KVP<K, V> {
+
+    private K key;
+    private V value;
+
+
+    public KVP(K key, V value) {
+        this.key   = key;
+        this.value = value;
+    }
+
+
+    public K getKey() {
+        return key;
+    }
+
+
+    public V getValue() {
+        return value;
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/model/User.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,43 @@
+package de.intevation.artifacts.common.model;
+
+import java.io.Serializable;
+
+/**
+ * An interface that describes a user of the system.
+ *
+ * @author <a href="mailto:ingo.weinzierl@intevation.de">Ingo Weinzierl</a>
+ */
+public interface User extends Serializable {
+
+    /**
+     * This method returns the firstname of the user.
+     *
+     * @return the firstname.
+     */
+    public String getFirstName();
+
+
+    /**
+     * Sets the user's firstname.
+     *
+     * @param firstName The user's firstname.
+     */
+    public void setFirstName(String firstName);
+
+
+    /**
+     * This method returns the lastname of the user.
+     *
+     * @return the lastname.
+     */
+    public String getLastName();
+
+
+    /**
+     * Sets the user's lastname.
+     *
+     * @param lastName The user's lastname.
+     */
+    public void setLastName(String lastName);
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/ClientProtocolUtils.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,860 @@
+/*
+ * Copyright (c) 2011 by Intevation GmbH
+ *
+ * This program is free software under the LGPL (>=v2.1)
+ * Read the file LGPL.txt coming with the software for details
+ * or visit http://www.gnu.org/licenses/ if it does not exist.
+ */
+package de.intevation.artifacts.common.utils;
+
+import javax.xml.xpath.XPathConstants;
+
+import org.w3c.dom.Document;
+import org.w3c.dom.Element;
+import org.w3c.dom.Node;
+import org.w3c.dom.NodeList;
+
+import de.intevation.artifacts.common.ArtifactNamespaceContext;
+
+
+/**
+ * This class provides methods that help creating the artifact protocol
+ * documents DESCRIBE, FEED, ADVANCE and OUT.
+ *
+ * @author <a href="mailto:ingo.weinzierl@intevation.de">Ingo Weinzierl</a>
+ */
+public class ClientProtocolUtils {
+
+    /** The XPath to the current state in the DESCRIBE document. */
+    public static final String XPATH_CURRENT_STATE = "/art:result/art:state";
+
+    /** The XPath to the static UI part in the DESCRIBE document. */
+    public static final String XPATH_STATIC  = "/art:result/art:ui/art:static";
+
+    /** The XPath to the dynamic UI part in the DESCRIBE document. */
+    public static final String XPATH_DYNAMIC = "/art:result/art:ui/art:dynamic";
+
+    /** The XPath to the reachable states part in the DESCRIBE document. */
+    public static final String XPATH_STATES  =
+        "/art:result/art:reachable-states";
+
+    /** The XPath to the output modes in the DESCRIBE document. */
+    public static final String XPATH_OUTPUT_MODES  =
+        "/art:result/art:outputmodes/art:output";
+
+    /** The XPath to the select node relative to the dynamic UI node in the
+     * DESCRIBE document. */
+    public static final String XPATH_DATA_SELECT = "art:select";
+
+    /** The XPath to the choices nodes relative to the select node in the
+     * DESCRIBE document. */
+    public static final String XPATH_DATA_ITEMS = "art:choices/art:item";
+
+    /** The XPath that points to the min value of a range.*/
+    public static final String XPATH_MIN_NODE = "art:min/@art:value";
+
+    /** The XPath that points to the max value of a range.*/
+    public static final String XPATH_MAX_NODE = "art:max/@art:value";
+
+    /** The XPath that points to the default min value of a range.*/
+    public static final String XPATH_DEF_MIN = "art:min/@art:default";
+
+    /** The XPath that points to the default max value of a range.*/
+    public static final String XPATH_DEF_MAX = "art:max/@art:default";
+
+    /** The XPath to a label in the artifact's DESCRIBE document. */
+    public static final String XPATH_LABEL = "art:label/text()";
+
+    /** The XPath to a value in the artifact's DESCRIBE document. */
+    public static final String XPATH_VALUE = "art:value/text()";
+
+
+    /**
+     * It should not be necessary to create instances of this class.
+     */
+    private ClientProtocolUtils() {
+    }
+
+
+    /**
+     * This method creates a new CREATE document.
+     *
+     * @return the CREATE document.
+     */
+    public static Document newCreateDocument(String factory) {
+        return newCreateDocument(factory, null);
+    }
+
+
+    /**
+     * This method creates a new CREATE document.
+     *
+     * @return the CREATE document.
+     */
+    public static Document newCreateDocument(String factory, String uuid) {
+        return newCreateDocument(factory, uuid, null);
+    }
+
+    public static Document newCreateDocument(
+        String  factory,
+        String  uuid,
+        String  ids
+    ) {
+        return newCreateDocument(factory, uuid, ids, null);
+    }
+
+    /**
+     * This method creates a new CREATE document.
+     *
+     * @return the CREATE document.
+     */
+    public static Document newCreateDocument(
+        String         factory,
+        String         uuid,
+        String         ids,
+        CreationFilter filter
+    ) {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action = cr.create("action");
+        Element type   = cr.create("type");
+        Element fac    = cr.create("factory");
+
+        type.setAttribute("name", "create");
+        fac.setAttribute("name", factory);
+
+        action.appendChild(type);
+        action.appendChild(fac);
+
+        if (uuid != null) {
+            Element templ = cr.create("template");
+            templ.setAttribute("uuid", uuid);
+            action.appendChild(templ);
+        }
+
+        if (ids != null) {
+            Element id = cr.create("ids");
+            id.setAttribute("value", ids);
+            action.appendChild(id);
+        }
+
+        if (filter != null) {
+            action.appendChild(filter.toXML(cr));
+        }
+
+        doc.appendChild(action);
+
+        return doc;
+    }
+
+
+    /**
+     * This method creates a new FEED document.
+     *
+     * @param theUuid The identifier of the artifact.
+     * @param theHash The hash of the artifact.
+     * @param theData An array that contains key/value pairs that represent the
+     * data that should be included in the FEED document.
+     *
+     * @return the FEED document.
+     */
+    public static Document newFeedDocument(
+        String     theUuid,
+        String     theHash,
+        String[][] theData)
+    {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action = cr.create("action");
+        Element type   = cr.create("type");
+        Element uuid   = cr.create("uuid");
+        Element hash   = cr.create("hash");
+        Element data   = cr.create("data");
+
+        // XXX It is not nice that the type has no attribute namespace, but to
+        // be backward compatible, we don't change this now.
+        cr.addAttr(type, "name", "feed", false);
+        cr.addAttr(uuid, "value", theUuid, true);
+        cr.addAttr(hash, "value", theHash, true);
+
+        for (String[] kvp: theData) {
+            Element input = cr.create("input");
+            cr.addAttr(input, "name", kvp[0], true);
+            cr.addAttr(input, "value", kvp[1], true);
+
+            data.appendChild(input);
+        }
+
+        action.appendChild(type);
+        action.appendChild(uuid);
+        action.appendChild(hash);
+        action.appendChild(data);
+
+        doc.appendChild(action);
+
+        return doc;
+    }
+
+
+    /**
+     * This method creates a new DESCRIBE document.
+     *
+     * @param theUuid The identifier of the artifact.
+     * @param theHash The hash of the artifact.
+     * @param ui If true, the UI part is included.
+     *
+     * @return the DESCRIBE document.
+     */
+    public static Document newDescribeDocument(
+        String  theUuid,
+        String  theHash,
+        boolean incUI)
+    {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action = cr.create("action");
+        Element type   = cr.create("type");
+        Element uuid   = cr.create("uuid");
+        Element hash   = cr.create("hash");
+        Element ui     = cr.create("include-ui");
+
+        // XXX It is not nice that the type has no attribute namespace, but to
+        // be backward compatible, we don't change this now.
+        cr.addAttr(type, "name", "describe", false);
+        cr.addAttr(uuid, "value", theUuid, true);
+        cr.addAttr(hash, "value", theHash, true);
+
+        ui.setTextContent(incUI ? "true" : "false");
+
+        action.appendChild(type);
+        action.appendChild(uuid);
+        action.appendChild(hash);
+        action.appendChild(ui);
+
+        doc.appendChild(action);
+
+        return doc;
+    }
+
+
+    /**
+     * This method creates a new ADVANCE document.
+     *
+     * @param theUuid The identifier of the artifact.
+     * @param theHash The hash of the artifact.
+     * @param theTarget The target state identifier.
+     *
+     * @return the ADVANCE document.
+     */
+    public static Document newAdvanceDocument(
+        String theUuid,
+        String theHash,
+        String theTarget)
+    {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action = cr.create("action");
+        Element type   = cr.create("type");
+        Element uuid   = cr.create("uuid");
+        Element hash   = cr.create("hash");
+        Element target = cr.create("target");
+
+        // XXX It is not nice that the type has no attribute namespace, but to
+        // be backward compatible, we don't change this now.
+        cr.addAttr(type, "name", "advance", false);
+        cr.addAttr(uuid, "value", theUuid, true);
+        cr.addAttr(hash, "value", theHash, true);
+        cr.addAttr(target, "name", theTarget, true);
+
+        action.appendChild(type);
+        action.appendChild(uuid);
+        action.appendChild(hash);
+        action.appendChild(target);
+
+        doc.appendChild(action);
+
+        return doc;
+    }
+
+
+    /**
+     * This method creates a new document that is used to create new artifact
+     * collections in the artifact server.
+     *
+     * @param name <b>Optional</b> name of the collection.
+     *
+     * @return the document to create new collections.
+     */
+    public static Document newCreateCollectionDocument(String name) {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action     = cr.create("action");
+        Element type       = cr.create("type");
+        Element collection = cr.create("collection");
+        Element attribute  = cr.create("attribute");
+
+        cr.addAttr(type, "name", "create");
+        cr.addAttr(collection, "name", name != null ? name : "");
+
+        action.appendChild(type);
+        type.appendChild(collection);
+        collection.appendChild(attribute);
+
+        doc.appendChild(action);
+
+        return doc;
+    }
+
+
+    /**
+     * This method creates a new Document that is used to add an artifact to a
+     * collection in the artifact server.
+     *
+     * @param artId The identifier of the artifact that should be added.
+     * @param attr A document that contains attributes for the artifact's
+     * life in the collection.
+     *
+     * @return the document to add an artifact into a collection.
+     */
+    public static Document newAddArtifactDocument(String artId, Document attr) {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action    = cr.create("action");
+        Element type      = cr.create("type");
+        Element artifact  = cr.create("artifact");
+        Element attribute = cr.create("attribute");
+
+        cr.addAttr(artifact, "uuid", artId);
+        cr.addAttr(type, "name", "addartifact");
+
+        if (attr != null) {
+            attr.appendChild(attr);
+        }
+
+        action.appendChild(type);
+        type.appendChild(artifact);
+        artifact.appendChild(attribute);
+
+        doc.appendChild(action);
+
+        return doc;
+    }
+
+
+    /**
+     * Create a new Document that is used to remove an artifact from a
+     * collection in the artifact server.
+     *
+     * @param artId The identifier of the artifact that should be added.
+     *
+     * @return the document to add an artifact into a collection.
+     */
+    public static Document newRemoveArtifactDocument(String artId) {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action    = cr.create("action");
+        Element type      = cr.create("type");
+        Element artifact  = cr.create("artifact");
+
+        cr.addAttr(artifact, "uuid", artId);
+        cr.addAttr(type, "name", "removeartifact");
+
+        action.appendChild(type);
+        type.appendChild(artifact);
+
+        doc.appendChild(action);
+
+        return doc;
+    }
+
+
+    /**
+     * This method creates a new Document that is used to trigger the DESCRIBE
+     * operation of a collection in the artifact server.
+     *
+     * @param uuid The identifier of the collection that should be described.
+     *
+     * @return the document to describe a collection.
+     */
+    public static Document newDescribeCollectionDocument(String uuid) {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action    = cr.create("action");
+        Element type      = cr.create("type");
+        cr.addAttr(type, "name", "describe");
+
+        action.appendChild(type);
+
+        doc.appendChild(action);
+
+        return doc;
+    }
+
+
+
+    /**
+     * This function builds a document that is used as request document of the
+     * out() operation of Collections.
+     *
+     * @param uuid The identifier of the collection.
+     * @param mode The name of the desired output mode.
+     * @param type The name of the desired output type.
+     *
+     * @return the request document.
+     */
+    public static Document newOutCollectionDocument(
+        String uuid,
+        String mode,
+        String type) {
+        return newOutCollectionDocument(uuid, mode, type, null);
+    }
+
+
+    /**
+     * This function builds a document that is used as request document of the
+     * out() operation of Collections. The document <i>attr</i> might be used to
+     * adjust some settings specific to the output.
+     *
+     * @param uuid The identifier of the collection.
+     * @param mode The name of the desired output mode.
+     * @param type The name of the desired output type.
+     * @param attr A document that contains settings specific to the output.
+     *
+     * @return the request document.
+     */
+    public static Document newOutCollectionDocument(
+        String   uuid,
+        String   mode,
+        String   type,
+        Document attr)
+    {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator cr = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action = cr.create("action");
+
+        cr.addAttr(action, "name", mode, true);
+        cr.addAttr(action, "type", type, true);
+
+        doc.appendChild(action);
+
+        if (attr != null) {
+            Node root = attr.getFirstChild();
+
+            if (root != null) {
+                action.appendChild(doc.importNode(root, true));
+            }
+        }
+
+        return doc;
+    }
+
+
+    /**
+     * This function creates a document that is used to set the attribute of a
+     * Collection.
+     *
+     * @param uuid The identifier of the Collection.
+     * @param attr The new attribute value for the Collection.
+     *
+     * @return the document that is used to set the attribute.
+     */
+    public static Document newSetAttributeDocument(
+        String uuid,
+        Document attr)
+    {
+        Node root = attr != null ? attr.getFirstChild() : null;
+
+        if (root == null) {
+            return null;
+        }
+
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action     = ec.create("action");
+        Element type       = ec.create("type");
+        Element collection = ec.create("collection");
+
+        ec.addAttr(type, "name", "setattribute", false);
+        ec.addAttr(collection, "uuid", uuid, false);
+
+        doc.appendChild(action);
+        action.appendChild(type);
+        type.appendChild(collection);
+
+        collection.appendChild(doc.importNode(root, true));
+
+        return doc;
+    }
+
+    /**
+     * This function creates a document that is used to set the attribute of a
+     * CollectionItem.
+     *
+     * @param uuid The identifier of the CollectionItem.
+     * @param attr The new attribute value for the CollectionItem.
+     *
+     * @return the document that is used to set the attribute.
+     */
+    public static Document newSetItemAttributeDocument(
+        String uuid,
+        Document attr)
+    {
+        Node root = attr.getFirstChild();
+
+        if (root == null) {
+            return null;
+        }
+
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action   = ec.create("action");
+        Element type     = ec.create("type");
+        Element artifact = ec.create("artifact");
+
+        ec.addAttr(type, "name", "setitemattribute");
+        ec.addAttr(artifact, "uuid", uuid);
+
+        doc.appendChild(action);
+        action.appendChild(type);
+        type.appendChild(artifact);
+
+        artifact.appendChild(doc.importNode(root, true));
+
+        return doc;
+    }
+
+
+    /**
+     * This function creates a document that is used to set the time-to-live
+     * of a collection.
+     *
+     * @param ttl The ttl for the Collection.
+     *
+     * @return the document that is used to set the time-to-live.
+     */
+    public static Document newSetCollectionTTLDocument(String ttl) {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action = ec.create("action");
+        Element type   = ec.create("type");
+        Element ttlEl  = ec.create("ttl");
+
+        ec.addAttr(type, "name", "settimetolive");
+        ec.addAttr(ttlEl, "value", ttl);
+
+        doc.appendChild(action);
+        action.appendChild(type);
+        type.appendChild(ttlEl);
+
+        return doc;
+    }
+
+
+    /**
+     * This function creates a document that is used to set the name of a
+     * collection.
+     *
+     * @param name The name for the Collection.
+     *
+     * @return the document that is used to set the name of a collection.
+     */
+    public static Document newSetCollectionNameDocument(String name) {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action = ec.create("action");
+        Element type   = ec.create("type");
+        Element coll   = ec.create("collection");
+
+        ec.addAttr(type, "name", "setname");
+        ec.addAttr(coll, "name", name);
+
+        doc.appendChild(action);
+        action.appendChild(type);
+        type.appendChild(coll);
+
+        return doc;
+    }
+
+
+    /**
+     * This function creates a document that is used to delete an existing
+     * collection.
+     *
+     * @return the document that is used to delete an existing collection.
+     */
+    public static Document newDeleteCollectionDocument() {
+        Document doc = XMLUtils.newDocument();
+
+        XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator(
+            doc,
+            ArtifactNamespaceContext.NAMESPACE_URI,
+            ArtifactNamespaceContext.NAMESPACE_PREFIX);
+
+        Element action = ec.create("action");
+        Element type   = ec.create("type");
+
+        ec.addAttr(type, "name", "delete");
+
+        doc.appendChild(action);
+        action.appendChild(type);
+
+        return doc;
+    }
+
+
+    /**
+     * Returns string value found by {@link XPATH_LABEL} relative to
+     * <i>node</i>.
+     *
+     * @param node A node.
+     *
+     * @return the string value found by {@link XPATH_LABEL}.
+     */
+    public static String getLabel(Node node) {
+        return (String) XMLUtils.xpath(
+            node,
+            XPATH_LABEL,
+            XPathConstants.STRING,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    /**
+     * Returns string value found by {@link XPATH_VALUE} relative to
+     * <i>node</i>.
+     *
+     * @param node A node.
+     *
+     * @return the string value found by {@link XPATH_VALUE}.
+     */
+    public static String getValue(Node node) {
+        return (String) XMLUtils.xpath(
+            node,
+            XPATH_VALUE,
+            XPathConstants.STRING,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    /**
+     * This method returns the static UI part of the artifact's DESCRIBE
+     * document.
+     *
+     * @param description The document returned by the artifact server's
+     * DESCRIBE operation.
+     *
+     * @return the static UI node.
+     */
+    public static Node getStaticUI(Document description) {
+        return (Node) XMLUtils.xpath(
+            description,
+            XPATH_STATIC,
+            XPathConstants.NODE,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    /**
+     * This method returns the dynamic UI part of the artifact's DESCRIBE
+     * document.
+     *
+     * @param description The document returned by the artifact server's
+     * DESCRIBE operation.
+     *
+     * @return the dynamic UI node.
+     */
+    public static Node getDynamicUI(Document description) {
+        return (Node) XMLUtils.xpath(
+            description,
+            XPATH_DYNAMIC,
+            XPathConstants.NODE,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    /**
+     * This method returns the current state node contained in the DESCRIBE
+     * document.
+     *
+     * @param description The document returned by the artifact server's
+     * DESCRIBE operation.
+     *
+     * @return the node containing information about the current state.
+     */
+    public static Node getCurrentState(Document description) {
+        return (Node) XMLUtils.xpath(
+            description,
+            XPATH_CURRENT_STATE,
+            XPathConstants.NODE,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    /**
+     * This method returns the node that contains information about the
+     * reachable states of the artifact in the artifact's DESCRIBE document.
+     *
+     * @param description The document returned by the artifact server's
+     * DESCRIBE operation.
+     *
+     * @return the node that contains the reachable states.
+     */
+    public static Node getReachableStates(Document description) {
+        return (Node) XMLUtils.xpath(
+            description,
+            XPATH_STATES,
+            XPathConstants.NODE,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    /**
+     * This method returns the output mode nodes of the DESCRIBE document.
+     *
+     * @param description The document returned by the artifact server's
+     * DESCRIBE operation.
+     *
+     * @return the node that contains the output modes.
+     */
+    public static NodeList getOutputModes(Document description) {
+        return (NodeList) XMLUtils.xpath(
+            description,
+            XPATH_OUTPUT_MODES,
+            XPathConstants.NODESET,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    /**
+     * Returns the node found by {@link XPATH_DATA_SELECT}.
+     *
+     * @param dynamicNode The dynamic UI node of the DESCRIBE document.
+     *
+     * @return the select node found in the dynamic UI node.
+     */
+    public static NodeList getSelectNode(Node dynamicNode) {
+        return (NodeList) XMLUtils.xpath(
+            dynamicNode,
+            XPATH_DATA_SELECT,
+            XPathConstants.NODESET,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    /**
+     * Returns the items that could be found in the <i>node</i>.
+     *
+     * @param node A select node.
+     *
+     * @return the choices nodes as node list.
+     */
+    public static NodeList getItemNodes(Node node) {
+        return (NodeList) XMLUtils.xpath(
+            node,
+            XPATH_DATA_ITEMS,
+            XPathConstants.NODESET,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    public static String getMinNode(Node parent) {
+        return (String) XMLUtils.xpath(
+            parent,
+            XPATH_MIN_NODE,
+            XPathConstants.STRING,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    public static String getMaxNode(Node parent) {
+        return (String) XMLUtils.xpath(
+            parent,
+            XPATH_MAX_NODE,
+            XPathConstants.STRING,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    public static String getDefMin(Node parent) {
+        return (String) XMLUtils.xpath(
+            parent,
+            XPATH_DEF_MIN,
+            XPathConstants.STRING,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+
+
+    public static String getDefMax(Node parent) {
+        return (String) XMLUtils.xpath(
+            parent,
+            XPATH_DEF_MAX,
+            XPathConstants.STRING,
+            ArtifactNamespaceContext.INSTANCE);
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/Config.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,276 @@
+/*
+ * Copyright (c) 2010 by Intevation GmbH
+ *
+ * This program is free software under the LGPL (>=v2.1)
+ * Read the file LGPL.txt coming with the software for details
+ * or visit http://www.gnu.org/licenses/ if it does not exist.
+ */
+
+package de.intevation.artifacts.common.utils;
+
+import java.io.File;
+import java.io.IOException;
+
+import javax.xml.namespace.QName;
+
+import javax.xml.parsers.DocumentBuilderFactory;
+import javax.xml.parsers.ParserConfigurationException;
+
+import javax.xml.xpath.XPathConstants;
+
+import org.apache.log4j.Logger;
+
+import org.w3c.dom.Document;
+import org.w3c.dom.Node;
+import org.w3c.dom.NodeList;
+
+import org.xml.sax.SAXException;
+
+/**
+ * The central access to the configuration of the artifact database.
+ * This class provides some static methods to access the central
+ * configuration XML file via XPath.
+ *
+ * @author <a href="mailto:sascha.teichmann@intevation.de">Sascha L. Teichmann</a>
+ */
+public final class Config
+{
+    private static Logger logger = Logger.getLogger(Config.class);
+
+    /**
+     * System property name where to find the configuration directory.
+     */
+    public static final String CONFIG_DIR = "artifact.database.dir";
+
+    /**
+     * Default path to the configuration directory if none
+     * was specified by the CONFIG_DIR system property.
+     */
+    public static final File CONFIG_DIR_DEFAULT =
+        new File(new File(System.getProperty("user.home",
+            System.getProperty("user.dir", "."))), ".artifactdb");
+
+    /**
+     * Name of the central configuration XML file.
+     */
+    public static final String CONFIG_FILE  = "conf.xml";
+
+    /**
+     * Name of the configuration filename alias to be use
+     * within the configuration. This alias is replaced by
+     * the real path.
+     */
+    public static final String CONFIG_DIR_PLACEHOLDER =
+        "${artifacts.config.dir}";
+
+    private static Document config;
+
+    private Config() {
+    }
+
+    /**
+     * Singleton access to the central XML configuration document.
+     * @return The central XML configuration document.
+     */
+    public static synchronized final Document getConfig() {
+        if (config == null) {
+            config = loadConfig();
+        }
+        return config;
+    }
+
+    /**
+     * Returns the path to the configuartion directory. If a path
+     * was specified via the CONFIG_DIR system property this one
+     * is used. Else it falls back to default configuration path.
+     * @return The path to the configuartion directory.
+     */
+    public static File getConfigDirectory() {
+        String configDirString = System.getProperty(CONFIG_DIR);
+
+        File configDir = configDirString != null
+            ? new File(configDirString)
+            : CONFIG_DIR_DEFAULT;
+
+        if (!configDir.isDirectory()) {
+            logger.warn("'" + configDir + "' is not a directory.");
+            configDir = CONFIG_DIR_DEFAULT;
+        }
+
+        return configDir;
+    }
+
+    /**
+     * Replaces the CONFIG_DIR_PLACEHOLDER alias with the real path
+     * of the configuration directory.
+     * @param path The path containing the CONFIG_DIR_PLACEHOLDER placeholder.
+     * @return The path where the CONFIG_DIR_PLACEHOLDER placeholders are
+     * replaced by the real path name.
+     */
+    public static String replaceConfigDir(String path) {
+        String configDir = getConfigDirectory().getAbsolutePath();
+        return path.replace(CONFIG_DIR_PLACEHOLDER, configDir);
+    }
+
+    private static Document loadConfig() {
+
+        File configDir = getConfigDirectory();
+
+        File file = new File(configDir, CONFIG_FILE);
+
+        if (!file.canRead() && !file.isFile()) {
+            logger.error("Cannot read config file '"
+                + file + "'.");
+            return null;
+        }
+
+        try {
+            DocumentBuilderFactory factory =
+                DocumentBuilderFactory.newInstance();
+            factory.setValidating(false); // XXX: This may change in future.
+            return factory.newDocumentBuilder().parse(file);
+        }
+        catch (SAXException se) {
+            logger.error(se.getLocalizedMessage(), se);
+        }
+        catch (ParserConfigurationException pce) {
+            logger.error(pce.getLocalizedMessage(), pce);
+        }
+        catch (IOException ioe) {
+            logger.error(ioe.getLocalizedMessage());
+        }
+
+        return null;
+    }
+
+    /**
+     * Convenience method to search within a given document tree via XPath.
+     * See {@link XMLUtils#xpath(Object, String, QName) } for details.
+     * @param root The object which is used as the root of the tree to
+     * be searched in.
+     * @param query The XPath query.
+     * @param returnType The type of the result.
+     * @return The result of type 'returnTyp' or null if something went
+     * wrong during XPath evaluation.
+     */
+    public static final Object getXPath(
+        Object root, String query, QName returnType
+    ) {
+        return XMLUtils.xpath(root, query, returnType);
+    }
+
+    /**
+     * Convenience method to search within the central configuration via XPath.
+     * See {@link XMLUtils#xpath(Object, String, QName) } for details.
+     * @param query The XPath query.
+     * @param returnType The type of the result.
+     * @return The result of type 'returnTyp' or null if something went
+     * wrong during XPath evaluation.
+     */
+    public static final Object getXPath(String query, QName returnType) {
+        return XMLUtils.xpath(getConfig(), query, returnType);
+    }
+
+    /**
+     * Convenience method to search for a node list within the central
+     * configuation document via XPath.
+     * @param query The XPath query.
+     * @return The queried node list or null if something went
+     * wrong during XPath evaluation.
+     */
+    public static final NodeList getNodeSetXPath(String query) {
+        return (NodeList)getXPath(query, XPathConstants.NODESET);
+    }
+
+    /**
+     * Convenience method to search for a node within the central
+     * configuation document via XPath.
+     * @param query The XPath query.
+     * @return The queried node or null if something went
+     * wrong during XPath evaluation.
+     */
+    public static final Node getNodeXPath(String query) {
+        return (Node)getXPath(query, XPathConstants.NODE);
+    }
+
+    /**
+     * Convenience method to search for a string within the central
+     * configuation document via XPath.
+     * @param xpath The XPath query.
+     * @return The queried string or null if something went
+     * wrong during XPath evaluation.
+     */
+    public static final String getStringXPath(String xpath) {
+        return getStringXPath(xpath, null);
+    }
+
+    /**
+     * Convenience method to search for a string within the central
+     * configuation document via XPath.
+     * @param query The XPath query.
+     * @param def The string to be returned if the search has no results.
+     * @return The queried string or the default value if something went
+     * wrong during XPath evaluation.
+     */
+    public static final String getStringXPath(String query, String def) {
+        String s = (String)getXPath(query, XPathConstants.STRING);
+        return s == null || s.length() == 0
+            ? def
+            : s;
+    }
+
+    /**
+     * Convenience method to search for a node list within a given tree
+     * via XPath.
+     * @param root The root of the tree to be searched in.
+     * @param query The XPath query.
+     * @return The queried node list or null if something went
+     * wrong during XPath evaluation.
+     */
+    public static final NodeList getNodeSetXPath(Object root, String query) {
+        return (NodeList)getXPath(root, query, XPathConstants.NODESET);
+    }
+
+    /**
+     * Convenience method to search for a node within a given tree
+     * via XPath.
+     * @param root The root of the tree to be searched in.
+     * @param query The XPath query.
+     * @return The queried node or null if something went
+     * wrong during XPath evaluation.
+     */
+    public static final Node getNodeXPath(Object root, String query) {
+        return (Node)getXPath(root, query, XPathConstants.NODE);
+    }
+
+    /**
+     * Convenience method to search for a string within a given tree
+     * via XPath.
+     * @param root The root of the tree to be searched in.
+     * @param xpath The XPath query.
+     * @return The queried string or null if something went
+     * wrong during XPath evaluation.
+     */
+    public static final String getStringXPath(Object root, String xpath) {
+        return getStringXPath(root, xpath, null);
+    }
+
+    /**
+     * Convenience method to search for a string within a given tree
+     * via XPath.
+     * @param root The root of the tree to be searched in.
+     * @param query xpath The XPath query.
+     * @param def The string to be returned if the search has no results.
+     * @return The queried string or the default value if something went
+     * wrong during XPath evaluation.
+     */
+    public static final String getStringXPath(
+        Object root, String query, String def
+    ) {
+        String s = (String)getXPath(root, query, XPathConstants.STRING);
+        return s == null || s.length() == 0
+            ? def
+            : s;
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/CreationFilter.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,61 @@
+package de.intevation.artifacts.common.utils;
+
+import java.util.List;
+import java.util.Map;
+import java.util.HashMap;
+
+import org.w3c.dom.Element;
+
+public class CreationFilter
+{
+    public static class Facet {
+
+        protected String name;
+        protected String index;
+
+        public Facet() {
+        }
+
+        public Facet(String name, String index) {
+            this.name  = name;
+            this.index = index;
+        }
+
+        public String getName() {
+            return name;
+        }
+
+        public String getIndex() {
+            return index;
+        }
+    }
+
+    protected Map<String, List<Facet>> outs;
+
+    public CreationFilter() {
+        outs = new HashMap<String, List<Facet>>();
+    }
+
+    public void add(String out, List<Facet> facets) {
+        outs.put(out, facets);
+    }
+
+    public Element toXML(XMLUtils.ElementCreator ec) {
+        Element filter = ec.create("filter");
+
+        for (Map.Entry<String, List<Facet>> entry: outs.entrySet()) {
+            Element out = ec.create("out");
+            out.setAttribute("name", entry.getKey());
+            for (Facet facet: entry.getValue()) {
+                Element f = ec.create("facet");
+                f.setAttribute("name", facet.getName());
+                f.setAttribute("index", facet.getIndex());
+                out.appendChild(f);
+            }
+            filter.appendChild(out);
+        }
+
+        return filter;
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/DateUtils.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,30 @@
+package de.intevation.artifacts.common.utils;
+
+import java.util.Calendar;
+import java.util.Date;
+
+
+public class DateUtils {
+
+    private DateUtils() {
+    }
+
+
+    /**
+     * This function extracts the year as int value from <i>date</i>.
+     *
+     * @param date The source date.
+     *
+     * @return the year as integer or -1 if date is empty.
+     */
+    public static int getYearFromDate(Date date) {
+        if (date == null) {
+            return -1;
+        }
+
+        Calendar cal = Calendar.getInstance();
+        cal.setTime(date);
+
+        return cal.get(Calendar.YEAR);
+    }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/FileTools.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,613 @@
+/*
+ * Copyright (c) 2010, 2011 by Intevation GmbH
+ *
+ * This program is free software under the LGPL (>=v2.1)
+ * Read the file LGPL.txt coming with the software for details
+ * or visit http://www.gnu.org/licenses/ if it does not exist.
+ */
+package de.intevation.artifacts.common.utils;
+
+import java.io.BufferedInputStream;
+import java.io.File;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.FileInputStream;
+import java.io.FileOutputStream;
+import java.io.OutputStream;
+import java.io.BufferedOutputStream;
+import java.nio.channels.FileChannel;
+
+import java.util.Deque;
+import java.util.ArrayDeque;
+import java.util.List;
+import java.util.Set;
+import java.util.HashSet;
+import java.util.ArrayList;
+import java.util.Enumeration;
+import java.util.zip.ZipFile;
+import java.util.zip.ZipEntry;
+import java.util.zip.ZipOutputStream;
+
+import org.apache.log4j.Logger;
+
+import java.security.MessageDigest;
+import java.security.NoSuchAlgorithmException;
+
+public class FileTools
+{
+    private static Logger log = Logger.getLogger(FileTools.class);
+
+    public static final String DIGEST =
+        System.getProperty("artifacts.common.file.cmp.digest", "MD5");
+
+    private FileTools() {
+    }
+
+
+    /** Remove everything after dot from name. */
+    public static final String removeExtension(String name) {
+        int index = name.lastIndexOf('.');
+        return index == -1
+            ? name
+            : name.substring(0, index);
+    }
+
+
+    public static File getDirectory(String path, String name) {
+        if (path == null || name == null) {
+            return null;
+        }
+
+        File dir = new File(path, name);
+
+        if (!dir.exists()) {
+            log.debug(
+                "Directory '" + dir.getAbsolutePath() + "' doesn't " +
+                "exist. Try to create it.");
+
+            return dir.mkdir() ? dir : null;
+        }
+        else {
+            return dir.isDirectory() ? dir : null;
+        }
+    }
+
+    public static File repair(File file) {
+        file = file.getAbsoluteFile();
+        if (file.exists()) {
+            return file;
+        }
+        Deque<String> parts = new ArrayDeque<String>();
+        File curr = file;
+        while (curr != null) {
+            String name = curr.getName();
+            if (name.length() > 0) {
+                parts.push(curr.getName());
+            }
+            curr = curr.getParentFile();
+        }
+
+        curr = null;
+        OUTER: while (!parts.isEmpty()) {
+            String f = parts.pop();
+            log.debug("fixing: '" + f + "'");
+            if (f.equals(".") || f.equals("..")) {
+                // No need to fix . or ..
+                continue;
+            }
+            if (curr == null) {
+                // XXX: Not totaly correct because there
+                // more than one root on none unix systems.
+                for (File root: File.listRoots()) {
+                    File [] files = root.listFiles();
+                    if (files == null) {
+                        log.warn("cannot list '" + root);
+                        continue;
+                    }
+                    for (File candidate: files) {
+                        if (candidate.getName().equalsIgnoreCase(f)) {
+                            curr = new File(root, candidate.getName());
+                            continue OUTER;
+                        }
+                    }
+                }
+                break;
+            }
+            else {
+                File [] files = curr.listFiles();
+                if (files == null) {
+                    log.warn("cannot list: '" + curr + "'");
+                    return file;
+                }
+                for (File candidate: files) {
+                    if (candidate.getName().equalsIgnoreCase(f)) {
+                        curr = new File(curr, candidate.getName());
+                        continue OUTER;
+                    }
+                }
+                curr = null;
+                break;
+            }
+        }
+
+        if (curr == null) {
+            log.warn("cannot repair path '" + file + "'");
+            return file;
+        }
+
+        return curr;
+    }
+
+    /** Object that can calculate hash of file, compare two hashed files etc. */
+    public static class HashedFile
+    implements Comparable<HashedFile>
+    {
+        protected File    file;
+        protected long    length;
+        protected byte [] hash;
+
+        public HashedFile(File file) {
+            this.file = file;
+            length = file.length();
+        }
+
+        public File getFile() {
+            return file;
+        }
+
+        protected byte [] getHash() {
+            if (hash == null) {
+                InputStream in = null;
+
+                try {
+                    in = new FileInputStream(file);
+
+                    MessageDigest digest = MessageDigest.getInstance(DIGEST);
+
+                    byte [] buf = new byte[40*1024];
+                    int r;
+
+                    while ((r = in.read(buf)) >= 0) {
+                        digest.update(buf, 0, r);
+                    }
+
+                    hash = digest.digest();
+                }
+                catch (IOException ioe) {
+                    log.error(ioe);
+                    hash = new byte[0];
+                }
+                catch (NoSuchAlgorithmException nsae) {
+                    log.error(nsae);
+                    hash = new byte[0];
+                }
+                finally {
+                    if (in != null) {
+                        try {
+                            in.close();
+                        }
+                        catch (IOException ioe) {
+                            log.error(ioe);
+                        }
+                    }
+                }
+            }
+            return hash;
+        }
+
+        @Override
+        public int compareTo(HashedFile other) {
+            if (length < other.length) return -1;
+            if (length > other.length) return +1;
+            return compare(getHash(), other.getHash());
+        }
+
+        private static int compare(byte [] a, byte [] b) {
+            if (a.length < b.length) return -1;
+            if (a.length > b.length) return +1;
+            for (int i = 0; i < a.length; ++i) {
+                int x = a[i] & 0xff;
+                int y = b[i] & 0xff;
+                if (x < y) return -1;
+                if (x > y) return +1;
+            }
+            return 0;
+        }
+
+        @Override
+        public boolean equals(Object other) {
+            return other instanceof HashedFile
+                && ((HashedFile)other).compareTo(this) == 0;
+        }
+
+        @Override
+        public int hashCode() {
+            return (int)(length ^ (length >>> 32));
+        }
+    } // class HashedFile
+
+    public static List<File> uniqueFiles(List<File> files) {
+
+        Set<HashedFile> set = new HashSet<HashedFile>();
+
+        for (File file: files) {
+            if (!set.add(new HashedFile(file))) {
+                log.warn("file '" + file + "' is a duplicate.");
+            }
+        }
+
+        ArrayList<File> out = new ArrayList<File>(set.size());
+        for (HashedFile hf: set) {
+            out.add(hf.file);
+        }
+
+        return out;
+    }
+
+    public interface FileVisitor {
+        boolean visit(File file);
+    } // Visitor
+
+    public static void walkTree(File root, FileVisitor visitor) {
+
+        Deque<File> stack = new ArrayDeque<File>();
+
+        stack.push(root);
+
+        while (!stack.isEmpty()) {
+            File current = stack.pop();
+            if (!visitor.visit(current)) break;
+            if (current.isDirectory()) {
+                File [] subs = current.listFiles();
+                if (subs != null) {
+                    for (File f: subs) {
+                        stack.push(f);
+                    }
+                }
+            }
+        }
+    }
+
+    /**
+     * Deletes everything in a directory.
+     *
+     * @param dir The directory.
+     */
+    public final static void deleteContent(File dir) {
+        if (dir == null || !dir.isDirectory()) {
+            return;
+        }
+
+        File[] files = dir.listFiles();
+        if (files != null) {
+            for (File file: files) {
+                deleteRecursive(file);
+            }
+        }
+
+        return;
+    }
+
+    /**
+     * Delete <i>file</i> and everything in <i>file</i> if it is a directory.
+     *
+     * @param file The file or directory.
+     * @return true, if deletion was successful - otherwise false.
+     */
+    public final static boolean deleteRecursive(File file) {
+
+        if (file == null) {
+            return false;
+        }
+
+        if (file.isDirectory()) {
+            File [] files = file.listFiles();
+            if (files != null) {
+                for (File sub: files) {
+                    if (!deleteRecursive(sub)) {
+                        return false;
+                    }
+                }
+            }
+        }
+
+        return file.delete();
+    }
+
+    /**
+     * Put the given file or directory into a zip archive.
+     *
+     * @param file The file or directory.
+     * @param outputStream The stream to write the archive to.
+     * @throws IOException if an error occured while zip creation or writing to
+     * output stream.
+     */
+    public static void createZipArchive(
+        File         file,
+        OutputStream outputStream
+    )
+    throws IOException
+    {
+        ZipOutputStream out = new ZipOutputStream(outputStream);
+
+        if (file.isFile()) {
+            copyFileToZip("", file, out);
+        }
+        else if (file.isDirectory()) {
+
+            Deque<PrefixDir> stack = new ArrayDeque<PrefixDir>();
+            stack.push(new PrefixDir(file.getName() + "/", file));
+
+            while (!stack.isEmpty()) {
+                PrefixDir pd = stack.pop();
+
+                ZipEntry dirEntry = new ZipEntry(pd.prefix);
+                out.putNextEntry(dirEntry);
+                out.closeEntry();
+
+                File [] files = pd.dir.listFiles();
+                if (files != null) {
+                    for (File sub: files) {
+                        if (sub.isDirectory()) {
+                            stack.push(new PrefixDir(
+                                pd.prefix + sub.getName() + "/",
+                                sub));
+                        }
+                        else if (sub.isFile()) {
+                            copyFileToZip(pd.prefix, sub, out);
+                        }
+                    }
+                }
+            }
+        }
+
+        out.finish();
+    }
+
+
+    public static void extractArchive(File archive, File destDir)
+    throws IOException {
+        if (!destDir.exists()) {
+            destDir.mkdir();
+        }
+
+        ZipFile zipFile = new ZipFile(archive);
+        try {
+            Enumeration<? extends ZipEntry> entries = zipFile.entries();
+
+            byte [] buffer = new byte[16384];
+
+            while (entries.hasMoreElements()) {
+                ZipEntry entry = entries.nextElement();
+
+                String entryFileName = entry.getName();
+
+                File dir = buildDirectoryHierarchyFor(entryFileName, destDir);
+                if (!dir.exists()) {
+                    dir.mkdirs();
+                }
+
+                if (!entry.isDirectory()) {
+                    BufferedInputStream bis = new BufferedInputStream(
+                        zipFile.getInputStream(entry));
+                    try {
+                        BufferedOutputStream bos = new BufferedOutputStream(
+                            new FileOutputStream(new File(destDir, entryFileName)));
+
+                        try {
+                            int len;
+                            while ((len = bis.read(buffer)) > 0) {
+                                bos.write(buffer, 0, len);
+                            }
+                            bos.flush();
+                        }
+                        finally {
+                            bos.close();
+                        }
+                    }
+                    finally {
+                        bis.close();
+                    }
+                } // is file
+            }
+        }
+        finally {
+            zipFile.close();
+        }
+    }
+
+    private static File buildDirectoryHierarchyFor(
+        String entryName,
+        File destDir)
+    {
+        int lastIndex = entryName.lastIndexOf('/');
+        String internalPathToEntry = entryName.substring(0, lastIndex + 1);
+        return new File(destDir, internalPathToEntry);
+    }
+
+    /**
+     * A class representing a directory with a prefix.
+     */
+    private static final class PrefixDir {
+
+        String prefix;
+        File   dir;
+
+        public PrefixDir(String prefix, File dir) {
+            this.prefix = prefix;
+            this.dir    = dir;
+        }
+
+    } // class PrefixDir
+
+    /**
+     * Write a file to zip archive.
+     *
+     * @param prefix A prefix.
+     * @param file The file.
+     * @param out The output stream.
+     * @throws IOException if an error occured while writing to zip output
+     * stream.
+     */
+    private static void copyFileToZip(
+        String          prefix,
+        File            file,
+        ZipOutputStream out
+    )
+    throws IOException
+    {
+        String   entryName = prefix + file.getName();
+        ZipEntry entry     = new ZipEntry(entryName);
+        out.putNextEntry(entry);
+        InputStream in = null;
+        try {
+            in =
+                new BufferedInputStream(
+                new FileInputStream(file), 20*1024);
+
+            byte [] buf = new byte[2048];
+
+            int r;
+            while ((r = in.read(buf)) > 0) {
+                out.write(buf, 0, r);
+            }
+        }
+        finally {
+            if (in != null) {
+                try { in.close(); }
+                catch (IOException ioe) {}
+            }
+        }
+        out.closeEntry();
+    }
+
+
+    /**
+     * Copies a <i>src</i> file to <i>target</i>.
+     *
+     * @param src A file (not a directory) that should be copied.
+     * @param target The destination. This might be a file or a directory.
+     *
+     * @return true, if <i>src</i> has been successfully copied; otherwise
+     * false.
+     */
+    public static boolean copyFile(File src, File target)
+    throws IOException
+    {
+        if (src == null || !src.exists()) {
+            log.warn("Source file does not exist!");
+            return false;
+        }
+
+        if (!src.canRead()) {
+            log.warn("Cannot read Source file!");
+            return false;
+        }
+
+        if (src.isDirectory()) {
+            log.warn("Source is a directory!");
+            return false;
+        }
+
+        if (target.isDirectory()) {
+            target = new File(target, src.getName());
+        }
+
+        FileInputStream  in  = null;
+        FileOutputStream out = null;
+
+        try {
+            in  = new FileInputStream(src);
+            out = new FileOutputStream(target);
+
+            FileChannel inChannel  = in.getChannel();
+            FileChannel outChannel = out.getChannel();
+
+            inChannel.transferTo(0l, inChannel.size(), outChannel);
+
+            return true;
+        }
+        catch (IOException ioe) {
+            log.warn(ioe, ioe);
+        }
+        finally {
+            if (in != null) {
+                try {
+                    in.close();
+                }
+                catch (IOException ioe) { /* do nothing here */ }
+            }
+
+            if (out != null) {
+                try {
+                    out.close();
+                }
+                catch (IOException ioe) { /* do nothing here */ }
+            }
+        }
+
+        return false;
+    }
+
+
+    /**
+     * Copies a directory <i>source</i> to a destination path <i>dest</i>.
+     *
+     * @param source A directory that should be copied.
+     * @param dest A destination directory which is created if it is not
+     * existing yet.
+     *
+     * @return true, if the directory has been successfully copied; otherwise
+     * false.
+     */
+    public static boolean copyDirectory(final File source, final File dest) {
+        if (source == null || !source.exists()) {
+            log.warn("Source directory does not exist!");
+            return false;
+        }
+
+        if (!source.isDirectory()) {
+            log.warn("Source is not a directory!");
+            return false;
+        }
+
+        if (dest == null) {
+            log.warn("Destination directory is null!");
+            return false;
+        }
+
+        if (!dest.exists()) {
+            if (!dest.mkdir()) {
+                log.warn("Cannot create destination directory!");
+                return false;
+            }
+        }
+
+        File[] children = source.listFiles();
+        int    failed   = 0;
+
+        if (children != null && children.length > 0) {
+            for (File child: children) {
+                if (child.isFile()) {
+                    try {
+                        if (!copyFile(child, dest)) {
+                            failed++;
+                        }
+                    }
+                    catch (IOException ioe) {
+                        log.warn(ioe, ioe);
+                        failed++;
+                    }
+                }
+                else if (child.isDirectory()) {
+                    copyDirectory(child, new File(dest, child.getName()));
+                }
+            }
+        }
+
+        log.debug("Failed to copy " + failed + " files.");
+
+        return true;
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/JSON.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,445 @@
+package de.intevation.artifacts.common.utils;
+
+import java.util.Map;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.ArrayList;
+import java.util.Iterator;
+
+import java.io.IOException;
+import java.io.PushbackInputStream;
+import java.io.InputStream;
+import java.io.PrintWriter;
+import java.io.ByteArrayInputStream;
+import java.io.StringWriter;
+
+import java.nio.charset.Charset;
+import java.nio.charset.UnsupportedCharsetException;
+
+public final class JSON
+{
+    private JSON() {
+    }
+
+    private static final boolean isDigit(int c) {
+        return c >= '0' && c <= '9';
+    }
+
+    public static final boolean isWhitespace(int c) {
+        return c == ' '  || c == '\n' || c == '\r'
+            || c == '\t' || c == '\f';
+    }
+
+    private static final void match(int c, int x) throws IOException {
+        if (c != x) {
+            throw new IOException(
+                "Expecting '" + (char)c + "' found '" + (char)x + "'");
+        }
+    }
+
+    private static final int eof(InputStream in)
+    throws IOException
+    {
+        int c = in.read();
+        if (c == -1) {
+            throw new IOException("EOF unexpected.");
+        }
+        return c;
+    }
+
+    private static final int whitespace(InputStream in)
+    throws IOException
+    {
+        int c;
+        while (isWhitespace(c = eof(in)));
+        return c;
+    }
+
+    private static final int parseHex(String hex) throws IOException {
+        try {
+            return Integer.parseInt(hex, 16);
+        }
+        catch (NumberFormatException nfe) {
+            throw new IOException("'" + hex + "' is not a hex string.");
+        }
+    }
+
+    public static final String jsonString(String string) {
+        StringBuilder sb = new StringBuilder(string.length()+2);
+
+        sb.append('"');
+
+        for (int i = 0, N = string.length(); i < N; ++i) {
+            char c = string.charAt(i);
+            switch (c) {
+                case '"':  sb.append("\\\""); break;
+                case '\t': sb.append("\\t"); break;
+                case '\r': sb.append("\\r"); break;
+                case '\n': sb.append("\\n"); break;
+                case '\b': sb.append("\\b"); break;
+                case '\f': sb.append("\\f"); break;
+                default:
+                    if (c >= 128) {
+                        sb.append("\\u");
+                        String hex = Integer.toHexString((int)c);
+                        for (int j = 4-hex.length(); j > 0; --j) {
+                            sb.append('0');
+                        }
+                        sb.append(hex);
+                    }
+                    else {
+                        sb.append(c);
+                    }
+            }
+        }
+
+        sb.append('"');
+
+        return sb.toString();
+    }
+
+    public static String toJSONString(Map<String, Object> map) {
+        StringWriter sw = new StringWriter();
+        PrintWriter pw = new PrintWriter(sw);
+        write(pw, map);
+        pw.flush();
+        return sw.toString();
+    }
+
+
+    public static void write(PrintWriter out, Map<String, Object> map) {
+        writeObject(out, map);
+    }
+
+    private static void writeValue(PrintWriter out, Object value) {
+        if (value instanceof Map) {
+            writeObject(out, (Map)value);
+        }
+        else if (value instanceof List) {
+            writeList(out, (List)value);
+        }
+        else if (value instanceof Number) {
+            out.print(value);
+        }
+        else if (value instanceof Boolean) {
+            out.print(((Boolean)value) ? "true" : "false");
+        }
+        else if (value == null) {
+            out.print("null");
+        }
+        else {
+            out.print(jsonString(value.toString()));
+        }
+    }
+
+    private static void writeObject(PrintWriter out, Map map) {
+
+        out.print('{');
+        Iterator iter = map.entrySet().iterator();
+        while (iter.hasNext()) {
+            Map.Entry entry = (Map.Entry)iter.next();
+            out.print(jsonString(entry.getKey().toString()));
+            out.print(':');
+            writeValue(out, entry.getValue());
+            if (iter.hasNext()) {
+                out.print(',');
+            }
+        }
+        out.print('}');
+    }
+
+    private static void writeList(PrintWriter out, List list) {
+        out.print('[');
+        Iterator iter = list.iterator();
+        while (iter.hasNext()) {
+            writeValue(out, iter.next());
+            if (iter.hasNext()) {
+                out.print(',');
+            }
+        }
+        out.print(']');
+    }
+
+    public static Map<String, Object> parse(String in)
+    throws IOException
+    {
+        return parse(asInputStream(in));
+    }
+
+    private static InputStream asInputStream(String in) {
+        byte [] bytes;
+        try {
+            bytes = in.getBytes(Charset.forName("US-ASCII"));
+        }
+        catch (UnsupportedCharsetException uce) {
+            // Should not happen.
+            bytes = in.getBytes();
+        }
+        return new ByteArrayInputStream(bytes);
+    }
+
+    public static Map<String, Object> parse(InputStream in)
+    throws IOException
+    {
+        return parseObject(new PushbackInputStream(in, 1));
+    }
+
+    public static Map<String, Object> parse(PushbackInputStream in)
+    throws IOException
+    {
+        return parseObject(in);
+    }
+
+    private static final String parseString(
+        PushbackInputStream in
+    )
+    throws IOException
+    {
+        StringBuilder sb = new StringBuilder();
+
+        int mode = 0;
+
+        char [] hex = new char[4];
+
+        match('"', eof(in));
+
+        OUT: for (int c = eof(in);; c = eof(in)) {
+
+            switch (mode) {
+                case 0:
+                    if (c == '"') {
+                        break OUT;
+                    }
+                    if (c == '\\') {
+                        mode = 1;
+                    }
+                    else {
+                        sb.append((char)c);
+                    }
+                    break;
+                case 1:
+                    switch (c) {
+                        case 'u':
+                            mode = 2;
+                            continue;
+                        case 'b':
+                            sb.append('\b');
+                            break;
+                        case 'f':
+                            sb.append('\f');
+                            break;
+                        case 'n':
+                            sb.append('\n');
+                            break;
+                        case 'r':
+                            sb.append('\r');
+                            break;
+                        case 't':
+                            sb.append('\t');
+                            break;
+                        default:
+                            sb.append((char)c);
+                    }
+                    mode = 0;
+                    break;
+                case 2:
+                    hex[0] = (char)c;
+                    mode = 3;
+                    break;
+                case 3:
+                    hex[1] = (char)c;
+                    mode = 4;
+                    break;
+                case 4:
+                    hex[2] = (char)c;
+                    mode = 5;
+                    break;
+                case 5:
+                    hex[3] = (char)c;
+                    sb.append((char)parseHex(new String(hex)));
+                    mode = 0;
+                    break;
+            }
+        }
+        return sb.toString();
+    }
+
+    private static final Boolean parseTrue(InputStream in)
+    throws IOException
+    {
+        match('t', eof(in));
+        match('r', eof(in));
+        match('u', eof(in));
+        match('e', eof(in));
+        return Boolean.TRUE;
+    }
+
+    private static final Boolean parseFalse(InputStream in)
+    throws IOException
+    {
+        match('f', eof(in));
+        match('a', eof(in));
+        match('l', eof(in));
+        match('s', eof(in));
+        match('e', eof(in));
+        return Boolean.FALSE;
+    }
+
+    private static final Object parseNull(InputStream in)
+    throws IOException
+    {
+        match('n', eof(in));
+        match('u', eof(in));
+        match('l', eof(in));
+        match('l', eof(in));
+        return null;
+    }
+
+    private static final Number parseNumber(PushbackInputStream in)
+    throws IOException
+    {
+        StringBuilder sb = new StringBuilder();
+
+        boolean isInteger = true;
+
+        int c;
+        OUT: for (;;) {
+            switch (c = eof(in)) {
+                case '0': case '1': case '2': case '3': case '4':
+                case '5': case '6': case '7': case '8': case '9':
+                case '-': case '+':
+                    sb.append((char)c);
+                    break;
+                case '.': case 'e': case 'E':
+                    isInteger = false;
+                    sb.append((char)c);
+                    break;
+                default:
+                    in.unread(c);
+                    break OUT;
+            }
+        }
+
+        try {
+            if (isInteger) {
+                return sb.length() > 9
+                    ? (Number)Long   .valueOf(sb.toString())
+                    : (Number)Integer.valueOf(sb.toString());
+            }
+            return (Number)Double.valueOf(sb.toString());
+        }
+        catch (NumberFormatException nfe) {
+            throw new IOException("Not a number '" + sb + "'");
+        }
+    }
+
+    private static List<Object> parseList(PushbackInputStream in)
+    throws IOException
+    {
+        List<Object> list = new ArrayList<Object>();
+        match('[', whitespace(in));
+        int c = whitespace(in);
+        if (c == ']') {
+            return list;
+        }
+
+        for (;; c = whitespace(in)) {
+            Object value;
+            in.unread(c);
+            switch (c) {
+                case '{':
+                    value = parseObject(in);
+                    break;
+                case '[':
+                    value = parseList(in);
+                    break;
+                case '"':
+                    value = parseString(in);
+                    break;
+                case 't':
+                    value = parseTrue(in);
+                    break;
+                case 'f':
+                    value = parseFalse(in);
+                    break;
+                case 'n':
+                    value = parseNull(in);
+                    break;
+                default:
+                    value = parseNumber(in);
+            }
+            list.add(value);
+
+            if ((c = whitespace(in)) == ']') break;
+            match(',', c);
+        }
+        return list;
+    }
+
+    private static void parsePair(
+        PushbackInputStream in,
+        Map<String, Object> pairs
+    )
+    throws IOException
+    {
+        in.unread(whitespace(in));
+        String string = parseString(in);
+        match(':', whitespace(in));
+
+        Object value;
+
+        int c = whitespace(in);
+        in.unread(c);
+        switch (c) {
+            case '{':
+                value = parseObject(in);
+                break;
+            case '[':
+                value = parseList(in);
+                break;
+            case '"':
+                value = parseString(in);
+                break;
+            case 't':
+                value = parseTrue(in);
+                break;
+            case 'f':
+                value = parseFalse(in);
+                break;
+            case 'n':
+                value = parseNull(in);
+                break;
+            default:
+                value = parseNumber(in);
+        }
+        pairs.put(string, value);
+    }
+
+    private static Map<String, Object> parseObject(PushbackInputStream in)
+    throws IOException
+    {
+        Map<String, Object> pairs = new LinkedHashMap<String, Object>();
+
+        int c = whitespace(in);
+        match('{', c);
+
+        if ((c = whitespace(in)) == '}') {
+            return pairs;
+        }
+
+        in.unread(c);
+
+        for (;;) {
+            parsePair(in, pairs);
+
+            if ((c = whitespace(in)) == '}') {
+                break;
+            }
+
+            if (c == '}') break;
+            match(',', c);
+        }
+
+        return pairs;
+    }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/LRUCache.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,26 @@
+package de.intevation.artifacts.common.utils;
+
+import java.util.Map;
+import java.util.LinkedHashMap;
+
+public class LRUCache<K, V>
+extends      LinkedHashMap<K, V>
+{
+    public static final int DEFAULT_MAX_CAPACITY = 25;
+
+    private int maxCapacity;
+
+    public LRUCache() {
+        this(DEFAULT_MAX_CAPACITY);
+    }
+
+    public LRUCache(int maxCapacity) {
+        this.maxCapacity = maxCapacity;
+    }
+
+    @Override
+    protected boolean removeEldestEntry(Map.Entry<K, V> eldest) {
+        return size() > maxCapacity;
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/MapXPathVariableResolver.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,48 @@
+/*
+ * Copyright (c) 2011 by Intevation GmbH
+ *
+ * This program is free software under the LGPL (>=v2.1)
+ * Read the file LGPL.txt coming with the software for details
+ * or visit http://www.gnu.org/licenses/ if it does not exist.
+ */
+package de.intevation.artifacts.common.utils;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import javax.xml.namespace.QName;
+import javax.xml.xpath.XPathVariableResolver;
+
+
+/**
+ *  @author <a href="mailto:ingo.weinzierl@intevation.de">Ingo Weinzierl</a>
+ */
+public class MapXPathVariableResolver implements XPathVariableResolver {
+
+    protected Map<String, String> variables;
+
+
+    public MapXPathVariableResolver() {
+        this.variables = new HashMap<String, String>();
+    }
+
+
+    public MapXPathVariableResolver(Map<String, String> variables) {
+        this.variables = variables;
+    }
+
+
+    public void addVariable(String name, String value) {
+        if (name != null && value != null) {
+            variables.put(name, value);
+        }
+    }
+
+
+    @Override
+    public Object resolveVariable(QName variableName) {
+        String key = variableName.getLocalPart();
+        return variables.get(key);
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/StringUtils.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,148 @@
+/*
+ * Copyright (c) 2010 by Intevation GmbH
+ *
+ * This program is free software under the LGPL (>=v2.1)
+ * Read the file LGPL.txt coming with the software for details
+ * or visit http://www.gnu.org/licenses/ if it does not exist.
+ */
+
+package de.intevation.artifacts.common.utils;
+
+import java.io.UnsupportedEncodingException;
+
+import java.util.UUID;
+
+import org.apache.commons.codec.DecoderException;
+
+import org.apache.commons.codec.binary.Hex;
+
+import org.apache.log4j.Logger;
+
+/**
+ * Commonly used string functions.
+ *
+ * @author <a href="mailto:sascha.teichmann@intevation.de">Sascha L. Teichmann</a>
+ */
+public final class StringUtils
+{
+    private static Logger logger = Logger.getLogger(StringUtils.class);
+
+    /**
+     * Generated a random UUIDv4 in form of a string.
+     * @return the UUID
+     */
+    public static final String newUUID() {
+        return UUID.randomUUID().toString();
+    }
+
+    /**
+     * Checks if a given string is a valid UUID.
+     * @param uuid The string to test.
+     * @return true if the string is a valid UUID else false.
+     */
+    public static final boolean checkUUID(String uuid) {
+        try {
+            UUID.fromString(uuid);
+        }
+        catch (IllegalArgumentException iae) {
+            logger.warn(iae.getLocalizedMessage());
+            return false;
+        }
+        return true;
+    }
+
+    /**
+     * Returns the UTF-8 byte array representation of a given string.
+     * @param s The string to be transformed.
+     * @return The byte array representation.
+     */
+    public static final byte [] getUTF8Bytes(String s) {
+        try {
+            return s.getBytes("UTF-8");
+        }
+        catch (UnsupportedEncodingException usee) {
+            logger.error(usee.getLocalizedMessage(), usee);
+            return s.getBytes();
+        }
+    }
+
+    /**
+     * Tries to convert a Base64 encoded string into the
+     * corresponing byte array.
+     * @param s The Base64 encoded string
+     * @return The byte array representation or null if
+     * an decoding error occurs.
+     */
+    public static final byte [] decodeHex(String s) {
+        try {
+            return Hex.decodeHex(s.toCharArray());
+        }
+        catch (DecoderException de) {
+            return null;
+        }
+    }
+
+    public static final String repeat(String s, int count, String sep) {
+        if (count <= 0) {
+            return "";
+        }
+        StringBuilder sb = new StringBuilder(s);
+        for (--count; count > 0; --count) {
+            sb.append(sep).append(s);
+        }
+        return sb.toString();
+    }
+
+    public static final String repeat(char c, int count, char sep) {
+        if (count <= 0) {
+            return "";
+        }
+        StringBuilder sb = new StringBuilder(2*count-1).append(c);
+        for (--count; count > 0; --count) {
+            sb.append(sep).append(c);
+        }
+        return sb.toString();
+    }
+
+    public static final String [] toUpperCase(String [] s) {
+        if (s == null) {
+            return null;
+        }
+        String [] d = new String[s.length];
+        for (int i = 0; i < s.length; ++i) {
+            if (s[i] != null) {
+                d[i] = s[i].toUpperCase();
+            }
+        }
+        return d;
+    }
+
+    public static String join(String sep, String [] strings) {
+        StringBuilder sb = new StringBuilder();
+        for (int i = 0; i < strings.length; ++i) {
+            if (i > 0) sb.append(sep);
+            sb.append(strings[i]);
+        }
+        return sb.toString();
+    }
+
+    public static final String [] join(String [] a, String [] b) {
+        if (a == null && b == null) return null;
+        if (a == null) return b;
+        if (b == null) return a;
+        String [] dst = new String[a.length + b.length];
+        System.arraycopy(a, 0, dst, 0, a.length);
+        System.arraycopy(b, 0, dst, a.length, b.length);
+        return dst;
+    }
+
+    public static final boolean contains(String needle, String [] haystack) {
+        for (String stray: haystack) {
+            if (needle.equals(stray)) {
+                return true;
+            }
+        }
+        return false;
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/XMLUtils.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,617 @@
+/*
+ * Copyright (c) 2010 by Intevation GmbH
+ *
+ * This program is free software under the LGPL (>=v2.1)
+ * Read the file LGPL.txt coming with the software for details
+ * or visit http://www.gnu.org/licenses/ if it does not exist.
+ */
+
+package de.intevation.artifacts.common.utils;
+
+import java.util.List;
+import java.util.ArrayList;
+import java.util.Map;
+import java.util.LinkedHashMap;
+import java.util.zip.GZIPInputStream;
+import java.util.zip.GZIPOutputStream;
+
+import java.io.ByteArrayInputStream;
+import java.io.FileInputStream;
+import java.io.BufferedInputStream;
+import java.io.ByteArrayOutputStream;
+import java.io.File;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.OutputStream;
+import java.io.StringWriter;
+
+import javax.xml.namespace.NamespaceContext;
+import javax.xml.namespace.QName;
+
+import javax.xml.parsers.DocumentBuilderFactory;
+import javax.xml.parsers.ParserConfigurationException;
+
+import javax.xml.transform.Transformer;
+import javax.xml.transform.TransformerConfigurationException;
+import javax.xml.transform.TransformerException;
+import javax.xml.transform.TransformerFactory;
+import javax.xml.transform.TransformerFactoryConfigurationError;
+
+import javax.xml.transform.dom.DOMSource;
+
+import javax.xml.transform.stream.StreamResult;
+
+import javax.xml.xpath.XPath;
+import javax.xml.xpath.XPathConstants;
+import javax.xml.xpath.XPathExpressionException;
+import javax.xml.xpath.XPathFactory;
+import javax.xml.xpath.XPathVariableResolver;
+
+import org.apache.log4j.Logger;
+
+import org.w3c.dom.Attr;
+import org.w3c.dom.Document;
+import org.w3c.dom.Element;
+import org.w3c.dom.Node;
+
+import org.xml.sax.SAXException;
+
+/**
+ * Some helper functions to ease work with XML concering namespaces, XPATH
+ * and so on.
+ *  @author <a href="mailto:sascha.teichmann@intevation.de">Sascha L. Teichmann</a>
+ */
+public final class XMLUtils
+{
+    /** W3C URL of XForms. */
+    public static final String XFORM_URL    = "http://www.w3.org/2002/xforms";
+
+    /** W3C prefix of XForms. */
+    public static final String XFORM_PREFIX = "xform";
+
+    /** Logger for this class. */
+    private static Logger logger = Logger.getLogger(XMLUtils.class);
+
+    private XMLUtils() {
+    }
+
+    /**
+     * Helper class to generate elements and attributes with
+     * namespaces.
+     */
+    public static class ElementCreator
+    {
+        /** Owner document of the elements to be created. */
+        protected Document document;
+
+        /** Namespace to be used. */
+        protected String   ns;
+
+        /** Prefix to be used. */
+        protected String   prefix;
+
+        /**
+         * Constructor to create an element/attribute creator
+         * with a given namespace and namespace prefix using a
+         * given owner document.
+         * @param document The owning document
+         * @param ns       The namespace
+         * @param prefix   The namespace prefix
+         */
+        public ElementCreator(Document document, String ns, String prefix) {
+            this.document = document;
+            this.ns       = ns;
+            this.prefix   = prefix;
+        }
+
+        /**
+         * Creates a new element using the owning document with
+         * the this creators namespace and namespace prefix.
+         * @param name The name of the element
+         * @return     The new element
+         */
+        public Element create(String name) {
+            Element element = document.createElementNS(ns, name);
+            element.setPrefix(prefix);
+            return element;
+        }
+
+        /**
+         * Adds a new attribute and its value to a given element.
+         * It does not set the namespace prefix.
+         * @param element The element to add the attribute to
+         * @param name    The name of the attribute
+         * @param value   The value of the attribute
+         */
+        public void addAttr(Element element, String name, String value) {
+            addAttr(element, name, value, false);
+        }
+
+        /**
+         * Adds a new attribute and its value to a given element.
+         * If the namespace prefix is used is decided by the 'addPrefix' flag.
+         * @param element The element to add the attribute to
+         * @param name    The name of the attribute
+         * @param value   The value of the attribute
+         * @param addPrefix If true the creators namespace prefix is
+         * set on the attribute.
+         */
+        public void addAttr(
+            Element element,
+            String  name,
+            String  value,
+            boolean addPrefix
+        ) {
+            if (addPrefix) {
+                Attr attr = document.createAttributeNS(ns, name);
+                attr.setValue(value);
+                attr.setPrefix(prefix);
+
+                element.setAttributeNode(attr);
+            }
+            else {
+                element.setAttribute(name, value);
+            }
+        }
+    } // class ElementCreator
+
+    /**
+     * Creates a new XML document
+     * @return the new XML document ot null if something went wrong during
+     * creation.
+     */
+    public static final Document newDocument() {
+        DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
+        factory.setNamespaceAware(true);
+
+        try {
+            return factory.newDocumentBuilder().newDocument();
+        }
+        catch (ParserConfigurationException pce) {
+            logger.error(pce.getLocalizedMessage(), pce);
+        }
+        return null;
+    }
+
+
+    /**
+     * Create xml/string representation of element (nested in otherwise empty
+     * document).
+     * @param element element to inspect in string.
+     * @return string with xml representation of element.
+     */
+    public final static String toString(Node node) {
+        Document doc = newDocument();
+        doc.appendChild(doc.importNode(node,true));
+        return toString(doc);
+    }
+
+
+    /**
+     * Loads a XML document namespace aware from a file
+     * @param file The file to load.
+     * @return the XML document or null if something went wrong
+     * during loading.
+     */
+    public static final Document parseDocument(File file) {
+        InputStream inputStream = null;
+        try {
+            inputStream = new BufferedInputStream(new FileInputStream(file));
+            return parseDocument(inputStream);
+        }
+        catch (IOException ioe) {
+            logger.error(ioe.getLocalizedMessage(), ioe);
+        }
+        finally {
+            if (inputStream != null) {
+                try { inputStream.close(); }
+                catch (IOException ioe) {}
+            }
+        }
+        return null;
+    }
+
+    /**
+     * Parses a String to a xml document.
+     *
+     * @param string The xml string
+     * @return the XML document or null if something went wrong.
+     */
+    public static final Document parseDocument(String string) {
+        InputStream inputStream = new ByteArrayInputStream(string.getBytes());
+        return parseDocument(inputStream);
+    }
+
+
+    public static final Document parseDocument(InputStream inputStream) {
+        return parseDocument(inputStream, Boolean.TRUE);
+    }
+
+    public static final Document parseDocument(
+        InputStream inputStream,
+        Boolean     namespaceAware
+    ) {
+        DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
+
+        if (namespaceAware != null) {
+            factory.setNamespaceAware(namespaceAware.booleanValue());
+        }
+
+        try {
+            return factory.newDocumentBuilder().parse(inputStream);
+        }
+        catch (ParserConfigurationException pce) {
+            logger.error(pce.getLocalizedMessage(), pce);
+        }
+        catch (SAXException se) {
+            logger.error(se.getLocalizedMessage(), se);
+        }
+        catch (IOException ioe) {
+            logger.error(ioe.getLocalizedMessage(), ioe);
+        }
+        return null;
+    }
+
+    /**
+     * Creates a new XPath without a namespace context.
+     * @return the new XPath.
+     */
+    public static final XPath newXPath() {
+        return newXPath(null, null);
+    }
+
+    /**
+     * Creates a new XPath with a given namespace context.
+     * @param namespaceContext The namespace context to be used or null
+     * if none should be used.
+     * @return The new XPath
+     */
+    public static final XPath newXPath(
+        NamespaceContext      namespaceContext,
+        XPathVariableResolver resolver)
+    {
+        XPathFactory factory = XPathFactory.newInstance();
+        XPath        xpath   = factory.newXPath();
+        if (namespaceContext != null) {
+            xpath.setNamespaceContext(namespaceContext);
+        }
+
+        if (resolver != null) {
+            xpath.setXPathVariableResolver(resolver);
+        }
+        return xpath;
+    }
+
+    /**
+     * Evaluates an XPath query on a given object and returns the result
+     * as a given type. No namespace context is used.
+     * @param root  The object which is used as the root of the tree to
+     * be searched in.
+     * @param query The XPath query
+     * @param returnTyp The type of the result.
+     * @return The result of type 'returnTyp' or null if something
+     * went wrong during XPath evaluation.
+     */
+    public static final Object xpath(
+        Object root,
+        String query,
+        QName  returnTyp
+    ) {
+        return xpath(root, query, returnTyp, null);
+    }
+
+    /**
+     * Evaluates an XPath query on a given object and returns the result
+     * as a string. A given namespace context is used.
+     * @param root  The object which is used as the root of the tree to
+     * be searched in.
+     * @param query The XPath query
+     * @param namespaceContext The namespace context to be used or null
+     * if none should be used.
+     * @return The result of the query or null if something went wrong
+     * during XPath evaluation.
+     */
+    public static final String xpathString(
+        Object root, String query, NamespaceContext namespaceContext
+    ) {
+        return (String)xpath(
+            root, query, XPathConstants.STRING, namespaceContext);
+    }
+
+    /**
+     * Evaluates an XPath query on a given object and returns the result
+     * as a given type. Optionally a namespace context is used.
+     * @param root The object which is used as the root of the tree to
+     * be searched in.
+     * @param query The XPath query
+     * @param returnType The type of the result.
+     * @param namespaceContext The namespace context to be used or null
+     * if none should be used.
+     * @return The result of type 'returnTyp' or null if something
+     * went wrong during XPath evaluation.
+     */
+    public static final Object xpath(
+        Object           root,
+        String           query,
+        QName            returnType,
+        NamespaceContext namespaceContext
+    ) {
+        return xpath(root, query, returnType, namespaceContext, null);
+    }
+
+    public static final Object xpath(
+        Object           root,
+        String           query,
+        QName            returnType,
+        NamespaceContext namespaceContext,
+        Map<String, String> variables)
+    {
+        if (root == null) {
+            return null;
+        }
+
+        XPathVariableResolver resolver = variables != null
+            ? new MapXPathVariableResolver(variables)
+            : null;
+
+        try {
+            XPath xpath = newXPath(namespaceContext, resolver);
+            if (xpath != null) {
+                return xpath.evaluate(query, root, returnType);
+            }
+        }
+        catch (XPathExpressionException xpee) {
+            logger.error(xpee.getLocalizedMessage(), xpee);
+        }
+
+        return null;
+    }
+
+    /**
+     * Streams out an XML document to a given output stream.
+     * @param document The document to be streamed out.
+     * @param out      The output stream to be used.
+     * @return true if operation succeeded else false.
+     */
+    public static boolean toStream(Document document, OutputStream out) {
+        try {
+            Transformer transformer =
+                TransformerFactory.newInstance().newTransformer();
+            DOMSource    source = new DOMSource(document);
+            StreamResult result = new StreamResult(out);
+            transformer.transform(source, result);
+            return true;
+        }
+        catch (TransformerConfigurationException tce) {
+            logger.error(tce.getLocalizedMessage(), tce);
+        }
+        catch (TransformerFactoryConfigurationError tfce) {
+            logger.error(tfce.getLocalizedMessage(), tfce);
+        }
+        catch (TransformerException te) {
+            logger.error(te.getLocalizedMessage(), te);
+        }
+
+        return false;
+    }
+
+    public static String toString(Document document) {
+        try {
+            Transformer transformer =
+                TransformerFactory.newInstance().newTransformer();
+            DOMSource    source = new DOMSource(document);
+            StringWriter out    = new StringWriter();
+            StreamResult result = new StreamResult(out);
+            transformer.transform(source, result);
+            out.flush();
+            return out.toString();
+        }
+        catch (TransformerConfigurationException tce) {
+            logger.error(tce.getLocalizedMessage(), tce);
+        }
+        catch (TransformerFactoryConfigurationError tfce) {
+            logger.error(tfce.getLocalizedMessage(), tfce);
+        }
+        catch (TransformerException te) {
+            logger.error(te.getLocalizedMessage(), te);
+        }
+
+        return null;
+    }
+
+    public static byte [] toByteArray(Document document) {
+        return toByteArray(document, false);
+    }
+
+    /**
+     * Transforms an XML document into a byte array.
+     * @param document The document to be streamed out.
+     * @param compress The document should be compressed, too.
+     * @return the byte array or null if operation failed or
+     * document is null.
+     */
+    public static byte [] toByteArray(Document document, boolean compress) {
+        if (document != null) {
+            try {
+                ByteArrayOutputStream baos = new ByteArrayOutputStream();
+                OutputStream out = compress
+                    ? new GZIPOutputStream(baos)
+                    : baos;
+                boolean success = toStream(document, out);
+                out.flush();
+                out.close();
+                return success
+                    ? baos.toByteArray()
+                    : null;
+            }
+            catch (IOException ioe) {
+                logger.error(ioe);
+            }
+        }
+        return null;
+    }
+
+    public static Document fromByteArray(byte [] data) {
+        return fromByteArray(data, false);
+    }
+
+    public static Document fromByteArray(byte [] data, boolean decompress) {
+        if (data != null) {
+            InputStream in = new ByteArrayInputStream(data);
+            try {
+                if (decompress) {
+                    in = new GZIPInputStream(in);
+                }
+                return parseDocument(in);
+            }
+            catch (IOException ioe) {
+                logger.error(ioe);
+            }
+            finally {
+                try {
+                    in.close();
+                }
+                catch (IOException ioe) {
+                    logger.error(ioe);
+                }
+            }
+        }
+        return null;
+    }
+
+    private static class BuildResult {
+        List<Node>          children;
+        Map<String, String> attributes;
+        BuildResult() {
+            children   = new ArrayList<Node>();
+            attributes = new LinkedHashMap<String, String>();
+        }
+
+        void setAttributes(Element element) {
+            for (Map.Entry<String, String> entry: attributes.entrySet()) {
+                element.setAttribute(entry.getKey(), entry.getValue());
+            }
+        }
+
+        void finish(Element element) {
+            setAttributes(element);
+            for (Node child: children) {
+                element.appendChild(child);
+            }
+        }
+
+        void add(Node node) {
+            children.add(node);
+        }
+
+        void add(String key, Object value) {
+            attributes.put(key, value != null ? value.toString() : "null");
+        }
+
+        int numChildren() {
+            return children.size();
+        }
+
+        Node firstChild() {
+            return children.get(0);
+        }
+    } // class BuildResult
+
+    private static BuildResult recursiveBuild(
+        List     list,
+        Document document
+    ) {
+        BuildResult result = new BuildResult();
+        for (Object entry: list) {
+            if (entry instanceof Map) {
+                BuildResult subResult = recursiveBuild(
+                    (Map<String, Object>)entry, document);
+                if (subResult.numChildren() == 1) {
+                    result.add(subResult.firstChild());
+                }
+                else {
+                    Element element = document.createElement("map");
+                    subResult.finish(element);
+                    result.add(element);
+                }
+            }
+            else if (entry instanceof List) {
+                Element element = document.createElement("list");
+                BuildResult subResult = recursiveBuild((List)entry, document);
+                subResult.finish(element);
+                result.add(element);
+            }
+            else {
+                Element element = document.createElement("entry");
+                element.setAttribute(
+                    "value",
+                    entry != null ? entry.toString() : "null");
+            }
+        }
+        return result;
+    }
+
+    private static BuildResult recursiveBuild(
+        Map<String, Object> map,
+        Document            document
+    ) {
+        BuildResult result = new BuildResult();
+
+        List<Node> nodes = new ArrayList<Node>();
+        for (Map.Entry<String, Object> entry: map.entrySet()) {
+            Object value = entry.getValue();
+            if (value instanceof Map) {
+                Element element = document.createElement(entry.getKey());
+                BuildResult subResult = recursiveBuild(
+                    (Map<String, Object>)value, document);
+                subResult.finish(element);
+                result.add(element);
+            }
+            else if (value instanceof List) {
+                Element element = document.createElement(entry.getKey());
+                BuildResult subResult = recursiveBuild((List)value, document);
+                subResult.finish(element);
+                result.add(element);
+            }
+            else {
+                result.add(entry.getKey(), value);
+            }
+        }
+        return result;
+    }
+
+    public static Document jsonToXML(String input) {
+        Document document = newDocument();
+
+        if (document == null) {
+            return null;
+        }
+
+        Map<String, Object> map;
+        try {
+            map = JSON.parse(input);
+        }
+        catch (IOException ioe) {
+            logger.error(ioe);
+            return null;
+        }
+
+        BuildResult roots = recursiveBuild(map, document);
+
+        int N = roots.children.size();
+
+        if (N == 1) {
+            document.appendChild(roots.children.get(0));
+        }
+        else if (N > 1) {
+            Node root = document.createElement("root");
+            for (int i = 0; i < N; ++i) {
+                root.appendChild(roots.children.get(i));
+            }
+            document.appendChild(root);
+        }
+
+        return document;
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/artifacts-common/src/main/java/org/dive4elements/artifacts/common/utils/XSLTransformer.java	Thu Apr 25 10:53:15 2013 +0200
@@ -0,0 +1,73 @@
+package de.intevation.artifacts.common.utils;
+
+import java.io.InputStream;
+import java.io.StringWriter;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import javax.xml.transform.Source;
+import javax.xml.transform.Transformer;
+import javax.xml.transform.TransformerConfigurationException;
+import javax.xml.transform.TransformerException;
+import javax.xml.transform.TransformerFactory;
+
+import javax.xml.transform.dom.DOMSource;
+
+import javax.xml.transform.stream.StreamResult;
+import javax.xml.transform.stream.StreamSource;
+
+import org.apache.log4j.Logger;
+
+import org.w3c.dom.Node;
+
+public class XSLTransformer {
+
+    private static Logger log = Logger.getLogger(XSLTransformer.class);
+
+    protected Map<String, Object> parameters;
+
+    public XSLTransformer() {
+    }
+
+    public String transform(Node source, InputStream transform) {
+
+        try {
+            Source templateSource = new StreamSource(transform);
+            TransformerFactory xformFactory =
+                TransformerFactory.newInstance();
+            Transformer transformer =
+                xformFactory.newTransformer(templateSource);
+
+            if (parameters != null) {
+                for (Map.Entry<String, Object> entry: parameters.entrySet()) {
+                    transformer.setParameter(entry.getKey(), entry.getValue());
+                }
+            }
+
+            StringWriter result = new StringWriter();
+
+            DOMSource    src = new DOMSource(source);
+            StreamResult dst = new StreamResult(result);
+            transformer.transform(src, dst);
+
+            return result.toString();
+        }
+        catch (TransformerConfigurationException tce) {
+            log.error(tce, tce);
+        }
+        catch (TransformerException te) {
+            log.error(te, te);
+        }
+
+        return null;
+    }
+
+    public void addParameter(String key, Object value) {
+        if (parameters == null) {
+            parameters = new HashMap<String, Object>();
+        }
+        parameters.put(key, value);
+    }
+}
+// vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :

http://dive4elements.wald.intevation.org