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

org.exist.numbering.DLN Maven / Gradle / Ivy

/*
 *  eXist Open Source Native XML Database
 *  Copyright (C) 2001-06 The eXist Project
 *  http://exist-db.org
 *
 *  This program is free software; you can redistribute it and/or
 *  modify it under the terms of the GNU Lesser General Public License
 *  as published by the Free Software Foundation; either version 2
 *  of the License, or (at your option) any later version.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU Lesser General Public License for more details.
 *
 *  You should have received a copy of the GNU Lesser General Public
 *  License along with this library; if not, write to the Free Software
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 *
 *  $Id$
 */
package org.exist.numbering;

import java.io.IOException;
import org.exist.security.MessageDigester;
import org.exist.storage.io.VariableByteInput;
import org.exist.storage.io.VariableByteOutputStream;

/**
 * Represents a node id in the form of a dynamic level number (DLN). DLN's are
 * hierarchical ids, which borrow from Dewey's decimal classification. Examples for
 * node ids: 1, 1.1, 1.2, 1.2.1, 1.2.2, 1.3. In this case, 1 represents the root node, 1.1 is
 * the first node on the second level, 1.2 the second, and so on.
 * 
 * To support efficient insertion of new nodes between existing nodes, we use the
 * concept of sublevel ids. Between two nodes 1.1 and 1.2, a new node can be inserted
 * as 1.1/1, where the / is the sublevel separator. The / does not start a new level. 1.1 and 
 * 1.1/1 are thus on the same level of the tree.
 * 
 * In the binary encoding, the '.' is represented by a 0-bit while '/' is written as a 1-bit.
 */
public class DLN extends DLNBase implements NodeId {

    /**
     * Constructs a new DLN with a single id with value 1.
     *
     */
    public DLN() {
        this(1);
    }

    /**
     * Constructs a new DLN by parsing the string argument.
     * In the string, levels are separated by a '.', sublevels by
     * a '/'. For example, '1.2/1' or '1.2/1.2' are valid ids.
     * 
     * @param s string represenation of a DLN
     */
    public DLN(final String s) {
        bits = new byte[1];
        final StringBuilder buf = new StringBuilder(16);
        boolean subValue = false;
        for(int p = 0; p < s.length(); p++) {
            final char ch = s.charAt(p);
            if(ch == '.' || ch == '/') {
                addLevelId(Integer.parseInt(buf.toString()), subValue);
                subValue = ch == '/';
                buf.setLength(0);
            } else {
                buf.append(ch);
            }
        }
        if(buf.length() > 0) {
            addLevelId(Integer.parseInt(buf.toString()), subValue);
        }
    }

    /**
     * Constructs a new DLN, using the passed id as its
     * single level value.
     * 
     * @param id the value for the initial first level of this DLN
     */
    public DLN(final int id) {
        bits = new byte[1];
        addLevelId(id, false);
    }

    /**
     * Constructs a new DLN by copying the data of the
     * passed DLN.
     * 
     * @param other the DLN to copy data from
     */
    public DLN(final DLN other) {
        super(other);
    }

    /**
     * Reads a DLN from the given byte[].
     * 
     * @param units number of bits to read
     * @param data the byte[] to read from
     * @param startOffset the start offset to start reading at
     */
    public DLN(final int units, final byte[] data, final int startOffset) {
        super(units, data, startOffset);
    }

    /**
     * Reads a DLN from the given {@link VariableByteInput} stream.
     * 
     * @see #write(VariableByteOutputStream)
     * @param bitCnt total number of bits to read
     * @param is the input stream to read from
     * @throws IOException in case of an error reading the DLN
     */
    public DLN(final short bitCnt, final VariableByteInput is) throws IOException {
        super(bitCnt, is);
    }

    public DLN(final byte prefixLen, final DLN previous, final short bitCnt, final VariableByteInput is) throws IOException {
        super(prefixLen, previous, bitCnt, is);
    }

    /**
     * Create a new DLN by copying nbits bits from the given 
     * byte[].
     * 
     * @param data the byte[] to read bits from
     * @param nbits number of bits to read
     */
    protected DLN(final byte[] data, final int nbits) {
        super(data, nbits);
    }

