/* * Copyright (C) 2008-2009, Google Inc. * Copyright (C) 2007, Robin Rosenberg * Copyright (C) 2008, Shawn O. Pearce * and other copyright owners as documented in the project's IP log. * * This program and the accompanying materials are made available * under the terms of the Eclipse Distribution License v1.0 which * accompanies this distribution, is reproduced below, and is * available at http://www.eclipse.org/org/documents/edl-v10.php * * All rights reserved. * * Redistribution and use in source and binary forms, with or * without modification, are permitted provided that the following * conditions are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * - Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials provided * with the distribution. * * - Neither the name of the Eclipse Foundation, Inc. nor the * names of its contributors may be used to endorse or promote * products derived from this software without specific prior * written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ package org.eclipse.jgit.treewalk; import java.io.IOException; import java.nio.ByteBuffer; import java.nio.CharBuffer; import org.eclipse.jgit.attributes.AttributesHandler; import org.eclipse.jgit.attributes.AttributesNode; import org.eclipse.jgit.errors.CorruptObjectException; import org.eclipse.jgit.errors.IncorrectObjectTypeException; import org.eclipse.jgit.lib.Constants; import org.eclipse.jgit.lib.FileMode; import org.eclipse.jgit.lib.MutableObjectId; import org.eclipse.jgit.lib.ObjectId; import org.eclipse.jgit.lib.ObjectReader; import org.eclipse.jgit.util.Paths; /** * Walks a Git tree (directory) in Git sort order. *

* A new iterator instance should be positioned on the first entry, or at eof. * Data for the first entry (if not at eof) should be available immediately. *

* Implementors must walk a tree in the Git sort order, which has the following * odd sorting: *

    *
  1. A.c
  2. *
  3. A/c
  4. *
  5. A0c
  6. *
*

* In the second item, A is the name of a subtree and * c is a file within that subtree. The other two items are files * in the root level tree. * * @see CanonicalTreeParser */ public abstract class AbstractTreeIterator { /** Default size for the {@link #path} buffer. */ protected static final int DEFAULT_PATH_SIZE = 128; /** A dummy object id buffer that matches the zero ObjectId. */ protected static final byte[] zeroid = new byte[Constants.OBJECT_ID_LENGTH]; /** * Iterator for the parent tree; null if we are the root iterator. *

* Used by {@link TreeWalk} and {@link AttributesHandler} * * @since 4.3 */ public final AbstractTreeIterator parent; /** The iterator this current entry is path equal to. */ AbstractTreeIterator matches; /** * Parsed rules of .gitattributes file if it exists. * * @since 4.2 */ protected AttributesNode attributesNode; /** * Number of entries we moved forward to force a D/F conflict match. * * @see NameConflictTreeWalk */ int matchShift; /** * Mode bits for the current entry. *

* A numerical value from FileMode is usually faster for an iterator to * obtain from its data source so this is the preferred representation. * * @see org.eclipse.jgit.lib.FileMode */ protected int mode; /** * Path buffer for the current entry. *

* This buffer is pre-allocated at the start of walking and is shared from * parent iterators down into their subtree iterators. The sharing allows * the current entry to always be a full path from the root, while each * subtree only needs to populate the part that is under their control. */ protected byte[] path; /** * Position within {@link #path} this iterator starts writing at. *

* This is the first offset in {@link #path} that this iterator must * populate during {@link #next}. At the root level (when {@link #parent} * is null) this is 0. For a subtree iterator the index before this position * should have the value '/'. */ protected final int pathOffset; /** * Total length of the current entry's complete path from the root. *

* This is the number of bytes within {@link #path} that pertain to the * current entry. Values at this index through the end of the array are * garbage and may be randomly populated from prior entries. */ protected int pathLen; /** * Create a new iterator with no parent. */ protected AbstractTreeIterator() { parent = null; path = new byte[DEFAULT_PATH_SIZE]; pathOffset = 0; } /** * Create a new iterator with no parent and a prefix. *

* The prefix path supplied is inserted in front of all paths generated by * this iterator. It is intended to be used when an iterator is being * created for a subsection of an overall repository and needs to be * combined with other iterators that are created to run over the entire * repository namespace. * * @param prefix * position of this iterator in the repository tree. The value * may be null or the empty string to indicate the prefix is the * root of the repository. A trailing slash ('/') is * automatically appended if the prefix does not end in '/'. */ protected AbstractTreeIterator(final String prefix) { parent = null; if (prefix != null && prefix.length() > 0) { final ByteBuffer b; b = Constants.CHARSET.encode(CharBuffer.wrap(prefix)); pathLen = b.limit(); path = new byte[Math.max(DEFAULT_PATH_SIZE, pathLen + 1)]; b.get(path, 0, pathLen); if (path[pathLen - 1] != '/') path[pathLen++] = '/'; pathOffset = pathLen; } else { path = new byte[DEFAULT_PATH_SIZE]; pathOffset = 0; } } /** * Create a new iterator with no parent and a prefix. *

* The prefix path supplied is inserted in front of all paths generated by * this iterator. It is intended to be used when an iterator is being * created for a subsection of an overall repository and needs to be * combined with other iterators that are created to run over the entire * repository namespace. * * @param prefix * position of this iterator in the repository tree. The value * may be null or the empty array to indicate the prefix is the * root of the repository. A trailing slash ('/') is * automatically appended if the prefix does not end in '/'. */ protected AbstractTreeIterator(final byte[] prefix) { parent = null; if (prefix != null && prefix.length > 0) { pathLen = prefix.length; path = new byte[Math.max(DEFAULT_PATH_SIZE, pathLen + 1)]; System.arraycopy(prefix, 0, path, 0, pathLen); if (path[pathLen - 1] != '/') path[pathLen++] = '/'; pathOffset = pathLen; } else { path = new byte[DEFAULT_PATH_SIZE]; pathOffset = 0; } } /** * Create an iterator for a subtree of an existing iterator. * * @param p * parent tree iterator. */ protected AbstractTreeIterator(final AbstractTreeIterator p) { parent = p; path = p.path; pathOffset = p.pathLen + 1; try { path[pathOffset - 1] = '/'; } catch (ArrayIndexOutOfBoundsException e) { growPath(p.pathLen); path[pathOffset - 1] = '/'; } } /** * Create an iterator for a subtree of an existing iterator. *

* The caller is responsible for setting up the path of the child iterator. * * @param p * parent tree iterator. * @param childPath * path array to be used by the child iterator. This path must * contain the path from the top of the walk to the first child * and must end with a '/'. * @param childPathOffset * position within childPath where the child can * insert its data. The value at * childPath[childPathOffset-1] must be '/'. */ protected AbstractTreeIterator(final AbstractTreeIterator p, final byte[] childPath, final int childPathOffset) { parent = p; path = childPath; pathOffset = childPathOffset; } /** * Grow the path buffer larger. * * @param len * number of live bytes in the path buffer. This many bytes will * be moved into the larger buffer. */ protected void growPath(final int len) { setPathCapacity(path.length << 1, len); } /** * Ensure that path is capable to hold at least {@code capacity} bytes * * @param capacity * the amount of bytes to hold * @param len * the amount of live bytes in path buffer */ protected void ensurePathCapacity(final int capacity, final int len) { if (path.length >= capacity) return; final byte[] o = path; int current = o.length; int newCapacity = current; while (newCapacity < capacity && newCapacity > 0) newCapacity <<= 1; setPathCapacity(newCapacity, len); } /** * Set path buffer capacity to the specified size * * @param capacity * the new size * @param len * the amount of bytes to copy */ private void setPathCapacity(int capacity, int len) { final byte[] o = path; final byte[] n = new byte[capacity]; System.arraycopy(o, 0, n, 0, len); for (AbstractTreeIterator p = this; p != null && p.path == o; p = p.parent) p.path = n; } /** * Compare the path of this current entry to another iterator's entry. * * @param p * the other iterator to compare the path against. * @return -1 if this entry sorts first; 0 if the entries are equal; 1 if * p's entry sorts first. */ public int pathCompare(final AbstractTreeIterator p) { return pathCompare(p, p.mode); } int pathCompare(final AbstractTreeIterator p, final int pMode) { // Its common when we are a subtree for both parents to match; // when this happens everything in path[0..cPos] is known to // be equal and does not require evaluation again. // int cPos = alreadyMatch(this, p); return pathCompare(p.path, cPos, p.pathLen, pMode, cPos); } /** * Seek the iterator on a file, if present. * * @param name * file name to find (will not find a directory). * @return true if the file exists in this tree; false otherwise. * @throws org.eclipse.jgit.errors.CorruptObjectException * tree is invalid. * @since 4.2 */ public boolean findFile(String name) throws CorruptObjectException { return findFile(Constants.encode(name)); } /** * Seek the iterator on a file, if present. * * @param name * file name to find (will not find a directory). * @return true if the file exists in this tree; false otherwise. * @throws org.eclipse.jgit.errors.CorruptObjectException * tree is invalid. * @since 4.2 */ public boolean findFile(byte[] name) throws CorruptObjectException { for (; !eof(); next(1)) { int cmp = pathCompare(name, 0, name.length, 0, pathOffset); if (cmp == 0) { return true; } else if (cmp > 0) { return false; } } return false; } /** * Compare the path of this current entry to a raw buffer. * * @param buf * the raw path buffer. * @param pos * position to start reading the raw buffer. * @param end * one past the end of the raw buffer (length is end - pos). * @param pathMode * the mode of the path. * @return -1 if this entry sorts first; 0 if the entries are equal; 1 if * p's entry sorts first. */ public int pathCompare(byte[] buf, int pos, int end, int pathMode) { return pathCompare(buf, pos, end, pathMode, 0); } private int pathCompare(byte[] b, int bPos, int bEnd, int bMode, int aPos) { return Paths.compare( path, aPos, pathLen, mode, b, bPos, bEnd, bMode); } private static int alreadyMatch(AbstractTreeIterator a, AbstractTreeIterator b) { for (;;) { final AbstractTreeIterator ap = a.parent; final AbstractTreeIterator bp = b.parent; if (ap == null || bp == null) return 0; if (ap.matches == bp.matches) return a.pathOffset; a = ap; b = bp; } } /** * Check if the current entry of both iterators has the same id. *

* This method is faster than {@link #getEntryObjectId()} as it does not * require copying the bytes out of the buffers. A direct {@link #idBuffer} * compare operation is performed. * * @param otherIterator * the other iterator to test against. * @return true if both iterators have the same object id; false otherwise. */ public boolean idEqual(final AbstractTreeIterator otherIterator) { return ObjectId.equals(idBuffer(), idOffset(), otherIterator.idBuffer(), otherIterator.idOffset()); } /** * Whether the entry has a valid ObjectId. * * @return {@code true} if the entry has a valid ObjectId. */ public abstract boolean hasId(); /** * Get the object id of the current entry. * * @return an object id for the current entry. */ public ObjectId getEntryObjectId() { return ObjectId.fromRaw(idBuffer(), idOffset()); } /** * Obtain the ObjectId for the current entry. * * @param out * buffer to copy the object id into. */ public void getEntryObjectId(final MutableObjectId out) { out.fromRaw(idBuffer(), idOffset()); } /** * Get the file mode of the current entry. * * @return the file mode of the current entry. */ public FileMode getEntryFileMode() { return FileMode.fromBits(mode); } /** * Get the file mode of the current entry as bits. * * @return the file mode of the current entry as bits. */ public int getEntryRawMode() { return mode; } /** * Get path of the current entry, as a string. * * @return path of the current entry, as a string. */ public String getEntryPathString() { return TreeWalk.pathOf(this); } /** * Get the current entry path buffer. *

* Note that the returned byte[] has to be used together with * {@link #getEntryPathLength()} (only use bytes up to this length). * * @return the internal buffer holding the current path. */ public byte[] getEntryPathBuffer() { return path; } /** * Get length of the path in {@link #getEntryPathBuffer()}. * * @return length of the path in {@link #getEntryPathBuffer()}. */ public int getEntryPathLength() { return pathLen; } /** * Get the current entry's path hash code. *

* This method computes a hash code on the fly for this path, the hash is * suitable to cluster objects that may have similar paths together. * * @return path hash code; any integer may be returned. */ public int getEntryPathHashCode() { int hash = 0; for (int i = Math.max(0, pathLen - 16); i < pathLen; i++) { byte c = path[i]; if (c != ' ') hash = (hash >>> 2) + (c << 24); } return hash; } /** * Get the byte array buffer object IDs must be copied out of. *

* The id buffer contains the bytes necessary to construct an ObjectId for * the current entry of this iterator. The buffer can be the same buffer for * all entries, or it can be a unique buffer per-entry. Implementations are * encouraged to expose their private buffer whenever possible to reduce * garbage generation and copying costs. * * @return byte array the implementation stores object IDs within. * @see #getEntryObjectId() */ public abstract byte[] idBuffer(); /** * Get the position within {@link #idBuffer()} of this entry's ObjectId. * * @return offset into the array returned by {@link #idBuffer()} where the * ObjectId must be copied out of. */ public abstract int idOffset(); /** * Create a new iterator for the current entry's subtree. *

* The parent reference of the iterator must be this, * otherwise the caller would not be able to exit out of the subtree * iterator correctly and return to continue walking this. * * @param reader * reader to load the tree data from. * @return a new parser that walks over the current subtree. * @throws org.eclipse.jgit.errors.IncorrectObjectTypeException * the current entry is not actually a tree and cannot be parsed * as though it were a tree. * @throws java.io.IOException * a loose object or pack file could not be read. */ public abstract AbstractTreeIterator createSubtreeIterator( ObjectReader reader) throws IncorrectObjectTypeException, IOException; /** * Create a new iterator as though the current entry were a subtree. * * @return a new empty tree iterator. */ public EmptyTreeIterator createEmptyTreeIterator() { return new EmptyTreeIterator(this); } /** * Create a new iterator for the current entry's subtree. *

* The parent reference of the iterator must be this, otherwise * the caller would not be able to exit out of the subtree iterator * correctly and return to continue walking this. * * @param reader * reader to load the tree data from. * @param idBuffer * temporary ObjectId buffer for use by this method. * @return a new parser that walks over the current subtree. * @throws org.eclipse.jgit.errors.IncorrectObjectTypeException * the current entry is not actually a tree and cannot be parsed * as though it were a tree. * @throws java.io.IOException * a loose object or pack file could not be read. */ public AbstractTreeIterator createSubtreeIterator( final ObjectReader reader, final MutableObjectId idBuffer) throws IncorrectObjectTypeException, IOException { return createSubtreeIterator(reader); } /** * Position this iterator on the first entry. * * The default implementation of this method uses {@code back(1)} until * {@code first()} is true. This is most likely not the most efficient * method of repositioning the iterator to its first entry, so subclasses * are strongly encouraged to override the method. * * @throws org.eclipse.jgit.errors.CorruptObjectException * the tree is invalid. */ public void reset() throws CorruptObjectException { while (!first()) back(1); } /** * Is this tree iterator positioned on its first entry? *

* An iterator is positioned on the first entry if back(1) * would be an invalid request as there is no entry before the current one. *

* An empty iterator (one with no entries) will be * first() && eof(). * * @return true if the iterator is positioned on the first entry. */ public abstract boolean first(); /** * Is this tree iterator at its EOF point (no more entries)? *

* An iterator is at EOF if there is no current entry. * * @return true if we have walked all entries and have none left. */ public abstract boolean eof(); /** * Move to next entry, populating this iterator with the entry data. *

* The delta indicates how many moves forward should occur. The most common * delta is 1 to move to the next entry. *

* Implementations must populate the following members: *

* as well as any implementation dependent information necessary to * accurately return data from {@link #idBuffer()} and {@link #idOffset()} * when demanded. * * @param delta * number of entries to move the iterator by. Must be a positive, * non-zero integer. * @throws org.eclipse.jgit.errors.CorruptObjectException * the tree is invalid. */ public abstract void next(int delta) throws CorruptObjectException; /** * Move to prior entry, populating this iterator with the entry data. *

* The delta indicates how many moves backward should occur.The most common * delta is 1 to move to the prior entry. *

* Implementations must populate the following members: *

* as well as any implementation dependent information necessary to * accurately return data from {@link #idBuffer()} and {@link #idOffset()} * when demanded. * * @param delta * number of entries to move the iterator by. Must be a positive, * non-zero integer. * @throws org.eclipse.jgit.errors.CorruptObjectException * the tree is invalid. */ public abstract void back(int delta) throws CorruptObjectException; /** * Advance to the next tree entry, populating this iterator with its data. *

* This method behaves like seek(1) but is called by * {@link org.eclipse.jgit.treewalk.TreeWalk} only if a * {@link org.eclipse.jgit.treewalk.filter.TreeFilter} was used and ruled * out the current entry from the results. In such cases this tree iterator * may perform special behavior. * * @throws org.eclipse.jgit.errors.CorruptObjectException * the tree is invalid. */ public void skip() throws CorruptObjectException { next(1); } /** * Indicates to the iterator that no more entries will be read. *

* This is only invoked by TreeWalk when the iteration is aborted early due * to a {@link org.eclipse.jgit.errors.StopWalkException} being thrown from * within a TreeFilter. */ public void stopWalk() { // Do nothing by default. Most iterators do not care. } /** * Whether the iterator implements {@link #stopWalk()}. * * @return {@code true} if the iterator implements {@link #stopWalk()}. * @since 4.2 */ protected boolean needsStopWalk() { return false; } /** * Get the length of the name component of the path for the current entry. * * @return the length of the name component of the path for the current * entry. */ public int getNameLength() { return pathLen - pathOffset; } /** * JGit internal API for use by * {@link org.eclipse.jgit.dircache.DirCacheCheckout} * * @return start of name component part within {@link #getEntryPathBuffer()} * @since 2.0 */ public int getNameOffset() { return pathOffset; } /** * Get the name component of the current entry path into the provided * buffer. * * @param buffer * the buffer to get the name into, it is assumed that buffer can * hold the name * @param offset * the offset of the name in the buffer * @see #getNameLength() */ public void getName(byte[] buffer, int offset) { System.arraycopy(path, pathOffset, buffer, offset, pathLen - pathOffset); } /** {@inheritDoc} */ @SuppressWarnings("nls") @Override public String toString() { return getClass().getSimpleName() + "[" + getEntryPathString() + "]"; //$NON-NLS-1$ } /** * Whether or not this Iterator is iterating through the working tree. * * @return whether or not this Iterator is iterating through the working * tree * @since 4.3 */ public boolean isWorkTree() { return false; } }