changeset 520:1ee452f31187

Experimental support for inverse direction history walking. Refactored/streamlined cancellation in HgLogCommand and down the stack
author Artem Tikhomirov <tikhomirov.artem@gmail.com>
date Fri, 21 Dec 2012 21:20:26 +0100 (2012-12-21)
parents 934037edbea0
children 59e555c85da0
files src/org/tmatesoft/hg/core/ChangesetTransformer.java src/org/tmatesoft/hg/core/HgLogCommand.java src/org/tmatesoft/hg/internal/AdapterPlug.java src/org/tmatesoft/hg/internal/BatchRangeHelper.java src/org/tmatesoft/hg/internal/IterateControlMediator.java src/org/tmatesoft/hg/internal/LifecycleBridge.java src/org/tmatesoft/hg/internal/LifecycleProxy.java src/org/tmatesoft/hg/internal/RevlogStream.java src/org/tmatesoft/hg/repo/HgChangelog.java test/org/tmatesoft/hg/test/ErrorCollectorExt.java test/org/tmatesoft/hg/test/TestAuxUtilities.java test/org/tmatesoft/hg/test/TestHistory.java
diffstat 12 files changed, 698 insertions(+), 143 deletions(-) [+]
line wrap: on
line diff
--- a/src/org/tmatesoft/hg/core/ChangesetTransformer.java	Thu Dec 20 20:21:59 2012 +0100
+++ b/src/org/tmatesoft/hg/core/ChangesetTransformer.java	Fri Dec 21 21:20:26 2012 +0100
@@ -18,6 +18,8 @@
 
 import java.util.Set;
 
+import org.tmatesoft.hg.internal.Lifecycle;
+import org.tmatesoft.hg.internal.LifecycleBridge;
 import org.tmatesoft.hg.internal.PathPool;
 import org.tmatesoft.hg.repo.HgChangelog;
 import org.tmatesoft.hg.repo.HgChangelog.RawChangeset;
@@ -37,14 +39,12 @@
  * @author Artem Tikhomirov
  * @author TMate Software Ltd.
  */
-/*package-local*/ class ChangesetTransformer implements HgChangelog.Inspector, Adaptable, CancelSupport {
+/*package-local*/ class ChangesetTransformer implements HgChangelog.Inspector, Adaptable {
 	private final HgChangesetHandler handler;
-	private final ProgressSupport progressHelper;
-	private final CancelSupport cancelHelper;
+	private final LifecycleBridge lifecycleBridge;
 	private final Transformation t;
 	private Set<String> branches;
 	private HgCallbackTargetException failure;
-	private CancelledException cancellation;
 
 	// repo and delegate can't be null, parent walker can
 	// ps and cs can't be null
@@ -58,11 +58,9 @@
 		HgStatusCollector statusCollector = new HgStatusCollector(hgRepo);
 		t = new Transformation(statusCollector, pw);
 		handler = delegate;
-		// we let HgChangelog#range deal with progress (pipe through getAdapter)
-		// but use own cancellation (which involves CallbackTargetException as well, and preserves original cancellation 
-		// exception in case clients care)
-		cancelHelper = cs;
-		progressHelper = ps;
+		// lifecycleBridge takes care of progress and cancellation, plus
+		// gives us explicit way to stop iteration (once HgCallbackTargetException) comes.
+		lifecycleBridge = new LifecycleBridge(ps, cs);
 	}
 	
 	public void next(int revisionNumber, Nodeid nodeid, RawChangeset cset) {
@@ -73,23 +71,21 @@
 		HgChangeset changeset = t.handle(revisionNumber, nodeid, cset);
 		try {
 			handler.cset(changeset);
-			cancelHelper.checkCancelled();
+			lifecycleBridge.nextStep();
 		} catch (HgCallbackTargetException ex) {
 			failure = ex.setRevision(nodeid).setRevisionIndex(revisionNumber);
-		} catch (CancelledException ex) {
-			cancellation = ex;
+			lifecycleBridge.stop();
 		}
 	}
 	
 	public void checkFailure() throws HgCallbackTargetException, CancelledException {
 		if (failure != null) {
 			HgCallbackTargetException toThrow = failure;
-			failure = null; // just in (otherwise unexpected) case this instance would get reused
 			throw toThrow;
 		}
-		if (cancellation != null) {
-			CancelledException toThrow = cancellation;
-			cancellation = null;
+		if (lifecycleBridge.isCancelled()) {
+			CancelledException toThrow = lifecycleBridge.getCancelOrigin();
+			assert toThrow != null;
 			throw toThrow;
 		}
 	}
@@ -121,18 +117,11 @@
 		}
 	}
 
-	public void checkCancelled() throws CancelledException {
-		if (failure != null || cancellation != null) {
-			// stop HgChangelog.Iterator. Our exception is for the purposes of cancellation only,
-			// the one we have stored (this.cancellation) is for user
-			throw new CancelledException(); 
+	public <T> T getAdapter(Class<T> adapterClass) {
+		if (adapterClass == Lifecycle.class) {
+			return adapterClass.cast(lifecycleBridge);
 		}
-	}
-
-	public <T> T getAdapter(Class<T> adapterClass) {
-		if (adapterClass == ProgressSupport.class) {
-			return adapterClass.cast(progressHelper);
-		}
-		return null;
+		// just in case there are more adapters in future
+		return Adaptable.Factory.getAdapter(handler, adapterClass, null);
 	}
 }
\ No newline at end of file
--- a/src/org/tmatesoft/hg/core/HgLogCommand.java	Thu Dec 20 20:21:59 2012 +0100
+++ b/src/org/tmatesoft/hg/core/HgLogCommand.java	Fri Dec 21 21:20:26 2012 +0100
@@ -34,9 +34,13 @@
 import java.util.Set;
 import java.util.TreeSet;
 
+import org.tmatesoft.hg.internal.AdapterPlug;
+import org.tmatesoft.hg.internal.BatchRangeHelper;
+import org.tmatesoft.hg.internal.Experimental;
 import org.tmatesoft.hg.internal.IntMap;
 import org.tmatesoft.hg.internal.IntVector;
 import org.tmatesoft.hg.internal.Lifecycle;
+import org.tmatesoft.hg.internal.LifecycleProxy;
 import org.tmatesoft.hg.repo.HgChangelog;
 import org.tmatesoft.hg.repo.HgChangelog.RawChangeset;
 import org.tmatesoft.hg.repo.HgDataFile;
@@ -287,19 +291,41 @@
 			throw new HgBadArgumentException(String.format("Bad value %d for start revision for range [%1$d..%d]", startRev, lastCset), null);
 		}
 		final ProgressSupport progressHelper = getProgressSupport(handler);