    /**
     * Returns a new DLN representing the first child
     * node of this node.
     *
     * @return new child node id
     */
    @Override
    public NodeId newChild() {
        final DLN child = new DLN(this);
        child.addLevelId(1, false);
        return child;
    }

    /**
     * Returns a new DLN representing the next following
     * sibling of this node.
     *
     * @return new sibling node id.
     */
    @Override
    public NodeId nextSibling() {
        final DLN sibling = new DLN(this);
        sibling.incrementLevelId();
        return sibling;
    }

    @Override
    public NodeId precedingSibling() {
        final DLN sibling = new DLN(this);
        sibling.decrementLevelId();
        return sibling;
    }

    @Override
    public NodeId getChild(final int child) {
        final DLN nodeId = new DLN(this);
        nodeId.addLevelId(child, false);
        return nodeId;
    }

    @Override
    public NodeId insertNode(final NodeId right) {
        final DLN rightNode = (DLN) right;
        if (right == null) {
            return nextSibling();
        }
        final int lastLeft = lastLevelOffset();
        final int lastRight = rightNode.lastLevelOffset();
        final int lenLeft = getSubLevelCount(lastLeft);
        final int lenRight = rightNode.getSubLevelCount(lastRight);
        final DLN newNode;
        if(lenLeft > lenRight) {
            newNode = new DLN(this);
            newNode.incrementLevelId();
        } else if(lenLeft < lenRight) {
            newNode = (DLN) rightNode.insertBefore(); 
        } else {
            newNode = new DLN(this);
            newNode.addLevelId(1, true);
        }
        return newNode;
    }

    @Override
    public NodeId insertBefore() {
        final int lastPos = lastFieldPosition();
        final int lastId = getLevelId(lastPos);
        final DLN newNode = new DLN(this);
        //System.out.println("insertBefore: " + newNode.toString() + " = " + newNode.bitIndex);
        if (lastId == 1) {
            newNode.setLevelId(lastPos, 0);
            newNode.addLevelId(35, true);
        } else {
            newNode.setLevelId(lastPos, lastId - 1);
            newNode.compact();
            //System.out.println("newNode: " + newNode.toString() + " = " + newNode.bitIndex + "; last = " + lastPos);
        }
        return newNode;
    }

    @Override
    public NodeId append(final NodeId otherId) {
        final DLN other = (DLN) otherId;
        final DLN newId = new DLN(this);
        int offset = 0;
        while(offset <= other.bitIndex) {
            boolean subLevel = false;
            if (offset > 0) {
                subLevel = ((other.bits[offset >> UNIT_SHIFT] & (1 << ((7 - offset++) & 7))) != 0);
            }
            final int id = other.getLevelId(offset);
            newId.addLevelId(id, subLevel);
            offset += DLNBase.getUnitsRequired(id) * BITS_PER_UNIT;
        }
        return newId;
    }

    /**
     * Returns a new DLN representing the parent of the
     * current node. If the current node is the root element
     * of the document, the method returns 
     * {@link NodeId#DOCUMENT_NODE}. If the current node
     * is the document node, null is returned.
     * 
     * @see NodeId#getParentId()
     */
    @Override
    public NodeId getParentId() {
        if(this == DOCUMENT_NODE) {
            return null;
        }
        
        final int last = lastLevelOffset();
        if (last == 0) {
            return DOCUMENT_NODE;
        }
        
        return new DLN(bits, last - 1);
    }

    @Override
    public boolean isDescendantOf(final NodeId ancestor) {
        final DLN other = (DLN) ancestor;
        return startsWith(other) && bitIndex > other.bitIndex
            && isLevelSeparator(other.bitIndex + 1);
    }

    @Override
    public boolean isDescendantOrSelfOf(final NodeId other) {
        final DLN ancestor = (DLN) other;
        return startsWith(ancestor) &&
            (bitIndex == ancestor.bitIndex || isLevelSeparator((ancestor).bitIndex + 1));
    }

    @Override
    public boolean isChildOf(final NodeId parent) {
        final DLN other = (DLN) parent;
        if(!startsWith(other)) {
            return false;
        }
        final int levels = getLevelCount(other.bitIndex + 2);
        return levels == 1;
    }

