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

org.apache.openejb.util.SuperProperties Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.apache.openejb.util;

import org.w3c.dom.Comment;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.xml.sax.EntityResolver;
import org.xml.sax.ErrorHandler;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.SAXParseException;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.PrintStream;
import java.io.PrintWriter;
import java.io.StringReader;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Enumeration;
import java.util.Hashtable;
import java.util.InvalidPropertiesFormatException;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Properties;
import java.util.Set;

/**
 * Properties is a Hashtable where the keys and values must be Strings. Each Properties can have a default
 * Properties which specifies the default values which are used if the key is not in this Properties.
 *
 * @see Hashtable
 * @see System#getProperties
 */
public class SuperProperties extends Properties {

    private static final int EOF = -1;
    private static final int LINE_ENDING = -4200;
    private static final int ENCODED_EQUALS = -5000;
    private static final int ENCODED_COLON = -5001;
    private static final int ENCODED_SPACE = -5002;
    private static final int ENCODED_TAB = -5003;
    private static final int ENCODED_NEWLINE = -5004;
    private static final int ENCODED_CARRIAGE_RETURN = -5005;
    private static final Class STRING = String.class;
    private static final String PROP_DTD_NAME = "http://java.sun.com/dtd/properties.dtd";

    private static final String PROP_DTD = ""
            + "    "
            + "    "
            + "    "
            + "    "
            + "    ";


    /**
     * Actual property values.
     */
    protected LinkedHashMap properties = new LinkedHashMap();

    /**
     * Comments for individual the properties.
     */
    protected LinkedHashMap comments = new LinkedHashMap();

    /**
     * Attributes for the properties.
     */
    protected LinkedHashMap> attributes = new LinkedHashMap>();

    /**
     * The default property values.
     */
    protected Properties defaults;

    /**
     * Are lookups case insensitive?
     */
    protected boolean caseInsensitive;

    /**
     * The text between a key and the value.
     */
    protected String keyValueSeparator = "=";

    /**
     * The line separator to use when storing.  Defaults to system line separator.
     */
    protected String lineSeparator = System.getProperty("line.separator");

    /**
     * Number of spaces to indent each line of the properties file.
     */
    protected String indent = "";

    /**
     * Number of spaces to indent comment after '#' character.
     */
    protected String commentIndent = " ";

    /**
     * Should there be a blank line between properties.
     */
    protected boolean spaceBetweenProperties = true;

    /**
     * Should there be a blank line between a comment and the property.
     */
    protected boolean spaceAfterComment;

    /**
     * Used for loadFromXML.
     */
    private DocumentBuilder builder;

    /**
     * Constructs a new Properties object.
     */
    public SuperProperties() {
        super();
    }

    /**
     * Constructs a new Properties object using the specified default properties.
     *
     * @param properties the default properties
     */
    public SuperProperties(final Properties properties) {
        super(properties);
        defaults = properties;
    }

    /**
     * Are lookups case insensitive?
     *
     * @return true if lookups are insensitive
     */
    public boolean isCaseInsensitive() {
        return caseInsensitive;
    }

    /**
     * Sets the sensitive of lookups.
     *
     * @param caseInsensitive if looks are insensitive
     */
    public void setCaseInsensitive(final boolean caseInsensitive) {
        this.caseInsensitive = caseInsensitive;
    }

    public SuperProperties caseInsensitive(final boolean caseInsensitive) {
        setCaseInsensitive(caseInsensitive);
        return this;
    }

    /**
     * Gets the text that separates keys and values.
     * The default is "=".
     *
     * @return the text that separates keys and values
     */
    public String getKeyValueSeparator() {
        return keyValueSeparator;
    }

    /**
     * Sets the text that separates keys and values.
     *
     * @param keyValueSeparator the text that separates keys and values
     */
    public void setKeyValueSeparator(final String keyValueSeparator) {
        if (keyValueSeparator == null) {
            throw new NullPointerException("keyValueSeparator is null");
        }
        if (keyValueSeparator.length() == 0) {
            throw new NullPointerException("keyValueSeparator is empty");
        }
        this.keyValueSeparator = keyValueSeparator;
    }