+		final int BATCH_SIZE = 100;
 		try {
 			count = 0;
 			HgParentChildMap<HgChangelog> pw = getParentHelper(file == null); // leave it uninitialized unless we iterate whole repo
 			// ChangesetTransfrom creates a blank PathPool, and #file(String, boolean) above 
 			// may utilize it as well. CommandContext? How about StatusCollector there as well?
 			csetTransform = new ChangesetTransformer(repo, handler, pw, progressHelper, getCancelSupport(handler, true));
+			// FilteringInspector is responsible to check command arguments: users, branches, limit, etc.
+			// prior to passing cset to next Inspector, which is either (a) collector to reverse cset order, then invokes 
+			// transformer from (b), below, with alternative cset order or (b) transformer to hi-level csets. 
 			FilteringInspector filterInsp = new FilteringInspector();
 			filterInsp.changesets(startRev, lastCset);
 			if (file == null) {
-				progressHelper.start(endRev - startRev + 1);
-				repo.getChangelog().range(startRev, endRev, filterInsp);
-				csetTransform.checkFailure();
+				progressHelper.start(lastCset - startRev + 1);
+				if (iterateDirection == IterateDirection.FromOldToNew) {
+					filterInsp.delegateTo(csetTransform);
+					repo.getChangelog().range(startRev, lastCset, filterInsp);
+					csetTransform.checkFailure();
+				} else {
+					assert iterateDirection == IterateDirection.FromNewToOld;
+					BatchRangeHelper brh = new BatchRangeHelper(startRev, lastCset, BATCH_SIZE, true);
+					BatchChangesetInspector batchInspector = new BatchChangesetInspector(Math.min(lastCset-startRev+1, BATCH_SIZE));
+					filterInsp.delegateTo(batchInspector);
+					while (brh.hasNext()) {
+						brh.next();
+						repo.getChangelog().range(brh.start(), brh.end(), filterInsp);
+						for (BatchChangesetInspector.BatchRecord br : batchInspector.iterate(true)) {
+							csetTransform.next(br.csetIndex, br.csetRevision, br.cset);
+							csetTransform.checkFailure();
+						}
+						batchInspector.reset();
+					}
+				}
 			} else {
+				filterInsp.delegateTo(csetTransform);
 				final HgFileRenameHandlerMixin withCopyHandler = Adaptable.Factory.getAdapter(handler, HgFileRenameHandlerMixin.class, null);
 				List<Pair<HgDataFile, Nodeid>> fileRenames = buildFileRenamesQueue();
 				progressHelper.start(-1/*XXX enum const, or a dedicated method startUnspecified(). How about startAtLeast(int)?*/);
@@ -309,10 +335,12 @@
 					HgDataFile fileNode = curRename.first();
 					if (followAncestry) {
 						TreeBuildInspector treeBuilder = new TreeBuildInspector(followAncestry);
+						@SuppressWarnings("unused")
 						List<HistoryNode> fileAncestry = treeBuilder.go(fileNode, curRename.second());
 						int[] commitRevisions = narrowChangesetRange(treeBuilder.getCommitRevisions(), startRev, lastCset);
 						if (iterateDirection == IterateDirection.FromOldToNew) {
 							repo.getChangelog().range(filterInsp, commitRevisions);
+							csetTransform.checkFailure();
 						} else {
 							assert iterateDirection == IterateDirection.FromNewToOld;
 							// visit one by one in the opposite direction
@@ -325,8 +353,24 @@
 						// report complete file history (XXX may narrow range with [startRev, endRev], but need to go from file rev to link rev)
 						int fileStartRev = 0; //fileNode.getChangesetRevisionIndex(0) >= startRev
 						int fileEndRev = fileNode.getLastRevision();
-						fileNode.history(fileStartRev, fileEndRev, filterInsp);
-						csetTransform.checkFailure();
+						if (iterateDirection == IterateDirection.FromOldToNew) {
+							fileNode.history(fileStartRev, fileEndRev, filterInsp);
+							csetTransform.checkFailure();
+						} else {
+							assert iterateDirection == IterateDirection.FromNewToOld;
+							BatchRangeHelper brh = new BatchRangeHelper(fileStartRev, fileEndRev, BATCH_SIZE, true);
+							BatchChangesetInspector batchInspector = new BatchChangesetInspector(Math.min(fileEndRev-fileStartRev+1, BATCH_SIZE));
+							filterInsp.delegateTo(batchInspector);
+							while (brh.hasNext()) {
+								brh.next();
+								fileNode.history(brh.start(), brh.end(), filterInsp);
+								for (BatchChangesetInspector.BatchRecord br : batchInspector.iterate(true /*iterateDirection == IterateDirection.FromNewToOld*/)) {
+									csetTransform.next(br.csetIndex, br.csetRevision, br.cset);
+									csetTransform.checkFailure();
+								}
+								batchInspector.reset();
+							}
+						}
 					}
 					if (followRenames && withCopyHandler != null && nameIndex + 1 < fileRenamesSize) {
 						Pair<HgDataFile, Nodeid> nextRename = fileRenames.get(nameIndex+1);
@@ -354,6 +398,46 @@
 		}
 	}
 	
+	private static class BatchChangesetInspector extends AdapterPlug implements HgChangelog.Inspector {
+		private static class BatchRecord {
+			public final int csetIndex;
+			public final Nodeid csetRevision;
+			public final RawChangeset cset;
+			
+			public BatchRecord(int index, Nodeid nodeid, RawChangeset changeset) {
+				csetIndex = index;
+				csetRevision = nodeid;
+				cset = changeset;
+			}
+		}
+		private final ArrayList<BatchRecord> batch;
+
+		public BatchChangesetInspector(int batchSizeHint) {
+			batch = new ArrayList<BatchRecord>(batchSizeHint);
+		}
+
+		public BatchChangesetInspector reset() {
+			batch.clear();
+			return this;
+		}
+		
+		public void next(int revisionIndex, Nodeid nodeid, RawChangeset cset) {
+			batch.add(new BatchRecord(revisionIndex, nodeid, cset.clone()));
+		}
+		
+		public Iterable<BatchRecord> iterate(final boolean reverse) {
+			return new Iterable<BatchRecord>() {
+				
+				public Iterator<BatchRecord> iterator() {
+					return reverse ? new ReverseIterator<BatchRecord>(batch) : batch.iterator();
+				}
+			};
+		}
+		
+		// alternative would be dispatch(HgChangelog.Inspector) and dispatchReverse()
+		// methods, but progress and cancellation might get messy then
+	}
+	
 //	public static void main(String[] args) {
 //		int[] r = new int[] {17, 19, 21, 23, 25, 29};
 //		System.out.println(Arrays.toString(narrowChangesetRange(r, 0, 45)));
@@ -454,6 +538,21 @@
 		progressHelper.done();
 	}
 	
+	/**
+	 * DO NOT USE THIS METHOD, DEBUG PURPOSES ONLY!!!
+	 */
+	@Experimental(reason="Work in progress")
+	public HgLogCommand debugSwitch1() {
+		// FIXME can't expose iteration direction unless general iteration (changelog, not a file) supports it, too.
+		// however, need to test the code already there, hence this debug switch
+		if (iterateDirection == IterateDirection.FromOldToNew) {
+			iterateDirection = IterateDirection.FromNewToOld;
+		} else {
+			iterateDirection = IterateDirection.FromOldToNew;
+		}
+		return this;
+	}
+	
 	private IterateDirection iterateDirection = IterateDirection.FromOldToNew;
 
 	private static class ReverseIterator<E> implements Iterator<E> {
@@ -820,16 +919,29 @@
 
 	//
 	
-	private class FilteringInspector implements HgChangelog.Inspector, Lifecycle {
+	private class FilteringInspector extends AdapterPlug implements HgChangelog.Inspector, Adaptable {
 	
-		private Callback lifecycle;
 		private int firstCset = BAD_REVISION, lastCset = BAD_REVISION;
+		private HgChangelog.Inspector delegate;
+		// we use lifecycle to stop when limit is reached.
+		// delegate, however, may use lifecycle, too, so give it a chance
+		private LifecycleProxy lifecycleProxy;
 		
 		// limit to changesets in this range only
 		public void changesets(int start, int end) {
 			firstCset = start;
 			lastCset = end;
 		}
+		
+		public void delegateTo(HgChangelog.Inspector inspector) {
+			delegate = inspector;
+			// let delegate control life cycle, too
+			if (lifecycleProxy == null) {
+				super.attachAdapter(Lifecycle.class, lifecycleProxy = new LifecycleProxy(inspector));
+			} else {
+				lifecycleProxy.init(inspector);
+			}
+		}
 
 		public void next(int revisionNumber, Nodeid nodeid, RawChangeset cset) {
 			if (limit > 0 && count >= limit) {
@@ -862,22 +974,14 @@
 			if (date != null) {
 				// TODO post-1.0 implement date support for log
 			}
-			csetTransform.next(revisionNumber, nodeid, cset);
-			if (++count >= limit) {
-				if (lifecycle != null) { // FIXME support Lifecycle delegation
-				lifecycle.stop();
-				}
+			delegate.next(revisionNumber, nodeid, cset);
+			count++;
+			if (limit > 0 && count >= limit) {
+				lifecycleProxy.stop();
 			}
 		}
+	}
 
-		public void start(int count, Callback callback, Object token) {
-			lifecycle = callback;
-		}
-
-		public void finish(Object token) {
-		}
-	}
-	
 	private HgParentChildMap<HgChangelog> getParentHelper(boolean create) throws HgInvalidControlFileException {
 		if (parentHelper == null && create) {
 			parentHelper = new HgParentChildMap<HgChangelog>(repo.getChangelog());
@@ -885,8 +989,7 @@
 		}
 		return parentHelper;
 	}
-
-
+	
 	public static class CollectHandler implements HgChangesetHandler {
 		private final List<HgChangeset> result = new LinkedList<HgChangeset>();
 
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/org/tmatesoft/hg/internal/AdapterPlug.java	Fri Dec 21 21:20:26 2012 +0100
@@ -0,0 +1,59 @@
+/*
+ * Copyright (c) 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.internal;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import org.tmatesoft.hg.util.Adaptable;
+
+/**
+ * Implementation of {@link Adaptable} that allows to add ("plug")
+ * adapters as needed
+ * 
+ * @author Artem Tikhomirov
+ * @author TMate Software Ltd.
+ */
+public class AdapterPlug implements Adaptable {
+	private final Map<Class<?>, Object> adapters = new HashMap<Class<?>, Object>();
+	private final List<Class<?>> adapterUses = new ArrayList<Class<?>>();
+	
+	public <T> void attachAdapter(Class<T> adapterClass, T instance) {
+		adapters.put(adapterClass, instance);
+	}
+
+	public <T> T getAdapter(Class<T> adapterClass) {
+		Object instance = adapters.get(adapterClass);
+		if (instance != null) {
+			adapterUses.add(adapterClass);
+			return adapterClass.cast(instance);
+		}
+		return null;
+	}
+	
+	public int getAdapterUse(Class<?> adapterClass) {
+		int uses = 0;
+		for (Class<?> c : adapterUses) {
+			if (c == adapterClass) {
+				uses++;
+			}
+		}
+		return uses;
+	}
+}
\ No newline at end of file
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/org/tmatesoft/hg/internal/BatchRangeHelper.java	Fri Dec 21 21:20:26 2012 +0100
@@ -0,0 +1,138 @@
+/*
+ * Copyright (c) 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.internal;
+
+import java.util.Arrays;
+import java.util.NoSuchElementException;
+
+/**
+ * Helper to break given range [start..end] (inclusive bounds) to series of ranges,
+ * all but last are of batchSize length, and the last one is at most of batchSize+batchSizeTolerance length.
+ * 
+ * Range is [{@link #start()rangeStart}..{@link #end() rangeEnd}], where rangeStart is less or equal to rangeEnd.
+ * 
+ * When reverse range iteration is requested, original range is iterated from end to start, but the subranges 
+ * boundaries are in natural order. i.e. for 0..100, 10 first subrange would be [91..100], not [100..91]. This 
+ * helps clients of this class to get [start()..end()] in natural order regardless of iteration direction.
+ * 
+ * Note, this class (and its treatment of inclusive boundaries) is designed solely for use with methods that navigate
+ * revlogs and take (start,end) pair of inclusive range.
+ *  
+ * @author Artem Tikhomirov
+ * @author TMate Software Ltd.
+ */
+public class BatchRangeHelper {
+
+	private final int rangeCount;
+	private final int rangeDelta;
+	private final int nextValueDelta;
+	private final int firstBoundary, lastBoundary;
+	private int rangeIndex, rangeValue, rangeStart, rangeEnd;
+
+	public BatchRangeHelper(int start, int end, int batchSize, boolean reverse) {
+		this(start, end, batchSize, batchSize/5, reverse); 
+	}
+
+	public BatchRangeHelper(int start, int end, int batchSize, int batchSizeTolerance, boolean reverse) {
+		assert end >= start;
+		assert start >= 0;
+		assert batchSize > 0;
+		assert batchSizeTolerance >= 0;
+		final int totalElements = end-start+1;
+		int batchRangeCount = totalElements / batchSize;
+		// batchRangeCount == 0, totalElements > 0 => need at least 1 range
+		if (batchRangeCount == 0 || batchRangeCount*batchSize+batchSizeTolerance < totalElements) {
+			batchRangeCount++;
+		}
+		rangeCount = batchRangeCount;
+		rangeDelta = batchSize-1; // ranges are inclusive, and always grow naturally.
+		nextValueDelta = reverse ? -batchSize : batchSize;
+		firstBoundary = reverse ? end-rangeDelta : start;
+		lastBoundary = reverse ? start : end;
+		reset();
+	}
+
+	public boolean hasNext() {
+		return rangeIndex < rangeCount;
+	}
+	
+	public void next() {
+		if (!hasNext()) {
+			throw new NoSuchElementException();
+		}
+		rangeStart = rangeValue;
+		rangeEnd = rangeValue + rangeDelta;
+		rangeValue += nextValueDelta;
+		if (++rangeIndex >= rangeCount) {
+			if (nextValueDelta < 0) {
+				// reverse iteration, lastBoundary represents start
+				rangeStart = lastBoundary;
+			} else {
+				// lastBoundary represents end
+				rangeEnd = lastBoundary;
+			}
+		}
+	}
+	
+	public int start() {
+		return rangeStart;
+	}
+	
+	public int end() {
+		return rangeEnd;
+	}
+	
+	public BatchRangeHelper reset() {
+		rangeValue = firstBoundary;
+		rangeIndex = 0;
+		return this;
+	}
+
+	public int[] toArray() {
+		int[] rv = new int[rangeCount*2];
+		reset();
+		int i = 0;
+		while (hasNext()) {
+			next();
+			rv[i++] = start();
+			rv[i++] = end();
+		}
+		reset();
+		return rv;
+	}
+
+	public static void main(String[] args) {
+		System.out.println("With remainder within tolerance");
+		System.out.println(Arrays.toString(new BatchRangeHelper(0, 102, 10, 4, false).toArray()));
+		System.out.println(Arrays.toString(new BatchRangeHelper(0, 102, 10, 4, true).toArray()));
+		System.out.println("With remainder out of tolerance");
+		System.out.println(Arrays.toString(new BatchRangeHelper(0, 102, 10, 2, false).toArray()));
+		System.out.println(Arrays.toString(new BatchRangeHelper(0, 102, 10, 2, true).toArray()));
+		System.out.println("Range smaller than batch");
+		System.out.println(Arrays.toString(new BatchRangeHelper(1, 9, 10, false).toArray()));
+		System.out.println(Arrays.toString(new BatchRangeHelper(1, 9, 10, true).toArray()));
+		System.out.println("Range smaller than batch and smaller than tolerance");
+		System.out.println(Arrays.toString(new BatchRangeHelper(1, 9, 10, 20, false).toArray()));
+		System.out.println(Arrays.toString(new BatchRangeHelper(1, 9, 10, 20, true).toArray()));
+		System.out.println("Zero tolerance");
+		System.out.println(Arrays.toString(new BatchRangeHelper(0, 100, 10, 0, false).toArray()));
+		System.out.println(Arrays.toString(new BatchRangeHelper(0, 100, 10, 0, true).toArray()));
+		System.out.println("Right to boundary");
+		System.out.println(Arrays.toString(new BatchRangeHelper(1, 100, 10, false).toArray()));
+		System.out.println(Arrays.toString(new BatchRangeHelper(1, 100, 10, true).toArray()));
+	}
+}
--- a/src/org/tmatesoft/hg/internal/IterateControlMediator.java	Thu Dec 20 20:21:59 2012 +0100
+++ b/src/org/tmatesoft/hg/internal/IterateControlMediator.java	Fri Dec 21 21:20:26 2012 +0100
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2011 TMate Software Ltd
+ * 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
@@ -28,7 +28,8 @@
 public class IterateControlMediator {
 
 	private final CancelSupport src;
-	private Callback receiver;
+	private final Callback receiver;
+	private CancelledException cancellation;
 
 	public IterateControlMediator(CancelSupport source, Lifecycle.Callback target) {
 		assert target != null;
@@ -40,11 +41,15 @@
 		if (src == null) {
 			return false;
 		}
+		if (cancellation != null) {
+			return true;
+		}
 		try {
 			src.checkCancelled();
 			return false;
 		} catch (CancelledException ex) {
 			receiver.stop();
+			cancellation = ex;
 			return true;
 		}
 	}
@@ -52,4 +57,8 @@
 	public void stop() {
 		receiver.stop();
 	}
+	
+	public CancelledException getCancelException() {
+		return cancellation;
+	}
 }
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/org/tmatesoft/hg/internal/LifecycleBridge.java	Fri Dec 21 21:20:26 2012 +0100
@@ -0,0 +1,92 @@
+/*
+ * Copyright (c) 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.internal;
+
+import org.tmatesoft.hg.util.CancelSupport;
+import org.tmatesoft.hg.util.CancelledException;
+import org.tmatesoft.hg.util.ProgressSupport;
+
+/**
+ * Bridge low-level life-cycle ({@link Lifecycle}) API with high-level one ({@link ProgressSupport} and {@link CancelSupport}).
+ * 
+ * @author Artem Tikhomirov
+ * @author TMate Software Ltd.
+ */
+public class LifecycleBridge implements Lifecycle {
+	private final ProgressSupport progressHelper;
+	private final CancelSupport cancelSupport;
+	// may be null unless #start() is invoked
+	private Callback receiver;
+	private CancelledException cancellation;
+
+	
+	public LifecycleBridge(ProgressSupport progress, CancelSupport cancel) {
+		progressHelper = progress;
+		cancelSupport = cancel;
+	}
+
+	public void start(int count, Callback callback, Object token) {
+		receiver = callback;
+		if (progressHelper != null) {
+			progressHelper.start(count);
+		}
+	}
+
+	public void finish(Object token) {
+		if (progressHelper != null) {
+			progressHelper.done();
+		}
+		receiver = null;
+	}
+
+	// XXX SHALL work without start/finish sequence because
+	// HgLogCommand invokes ChangesetTransformer#next directly (i.e. not from
+	// inside a library's #range() or similar) to process changesets in unnatural order.
+	public void nextStep() {
+		if (progressHelper != null) {
+			progressHelper.worked(1);
+		}
+		if (cancelSupport == null) {
+			return;
+		}
+		try { 
+			cancelSupport.checkCancelled();
+		} catch (CancelledException ex) {
+			if (receiver != null) {
+				receiver.stop();
+			}
+			cancellation = ex;
+		}
+	}
+
+	public void stop() {
+		if (receiver != null) {
+			receiver.stop();
+		}
+	}
+	
+	/**
+	 * @return <code>true</code> iff {@link CancelledException} was thrown and catched. Forced stop doesn't count
+	 */
+	public boolean isCancelled() {
+		return cancellation != null;
+	}
+	
+	public CancelledException getCancelOrigin() {
+		return cancellation;
+	}
+}
\ No newline at end of file
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/org/tmatesoft/hg/internal/LifecycleProxy.java	Fri Dec 21 21:20:26 2012 +0100
@@ -0,0 +1,64 @@
+/*
+ * Copyright (c) 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.internal;
+
+import org.tmatesoft.hg.util.Adaptable;
+
+/**
+ * Save callback and delegate to another lifecycle instance, if any
+ * 
+ * @author Artem Tikhomirov
+ * @author TMate Software Ltd.
+ */
+public class LifecycleProxy implements Lifecycle {
+	
+	private Callback lifecycleCallback;
+	private Lifecycle target;
+	
+	public LifecycleProxy() {
+	}
+
+	public LifecycleProxy(Object delegate) {
+		init(delegate);
+	}
+	
+
+	public void start(int count, Callback callback, Object token) {
+		lifecycleCallback = callback;
+		if (target != null) {
+			target.start(count, callback, token);
+		}
+	}
+
+	public void finish(Object token) {
+		if (target != null) {
+			target.finish(token);
+		}
+		lifecycleCallback = null;
+	}
+
+	public void init(Object delegate) {
+		target = Adaptable.Factory.getAdapter(delegate, Lifecycle.class, null);
+	}
+
+	public void stop() {
+		assert lifecycleCallback != null;
+		if (lifecycleCallback != null) {
+			lifecycleCallback.stop();
+		}
+	}
+}
\ No newline at end of file
--- a/src/org/tmatesoft/hg/internal/RevlogStream.java	Thu Dec 20 20:21:59 2012 +0100
+++ b/src/org/tmatesoft/hg/internal/RevlogStream.java	Fri Dec 21 21:20:26 2012 +0100
@@ -29,6 +29,7 @@
 import org.tmatesoft.hg.repo.HgInvalidRevisionException;
 import org.tmatesoft.hg.repo.HgInvalidStateException;
 import org.tmatesoft.hg.repo.HgRepository;
+import org.tmatesoft.hg.util.Adaptable;
 
 
 /**
@@ -382,6 +383,7 @@
 		private final boolean needData;
 		private DataAccess daIndex = null, daData = null;
 		private Lifecycle.BasicCallback cb = null;
+		private Lifecycle lifecycleListener = null;
 		private int lastRevisionRead = BAD_REVISION;
 		private DataAccess lastUserData;
 		// next are to track two major bottlenecks - patch application and actual time spent in inspector 
@@ -399,28 +401,35 @@
 			if (needData && !inline) {
 				daData = getDataStream();
 			}
-			if (inspector instanceof Lifecycle) {
+			lifecycleListener = Adaptable.Factory.getAdapter(inspector, Lifecycle.class, null);
+			if (lifecycleListener != null) {
 				cb = new Lifecycle.BasicCallback();
-				((Lifecycle) inspector).start(totalWork, cb, cb);
+				lifecycleListener.start(totalWork, cb, cb);
 			}
 //			applyTime = inspectorTime = 0; // TIMING
 		}
 
+		// invoked only once per instance
 		public void finish() {
 			if (lastUserData != null) {
 				lastUserData.done();
 				lastUserData = null;
 			}
-			if (inspector instanceof Lifecycle) {
-				((Lifecycle) inspector).finish(cb);
+			if (lifecycleListener != null) {
+				lifecycleListener.finish(cb);
+				lifecycleListener = null;
+				cb = null;
+				
 			}
 			daIndex.done();
 			if (daData != null) {
 				daData.done();
+				daData = null;
 			}
 //			System.out.printf("applyTime:%d ms, inspectorTime: %d ms\n", applyTime, inspectorTime); // TIMING
 		}
 
+		// may be invoked few times per instance life
 		public boolean range(int start, int end) throws IOException {
 			byte[] nodeidBuf = new byte[20];
 			int i;
--- a/src/org/tmatesoft/hg/repo/HgChangelog.java	Thu Dec 20 20:21:59 2012 +0100
+++ b/src/org/tmatesoft/hg/repo/HgChangelog.java	Fri Dec 21 21:20:26 2012 +0100
@@ -33,10 +33,11 @@
 import org.tmatesoft.hg.core.Nodeid;
 import org.tmatesoft.hg.internal.Callback;
 import org.tmatesoft.hg.internal.DataAccess;
-import org.tmatesoft.hg.internal.IterateControlMediator;
 import org.tmatesoft.hg.internal.Lifecycle;
+import org.tmatesoft.hg.internal.LifecycleBridge;
 import org.tmatesoft.hg.internal.Pool;
 import org.tmatesoft.hg.internal.RevlogStream;
+import org.tmatesoft.hg.util.Adaptable;
 import org.tmatesoft.hg.util.CancelSupport;
 import org.tmatesoft.hg.util.ProgressSupport;
 
@@ -357,19 +358,33 @@
 		}
 	}
 
-	private static class RawCsetParser implements RevlogStream.Inspector, Lifecycle {
+	private static class RawCsetParser implements RevlogStream.Inspector, Adaptable {
 		
 		private final Inspector inspector;
 		private final Pool<String> usersPool;
 		private final RawChangeset cset = new RawChangeset();
-		private final ProgressSupport progressHelper;
-		private IterateControlMediator iterateControl;
+		// non-null when inspector uses high-level lifecycle entities (progress and/or cancel supports)
+		private final LifecycleBridge lifecycleStub;
+		// non-null when inspector relies on low-level lifecycle and is responsible
+		// to proceed any possible high-level entities himself.
+		private final Lifecycle inspectorLifecycle;
 
 		public RawCsetParser(HgChangelog.Inspector delegate) {
 			assert delegate != null;
 			inspector = delegate;
 			usersPool = new Pool<String>();
-			progressHelper = ProgressSupport.Factory.get(delegate);
+			inspectorLifecycle = Adaptable.Factory.getAdapter(delegate, Lifecycle.class, null);
+			if (inspectorLifecycle == null) {
+				ProgressSupport ph = Adaptable.Factory.getAdapter(delegate, ProgressSupport.class, null);
+				CancelSupport cs = Adaptable.Factory.getAdapter(delegate, CancelSupport.class, null);
+				if (cs != null || ph != null) {
+					lifecycleStub = new LifecycleBridge(ph, cs);
+				} else {
+					lifecycleStub = null;
+				}
+			} else {
+				lifecycleStub = null;
+			}
 		}
 
 		public void next(int revisionNumber, int actualLen, int baseRevision, int linkRevision, int parent1Revision, int parent2Revision, byte[] nodeid, DataAccess da) {
@@ -378,26 +393,30 @@
 				cset.init(data, 0, data.length, usersPool);
 				// XXX there's no guarantee for Changeset.Callback that distinct instance comes each time, consider instance reuse
 				inspector.next(revisionNumber, Nodeid.fromBinary(nodeid, 0), cset);
-				progressHelper.worked(1);
+				if (lifecycleStub != null) {
+					lifecycleStub.nextStep();
+				}
 			} catch (HgInvalidDataFormatException ex) {
 				throw ex.setRevisionIndex(revisionNumber);  
 			} catch (IOException ex) {
 				// XXX need better exception, perhaps smth like HgChangelogException (extends HgInvalidControlFileException)
 				throw new HgInvalidControlFileException("Failed reading changelog", ex, null).setRevisionIndex(revisionNumber);  
 			}
-			if (iterateControl != null) {
-				iterateControl.checkCancelled();
+		}
+		
+		public <T> T getAdapter(Class<T> adapterClass) {
+			if (adapterClass == Lifecycle.class) {
+				if (inspectorLifecycle != null) {
+					return adapterClass.cast(inspectorLifecycle);
+				}
+				// reveal interest in lifecycle only when either progress or cancel support is there
+				// and inspector itself doesn't respond to lifecycle request
+				// lifecycleStub may still be null here (no progress and cancel), it's ok to cast(null) 
+				return adapterClass.cast(lifecycleStub);
+				
 			}
+			return Adaptable.Factory.getAdapter(inspector, adapterClass, null);
 		}
 
-		public void start(int count, Callback callback, Object token) {
-			CancelSupport cs = CancelSupport.Factory.get(inspector, null);
-			iterateControl = cs == null ? null : new IterateControlMediator(cs, callback);
-			progressHelper.start(count);
-		}
-
-		public void finish(Object token) {
-			progressHelper.done();
-		}
 	}
 }
--- a/test/org/tmatesoft/hg/test/ErrorCollectorExt.java	Thu Dec 20 20:21:59 2012 +0100
+++ b/test/org/tmatesoft/hg/test/ErrorCollectorExt.java	Fri Dec 21 21:20:26 2012 +0100
@@ -59,7 +59,7 @@
 	public void assertTrue(final boolean value) {
 		assertTrue(null, value);
 	}
-
+	
 	public void assertTrue(final String reason, final boolean value) {
 		checkSucceeds(new Callable<Object>() {
 			public Object call() throws Exception {
@@ -72,4 +72,12 @@
 	public <T> void assertEquals(T expected, T actual) {
 		checkThat(null, actual, CoreMatchers.equalTo(expected));
 	}
+
+	public <T> void assertEquals(String reason, T expected, T actual) {
+		checkThat(reason, actual, CoreMatchers.equalTo(expected));
+	}
+
+	public void fail(String reason) {
+		addError(new AssertionError(reason));
+	}
 }
\ No newline at end of file
--- a/test/org/tmatesoft/hg/test/TestAuxUtilities.java	Thu Dec 20 20:21:59 2012 +0100
+++ b/test/org/tmatesoft/hg/test/TestAuxUtilities.java	Fri Dec 21 21:20:26 2012 +0100
@@ -147,83 +147,81 @@
 			}
 		}
 	}
+	
+	static class CancelAtValue {
+		public int lastSeen;
+		public final int stopValue;
+		protected final CancelImpl cancelImpl = new CancelImpl();
+
+		protected CancelAtValue(int value) {
+			stopValue = value;
+		}
+		
+		protected void nextValue(int value) {
+			lastSeen = value;
+			if (value == stopValue) {
+				cancelImpl.stop();
+			}
+		}
+	}
 
 	@Test
 	public void testChangelogCancelSupport() throws Exception {
 		HgRepository repository = Configuration.get().find("branches-1"); // any repo with more revisions
-		class InspectorImplementsCancel implements HgChangelog.Inspector, CancelSupport {
-			public final int when2stop;
-			public int lastVisitet = 0;
-			private final CancelImpl cancelImpl = new CancelImpl(); 
+		class InspectorImplementsCancel extends CancelAtValue implements HgChangelog.Inspector, CancelSupport {
 
 			public InspectorImplementsCancel(int limit) {
-				when2stop = limit;
+				super(limit);
 			}
 			
 			public void next(int revisionNumber, Nodeid nodeid, RawChangeset cset) {
-				lastVisitet = revisionNumber;
-				if (revisionNumber == when2stop) {
-					cancelImpl.stop();
-				}
+				nextValue(revisionNumber);
 			}
 
 			public void checkCancelled() throws CancelledException {
 				cancelImpl.checkCancelled();
 			}
 		};
-		class InspectorImplementsAdaptable implements HgChangelog.Inspector, Adaptable {
-			public final int when2stop;
-			public int lastVisitet = 0;
-			private final CancelImpl cancelImpl = new CancelImpl();
-			
+		class InspectorImplementsAdaptable extends CancelAtValue implements HgChangelog.Inspector, Adaptable {
 			public InspectorImplementsAdaptable(int limit) {
-				when2stop = limit;
+				super(limit);
 			}
 			
 			public void next(int revisionNumber, Nodeid nodeid, RawChangeset cset) {
-				lastVisitet = revisionNumber;
-				if (revisionNumber == when2stop) {
-					cancelImpl.stop();
-				}
+				nextValue(revisionNumber);
 			}
+
 			public <T> T getAdapter(Class<T> adapterClass) {
 				if (CancelSupport.class == adapterClass) {
 					return adapterClass.cast(cancelImpl);
 				}
 				return null;
 			}
-			
 		}
 		//
 		InspectorImplementsCancel insp1;
 		repository.getChangelog().all(insp1= new InspectorImplementsCancel(2));
-		Assert.assertEquals(insp1.when2stop, insp1.lastVisitet);
+		Assert.assertEquals(insp1.stopValue, insp1.lastSeen);
 		repository.getChangelog().all(insp1 = new InspectorImplementsCancel(12));
-		Assert.assertEquals(insp1.when2stop, insp1.lastVisitet);
+		Assert.assertEquals(insp1.stopValue, insp1.lastSeen);
 		//
 		InspectorImplementsAdaptable insp2;
 		repository.getChangelog().all(insp2= new InspectorImplementsAdaptable(3));
-		Assert.assertEquals(insp2.when2stop, insp2.lastVisitet);
+		Assert.assertEquals(insp2.stopValue, insp2.lastSeen);
 		repository.getChangelog().all(insp2 = new InspectorImplementsAdaptable(10));
-		Assert.assertEquals(insp2.when2stop, insp2.lastVisitet);
+		Assert.assertEquals(insp2.stopValue, insp2.lastSeen);
 	}
 	
 	@Test
 	public void testManifestCancelSupport() throws Exception {
 		HgRepository repository = Configuration.get().find("branches-1"); // any repo with as many revisions as possible
-		class InspectorImplementsAdaptable implements HgManifest.Inspector, Adaptable {
-			public final int when2stop;
-			public int lastVisitet = 0;
-			private final CancelImpl cancelImpl = new CancelImpl(); 
-
+		class InspectorImplementsAdaptable extends CancelAtValue implements HgManifest.Inspector, Adaptable {
 			public InspectorImplementsAdaptable(int limit) {
-				when2stop = limit;
+				super(limit);
 			}
 
 			public boolean begin(int mainfestRevision, Nodeid nid, int changelogRevision) {
-				if (++lastVisitet == when2stop) {
-					cancelImpl.stop();
-				}
+				nextValue(lastSeen+1);
 				return true;
 			}
 
@@ -244,9 +242,9 @@
 		}
 		InspectorImplementsAdaptable insp1;
 		repository.getManifest().walk(0, TIP, insp1= new InspectorImplementsAdaptable(3));
-		Assert.assertEquals(insp1.when2stop, insp1.lastVisitet);
+		Assert.assertEquals(insp1.stopValue, insp1.lastSeen);
 		repository.getManifest().walk(0, TIP, insp1 = new InspectorImplementsAdaptable(10));
-		Assert.assertEquals(insp1.when2stop, insp1.lastVisitet);
+		Assert.assertEquals(insp1.stopValue, insp1.lastSeen);
 	}
 	
 	@Test
--- a/test/org/tmatesoft/hg/test/TestHistory.java	Thu Dec 20 20:21:59 2012 +0100
+++ b/test/org/tmatesoft/hg/test/TestHistory.java	Fri Dec 21 21:20:26 2012 +0100
@@ -42,10 +42,13 @@
 import org.tmatesoft.hg.core.HgLogCommand;
 import org.tmatesoft.hg.core.HgLogCommand.CollectHandler;
 import org.tmatesoft.hg.core.Nodeid;
+import org.tmatesoft.hg.internal.AdapterPlug;
 import org.tmatesoft.hg.repo.HgLookup;
 import org.tmatesoft.hg.repo.HgRepository;
 import org.tmatesoft.hg.test.LogOutputParser.Record;
 import org.tmatesoft.hg.util.Adaptable;
+import org.tmatesoft.hg.util.CancelSupport;
+import org.tmatesoft.hg.util.CancelledException;
 import org.tmatesoft.hg.util.Pair;
 import org.tmatesoft.hg.util.Path;
 
@@ -93,7 +96,10 @@
 		changelogParser.reset();
 		eh.run("hg", "log", "--debug");
 		List<HgChangeset> r = new HgLogCommand(repo).execute();
-		report("hg log - COMPLETE REPO HISTORY", r, true); 
+		report("hg log - COMPLETE REPO HISTORY", r, true);
+		
+		r = new HgLogCommand(repo).debugSwitch1().execute();
+		report("hg log - COMPLETE REPO HISTORY, FROM NEW TO OLD", r, false);
 	}
 	
 	@Test
@@ -193,8 +199,21 @@
 		String lastRevOfFname1 = "369c0882d477c11424a62eb4b791e86d1d4b6769";
 		errorCollector.assertEquals(lastRevOfFname1, h.lastChangesetReportedAtRename.get(0).getNodeid().toString());
 		report("HgChangesetHandler(renames: true, ancestry:false)", h.getChanges(), true);
-		
-		// TODO direction
+		//
+		// Direction
+		h = new CollectWithRenameHandler();
+		new HgLogCommand(repo).file(fname2, true, false).debugSwitch1().execute(h);
+		// Identical rename shall be reported, at the same moment 
+		errorCollector.assertEquals(1, h.renames.size());
+		rename = h.renames.get(0);
+		errorCollector.assertEquals(fname1, rename.first().getPath().toString());
+		errorCollector.assertEquals(fname2, rename.second().getPath().toString());
+		errorCollector.assertEquals(1, h.lastChangesetReportedAtRename.size());
+		// new to old, recently reported would be the very first revision fname2 pops up
+		String firstRevOfFname2 = "27e7a69373b74d42e75f3211e56510ff17d01370";
+		errorCollector.assertEquals(firstRevOfFname2, h.lastChangesetReportedAtRename.get(0).getNodeid().toString());
+		report("HgChangesetHandler(renames: true, ancestry:false)", h.getChanges(), false);
+		//
 		// TODO TreeChangeHandler
 	}
 		
@@ -228,8 +247,12 @@
 		new HgLogCommand(repo).file(fname2, false, true).execute(h);
 		errorCollector.assertEquals(0, h.renames.size());
 		report("HgChangesetHandler(renames: false, ancestry:true)", h.getChanges(), fname2Follow, true, errorCollector);
-
-		// TODO direction
+		//
+		// Direction
+		h = new CollectWithRenameHandler();
+		new HgLogCommand(repo).file(fname2, false, true).debugSwitch1().execute(h);
+		report("HgChangesetHandler(renames: false, ancestry:true)", h.getChanges(), fname2Follow, false/*!!!*/, errorCollector);
+		//
 		// TODO TreeChangeHandler
 	}
 
@@ -256,10 +279,80 @@
 		errorCollector.assertEquals(fname1BranchRevision, h.lastChangesetReportedAtRename.get(0).getNodeid().toString());
 		// finally, match output
 		report("HgChangesetHandler(renames: true, ancestry:true)", h.getChanges(), true);
-		// TODO direction
+		//
+		// Switch direction and compare, order shall match that from console
+		h = new CollectWithRenameHandler();
+		new HgLogCommand(repo).file(fname2, true, true).debugSwitch1().execute(h);
+		// Identical rename event shall be reported
+		errorCollector.assertEquals(1, h.renames.size());
+		rename = h.renames.get(0);
+		errorCollector.assertEquals(fname1, rename.first().getPath().toString());
+		errorCollector.assertEquals(fname2, rename.second().getPath().toString());
+		// new to old, recently reported would be the very first revision fname2 pops up
+		String firstRevOfFname2 = "27e7a69373b74d42e75f3211e56510ff17d01370";
+		errorCollector.assertEquals(firstRevOfFname2, h.lastChangesetReportedAtRename.get(0).getNodeid().toString());
+		report("HgChangesetHandler(renames: true, ancestry:true)", h.getChanges(), false /*do not reorder console results !!!*/);
+		//
 		// TreeChangeHandler in #testChangesetTreeFollowRenameAndAncestry
 	}
 
+	/**
+	 * @see TestAuxUtilities#testChangelogCancelSupport()
+	 */
+	@Test
+	public void testLogCommandCancelSupport() throws Exception {
+		repo  = Configuration.get().find("branches-1"); // any repo with more revisions
+		class BaseCancel extends TestAuxUtilities.CancelAtValue implements HgChangesetHandler {
+			BaseCancel(int limit) {
+				super(limit);
+			}
+			public void cset(HgChangeset changeset) throws HgCallbackTargetException {
+				nextValue(changeset.getRevisionIndex());
+			}
+		};
+		class ImplementsCancel extends BaseCancel implements CancelSupport {
+			ImplementsCancel(int limit) {
+				super(limit);
+			}
+			public void checkCancelled() throws CancelledException {
+				cancelImpl.checkCancelled();
+			}
+		};
+		class AdaptsToCancel extends BaseCancel implements Adaptable {
+			AdaptsToCancel(int limit) {
+				super(limit);
+			}
+			public <T> T getAdapter(Class<T> adapterClass) {
+				if (adapterClass == CancelSupport.class) {
+					return adapterClass.cast(cancelImpl);
+				}
+				return null;
+			}
+		}
+
+		BaseCancel insp = new ImplementsCancel(3);
+		try {
+			new HgLogCommand(repo).execute(insp);
+			errorCollector.fail("CancelSupport as implemented iface");
+		} catch (CancelledException ex) {
+			errorCollector.assertEquals("CancelSupport as implemented iface", insp.stopValue, insp.lastSeen);
+		}
+		insp = new AdaptsToCancel(5);
+		try {
+			new HgLogCommand(repo).execute(insp);
+			errorCollector.fail("Adaptable to CancelSupport");
+		} catch (CancelledException ex) { 
+			errorCollector.assertEquals("Adaptable to CancelSupport", insp.stopValue, insp.lastSeen);
+		}
+		insp = new BaseCancel(9);
+		try {
+			new HgLogCommand(repo).set(insp.cancelImpl).execute(insp);
+			errorCollector.fail("cmd#set(CancelSupport)");
+		} catch (CancelledException e) {
+			errorCollector.assertEquals("cmd#set(CancelSupport)", insp.stopValue, insp.lastSeen);
+		}
+	}
+
 	private void report(String what, List<HgChangeset> r, boolean reverseConsoleResult) {
 		final List<Record> consoleResult = changelogParser.getResult();
 		report(what, r, consoleResult, reverseConsoleResult, errorCollector);
@@ -278,6 +371,8 @@
 				break;
 			}
 			Record cr = consoleResultItr.next();
+			// flags, not separate checkThat() because when lists are large, and do not match,
+			// number of failures may slow down test process significantly
 			int x = cs.getRevisionIndex() == cr.changesetIndex ? 0x1 : 0;
 			x |= cs.getDate().toString().equals(cr.date) ? 0x2 : 0;
 			x |= cs.getNodeid().toString().equals(cr.changesetNodeid) ? 0x4 : 0;
@@ -398,34 +493,6 @@
 
 	////
 	
-	private static class AdapterPlug implements Adaptable {
-		private final Map<Class<?>, Object> adapters = new HashMap<Class<?>, Object>();
-		private final List<Class<?>> adapterUses = new ArrayList<Class<?>>();
-		
-		public <T> void attachAdapter(Class<T> adapterClass, T instance) {
-			adapters.put(adapterClass, instance);
-		}
-
-		public <T> T getAdapter(Class<T> adapterClass) {
-			Object instance = adapters.get(adapterClass);
-			if (instance != null) {
-				adapterUses.add(adapterClass);
-				return adapterClass.cast(instance);
-			}
-			return null;
-		}
-		
-		public int getAdapterUse(Class<?> adapterClass) {
-			int uses = 0;
-			for (Class<?> c : adapterUses) {
-				if (c == adapterClass) {
-					uses++;
-				}
-			}
-			return uses;
-		}
-	}
-	
 	private final class TreeCollectHandler extends AdapterPlug implements HgChangesetTreeHandler {
 		private final LinkedList<HgChangeset> cmdResult = new LinkedList<HgChangeset>();
 		private final boolean reverseResult;