All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.devonfw.tools.ide.configurator.merge.XmlMerger Maven / Gradle / Ivy

package com.devonfw.tools.ide.configurator.merge;

import java.io.File;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;

import org.w3c.dom.Attr;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.w3c.dom.Text;

import com.devonfw.tools.ide.configurator.resolve.VariableResolver;
import com.devonfw.tools.ide.logging.Log;

/**
 * Implementation of {@link FileTypeMerger} for XML.
 *
 * @since 3.0.0
 */
public class XmlMerger extends FileTypeMerger {

  private static final DocumentBuilder DOCUMENT_BUILDER;

  private static final TransformerFactory TRANSFORMER_FACTORY;

  static {
    try {
      DocumentBuilderFactory documentBuilderFactory = DocumentBuilderFactory.newInstance();
      documentBuilderFactory.setNamespaceAware(true);
      DOCUMENT_BUILDER = documentBuilderFactory.newDocumentBuilder();
      TRANSFORMER_FACTORY = TransformerFactory.newInstance();
    } catch (Exception e) {
      throw new IllegalStateException("Invalid XML DOM support in JDK.", e);
    }
  }

  /**
   * The constructor.
   */
  public XmlMerger() {

    super();
  }

  @Override
  public void merge(File setupFile, File updateFile, VariableResolver resolver, File workspaceFile) {

    Document document = null;
    boolean updateFileExists = updateFile.exists();
    if (workspaceFile.exists()) {
      if (!updateFileExists) {
        return; // nothing to do ...
      }
      document = load(workspaceFile);
    } else if (setupFile.exists()) {
      document = load(setupFile);
    }
    if (updateFileExists) {
      if (document == null) {
        document = load(updateFile);
      } else {
        Document updateDocument = load(updateFile);
        merge(updateDocument, document, true, true);
      }
    }
    resolve(document, resolver, false);
    save(document, workspaceFile);
  }

  private void merge(Document sourceDocument, Document targetDocument, boolean override, boolean add) {

    assert (override || add);
    merge(sourceDocument.getDocumentElement(), targetDocument.getDocumentElement(), override, add);
  }

  private void merge(Element sourceElement, Element targetElement, boolean override, boolean add) {

    merge(sourceElement.getAttributes(), targetElement, override, add);
    NodeList sourceChildNodes = sourceElement.getChildNodes();
    int length = sourceChildNodes.getLength();
    for (int i = 0; i < length; i++) {
      Node child = sourceChildNodes.item(i);
      if (child.getNodeType() == Node.ELEMENT_NODE) {

      } else if (child.getNodeType() == Node.TEXT_NODE) {

      } else if (child.getNodeType() == Node.CDATA_SECTION_NODE) {

      }
    }
  }

  private void merge(NamedNodeMap sourceAttributes, Element targetElement, boolean override, boolean add) {

    int length = sourceAttributes.getLength();
    for (int i = 0; i < length; i++) {
      Attr sourceAttribute = (Attr) sourceAttributes.item(i);
      String namespaceURI = sourceAttribute.getNamespaceURI();
      // String localName = sourceAttribute.getLocalName();
      String name = sourceAttribute.getName();
      Attr targetAttribute = targetElement.getAttributeNodeNS(namespaceURI, name);
      if (targetAttribute == null) {
        if (add) {
          // ridiculous but JDK does not provide namespace support by default...
          targetElement.setAttributeNS(namespaceURI, name, sourceAttribute.getValue());
          // targetElement.setAttribute(name, sourceAttribute.getValue());
        }
      } else if (override) {
        targetAttribute.setValue(sourceAttribute.getValue());
      }
    }
  }

  @Override
  public void inverseMerge(File workspaceFile, VariableResolver resolver, boolean addNewProperties, File updateFile) {

    if (!workspaceFile.exists() || !updateFile.exists()) {
      return;
    }
    Document updateDocument = load(updateFile);
    Document workspaceDocument = load(workspaceFile);
    merge(workspaceDocument, updateDocument, true, addNewProperties);
    resolve(updateDocument, resolver, true);
    save(updateDocument, updateFile);
    Log.debug("Saved changes in " + workspaceFile.getName() + " to: " + updateFile.getAbsolutePath());
  }

  /**
   * @param file the {@link File} to load.
   * @return the loaded XML {@link Document}.
   */
  public static Document load(File file) {

    try {
      return DOCUMENT_BUILDER.parse(file);
    } catch (Exception e) {
      throw new IllegalStateException("Failed to load XML from: " + file, e);
    }
  }

  /**
   * @param document the XML {@link Document} to save.
   * @param file the {@link File} to save to.
   */
  public static void save(Document document, File file) {

    ensureParentDirecotryExists(file);
    try {
      Transformer transformer = TRANSFORMER_FACTORY.newTransformer();
      DOMSource source = new DOMSource(document);
      StreamResult result = new StreamResult(file);
      transformer.transform(source, result);
    } catch (Exception e) {
      throw new IllegalStateException("Failed to save XML to file: " + file, e);
    }

  }

  private void resolve(Document document, VariableResolver resolver, boolean inverse) {

    NodeList nodeList = document.getElementsByTagName("*");
    for (int i = 0; i < nodeList.getLength(); i++) {
      Element element = (Element) nodeList.item(i);
      resolve(element, resolver, inverse);
    }
  }

  private void resolve(Element element, VariableResolver resolver, boolean inverse) {

    resolve(element.getAttributes(), resolver, inverse);
    NodeList nodeList = element.getChildNodes();

    for (int i = 0; i < nodeList.getLength(); i++) {
      Node node = nodeList.item(i);
      if (node instanceof Text) {
        Text text = (Text) node;
        String value = text.getNodeValue();
        String resolvedValue;
        if (inverse) {
          resolvedValue = resolver.inverseResolve(value);
        } else {
          resolvedValue = resolver.resolve(value);
        }
        text.setNodeValue(resolvedValue);
      }
    }
  }

  private void resolve(NamedNodeMap attributes, VariableResolver resolver, boolean inverse) {

    for (int i = 0; i < attributes.getLength(); i++) {
      Attr attribute = (Attr) attributes.item(i);
      String value = attribute.getValue();
      String resolvedValue;
      if (inverse) {
        resolvedValue = resolver.inverseResolve(value);
      } else {
        resolvedValue = resolver.resolve(value);
      }
      attribute.setValue(resolvedValue);
    }
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy