Mercurial > hg4j
changeset 296:02f2963c70fa
Issue 13: Tests for mixed-cased filenames in case-insensitive FS
author | Artem Tikhomirov <tikhomirov.artem@gmail.com> |
---|---|
date | Fri, 16 Sep 2011 21:00:29 +0200 (2011-09-16) |
parents | 981f9f50bb6c |
children | e7ca6f16d074 |
files | src/org/tmatesoft/hg/repo/HgInternals.java test/org/tmatesoft/hg/test/TestDirstate.java |
diffstat | 2 files changed, 107 insertions(+), 2 deletions(-) [+] |
line wrap: on
line diff
--- a/src/org/tmatesoft/hg/repo/HgInternals.java Fri Sep 16 05:35:32 2011 +0200 +++ b/src/org/tmatesoft/hg/repo/HgInternals.java Fri Sep 16 21:00:29 2011 +0200 @@ -55,7 +55,33 @@ } public void dumpDirstate() { - repo.loadDirstate(new PathPool(new PathRewrite.Empty())).dump(); + getDirstate().dump(); + } + + public HgDirstate getDirstate() { + return repo.loadDirstate(new PathPool(new PathRewrite.Empty())); + } + + // tests + public HgDirstate createDirstate(boolean caseSensitiveFileSystem) { + PathRewrite canonicalPath = null; + if (!caseSensitiveFileSystem) { + canonicalPath = new PathRewrite() { + + public CharSequence rewrite(CharSequence path) { + return path.toString().toLowerCase(); + } + }; + } + return new HgDirstate(repo, new File(repo.getRepositoryRoot(), "dirstate"), new PathPool(new PathRewrite.Empty()), canonicalPath); + } + + public Path[] checkKnown(HgDirstate dirstate, Path[] toCheck) { + Path[] rv = new Path[toCheck.length]; + for (int i = 0; i < toCheck.length; i++) { + rv[i] = dirstate.known(toCheck[i]); + } + return rv; } public boolean[] checkIgnored(String... toCheck) {
--- a/test/org/tmatesoft/hg/test/TestDirstate.java Fri Sep 16 05:35:32 2011 +0200 +++ b/test/org/tmatesoft/hg/test/TestDirstate.java Fri Sep 16 21:00:29 2011 +0200 @@ -16,12 +16,23 @@ */ package org.tmatesoft.hg.test; +import static java.lang.Character.*; +import static org.junit.Assert.assertTrue; +import static org.junit.Assert.fail; + +import java.util.TreeSet; + import org.junit.Assert; import org.junit.Test; import org.tmatesoft.hg.core.Nodeid; +import org.tmatesoft.hg.repo.HgDirstate; +import org.tmatesoft.hg.repo.HgDirstate.EntryKind; +import org.tmatesoft.hg.repo.HgDirstate.Record; +import org.tmatesoft.hg.repo.HgInternals; import org.tmatesoft.hg.repo.HgLookup; import org.tmatesoft.hg.repo.HgRepository; import org.tmatesoft.hg.util.Pair; +import org.tmatesoft.hg.util.Path; /** * @@ -58,9 +69,77 @@ Assert.assertEquals("default", repo.getWorkingCopyBranchName()); } - public void testMixedNameCaseHandling() { + @Test + public void testMixedNameCaseHandling() throws Exception { + // General idea: to check cases like // 1. dirstate: /a/b/c, FileIterator: /a/B/C // 2. dirstate: /a/B/C, FileIterator: /a/b/c // 2. dirstate: /a/B/C, FileIterator: /A/b/C + repo = Configuration.get().find("mixed-case"); + // Windows, case-insensitive file system + final HgInternals testAccess = new HgInternals(repo); + HgDirstate dirstate = testAccess.createDirstate(false); + final TreeSet<Path> entries = new TreeSet<Path>(); + dirstate.walk(new HgDirstate.Inspector() { + + public boolean next(EntryKind kind, Record entry) { + entries.add(entry.name()); + return true; + } + }); + Path[] expected = new Path[] { + Path.create("a/low/low"), + Path.create("a/low/UP"), + Path.create("a/UP/low"), + Path.create("a/UP/UP"), + }; + Path[] allLower = new Path[expected.length]; + Path[] allUpper = new Path[expected.length]; + Path[] mixedNonMatching = new Path[expected.length]; + for (int i = 0; i < expected.length; i++) { + assertTrue("prereq", entries.contains(expected[i])); + final String s = expected[i].toString(); + allLower[i] = Path.create(s.toLowerCase()); + allUpper[i] = Path.create(s.toUpperCase()); + char[] ss = s.toCharArray(); + for (int j = 0; j < ss.length; j++) { + if (isLetter(ss[j])) { + ss[j] = isLowerCase(ss[j]) ? toUpperCase(ss[j]) : toLowerCase(ss[j]); + } + } + Path mixed = Path.create(new String(ss)); + mixedNonMatching[i] = mixed; + } + // prereq + checkKnownInDirstate(testAccess, dirstate, expected, expected); + // all upper + checkKnownInDirstate(testAccess, dirstate, allUpper, expected); + // all lower + checkKnownInDirstate(testAccess, dirstate, allLower, expected); + // mixed + checkKnownInDirstate(testAccess, dirstate, mixedNonMatching, expected); + // + // check that in case-sensitive file system mangled names do not match + dirstate = testAccess.createDirstate(true); + // ensure read + dirstate.walk(new HgDirstate.Inspector() { + public boolean next(EntryKind kind, Record entry) { + return false; + } + }); + Path[] known = testAccess.checkKnown(dirstate, mixedNonMatching); + for (int i = 0; i < known.length; i++) { + if (known[i] != null) { + fail(expected[i] + " in case-sensitive dirstate matched " + known[i]); + } + } + + } + + private static void checkKnownInDirstate(HgInternals testAccess, HgDirstate dirstate, Path[] toCheck, Path[] expected) { + Path[] known = testAccess.checkKnown(dirstate, toCheck); + for (int i = 0; i < expected.length; i++) { + assertTrue(expected[i].equals(known[i])); + } } }