    /**
     * Gets the text that separates lines while storing.
     * The default is the system line.separator.
     *
     * @return the text that separates keys and values
     */
    public String getLineSeparator() {
        return lineSeparator;
    }

    /**
     * Sets the text that separates lines while storing
     *
     * @param lineSeparator the text that separates lines
     */
    public void setLineSeparator(final String lineSeparator) {
        if (lineSeparator == null) {
            throw new NullPointerException("lineSeparator is null");
        }
        if (lineSeparator.length() == 0) {
            throw new NullPointerException("lineSeparator is empty");
        }
        this.lineSeparator = lineSeparator;
    }

    /**
     * Gets the number of spaces to indent each line of the properties file.
     *
     * @return the number of spaces to indent each line of the properties file
     */
    public int getIndent() {
        return indent.length();
    }

    /**
     * Sets the number of spaces to indent each line of the properties file.
     *
     * @param indent the number of spaces to indent each line of the properties file
     */
    public void setIndent(final int indent) {
        final char[] chars = new char[indent];
        Arrays.fill(chars, ' ');
        this.indent = new String(chars);
    }

    /**
     * Gets the number of spaces to indent comment after '#' character.
     *
     * @return the number of spaces to indent comment after '#' character
     */
    public int getCommentIndent() {
        return commentIndent.length();
    }

    /**
     * Sets the number of spaces to indent comment after '#' character.
     *
     * @param commentIndent the number of spaces to indent comment after '#' character
     */
    public void setCommentIndent(final int commentIndent) {
        final char[] chars = new char[commentIndent];
        Arrays.fill(chars, ' ');
        this.commentIndent = new String(chars);
    }

    /**
     * Should a blank line be added between properties?
     *
     * @return true if a blank line should be added between properties; false otherwise
     */
    public boolean isSpaceBetweenProperties() {
        return spaceBetweenProperties;
    }

    /**
     * If true a blank line will be added between properties.
     *
     * @param spaceBetweenProperties if true a blank line will be added between properties
     */
    public void setSpaceBetweenProperties(final boolean spaceBetweenProperties) {
        this.spaceBetweenProperties = spaceBetweenProperties;
    }

    /**
     * Should there be a blank line between a comment and the property?
     *
     * @return true if a blank line should be added between a comment and the property
     */
    public boolean isSpaceAfterComment() {
        return spaceAfterComment;
    }

    /**
     * If true a blank line will be added between a comment and the property.
     *
     * @param spaceAfterComment if true a blank line will be added between a comment and the property
     */
    public void setSpaceAfterComment(final boolean spaceAfterComment) {
        this.spaceAfterComment = spaceAfterComment;
    }

    @Override
    public String getProperty(final String name) {
        final Object result = get(name);
        String property = STRING.isInstance(result) ? STRING.cast(result) : null;
        if (property == null && defaults != null) {
            property = defaults.getProperty(name);
        }
        return property;
    }

    @Override
    public String getProperty(final String name, final String defaultValue) {
        final Object result = get(name);
        String property = STRING.isInstance(result) ? STRING.cast(result) : null;
        if (property == null && defaults != null) {
            property = defaults.getProperty(name);
        }
        if (property == null) {
            return defaultValue;
        }
        return property;
    }

    @Override
    public synchronized Object setProperty(final String name, final String value) {
        return put(name, value);
    }

    /**
     * Searches for the comment associated with the specified property. If the property is not found, look
     * in the default properties. If the property is not found in the default properties, answer null.
     *
     * @param name the name of the property to find
     * @return the named property value
     */
    public String getComment(String name) {
        name = normalize(name);
        String comment = comments.get(name);
        if (comment == null && defaults instanceof SuperProperties) {
            comment = ((SuperProperties) defaults).getComment(name);
        }
        return comment;
    }