    @Override
    public int computeRelation(final NodeId ancestor) {
        final DLN other = (DLN) ancestor;
        if (other == NodeId.DOCUMENT_NODE) {
            return getLevelCount(0) == 1 ? IS_CHILD : IS_DESCENDANT;
        }
        
        if (startsWith(other)) {
            if (bitIndex == other.bitIndex) {
                return IS_SELF;
            }
            if (bitIndex > other.bitIndex && isLevelSeparator(other.bitIndex + 1)) {
                if (getLevelCount(other.bitIndex + 2) == 1) {
                    return IS_CHILD;
                }
                return IS_DESCENDANT;
            }
        }
        return -1;
    }

    @Override
    public boolean isSiblingOf(final NodeId sibling) {
        final NodeId parent = getParentId();
        return sibling.isChildOf(parent);
    }

    /**
     * Returns the level within the document tree at which
     * this node occurs.
     */
    @Override
    public int getTreeLevel() {
        return getLevelCount(0);
    }

    @Override
    public boolean equals(final NodeId other) {
        return super.equals((DLNBase) other);
    }

    @Override
    public int compareTo(final NodeId otherId) {
        if(otherId == null) {
            return 1;
        }
        final DLN other = (DLN) otherId;
        final int a1len = bits.length;
        final int a2len = other.bits.length;
        final int limit = a1len <= a2len ? a1len : a2len;
        final byte[] obits = other.bits;
        
        for(int i = 0; i < limit; i++) {
            final byte b1 = bits[i];
            final byte b2 = obits[i];
            if(b1 != b2) {
                return (b1 & 0xFF) - (b2 & 0xFF);
            }
        }
        return (a1len - a2len);
    }

    @Override
    public boolean after(final NodeId other, final boolean isFollowing) {
        if (compareTo(other) > 0) {
            if (isFollowing) {
                return !isDescendantOf(other);
            }
            return true;
        }
        return false;
    }

    @Override
    public boolean before(final NodeId other, final boolean isPreceding) {
        if (compareTo(other) < 0) {
            if (isPreceding) {
                return !other.isDescendantOf(this);
            }
            return true;
        }
        return false;
    }

    /**
     * Write the node id to a {@link VariableByteOutputStream}.
     *
     * @param os the output stream to write to
     * @throws IOException in case of write error
     */
    @Override
    public void write(final VariableByteOutputStream os) throws IOException {
        os.writeShort((short) units());
        os.write(bits, 0, bits.length);
    }

    @Override
    public NodeId write(final NodeId prevId, final VariableByteOutputStream os) throws IOException {
        int i = 0;
        if(prevId != null) {
            final DLN previous = (DLN) prevId;
            final int len = Math.min(bits.length, previous.bits.length);
            for( ; i < len; i++) {
                final byte b = bits[i];
                if (b != previous.bits[i]) {
                    break;
                }
            }
        }
        os.writeByte((byte) i);
        os.writeShort((short) units());
        os.write(bits, i, bits.length - i);
        return this;
    }

    public static void main(final String[] args) {
        final DLN left1 = new DLN("5.6.2.6");
        final DLN left2 = new DLN("5.6.2.7");
        final DLN right = new DLN("5.6.2.7.1");
        final byte[] nodeIdData1 = new byte[left1.size()];
        left1.serialize(nodeIdData1, 0);
        System.out.println(left1.units() + ": " +left1 + ": " + MessageDigester.byteArrayToHex(nodeIdData1) + Integer.toHexString(left1.units()));
        final byte[] nodeIdData2 = new byte[left2.size()];
        left2.serialize(nodeIdData2, 0);
        System.out.println(left2.units() + ": " + left2 + ": " + MessageDigester.byteArrayToHex(nodeIdData2) + Integer.toHexString(left2.units()));
        final byte[] nodeIdData3 = new byte[right.size()];
        right.serialize(nodeIdData3, 0);
        System.out.println(right.units() + ": " + right + ": " + MessageDigester.byteArrayToHex(nodeIdData3) + Integer.toHexString(right.units()));
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy