Mercurial > hg4j
view src/org/tmatesoft/hg/core/HgFileRevision.java @ 508:ca5202afea90
Support follow history option when walking file history tree
author | Artem Tikhomirov <tikhomirov.artem@gmail.com> |
---|---|
date | Wed, 12 Dec 2012 20:52:10 +0100 |
parents | b3b1db9301a2 |
children | 5dcb4581c8ef |
line wrap: on
line source
/* * Copyright (c) 2011-2012 TMate Software Ltd * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; version 2 of the License. * * 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 General Public License for more details. * * For information on how to redistribute this software under * the terms of a license other than GNU General Public License * contact TMate Software at support@hg4j.com */ package org.tmatesoft.hg.core; import org.tmatesoft.hg.repo.HgDataFile; import org.tmatesoft.hg.repo.HgManifest; import org.tmatesoft.hg.repo.HgManifest.Flags; import org.tmatesoft.hg.repo.HgRepository; import org.tmatesoft.hg.repo.HgRuntimeException; import org.tmatesoft.hg.util.ByteChannel; import org.tmatesoft.hg.util.CancelledException; import org.tmatesoft.hg.util.Pair; import org.tmatesoft.hg.util.Path; /** * Keeps together information about specific file revision * * @author Artem Tikhomirov * @author TMate Software Ltd. */ public final class HgFileRevision { private final HgRepository repo; private final Nodeid revision; private final Path path; private Path origin; private Boolean isCopy = null; // null means not yet known private Pair<Nodeid, Nodeid> parents; private Flags flags; // null unless set/extracted /** * New description of a file revision from a specific repository. * * <p>Although this constructor is public, and clients can use it to construct own file revisions to pass e.g. to commands, its use is discouraged. * * @param hgRepo repository * @param rev file revision * @param manifestEntryFlags file flags at this revision (optional, may be null) * @param p path of the file at the given revision */ public HgFileRevision(HgRepository hgRepo, Nodeid rev, HgManifest.Flags manifestEntryFlags, Path p) { if (hgRepo == null || rev == null || p == null) { // since it's package local, it is our code to blame for non validated arguments throw new IllegalArgumentException(); } repo = hgRepo; revision = rev; flags = manifestEntryFlags; path = p; } // this cons shall be used when we know whether p was a copy. Perhaps, shall pass Map<Path,Path> instead to stress orig argument is not optional HgFileRevision(HgRepository hgRepo, Nodeid rev, HgManifest.Flags flags, Path p, Path orig) { this(hgRepo, rev, flags, p); isCopy = Boolean.valueOf(orig == null); origin = orig; } public Path getPath() { return path; } public Nodeid getRevision() { return revision; } /** * Extract flags of the file as recorded in the manifest for this file revision * @return whether regular file, executable or a symbolic link * @throws HgRuntimeException subclass thereof to indicate issues with the library. <em>Runtime exception</em> */ public HgManifest.Flags getFileFlags() throws HgRuntimeException { if (flags == null) { /* * Note, for uses other than HgManifestCommand or HgChangesetFileSneaker, when no flags come through the cons, * it's possible to face next shortcoming: * Imagine csetA and csetB, with corresponding manifestA and manifestB, the file didn't change (revision/nodeid is the same) * but flag of the file has changed (e.g. became executable). Since HgFileRevision doesn't keep reference to * an actual manifest revision, but only file's, and it's likely the flags returned from this method would * yield result as from manifestA (i.e. no flag change in manifestB ever noticed). */ HgDataFile df = repo.getFileNode(path); int revIdx = df.getRevisionIndex(revision); flags = df.getFlags(revIdx); } return flags; } public boolean wasCopied() throws HgException { if (isCopy == null) { checkCopy(); } return isCopy.booleanValue(); } /** * @return <code>null</code> if {@link #wasCopied()} is <code>false</code>, name of the copy source otherwise. */ public Path getOriginIfCopy() throws HgException { if (wasCopied()) { return origin; } return null; } /** * Access revisions this file revision originates from. * Note, these revisions are records in the file history, not that of the whole repository (aka changeset revisions) * In most cases, only one parent revision would be present, only for merge revisions one can expect both. * * @return parent revisions of this file revision, with {@link Nodeid#NULL} for missing values. * @throws HgRuntimeException subclass thereof to indicate issues with the library. <em>Runtime exception</em> */ public Pair<Nodeid, Nodeid> getParents() throws HgRuntimeException { if (parents == null) { HgDataFile fn = repo.getFileNode(path); int revisionIndex = fn.getRevisionIndex(revision); int[] pr = new int[2]; byte[] p1 = new byte[20], p2 = new byte[20]; // XXX Revlog#parents is not the best method to use here // need smth that gives Nodeids (piped through Pool<Nodeid> from repo's context) fn.parents(revisionIndex, pr, p1, p2); parents = new Pair<Nodeid, Nodeid>(Nodeid.fromBinary(p1, 0), Nodeid.fromBinary(p2, 0)); } return parents; } public void putContentTo(ByteChannel sink) throws HgException, CancelledException { HgDataFile fn = repo.getFileNode(path); int revisionIndex = fn.getRevisionIndex(revision); fn.contentWithFilters(revisionIndex, sink); } private void checkCopy() throws HgException { HgDataFile fn = repo.getFileNode(path); if (fn.isCopy()) { if (fn.getRevision(0).equals(revision)) { // this HgFileRevision represents first revision of the copy isCopy = Boolean.TRUE; origin = fn.getCopySourceName(); return; } } isCopy = Boolean.FALSE; } }