changeset 705:b4242b7e7dfe

Merge command: implement conflict resolution alternatives
author Artem Tikhomirov <tikhomirov.artem@gmail.com>
date Thu, 15 Aug 2013 18:43:50 +0200 (2013-08-15)
parents 7743a9c10bfa
children cd5c87d96315
files cmdline/org/tmatesoft/hg/console/Merge.java src/org/tmatesoft/hg/core/HgAddRemoveCommand.java src/org/tmatesoft/hg/core/HgCheckoutCommand.java src/org/tmatesoft/hg/core/HgCommitCommand.java src/org/tmatesoft/hg/core/HgMergeCommand.java src/org/tmatesoft/hg/core/HgPullCommand.java src/org/tmatesoft/hg/core/HgRevertCommand.java src/org/tmatesoft/hg/internal/CommitFacility.java src/org/tmatesoft/hg/internal/DirstateBuilder.java src/org/tmatesoft/hg/internal/FileUtils.java src/org/tmatesoft/hg/internal/Internals.java src/org/tmatesoft/hg/internal/MergeStateBuilder.java src/org/tmatesoft/hg/internal/WorkingDirFileWriter.java src/org/tmatesoft/hg/repo/HgMergeState.java src/org/tmatesoft/hg/util/Path.java
diffstat 15 files changed, 478 insertions(+), 107 deletions(-) [+]
line wrap: on
line diff
--- a/cmdline/org/tmatesoft/hg/console/Merge.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/cmdline/org/tmatesoft/hg/console/Merge.java	Thu Aug 15 18:43:50 2013 +0200
@@ -56,8 +56,8 @@
 
 	static class Dump implements HgMergeCommand.Mediator {
 
-		public void same(HgFileRevision first, HgFileRevision second, Resolver resolver) throws HgCallbackTargetException {
-			System.out.printf("Unchanged %s:%s", first.getPath(), first.getRevision().shortNotation());
+		public void same(HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException {
+			System.out.printf("Unchanged %s:%s", rev.getPath(), rev.getRevision().shortNotation());
 		}
 
 		public void onlyA(HgFileRevision base, HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException {
--- a/src/org/tmatesoft/hg/core/HgAddRemoveCommand.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/core/HgAddRemoveCommand.java	Thu Aug 15 18:43:50 2013 +0200
@@ -18,7 +18,6 @@
 
 import java.util.LinkedHashSet;
 
-import org.tmatesoft.hg.internal.COWTransaction;
 import org.tmatesoft.hg.internal.DirstateBuilder;
 import org.tmatesoft.hg.internal.DirstateReader;
 import org.tmatesoft.hg.internal.Internals;
@@ -123,7 +122,7 @@
 				progress.worked(1);
 				cancellation.checkCancelled();
 			}
-			Transaction.Factory trFactory = new COWTransaction.Factory();
+			Transaction.Factory trFactory = implRepo.getTransactionFactory();
 			Transaction tr = trFactory.create(repo);
 			try {
 				dirstateBuilder.serialize(tr);
--- a/src/org/tmatesoft/hg/core/HgCheckoutCommand.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/core/HgCheckoutCommand.java	Thu Aug 15 18:43:50 2013 +0200
@@ -231,8 +231,8 @@
 				lastFileMode = workingDirWriter.fmode();
 				lastFileModificationTime = workingDirWriter.mtime();
 				return true;
-			} catch (IOException ex) {
-				failure = new HgIOException("Failed to write down file revision", ex, workingDirWriter.getDestinationFile());
+			} catch (HgIOException ex) {
+				failure = ex;
 			} catch (HgRuntimeException ex) {
 				failure = new HgLibraryFailureException(ex);
 			}
--- a/src/org/tmatesoft/hg/core/HgCommitCommand.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/core/HgCommitCommand.java	Thu Aug 15 18:43:50 2013 +0200
@@ -20,7 +20,6 @@
 
 import java.io.IOException;
 
-import org.tmatesoft.hg.internal.COWTransaction;
 import org.tmatesoft.hg.internal.CommitFacility;
 import org.tmatesoft.hg.internal.CompleteRepoLock;
 import org.tmatesoft.hg.internal.FileContentSupplier;
@@ -112,7 +111,8 @@
 				newRevision = Nodeid.NULL;
 				return new Outcome(Kind.Failure, "nothing to add");
 			}
-			CommitFacility cf = new CommitFacility(Internals.getInstance(repo), parentRevs[0], parentRevs[1]);
+			final Internals implRepo = Internals.getInstance(repo);
+			CommitFacility cf = new CommitFacility(implRepo, parentRevs[0], parentRevs[1]);
 			for (Path m : status.getModified()) {
 				HgDataFile df = repo.getFileNode(m);
 				cf.add(df, new WorkingCopyContent(df));
@@ -131,7 +131,7 @@
 			}
 			cf.branch(detectBranch());
 			cf.user(detectUser());
-			Transaction.Factory trFactory = new COWTransaction.Factory();
+			Transaction.Factory trFactory = implRepo.getTransactionFactory();
 			Transaction tr = trFactory.create(repo);
 			try {
 				newRevision = cf.commit(message, tr);
--- a/src/org/tmatesoft/hg/core/HgMergeCommand.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/core/HgMergeCommand.java	Thu Aug 15 18:43:50 2013 +0200
@@ -18,13 +18,23 @@
 
 import static org.tmatesoft.hg.repo.HgRepository.BAD_REVISION;
 
+import java.io.File;
+import java.io.FileInputStream;
+import java.io.IOException;
 import java.io.InputStream;
 
 import org.tmatesoft.hg.internal.Callback;
 import org.tmatesoft.hg.internal.CsetParamKeeper;
+import org.tmatesoft.hg.internal.DirstateBuilder;
+import org.tmatesoft.hg.internal.DirstateReader;
 import org.tmatesoft.hg.internal.Experimental;
+import org.tmatesoft.hg.internal.FileUtils;
+import org.tmatesoft.hg.internal.Internals;
 import org.tmatesoft.hg.internal.ManifestRevision;
+import org.tmatesoft.hg.internal.MergeStateBuilder;
 import org.tmatesoft.hg.internal.Pool;
+import org.tmatesoft.hg.internal.Transaction;
+import org.tmatesoft.hg.internal.WorkingDirFileWriter;
 import org.tmatesoft.hg.repo.HgChangelog;
 import org.tmatesoft.hg.repo.HgParentChildMap;
 import org.tmatesoft.hg.repo.HgRepository;
@@ -62,7 +72,7 @@
 		return this;
 	}
 
-	public void execute(Mediator mediator) throws HgCallbackTargetException, HgRepositoryLockException, HgLibraryFailureException, CancelledException {
+	public void execute(Mediator mediator) throws HgCallbackTargetException, HgRepositoryLockException, HgIOException, HgLibraryFailureException, CancelledException {
 		if (firstCset == BAD_REVISION || secondCset == BAD_REVISION || ancestorCset == BAD_REVISION) {
 			throw new IllegalArgumentException("Merge heads and their ancestors are not initialized");
 		}
@@ -71,65 +81,91 @@
 		try {
 			Pool<Nodeid> cacheRevs = new Pool<Nodeid>();
 			Pool<Path> cacheFiles = new Pool<Path>();
+
+			Internals implRepo = Internals.getInstance(repo);
+			final DirstateBuilder dirstateBuilder = new DirstateBuilder(implRepo);
+			dirstateBuilder.fillFrom(new DirstateReader(implRepo, new Path.SimpleSource(repo.getSessionContext().getPathFactory(), cacheFiles)));
+			final HgChangelog clog = repo.getChangelog();
+			dirstateBuilder.parents(clog.getRevision(firstCset), clog.getRevision(secondCset));
+			//
+			MergeStateBuilder mergeStateBuilder = new MergeStateBuilder(implRepo);
+
 			ManifestRevision m1, m2, ma;
 			m1 = new ManifestRevision(cacheRevs, cacheFiles).init(repo, firstCset);
 			m2 = new ManifestRevision(cacheRevs, cacheFiles).init(repo, secondCset);
 			ma = new ManifestRevision(cacheRevs, cacheFiles).init(repo, ancestorCset);
-			ResolverImpl resolver = new ResolverImpl();
-			for (Path f : m1.files()) {
-				Nodeid fileRevBase, fileRevA, fileRevB;
-				if (m2.contains(f)) {
-					fileRevA = m1.nodeid(f);
-					fileRevB = m2.nodeid(f);
-					fileRevBase = ma.contains(f) ? ma.nodeid(f) : null;
-					if (fileRevA.equals(fileRevB)) {
-						HgFileRevision fr = new HgFileRevision(repo, fileRevA, m1.flags(f), f);
-						mediator.same(fr, fr, resolver);
-					} else if (fileRevBase == fileRevA) {
-						assert fileRevBase != null;
-						HgFileRevision frBase = new HgFileRevision(repo, fileRevBase, ma.flags(f), f);
-						HgFileRevision frSecond= new HgFileRevision(repo, fileRevB, m2.flags(f), f);
-						mediator.fastForwardB(frBase, frSecond, resolver);
-					} else if (fileRevBase == fileRevB) {
-						assert fileRevBase != null;
-						HgFileRevision frBase = new HgFileRevision(repo, fileRevBase, ma.flags(f), f);
-						HgFileRevision frFirst = new HgFileRevision(repo, fileRevA, m1.flags(f), f);
-						mediator.fastForwardA(frBase, frFirst, resolver);
+			Transaction transaction = implRepo.getTransactionFactory().create(repo);
+			ResolverImpl resolver = new ResolverImpl(implRepo, dirstateBuilder, mergeStateBuilder);
+			try {
+				for (Path f : m1.files()) {
+					Nodeid fileRevBase, fileRevA, fileRevB;
+					if (m2.contains(f)) {
+						fileRevA = m1.nodeid(f);
+						fileRevB = m2.nodeid(f);
+						fileRevBase = ma.contains(f) ? ma.nodeid(f) : null;
+						if (fileRevA.equals(fileRevB)) {
+							HgFileRevision fr = new HgFileRevision(repo, fileRevA, m1.flags(f), f);
+							resolver.presentState(f, fr, fr);
+							mediator.same(fr, resolver);
+						} else if (fileRevBase == fileRevA) {
+							assert fileRevBase != null;
+							HgFileRevision frBase = new HgFileRevision(repo, fileRevBase, ma.flags(f), f);
+							HgFileRevision frSecond= new HgFileRevision(repo, fileRevB, m2.flags(f), f);
+							resolver.presentState(f, frBase, frSecond);
+							mediator.fastForwardB(frBase, frSecond, resolver);
+						} else if (fileRevBase == fileRevB) {
+							assert fileRevBase != null;
+							HgFileRevision frBase = new HgFileRevision(repo, fileRevBase, ma.flags(f), f);
+							HgFileRevision frFirst = new HgFileRevision(repo, fileRevA, m1.flags(f), f);
+							resolver.presentState(f, frFirst, frBase);
+							mediator.fastForwardA(frBase, frFirst, resolver);
+						} else {
+							HgFileRevision frBase = fileRevBase == null ? null : new HgFileRevision(repo, fileRevBase, ma.flags(f), f);
+							HgFileRevision frFirst = new HgFileRevision(repo, fileRevA, m1.flags(f), f);
+							HgFileRevision frSecond= new HgFileRevision(repo, fileRevB, m2.flags(f), f);
+							resolver.presentState(f, frFirst, frSecond);
+							mediator.resolve(frBase, frFirst, frSecond, resolver);
+						}
 					} else {
-						HgFileRevision frBase = fileRevBase == null ? null : new HgFileRevision(repo, fileRevBase, ma.flags(f), f);
-						HgFileRevision frFirst = new HgFileRevision(repo, fileRevA, m1.flags(f), f);
-						HgFileRevision frSecond= new HgFileRevision(repo, fileRevB, m2.flags(f), f);
-						mediator.resolve(frBase, frFirst, frSecond, resolver);
+						// m2 doesn't contain the file, either new in m1, or deleted in m2
+						HgFileRevision frFirst = new HgFileRevision(repo, m1.nodeid(f), m1.flags(f), f);
+						resolver.presentState(f, frFirst, null);
+						if (ma.contains(f)) {
+							// deleted in m2
+							HgFileRevision frBase = new HgFileRevision(repo, ma.nodeid(f), ma.flags(f), f);
+							mediator.onlyA(frBase, frFirst, resolver);
+						} else {
+							// new in m1
+							mediator.newInA(frFirst, resolver);
+						}
 					}
-				} else {
-					// m2 doesn't contain the file, either new in m1, or deleted in m2
-					HgFileRevision frFirst = new HgFileRevision(repo, m1.nodeid(f), m1.flags(f), f);
+					resolver.apply();
+				} // for m1 files
+				for (Path f : m2.files()) {
+					if (m1.contains(f)) {
+						continue;
+					}
+					HgFileRevision frSecond= new HgFileRevision(repo, m2.nodeid(f), m2.flags(f), f);
+					// file in m2 is either new or deleted in m1
+					resolver.presentState(f, null, frSecond);
 					if (ma.contains(f)) {
-						// deleted in m2
+						// deleted in m1
 						HgFileRevision frBase = new HgFileRevision(repo, ma.nodeid(f), ma.flags(f), f);
-						mediator.onlyA(frBase, frFirst, resolver);
+						mediator.onlyB(frBase, frSecond, resolver);
 					} else {
-						// new in m1
-						mediator.newInA(frFirst, resolver);
+						// new in m2
+						mediator.newInB(frSecond, resolver);
 					}
+					resolver.apply();
 				}
-				resolver.apply();
-			} // for m1 files
-			for (Path f : m2.files()) {
-				if (m1.contains(f)) {
-					continue;
-				}
-				HgFileRevision frSecond= new HgFileRevision(repo, m2.nodeid(f), m2.flags(f), f);
-				// file in m2 is either new or deleted in m1
-				if (ma.contains(f)) {
-					// deleted in m1
-					HgFileRevision frBase = new HgFileRevision(repo, ma.nodeid(f), ma.flags(f), f);
-					mediator.onlyB(frBase, frSecond, resolver);
-				} else {
-					// new in m2
-					mediator.newInB(frSecond, resolver);
-				}
-				resolver.apply();
+				resolver.serializeChanged(transaction);
+				transaction.commit();
+			} catch (HgRuntimeException ex) {
+				transaction.rollback();
+				throw ex;
+			} catch (HgIOException ex) {
+				transaction.rollback();
+				throw ex;
 			}
 		} catch (HgRuntimeException ex) {
 			throw new HgLibraryFailureException(ex);
@@ -160,18 +196,43 @@
 	}
 
 	/**
-	 * This is the way client code takes part in the merge process
+	 * This is the way client code takes part in the merge process. 
+	 * It's advised to subclass {@link MediatorBase} unless special treatment for regular cases is desired
 	 */
 	@Experimental(reason="Provisional API. Work in progress")
 	@Callback
 	public interface Mediator {
-		public void same(HgFileRevision first, HgFileRevision second, Resolver resolver) throws HgCallbackTargetException;
+		/**
+		 * file revisions are identical in both heads
+		 */
+		public void same(HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException;
+		/**
+		 * file left in first/left/A trunk only, deleted in second/right/B trunk
+		 */
 		public void onlyA(HgFileRevision base, HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException;
+		/**
+		 * file left in second/right/B trunk only, deleted in first/left/A trunk
+		 */
 		public void onlyB(HgFileRevision base, HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException;
+		/**
+		 * file is missing in ancestor revision and second/right/B trunk, introduced in first/left/A trunk
+		 */
 		public void newInA(HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException;
+		/**
+		 * file is missing in ancestor revision and first/left/A trunk, introduced in second/right/B trunk
+		 */
 		public void newInB(HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException;
+		/**
+		 * file was changed in first/left/A trunk, unchanged in second/right/B trunk
+		 */
 		public void fastForwardA(HgFileRevision base, HgFileRevision first, Resolver resolver) throws HgCallbackTargetException;
+		/**
+		 * file was changed in second/right/B trunk, unchanged in first/left/A trunk 
+		 */
 		public void fastForwardB(HgFileRevision base, HgFileRevision second, Resolver resolver) throws HgCallbackTargetException;
+		/**
+		 * File changed (or added, if base is <code>null</code>) in both trunks 
+		 */
 		public void resolve(HgFileRevision base, HgFileRevision first, HgFileRevision second, Resolver resolver) throws HgCallbackTargetException;
 	}
 
@@ -182,24 +243,170 @@
 	@Experimental(reason="Provisional API. Work in progress")
 	public interface Resolver {
 		public void use(HgFileRevision rev);
-		public void use(InputStream content);
+		/**
+		 * Replace current revision with stream content.
+		 * Note, callers are not expected to {@link InputStream#close()} this stream. 
+		 * It will be {@link InputStream#close() closed} at <b>Hg4J</b>'s discretion
+		 * not necessarily during invocation of this method. IOW, the library may decide to 
+		 * use this stream not right away, at some point of time later, and streams supplied
+		 * shall respect this.
+		 * 
+		 * @param content New content to replace current revision, shall not be <code>null</code> 
+		 * @throws IOException propagated exceptions from content
+		 */
+		public void use(InputStream content) throws IOException;
+		public void forget(HgFileRevision rev);
 		public void unresolved(); // record the file for later processing by 'hg resolve'
 	}
 
+	/**
+	 * Base mediator implementation, with regular resolution
+	 */
+	@Experimental(reason="Provisional API. Work in progress")
+	public abstract class MediatorBase implements Mediator {
+		public void same(HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException {
+			resolver.use(rev);
+		}
+		public void onlyA(HgFileRevision base, HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException {
+			resolver.use(rev);
+		}
+		public void onlyB(HgFileRevision base, HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException {
+			resolver.use(rev);
+		}
+		public void newInA(HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException {
+			resolver.use(rev);
+		}
+		public void newInB(HgFileRevision rev, Resolver resolver) throws HgCallbackTargetException {
+			resolver.use(rev);
+		}
+		public void fastForwardA(HgFileRevision base, HgFileRevision first, Resolver resolver) throws HgCallbackTargetException {
+			resolver.use(first);
+		}
+		public void fastForwardB(HgFileRevision base, HgFileRevision second, Resolver resolver) throws HgCallbackTargetException {
+			resolver.use(second);
+		}
+	}
+
 	private static class ResolverImpl implements Resolver {
-		void apply() {
+		
+		private final Internals repo;
+		private final DirstateBuilder dirstateBuilder;
+		private final MergeStateBuilder mergeStateBuilder;
+		private boolean changedDirstate;
+		private HgFileRevision revA;
+		private HgFileRevision revB;
+		private Path file;
+		// resolutions:
+		private HgFileRevision resolveUse, resolveForget;
+		private File resolveContent;
+		private boolean resolveMarkUnresolved;
+		
+		public ResolverImpl(Internals implRepo, DirstateBuilder dirstateBuilder, MergeStateBuilder mergeStateBuilder) {
+			repo = implRepo;
+			this.dirstateBuilder = dirstateBuilder;
+			this.mergeStateBuilder = mergeStateBuilder;
+			changedDirstate = false;
+		}
+		
+		void serializeChanged(Transaction tr) throws HgIOException {
+			if (changedDirstate) {
+				dirstateBuilder.serialize(tr);
+			}
+			mergeStateBuilder.serialize(tr);
+		}
+
+		void presentState(Path p, HgFileRevision revA, HgFileRevision revB) {
+			assert revA != null || revB != null;
+			file = p;
+			this.revA = revA;
+			this.revB = revB;
+			resolveUse = resolveForget = null;
+			resolveContent = null;
+			resolveMarkUnresolved = false;
+		}
+
+		void apply() throws HgIOException, HgRuntimeException {
+			if (resolveMarkUnresolved) {
+				mergeStateBuilder.unresolved(file);
+			} else if (resolveForget != null) {
+				if (resolveForget == revA) {
+					changedDirstate = true;
+					dirstateBuilder.recordRemoved(file);
+				}
+			} else if (resolveUse != null) {
+				if (resolveUse != revA) {
+					changedDirstate = true;
+					final WorkingDirFileWriter fw = new WorkingDirFileWriter(repo);
+					fw.processFile(resolveUse);
+					if (resolveUse == revB) {
+						dirstateBuilder.recordMergedFromP2(file);
+					} else {
+						dirstateBuilder.recordMerged(file, fw.fmode(), fw.mtime(), fw.bytesWritten());
+					}
+				} // if resolution is to use revA, nothing to do
+			} else if (resolveContent != null) {
+				changedDirstate = true;
+				// FIXME write content to file using transaction?
+				InputStream is;
+				try {
+					is = new FileInputStream(resolveContent);
+				} catch (IOException ex) {
+					throw new HgIOException("Failed to read temporary content", ex, resolveContent);
+				}
+				final WorkingDirFileWriter fw = new WorkingDirFileWriter(repo);
+				fw.processFile(file, is, revA == null ? revB.getFileFlags() : revA.getFileFlags());
+				// XXX if presentState(null, fileOnlyInB), and use(InputStream) - i.e.
+				// resolution is to add file with supplied content - shall I put 'Merged', MergedFromP2 or 'Added' into dirstate?
+				if (revA == null && revB != null) {
+					dirstateBuilder.recordMergedFromP2(file);
+				} else {
+					dirstateBuilder.recordMerged(file, fw.fmode(), fw.mtime(), fw.bytesWritten());
+				}
+			} else {
+				assert false;
+			}
 		}
 
 		public void use(HgFileRevision rev) {
-			// TODO Auto-generated method stub
+			if (rev == null) {
+				throw new IllegalArgumentException();
+			}
+			assert resolveContent == null;
+			assert resolveForget == null;
+			resolveUse = rev;
 		}
 
-		public void use(InputStream content) {
-			// TODO Auto-generated method stub
+		public void use(InputStream content) throws IOException {
+			if (content == null) {
+				throw new IllegalArgumentException();
+			}
+			assert resolveUse == null;
+			assert resolveForget == null;
+			try {
+				// cache new contents just to fail fast if there are troubles with content
+				final FileUtils fileUtils = new FileUtils(repo.getLog(), this);
+				resolveContent = fileUtils.createTempFile();
+				fileUtils.write(content, resolveContent);
+			} finally {
+				content.close();
+			}
+			// do not care deleting file in case of failure to allow analyze of the issue
+		}
+
+		public void forget(HgFileRevision rev) {
+			if (rev == null) {
+				throw new IllegalArgumentException();
+			}
+			if (rev != revA || rev != revB) {
+				throw new IllegalArgumentException("Can't forget revision which doesn't represent actual state in either merged trunk");
+			}
+			assert resolveUse == null;
+			assert resolveContent == null;
+			resolveForget = rev;
 		}
 
 		public void unresolved() {
-			// TODO Auto-generated method stub
+			resolveMarkUnresolved = true;
 		}
 	}
 }
--- a/src/org/tmatesoft/hg/core/HgPullCommand.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/core/HgPullCommand.java	Thu Aug 15 18:43:50 2013 +0200
@@ -21,7 +21,6 @@
 import java.util.List;
 
 import org.tmatesoft.hg.internal.AddRevInspector;
-import org.tmatesoft.hg.internal.COWTransaction;
 import org.tmatesoft.hg.internal.Internals;
 import org.tmatesoft.hg.internal.PhasesHelper;
 import org.tmatesoft.hg.internal.RepositoryComparator;
@@ -78,7 +77,7 @@
 			// add revisions to changelog, manifest, files
 			final Internals implRepo = HgInternals.getImplementationRepo(repo);
 			final AddRevInspector insp;
-			Transaction.Factory trFactory = new COWTransaction.Factory();
+			Transaction.Factory trFactory = implRepo.getTransactionFactory();
 			Transaction tr = trFactory.create(repo);
 			try {
 				incoming.inspectAll(insp = new AddRevInspector(implRepo, tr));
--- a/src/org/tmatesoft/hg/core/HgRevertCommand.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/core/HgRevertCommand.java	Thu Aug 15 18:43:50 2013 +0200
@@ -21,7 +21,6 @@
 import java.util.LinkedHashSet;
 import java.util.Set;
 
-import org.tmatesoft.hg.internal.COWTransaction;
 import org.tmatesoft.hg.internal.CsetParamKeeper;
 import org.tmatesoft.hg.internal.DirstateBuilder;
 import org.tmatesoft.hg.internal.DirstateReader;
@@ -160,7 +159,7 @@
 				progress.worked(1);
 				cancellation.checkCancelled();
 			}
-			Transaction.Factory trFactory = new COWTransaction.Factory();
+			Transaction.Factory trFactory = implRepo.getTransactionFactory();
 			Transaction tr = trFactory.create(repo);
 			try {
 				// TODO same code in HgAddRemoveCommand and similar in HgCommitCommand
--- a/src/org/tmatesoft/hg/internal/CommitFacility.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/internal/CommitFacility.java	Thu Aug 15 18:43:50 2013 +0200
@@ -212,7 +212,7 @@
 		}
 		// bring dirstate up to commit state, TODO share this code with HgAddRemoveCommand
 		final DirstateBuilder dirstateBuilder = new DirstateBuilder(repo);
-		dirstateBuilder.fillFrom(new DirstateReader(repo, new Path.SimpleSource()));
+		dirstateBuilder.fillFrom(new DirstateReader(repo, repo.getSessionContext().getPathFactory()));
 		for (Path p : removals) {
 			dirstateBuilder.recordRemoved(p);
 		}
--- a/src/org/tmatesoft/hg/internal/DirstateBuilder.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/internal/DirstateBuilder.java	Thu Aug 15 18:43:50 2013 +0200
@@ -92,6 +92,22 @@
 		removed.put(fname, n);
 	}
 	
+	public void recordMerged(Path fname, int fmode, int mtime, int bytesWritten) {
+		forget(fname);
+		merged.put(fname, new HgDirstate.Record(fmode, bytesWritten,mtime, fname, null));
+	}
+	
+	/**
+	 * From DirState wiki:
+	 * <p>"size is ... when the dirstate is in a merge state: -2 will *always* return dirty, it is used to mark a file that was cleanly picked from p2"
+	 * and  
+	 * <p>"Additional meta status...'np2': merged from other parent (status == 'n', size == -2)"
+	 */
+	public void recordMergedFromP2(Path fname) {
+		forget(fname);
+		normal.put(fname, new HgDirstate.Record(0, -2, -1, fname, null));
+	}
+	
 	private HgDirstate.Record forget(Path fname) {
 		HgDirstate.Record r;
 		if ((r = normal.remove(fname)) != null) {
--- a/src/org/tmatesoft/hg/internal/FileUtils.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/internal/FileUtils.java	Thu Aug 15 18:43:50 2013 +0200
@@ -105,7 +105,7 @@
 		fos.flush();
 		fos.close();
 	}
-
+	
 	public void closeQuietly(Closeable stream) {
 		closeQuietly(stream, null);
 	}
@@ -126,4 +126,9 @@
 			}
 		}
 	}
+
+	// nothing special, just a single place with common prefix
+	public File createTempFile() throws IOException {
+		return File.createTempFile("hg4j-", null);
+	}
 }
--- a/src/org/tmatesoft/hg/internal/Internals.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/internal/Internals.java	Thu Aug 15 18:43:50 2013 +0200
@@ -513,6 +513,10 @@
 	public RevlogStream resolveStoreFile(Path path) {
 		return streamProvider.getStoreFile(path, false);
 	}
+	
+	public Transaction.Factory getTransactionFactory() {
+		return new COWTransaction.Factory();
+	}
 
 	// marker method
 	public static IllegalStateException notImplemented() {
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/org/tmatesoft/hg/internal/MergeStateBuilder.java	Thu Aug 15 18:43:50 2013 +0200
@@ -0,0 +1,48 @@
+/*
+ * 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.internal;
+
+import org.tmatesoft.hg.core.HgIOException;
+import org.tmatesoft.hg.repo.HgMergeState;
+import org.tmatesoft.hg.util.Path;
+
+/**
+ * Constructs merge/state file
+ * 
+ * @see HgMergeState
+ * @author Artem Tikhomirov
+ * @author TMate Software Ltd.
+ */
+public class MergeStateBuilder {
+	
+	private final Internals repo;
+
+	public MergeStateBuilder(Internals implRepo) {
+		repo = implRepo;
+	}
+	
+	public void resolved() {
+		throw Internals.notImplemented();
+	}
+
+	public void unresolved(Path file) {
+		throw Internals.notImplemented();
+	}
+
+	public void serialize(Transaction tr) throws HgIOException {
+	}
+}
--- a/src/org/tmatesoft/hg/internal/WorkingDirFileWriter.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/internal/WorkingDirFileWriter.java	Thu Aug 15 18:43:50 2013 +0200
@@ -21,9 +21,12 @@
 import java.io.File;
 import java.io.FileOutputStream;
 import java.io.IOException;
+import java.io.InputStream;
 import java.nio.ByteBuffer;
 import java.nio.channels.FileChannel;
 
+import org.tmatesoft.hg.core.HgFileRevision;
+import org.tmatesoft.hg.core.HgIOException;
 import org.tmatesoft.hg.repo.HgDataFile;
 import org.tmatesoft.hg.repo.HgManifest;
 import org.tmatesoft.hg.repo.HgRuntimeException;
@@ -65,45 +68,94 @@
 	 * Executable bit is set if specified and filesystem supports it. 
 	 * @throws HgRuntimeException 
 	 */
-	public void processFile(HgDataFile df, int fileRevIndex, HgManifest.Flags flags) throws IOException, HgRuntimeException {
-		try {
-			prepare(df.getPath());
-			if (flags != HgManifest.Flags.Link) {
-				destChannel = new FileOutputStream(dest).getChannel();
-			} else {
-				linkChannel = new ByteArrayChannel();
-			}
-			df.contentWithFilters(fileRevIndex, this);
-		} catch (CancelledException ex) {
-			hgRepo.getSessionContext().getLog().dump(getClass(), Severity.Error, ex, "Our impl doesn't throw cancellation");
-		} finally {
-			if (flags != HgManifest.Flags.Link) {
-				destChannel.close();
-				destChannel = null;
-				// leave dest in case anyone enquires with #getDestinationFile
+	public void processFile(final HgDataFile df, final int fileRevIndex, HgManifest.Flags flags) throws HgIOException, HgRuntimeException {
+		processFile(df.getPath(), new Fetch() {
+			public void readInto(ByteChannel ch) {
+				try {
+					df.contentWithFilters(fileRevIndex, ch);
+				} catch (CancelledException ex) {
+					handleUnexpectedCancel(ex);
+				}
 			}
-		}
-		if (linkChannel != null && symlinkCap) {
-			assert flags == HgManifest.Flags.Link;
-			fileFlagsHelper.createSymlink(dest.getParentFile(), dest.getName(), linkChannel.toArray());
-		} else if (flags == HgManifest.Flags.Exec && execCap) {
-			fileFlagsHelper.setExecutableBit(dest.getParentFile(), dest.getName());
-		}
-		// Although HgWCStatusCollector treats 644 (`hg manifest -v`) and 664 (my fs) the same, it's better
-		// to detect actual flags here
-		fmode = flags.fsMode(); // default to one from manifest
-		if (fileFlagsHelper != null) {
-			// if neither execBit nor link is supported by fs, it's unlikely file mode is supported, too.
+		}, flags);
+	}
+	
+	public void processFile(final HgFileRevision fr) throws HgIOException, HgRuntimeException {
+		processFile(fr.getPath(), new Fetch() {
+			
+			public void readInto(ByteChannel ch) throws IOException, HgRuntimeException {
+				try {
+					fr.putContentTo(ch);
+				} catch (CancelledException ex) {
+					handleUnexpectedCancel(ex);
+				}
+			}
+		}, fr.getFileFlags());
+	}
+	
+	/**
+	 * Closes supplied content stream 
+	 */
+	public void processFile(Path fname, final InputStream content, HgManifest.Flags flags) throws HgIOException, HgRuntimeException {
+		processFile(fname, new Fetch() {
+			
+			public void readInto(ByteChannel ch) throws IOException, HgRuntimeException {
+				try {
+					 ByteBuffer bb = ByteBuffer.wrap(new byte[8*1024]);
+					 int r;
+					 while ((r = content.read(bb.array())) != -1) {
+						 bb.position(0).limit(r);
+						 for (int wrote = 0; wrote < r; ) {
+							 r -= wrote; 
+							 wrote = ch.write(bb);
+							 assert bb.remaining() == r - wrote;
+						 }
+					 }
+				} catch (CancelledException ex) {
+					handleUnexpectedCancel(ex);
+				}
+			}
+		}, flags);
+	}
+	
+	private interface Fetch {
+		void readInto(ByteChannel ch) throws IOException, HgRuntimeException;
+	}
+
+	private void processFile(Path fname, Fetch fetch, HgManifest.Flags flags) throws HgIOException, HgRuntimeException {
+		try {
+			byte[] symlinkContent = null;
 			try {
-				fmode = fileFlagsHelper.getFileMode(dest, fmode);
-			} catch (IOException ex) {
-				// Warn, we've got default value and can live with it
-				hgRepo.getSessionContext().getLog().dump(getClass(), Warn, ex, "Failed get file access rights");
+				prepare(fname, flags);
+				fetch.readInto(this);
+			} finally {
+				symlinkContent = close(fname, flags);
 			}
+			if (flags == HgManifest.Flags.Link && symlinkCap) {
+				assert symlinkContent != null;
+				fileFlagsHelper.createSymlink(dest.getParentFile(), dest.getName(), symlinkContent);
+			} else if (flags == HgManifest.Flags.Exec && execCap) {
+				fileFlagsHelper.setExecutableBit(dest.getParentFile(), dest.getName());
+			}
+			// Although HgWCStatusCollector treats 644 (`hg manifest -v`) and 664 (my fs) the same, it's better
+			// to detect actual flags here
+			fmode = flags.fsMode(); // default to one from manifest
+			if (fileFlagsHelper != null) {
+				// if neither execBit nor link is supported by fs, it's unlikely file mode is supported, too.
+				try {
+					fmode = fileFlagsHelper.getFileMode(dest, fmode);
+				} catch (IOException ex) {
+					// Warn, we've got default value and can live with it
+					hgRepo.getSessionContext().getLog().dump(getClass(), Warn, ex, "Failed get file access rights");
+				}
+			}
+		} catch (IOException ex) {
+			String msg = String.format("Failed to write file %s to the working directory", fname);
+			throw new HgIOException(msg, ex, dest);
 		}
 	}
 
-	public void prepare(Path fname) throws IOException {
+	private void prepare(Path fname, HgManifest.Flags flags) throws IOException {
 		String fpath = fname.toString();
 		dest = new File(hgRepo.getRepo().getWorkingDir(), fpath);
 		if (fpath.indexOf('/') != -1) {
@@ -113,6 +165,25 @@
 		linkChannel = null;
 		totalBytesWritten = 0;
 		fmode = 0;
+		if (flags != HgManifest.Flags.Link) {
+			destChannel = new FileOutputStream(dest).getChannel();
+		} else {
+			linkChannel = new ByteArrayChannel();
+		}
+	}
+	
+	private byte[] close(Path fname, HgManifest.Flags flags) throws IOException {
+		if (flags != HgManifest.Flags.Link) {
+			destChannel.close();
+			destChannel = null;
+			// leave dest in case anyone enquires with #getDestinationFile
+		}
+		if (linkChannel != null) {
+			final byte[] rv = linkChannel.toArray();
+			linkChannel = null;
+			return rv;
+		}
+		return null;
 	}
 
 	public int write(ByteBuffer buffer) throws IOException, CancelledException {
@@ -144,4 +215,8 @@
 	public int mtime() {
 		return (int) (dest.lastModified() / 1000);
 	}
+
+	private void handleUnexpectedCancel(CancelledException ex) {
+		hgRepo.getSessionContext().getLog().dump(WorkingDirFileWriter.class, Severity.Error, ex, "Our impl doesn't throw cancellation");
+	}
 }
--- a/src/org/tmatesoft/hg/repo/HgMergeState.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/repo/HgMergeState.java	Thu Aug 15 18:43:50 2013 +0200
@@ -218,7 +218,7 @@
 
 	/**
 	 * List of conflicts as recorded in the merge state information file. 
-	 * Note, this information is valid unless {@link #isStale()} is <code>true</code>.
+	 * Note, this information is not valid unless {@link #isStale()} is <code>true</code>.
 	 * 
 	 * @return non-<code>null</code> list with both resolved and unresolved conflicts.
 	 */
--- a/src/org/tmatesoft/hg/util/Path.java	Wed Aug 14 20:07:26 2013 +0200
+++ b/src/org/tmatesoft/hg/util/Path.java	Thu Aug 15 18:43:50 2013 +0200
@@ -214,6 +214,7 @@
 	public static class SimpleSource implements Source {
 		private final PathRewrite normalizer;
 		private final Convertor<Path> convertor;
+		private final Path.Source delegate;
 
 		public SimpleSource() {
 			this(new PathRewrite.Empty(), null);
@@ -224,12 +225,30 @@
 		}
 
 		public SimpleSource(PathRewrite pathRewrite, Convertor<Path> pathConvertor) {
+			assert pathRewrite != null;
 			normalizer = pathRewrite;
 			convertor = pathConvertor;
+			delegate = null;
+		}
+
+		public SimpleSource(Path.Source actual, Convertor<Path> pathConvertor) {
+			assert actual != null;
+			normalizer = null;
+			delegate = actual;
+			convertor = pathConvertor;
 		}
 
 		public Path path(CharSequence p) {
-			Path rv = Path.create(normalizer.rewrite(p));
+			// in fact, it's nicer to have sequence of sources, and a bunch of small
+			// Source implementations each responsible for specific aspect, like Convertor
+			// or delegation to another Source. However, these classes are just too small 
+			// to justify their existence
+			Path rv;
+			if (delegate != null) {
+				rv = delegate.path(p);
+			} else {
+				rv = Path.create(normalizer.rewrite(p));
+			}
 			if (convertor != null) {
 				return convertor.mangle(rv);
 			}