    /**
     * Sets the comment associated with a property.
     *
     * @param name    the property name; not null
     * @param comment the comment; not null
     */
    public void setComment(String name, final String comment) {
        if (name == null) {
            throw new NullPointerException("name is null");
        }
        if (comment == null) {
            throw new NullPointerException("comment is null");
        }

        name = normalize(name);
        comments.put(name, comment);
    }

    /**
     * Searches for the attributes associated with the specified property. If the property is not found, look
     * in the default properties. If the property is not found in the default properties, answer null.
     *
     * @param name the name of the property to find
     * @return the attributes for an existing property (not null); null for non-existant properties
     */
    public Map getAttributes(String name) {
        if (name == null) {
            throw new NullPointerException("name is null");
        }

        name = normalize(name);
        Map attributes = this.attributes.get(name);
        if (attributes == null && defaults instanceof SuperProperties) {
            attributes = ((SuperProperties) defaults).getAttributes(name);
        }
        return attributes;
    }

    @Override
    public void list(final PrintStream out) {
        if (out == null) {
            throw new NullPointerException();
        }
        final StringBuilder buffer = new StringBuilder(80);
        final Enumeration keys = propertyNames();
        while (keys.hasMoreElements()) {
            appendProperty(buffer, keys);
            out.println(buffer.toString());
            buffer.setLength(0);
        }
    }

    @Override
    public void list(final PrintWriter writer) {
        if (writer == null) {
            throw new NullPointerException();
        }
        final StringBuilder buffer = new StringBuilder(80);
        final Enumeration keys = propertyNames();
        while (keys.hasMoreElements()) {
            appendProperty(buffer, keys);
            writer.println(buffer.toString());
            buffer.setLength(0);
        }
    }

    private void appendProperty(final StringBuilder buffer, final Enumeration keys) {
        final String key = keys.nextElement().toString();
        buffer.append(key);
        buffer.append('=');
        String property = get(key).toString();
        if (property == null) {
            property = defaults.getProperty(key);
        }
        if (property.length() > 40) {
            buffer.append(property.substring(0, 37));
            buffer.append("...");
        } else {
            buffer.append(property);
        }
    }

