Mercurial > jhg
view src/org/tmatesoft/hg/core/HgAnnotateCommand.java @ 578:f97e81d13190
Throw exception if repository encountered is too old for Hg4J
author | Artem Tikhomirov <tikhomirov.artem@gmail.com> |
---|---|
date | Wed, 17 Apr 2013 16:02:52 +0200 |
parents | c4fd1037bc6f |
children | 0890628ed51e |
line wrap: on
line source
/* * Copyright (c) 2013 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 static org.tmatesoft.hg.repo.HgRepository.NO_REVISION; import java.util.Arrays; import org.tmatesoft.hg.internal.Callback; import org.tmatesoft.hg.internal.CsetParamKeeper; import org.tmatesoft.hg.internal.Experimental; import org.tmatesoft.hg.internal.FileAnnotation; import org.tmatesoft.hg.internal.FileAnnotation.LineDescriptor; import org.tmatesoft.hg.internal.FileAnnotation.LineInspector; import org.tmatesoft.hg.repo.HgBlameFacility.BlockData; import org.tmatesoft.hg.repo.HgBlameFacility; import org.tmatesoft.hg.repo.HgDataFile; import org.tmatesoft.hg.repo.HgRepository; import org.tmatesoft.hg.util.CancelledException; import org.tmatesoft.hg.util.Path; /** * WORK IN PROGRESS. UNSTABLE API * * 'hg annotate' counterpart, report origin revision and file line-by-line * * @author Artem Tikhomirov * @author TMate Software Ltd. */ @Experimental(reason="Work in progress. Unstable API") public class HgAnnotateCommand extends HgAbstractCommand<HgAnnotateCommand> { private final HgRepository repo; private final CsetParamKeeper annotateRevision; private Path file; private boolean followRename; public HgAnnotateCommand(HgRepository hgRepo) { repo = hgRepo; annotateRevision = new CsetParamKeeper(repo); annotateRevision.doSet(HgRepository.TIP); } public HgAnnotateCommand changeset(Nodeid nodeid) throws HgBadArgumentException { annotateRevision.set(nodeid); return this; } public HgAnnotateCommand changeset(int changelogRevIndex) throws HgBadArgumentException { annotateRevision.set(changelogRevIndex); return this; } /** * Select file to annotate, origin of renamed/copied file would be followed, too. * * @param filePath path relative to repository root * @return <code>this</code> for convenience */ public HgAnnotateCommand file(Path filePath) { return file(filePath, true); } /** * Select file to annotate. * * @param filePath path relative to repository root * @param followCopyRename true to follow copies/renames. * @return <code>this</code> for convenience */ public HgAnnotateCommand file(Path filePath, boolean followCopyRename) { file = filePath; followRename = followCopyRename; return this; } // TODO [1.1] set encoding and provide String line content from LineInfo public void execute(Inspector inspector) throws HgException, HgCallbackTargetException, CancelledException { if (inspector == null) { throw new IllegalArgumentException(); } if (file == null) { throw new HgBadArgumentException("Command needs file argument", null); } HgDataFile df = repo.getFileNode(file); if (!df.exists()) { return; } final int changesetStart = followRename ? 0 : df.getChangesetRevisionIndex(0); Collector c = new Collector(); FileAnnotation fa = new FileAnnotation(c); HgBlameFacility af = new HgBlameFacility(df); af.annotate(changesetStart, annotateRevision.get(), fa, HgIterateDirection.NewToOld); LineImpl li = new LineImpl(); for (int i = 0; i < c.lineRevisions.length; i++) { li.init(i+1, c.lineRevisions[i], c.line(i)); inspector.next(li); } } /** * Callback to receive annotated lines */ @Callback public interface Inspector { // start(FileDescriptor) throws HgCallbackTargetException; void next(LineInfo lineInfo) throws HgCallbackTargetException; // end(FileDescriptor) throws HgCallbackTargetException; } /** * Describes a line reported through {@link Inspector#next(LineInfo)} * * Clients shall not implement this interface */ public interface LineInfo { int getLineNumber(); int getChangesetIndex(); byte[] getContent(); } // FIXME there's no need in FileAnnotation.LineInspector, merge it here private static class Collector implements LineInspector { private int[] lineRevisions; private byte[][] lines; Collector() { } public void line(int lineNumber, int changesetRevIndex, BlockData lineContent, LineDescriptor ld) { if (lineRevisions == null) { lineRevisions = new int [ld.totalLines()]; Arrays.fill(lineRevisions, NO_REVISION); lines = new byte[ld.totalLines()][]; } lineRevisions[lineNumber] = changesetRevIndex; lines[lineNumber] = lineContent.asArray(); } public byte[] line(int i) { return lines[i]; } } private static class LineImpl implements LineInfo { private int ln; private int rev; private byte[] content; void init(int line, int csetRev, byte[] cnt) { ln = line; rev = csetRev; content = cnt; } public int getLineNumber() { return ln; } public int getChangesetIndex() { return rev; } public byte[] getContent() { return content; } } }