/*
 * Copyright 2011 by Graz University of Technology, Austria
 * MOCCA has been developed by the E-Government Innovation Center EGIZ, a joint
 * initiative of the Federal Chancellery Austria and Graz University of Technology.
 *
 * Licensed under the EUPL, Version 1.1 or - as soon they will be approved by
 * the European Commission - subsequent versions of the EUPL (the "Licence");
 * You may not use this work except in compliance with the Licence.
 * You may obtain a copy of the Licence at:
 * http://www.osor.eu/eupl/
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the Licence is distributed on an "AS IS" basis,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the Licence for the specific language governing permissions and
 * limitations under the Licence.
 *
 * This product combines work with different licenses. See the "NOTICE" text
 * file for details on the various modules and licenses.
 * The "NOTICE" text file is part of the distribution. Any derivative works
 * that you distribute must include a readable copy of the "NOTICE" text file.
 */
package at.gv.egiz.idlink;
import java.io.IOException;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import javax.xml.transform.Result;
import javax.xml.transform.Source;
import javax.xml.transform.Templates;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerConfigurationException;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.sax.SAXTransformerFactory;
import javax.xml.transform.stream.StreamSource;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.w3c.dom.Text;
import at.gv.egiz.bku.utils.urldereferencer.StreamData;
import at.gv.egiz.bku.utils.urldereferencer.URLDereferencer;
public class IdentityLinkTransformer {
  
  private final Logger log = LoggerFactory.getLogger(IdentityLinkTransformer.class);
  /**
   * The transformer factory.
   */
  private TransformerFactory factory = SAXTransformerFactory.newInstance();
  /**
   * The URLDereferencer used to dereference style-sheet URLs.
   */
  private URLDereferencer urlDereferencer;
  
  /**
   * @return the urlDereferencer
   */
  public URLDereferencer getUrlDereferencer() {
    return urlDereferencer;
  }
  /**
   * @param urlDereferencer the urlDereferencer to set
   */
  public void setUrlDereferencer(URLDereferencer urlDereferencer) {
    this.urlDereferencer = urlDereferencer;
  }
  /**
   * Sets the given domainIdentifier on the corresponding
   * node of the given idLink.
   * 
This method may be used to cope with a flaw in the IssuerTemplate-Stylesheets
   * used to transform a CompressedIdentitiyLink into an
   * IdentityLink. Some IssuerTemplate-Stylesheets do not
   * consider the pr:Type element value of the 
   * CompressedIdentityLink and render a pr:Type 
   * element value of urn:publicid:gv.at:baseid
   * into the IdentityLink structure. This method allows to
   * set the pr:Type element value on the given idLink
   * after the transformation.
   * 
IdentityLink element or one of it's ancestors.
   * Must not be null.
   * 
   * @param domainIdentifier the value to be set for the pr:Type element
   * 
   * @throws NullPointerException if idLink is null.
   */
  public static void setDomainIdentifier(Node idLink, String domainIdentifier) {
    
    Element element;
    if (idLink instanceof Element) {
      element = (Element) idLink;
    } else if (idLink instanceof Document) {
      element = ((Document) idLink).getDocumentElement();
    } else if (idLink != null) {
      Document document = idLink.getOwnerDocument();
      element = document.getDocumentElement();
    } else {
      throw new NullPointerException("Parameter 'idLink' must no be null.");
    }
    
    NodeList nodeList = element.getElementsByTagNameNS(
        "http://reference.e-government.gv.at/namespace/persondata/20020228#",
        "Type");
    for (int i = 0; i < nodeList.getLength(); i++) {
      if (nodeList.item(i) instanceof Element) {
        Element typeElement = (Element) nodeList.item(i);
        NodeList children = typeElement.getChildNodes();
        for (int j = 0; j < children.getLength(); j++) {
          if (children.item(j) instanceof Text) {
            ((Text) children.item(j)).setNodeValue(domainIdentifier);
          }
        }
      }
    }
    
  }
  
  /**
   * Mapping of issuer template URIs to transformation templates.
   */
  private Mapsource to result with
   * the given issuer template from the stylesheetURL.
   * 
   * @param stylesheetURL
   *          the URL of the issuer template to be used for transformation
   * @param source
   *          the compressed identity link source
   * @param result
   *          the transformed identity link result
   * 
   * @throws MalformedURLException
   *           if the given stylesheetURL is not a valid
   *           http or https URL.
   * @throws IOException
   *           if dereferencing the stylesheetURL fails.
   * @throws TransformerConfigurationException
   *           if creating a transformation template from the dereferenced
   *           stylesheet fails.
   * @throws TransformerException
   *           if transforming the identity link fails.
   */
  public void transformIdLink(String stylesheetURL, Source source, Result result) throws IOException, TransformerException {
    Templates templ = templates.get(stylesheetURL);
    
    if (templ == null) {
      // TODO: implement stylesheet cache
      URL url = new URL(stylesheetURL);
      if (!"http".equalsIgnoreCase(url.getProtocol()) && !"https".equalsIgnoreCase(url.getProtocol())) {
        throw new MalformedURLException("Protocol " + url.getProtocol() + " not supported for IssuerTemplate URL.");
      }
      
      StreamData data = urlDereferencer.dereference(url.toExternalForm());
      
      log.trace("Trying to create issuer template.");
      templ = factory.newTemplates(new StreamSource(data.getStream()));
      log.trace("Successfully created issuer template");
      templates.put(stylesheetURL, templ);
    }
    
    Transformer transformer = templ.newTransformer();
    
    transformer.transform(source, result);
    
  }
  
}