    @Override
    public synchronized void load(final InputStream in) throws IOException {
        // never null, when empty we are processing the white space at the beginning of the line
        StringBuilder key = new StringBuilder();
        // null when processing key
        StringBuilder value = null;
        // never null, contains the comment for the property or nothing if no comment
        StringBuilder comment = new StringBuilder();
        // never null, contains attributes for a property
        LinkedHashMap attributes = new LinkedHashMap();

        int indent = 0;
        boolean globalIndentSet = false;

        int commentIndent = -1;
        boolean globalCommentIndentSet = false;

        // true when processing the separator between a key and value
        boolean inSeparator = false;

        while (true) {
            int nextByte = decodeNextCharacter(in);
            if (nextByte == EOF) {
                break;
            }
            char nextChar = (char) (nextByte & 0xff);

            switch (nextByte) {
                case ' ':
                case '\t':
                    //
                    // End of key if parsing key
                    //
                    // if parsing the key, this is the end of the key
                    if (key.length() > 0 && value == null) {
                        inSeparator = true;
                        value = new StringBuilder();
                        continue;
                    }
                    break;
                case ':':
                case '=':
                    //
                    // End of key
                    //
                    if (inSeparator) {
                        inSeparator = false;
                        continue;
                    }
                    if (value == null) {
                        value = new StringBuilder();
                        continue;
                    }
                    break;
                case LINE_ENDING:
                    //
                    // End of Line
                    //
                    if (key.length() > 0) {
                        // add property
                        put(key.toString(), value == null ? "" : value.toString());
                        // add comment
                        if (comment.length() > 0) {
                            setComment(key.toString(), comment.toString());
                            comment = new StringBuilder();
                        }
                        // add attributes
                        this.attributes.put(normalize(key.toString()), attributes);
                        attributes = new LinkedHashMap();
                        // set line indent
                        if (!globalIndentSet) {
                            setIndent(indent);
                            globalIndentSet = true;
                        }
                        indent = 0;
                    }
                    key = new StringBuilder();
                    value = null;
                    continue;
                case '#':
                case '!':
                    //
                    // Comment
                    //
                    if (key.length() == 0) {
                        // set global line indent
                        if (!globalIndentSet) {
                            setIndent(indent);
                            globalIndentSet = true;
                        }
                        indent = 0;

                        // read comment Line
                        final StringBuilder commentLine = new StringBuilder();
                        int commentLineIndent = 0;
                        boolean inIndent = true;
                        while (true) {
                            nextByte = in.read();
                            if (nextByte < 0) {
                                break;
                            }
                            nextChar = (char) nextByte; // & 0xff

                            if (inIndent && nextChar == ' ') {
                                commentLineIndent++;
                                commentLine.append(' ');
                            } else if (inIndent && nextChar == '\t') {
                                commentLineIndent += 4;
                                commentLine.append("    ");
                            } else if (nextChar == '\r' || nextChar == '\n') {
                                break;
                            } else {
                                inIndent = false;
                                commentLine.append(nextChar);
                            }
                        }

                        // Determine indent
                        if (comment.length() == 0) {
                            // if this is a new comment block, the comment indent size for this
                            // block is based the first line of the comment
                            commentIndent = commentLineIndent;
                            if (!globalCommentIndentSet) {
                                setCommentIndent(commentIndent);
                                globalCommentIndentSet = true;
                            }
                        }
                        commentLineIndent = Math.min(commentIndent, commentLineIndent);

                        checkForAttributeOrAppend(comment, attributes, commentLine, commentLineIndent);
                        continue;
                    }
                    break;
            }

            if (nextByte >= 0 && Character.isWhitespace(nextChar)) {
                // count leading white space
                if (key.length() == 0) {
                    if (nextChar == '\t') {
                        indent += 4;
                    } else {
                        indent++;
                    }
                }

                // if key length == 0 or value length == 0
                if (key.length() == 0 || value == null || value.length() == 0) {
                    continue;
                }
            }

            // Decode encoded separator characters
            switch (nextByte) {
                case ENCODED_EQUALS:
                    nextChar = '=';
                    break;
                case ENCODED_COLON:
                    nextChar = ':';
                    break;
                case ENCODED_SPACE:
                    nextChar = ' ';
                    break;
                case ENCODED_TAB:
                    nextChar = '\t';
                    break;
                case ENCODED_NEWLINE:
                    nextChar = '\n';
                    break;
                case ENCODED_CARRIAGE_RETURN:
                    nextChar = '\r';
                    break;
            }

            inSeparator = false;
            if (value == null) {
                key.append(nextChar);
            } else {
                value.append(nextChar);
            }
        }

        // if buffer has data, there is a property we still need toadd
        if (key.length() > 0) {
            // add property
            put(key.toString(), value == null ? "" : value.toString());
            // add comment
            if (comment.length() > 0) {
                setComment(key.toString(), comment.toString());
            }
            // add attributes
            this.attributes.put(normalize(key.toString()), attributes);
            // set line indent
            if (!globalIndentSet) {
                setIndent(indent);
            }
        }
    }

    private void checkForAttributeOrAppend(final StringBuilder comment, final LinkedHashMap attributes, final StringBuilder commentLine, final int commentLineIndent) {
        if (commentLine.toString().trim().startsWith("@")) {
            // process property attribute
            final String attribute = commentLine.toString().trim().substring(1);
            final String[] parts = attribute.split("=", 2);
            final String attributeName = parts[0].trim();
            final String attributeValue = parts.length == 2 ? parts[1].trim() : "";
            attributes.put(attributeName, attributeValue);
        } else {
            // append comment
            if (comment.length() != 0) {
                comment.append(lineSeparator);
            }
            comment.append(commentLine.toString().substring(commentLineIndent));
        }
    }

    private int decodeNextCharacter(final InputStream in) throws IOException {
        boolean lineContinuation = false;
        boolean carriageReturnLineContinuation = false;
        boolean encoded = false;
        while (true) {
            // read character
            int nextByte = in.read();
            if (nextByte < 0) {
                return EOF;
            }
            char nextChar = (char) (nextByte & 0xff);

            // if line continuation character was '\r', we need to ignore an optional '\n'
            // immediately following the \r
            if (carriageReturnLineContinuation) {
                carriageReturnLineContinuation = false;
                if (nextChar == '\n') {
                    continue;
                }
            }

            // If escape sequence \x or line continuation, decode it
            if (nextChar == '\\') {
                // next character is the escaped character
                nextByte = in.read();
                if (nextByte < 0) {
                    // line continuation to end of stream
                    // sun vm returns 0 character for this case
                    nextChar = '\u0000';
                } else {
                    nextChar = (char) (nextByte & 0xff);
                }

                switch (nextChar) {
                    case '\r':
                        // line continuation using '\r', which optionally can have a following '\n'
                        carriageReturnLineContinuation = true;
                        // fall through
                    case '\n':
                        // line continuation
                        lineContinuation = true;
                        continue;
                    case 'u':
                        nextChar = readUnicode(in);
                        break;
                    default:
                        encoded = true;
                        nextChar = decodeEscapeChar(nextChar);
                        break;
                }
            } else {
                // if line ending character, we return the special value LINE_ENDING so
                // caller can differentiate between an encoded "\n" sequence and a real
                // line ending character in the file
                if (nextChar == '\n' || nextChar == '\r') {
                    return LINE_ENDING;
                }
            }

            // in a line continuation we ignore spaces and tabs until the first real character
            if (lineContinuation && (nextChar == ' ' || nextChar == '\t')) {
                continue;
            }

            if (encoded) {
                switch (nextChar) {
                    case '=':
                        return ENCODED_EQUALS;
                    case ':':
                        return ENCODED_COLON;
                    case ' ':
                        return ENCODED_SPACE;
                    case '\t':
                        return ENCODED_TAB;
                    case '\n':
                        return ENCODED_NEWLINE;
                    case '\r':
                        return ENCODED_CARRIAGE_RETURN;
                }
            }
            return nextChar;
        }
    }

    private char decodeEscapeChar(final char nextChar) {
        switch (nextChar) {
            case 'b':
                return '\b';
            case 'f':
                return '\f';
            case 'n':
                return '\n';
            case 'r':
                return '\r';
            case 't':
                return '\t';
            case 'u':
                throw new IllegalArgumentException("decodeEscapeChar can not decode an unicode sequence");
            default:
                return nextChar;
        }
    }

    private char readUnicode(final InputStream in) throws IOException {
        final char[] buf = new char[4];
        int unicode = 0;
        for (int i = 0; i < buf.length; i++) {
            final int nextByte = in.read();

            // we must get exactally 4 bytes
            if (nextByte < 0) {
                throw new IllegalArgumentException("Invalid unicode sequence: expected format \\uxxxx, but got \\u" + new String(buf, 0, i));
            }

            // convert to character
            final char nextChar = (char) (nextByte & 0xff);
            buf[i] = nextChar;

            // convert to digit
            final int nextDigit = Character.digit(nextChar, 16);

            // all bytes must be valid hex digits
            if (nextDigit < 0) {
                throw new IllegalArgumentException("Illegal character " + nextChar + " in unicode sequence \\u" + new String(buf, 0, i + 1));
            }


            unicode = (unicode << 4) + nextDigit;
        }

        return (char) unicode;
    }

    @Override
    public Enumeration propertyNames() {
        if (defaults == null) {
            return keys();
        }

        final Hashtable set = new Hashtable(defaults.size() + size());
        Enumeration keys = defaults.propertyNames();
        while (keys.hasMoreElements()) {
            set.put(keys.nextElement(), set);
        }
        keys = keys();
        while (keys.hasMoreElements()) {
            set.put(keys.nextElement(), set);
        }
        return set.keys();
    }

    @Override
    @SuppressWarnings({"deprecation"})
    public void save(final OutputStream out, final String comment) {
        try {
            store(out, comment);
        } catch (final IOException e) {
            // no-op
        }
    }

    @Override
    public synchronized void store(final OutputStream out, final String headComment) throws IOException {
        final OutputStreamWriter writer = new OutputStreamWriter(out, "ISO8859_1");
        if (headComment != null) {
            writer.write(indent);
            writer.write("#");
            writer.write(commentIndent);
            writer.write(headComment);
            writer.write(lineSeparator);
        }

        boolean firstProperty = true;
        final StringBuilder buffer = new StringBuilder(200);
        for (final Map.Entry entry : entrySet()) {
            final String key = entry.getKey().toString();
            final String value = entry.getValue().toString();

            if (!firstProperty && spaceBetweenProperties) {
                buffer.append(lineSeparator);
            }

            final String comment = comments.get(key);
            final Map attributes = this.attributes.get(key);
            if (comment != null || !attributes.isEmpty()) {
                dumpComment(buffer, comment, attributes, "#");
                if (spaceAfterComment) {
                    buffer.append(lineSeparator);
                }
            }

            // ${indent}${key}=${value}
            buffer.append(indent);
            dumpString(buffer, key, true);
            if (value != null && value.length() > 0) {
                buffer.append(keyValueSeparator);
                dumpString(buffer, value, false);
            }
            buffer.append(lineSeparator);

            writer.write(buffer.toString());
            buffer.setLength(0);

            firstProperty = false;
        }
        writer.flush();
    }

    private void dumpString(final StringBuilder buffer, final String string, final boolean key) {
        int i = 0;
        if (!key && i < string.length() && string.charAt(i) == ' ') {
            buffer.append("\\ ");
            i++;
        }

        for (; i < string.length(); i++) {
            final char ch = string.charAt(i);
            switch (ch) {
                case '\t':
                    buffer.append("\\t");
                    break;
                case '\n':
                    buffer.append("\\n");
                    break;
                case '\f':
                    buffer.append("\\f");
                    break;
                case '\r':
                    buffer.append("\\r");
                    break;
                default:
                    if ("\\".indexOf(ch) >= 0 || key && "#!=: ".indexOf(ch) >= 0) {
                        buffer.append('\\');
                    }
                    if (ch >= ' ' && ch <= '~') {
                        buffer.append(ch);
                    } else {
                        final String hex = Integer.toHexString(ch);
                        buffer.append("\\u");
                        for (int j = 0; j < 4 - hex.length(); j++) {
                            buffer.append("0");
                        }
                        buffer.append(hex);
                    }
            }
        }
    }

    private void dumpComment(final StringBuilder buffer, final String comment, final Map attributes, final String commentToken) {
        if (comment != null) {
            boolean startOfLine = true;

            char ch = 0;
            for (int i = 0; i < comment.length(); i++) {
                ch = comment.charAt(i);

                if (startOfLine) {
                    buffer.append(indent);
                    buffer.append(commentToken);
                    buffer.append(commentIndent);
                    startOfLine = false;
                }

                switch (ch) {
                    case '\r':
                        // if next character is not \n, this is the line break
                        if (i + 1 < comment.length() && comment.charAt(i + 1) != '\n') {
                            buffer.append(lineSeparator);
                            startOfLine = true;
                        }
                        break;
                    case '\n':
                        buffer.append(lineSeparator);
                        startOfLine = true;
                        break;
                    default:
                        buffer.append(ch);
                }
            }

            // if the last character written was not a line break, write one now
            if (ch != '\r' && ch != '\n') {
                buffer.append(lineSeparator);
            }
        }

        // ${indent}#${commentIndent}@${attributeName}=${attributeValue}
        for (final Map.Entry entry : attributes.entrySet()) {
            buffer.append(indent);
            buffer.append("#");
            buffer.append(commentIndent);
            buffer.append("@");
            buffer.append(entry.getKey());
            if (entry.getValue() != null && entry.getValue().length() > 0) {
                buffer.append("=");
                buffer.append(entry.getValue());
            }
            buffer.append(lineSeparator);
        }
    }

    @Override
    public synchronized void loadFromXML(final InputStream in) throws IOException {
        if (in == null) {
            throw new NullPointerException();
        }

        final DocumentBuilder builder = getDocumentBuilder();

        try {
            final Document doc = builder.parse(in);
            final NodeList entries = doc.getElementsByTagName("entry");
            if (entries == null) {
                return;
            }

            final int entriesListLength = entries.getLength();
            for (int i = 0; i < entriesListLength; i++) {
                final Element entry = (Element) entries.item(i);
                final String key = entry.getAttribute("key");
                final String value = entry.getTextContent();
                put(key, value);

                // search backwards for a comment
                for (Node node = entry.getPreviousSibling(); node != null && !(node instanceof Element); node = node.getPreviousSibling()) {
                    if (node instanceof Comment) {
                        final InputStream cin = new ByteArrayInputStream(((Comment) node).getData().getBytes());

                        // read comment line by line
                        final StringBuilder comment = new StringBuilder();
                        final LinkedHashMap attributes = new LinkedHashMap();

                        int nextByte;
                        char nextChar;
                        boolean firstLine = true;
                        int commentIndent = Integer.MAX_VALUE;
                        do {
                            // read one line
                            final StringBuilder commentLine = new StringBuilder();
                            int commentLineIndent = 0;
                            boolean inIndent = true;
                            while (true) {
                                nextByte = cin.read();
                                if (nextByte < 0) {
                                    break;
                                }
                                nextChar = (char) nextByte; // & 0xff
                                if (inIndent && nextChar == ' ') {
                                    commentLineIndent++;
                                    commentLine.append(' ');
                                } else if (inIndent && nextChar == '\t') {
                                    commentLineIndent += 4;
                                    commentLine.append("    ");
                                } else if (nextChar == '\r' || nextChar == '\n') {
                                    break;
                                } else {
                                    inIndent = false;
                                    commentLine.append(nextChar);
                                }
                            }

                            // Determine indent
                            if (!firstLine && commentIndent == Integer.MAX_VALUE && commentLine.length() > 0) {
                                // if this is a new comment block, the comment indent size for this
                                // block is based the first full line of the comment (ignoring the
                                // line with the ");
                buf.append(lineSeparator);

                if (spaceAfterComment) {
                    buf.append(lineSeparator);
                }
            }


            // property
            buf.append(indent);
            buf.append("");
            buf.append(substitutePredefinedEntries(value));
            buf.append("");
            buf.append(lineSeparator);

            firstProperty = false;
        }


        buf.append("").append(lineSeparator);

        osw.write(buf.toString());
        osw.flush();
    }

    private String substitutePredefinedEntries(final String s) {
        /*
        * substitution for predefined character entities
        * to use them safely in XML
        */
        return s.replaceAll("&", "&")
                .replaceAll("<", "<")
                .replaceAll(">", ">")
                .replaceAll("\u0027", "'")
                .replaceAll("\"", """);
    }

    //
    // Delegate all remaining methods to the properties object
    //

    @Override
    public boolean isEmpty() {
        return properties.isEmpty();
    }

    @Override
    public int size() {
        return properties.size();
    }

    @Override
    public Object get(Object key) {
        key = normalize(key);
        return properties.get(key);
    }

    @Override
    public Object put(Object key, final Object value) {
        key = normalize(key);
        if (STRING.isInstance(key)) {
            final String name = STRING.cast(key);
            if (!attributes.containsKey(name)) {
                attributes.put(name, new LinkedHashMap());
            }
        }
        return properties.put(key, value);
    }

    @Override
    public Object remove(Object key) {
        key = normalize(key);
        comments.remove(key);
        attributes.remove(key);
        return properties.remove(key);
    }

    @Override
    public void putAll(final Map t) {
        for (final Map.Entry entry : t.entrySet()) {
            put(entry.getKey(), entry.getValue());
        }
        if (t instanceof SuperProperties) {
            final SuperProperties superProperties = (SuperProperties) t;
            for (final Map.Entry entry : superProperties.comments.entrySet()) {
                comments.put(normalize(entry.getKey()), entry.getValue());
            }
            for (final Map.Entry> entry : superProperties.attributes.entrySet()) {
                attributes.put(normalize(entry.getKey()), entry.getValue());
            }
        }
    }

    /**
     * Returns an unmodifiable view of the keys.
     *
     * @return an unmodifiable view of the keys
     */
    @Override
    public Set keySet() {
        return Collections.unmodifiableSet(properties.keySet());
    }

    @Override
    public Enumeration keys() {
        return Collections.enumeration(properties.keySet());
    }

    /**
     * Returns an unmodifiable view of the values.
     *
     * @return an unmodifiable view of the values
     */
    @Override
    public Collection values() {
        return Collections.unmodifiableCollection(properties.values());
    }

    /**
     * Returns an unmodifiable view of the entries.
     *
     * @return an unmodifiable view of the entries
     */
    @Override
    public Set> entrySet() {
        return Collections.unmodifiableSet(properties.entrySet());
    }

    @Override
    public Enumeration elements() {
        return Collections.enumeration(properties.values());
    }

    @Override
    public boolean containsKey(Object key) {
        key = normalize(key);
        return properties.containsKey(key);
    }

    @Override
    public boolean containsValue(final Object value) {
        return properties.containsValue(value);
    }

    @Override
    public boolean contains(final Object value) {
        return properties.containsValue(value);
    }

    @Override
    public void clear() {
        properties.clear();
        comments.clear();
        attributes.clear();
    }

    @Override
    @SuppressWarnings({"unchecked"})
    public Object clone() {
        final SuperProperties clone = (SuperProperties) super.clone();
        clone.properties = (LinkedHashMap) properties.clone();
        clone.comments = (LinkedHashMap) comments.clone();
        clone.attributes = (LinkedHashMap>) attributes.clone();
        for (final Map.Entry> entry : clone.attributes.entrySet()) {
            entry.setValue((LinkedHashMap) entry.getValue().clone());
        }
        return clone;
    }

    @SuppressWarnings({"EqualsWhichDoesntCheckParameterClass"})
    public boolean equals(final Object o) {
        return properties.equals(o);
    }

    public int hashCode() {
        return properties.hashCode();
    }

    public String toString() {
        return properties.toString();
    }

    @Override
    protected void rehash() {
    }

    private Object normalize(final Object key) {

        if (STRING.isInstance(key)) {
            return normalize(STRING.cast(key));
        }
        return key;
    }

    private String normalize(final String property) {
        if (!caseInsensitive) {
            return property;
        }

        if (super.containsKey(property)) {
            return property;
        }

        String key = findKey(property, keySet());
        if (key != null) {
            return key;
        }

        if (defaults != null) {
            key = findKey(property, defaults.keySet());
            if (key != null) {
                return key;
            }
        }

        return property;
    }

    /**
     * Find property key or null
     *
     * @param property String
     * @param keySet   Set
     * @return String or null
     */
    private String findKey(final String property, final Set keySet) {
        for (final Object o : keySet) {
            if (String.class.isInstance(o)) {
                final String key = String.class.cast(o);
                if (key.equalsIgnoreCase(property)) {
                    return key;
                }
            }
        }
        return null;
    }
}