##// END OF EJS Templates
dispatch: protect against malicious 'hg serve --stdio' invocations (sec)...
dispatch: protect against malicious 'hg serve --stdio' invocations (sec) Some shared-ssh installations assume that 'hg serve --stdio' is a safe command to run for minimally trusted users. Unfortunately, the messy implementation of argument parsing here meant that trying to access a repo named '--debugger' would give the user a pdb prompt, thereby sidestepping any hoped-for sandboxing. Serving repositories over HTTP(S) is unaffected. We're not currently hardening any subcommands other than 'serve'. If your service exposes other commands to users with arbitrary repository names, it is imperative that you defend against repository names of '--debugger' and anything starting with '--config'. The read-only mode of hg-ssh stopped working because it provided its hook configuration to "hg serve --stdio" via --config parameter. This is banned for security reasons now. This patch switches it to directly call ui.setconfig(). If your custom hosting infrastructure relies on passing --config to "hg serve --stdio", you'll need to find a different way to get that configuration into Mercurial, either by using ui.setconfig() as hg-ssh does in this patch, or by placing an hgrc file someplace where Mercurial will read it. mitrandir@fb.com provided some extra fixes for the dispatch code and for hg-ssh in places that I overlooked.

File last commit:

r31876:94c1d3c1 default
r32050:77eaf953 4.1.3 stable
Show More
test-manifest.py
471 lines | 16.3 KiB | text/x-python | PythonLexer
Pulkit Goyal
tests: make test-manifest use absolute_import
r28929 from __future__ import absolute_import
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
Pulkit Goyal
tests: make test-manifest use absolute_import
r28929 import binascii
import itertools
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 import silenttestrunner
Pulkit Goyal
tests: make test-manifest use absolute_import
r28929 import unittest
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
Pulkit Goyal
tests: make test-manifest use absolute_import
r28929 from mercurial import (
manifest as manifestmod,
match as matchmod,
)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
Martin von Zweigbergk
test-manifest: create constant for empty manifest...
r24569 EMTPY_MANIFEST = ''
Martin von Zweigbergk
manifestv2: add support for reading new manifest format...
r24572 EMTPY_MANIFEST_V2 = '\0\n'
Martin von Zweigbergk
test-manifest: create constant for empty manifest...
r24569
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 HASH_1 = '1' * 40
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 BIN_HASH_1 = binascii.unhexlify(HASH_1)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 HASH_2 = 'f' * 40
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 BIN_HASH_2 = binascii.unhexlify(HASH_2)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 HASH_3 = '1234567890abcdef0987654321deadbeef0fcafe'
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 BIN_HASH_3 = binascii.unhexlify(HASH_3)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 A_SHORT_MANIFEST = (
'bar/baz/qux.py\0%(hash2)s%(flag2)s\n'
'foo\0%(hash1)s%(flag1)s\n'
) % {'hash1': HASH_1,
'flag1': '',
'hash2': HASH_2,
'flag2': 'l',
}
Martin von Zweigbergk
manifestv2: add support for reading new manifest format...
r24572 # Same data as A_SHORT_MANIFEST
A_SHORT_MANIFEST_V2 = (
'\0\n'
'\x00bar/baz/qux.py\0%(flag2)s\n%(hash2)s\n'
'\x00foo\0%(flag1)s\n%(hash1)s\n'
) % {'hash1': BIN_HASH_1,
'flag1': '',
'hash2': BIN_HASH_2,
'flag2': 'l',
}
# Same data as A_SHORT_MANIFEST
A_METADATA_MANIFEST = (
'\0foo\0bar\n'
'\x00bar/baz/qux.py\0%(flag2)s\0foo\0bar\n%(hash2)s\n' # flag and metadata
'\x00foo\0%(flag1)s\0foo\n%(hash1)s\n' # no flag, but metadata
) % {'hash1': BIN_HASH_1,
'flag1': '',
'hash2': BIN_HASH_2,
'flag2': 'l',
}
A_STEM_COMPRESSED_MANIFEST = (
'\0\n'
'\x00bar/baz/qux.py\0%(flag2)s\n%(hash2)s\n'
'\x04qux/foo.py\0%(flag1)s\n%(hash1)s\n' # simple case of 4 stem chars
'\x0az.py\0%(flag1)s\n%(hash1)s\n' # tricky newline = 10 stem characters
'\x00%(verylongdir)sx/x\0\n%(hash1)s\n'
'\xffx/y\0\n%(hash2)s\n' # more than 255 stem chars
) % {'hash1': BIN_HASH_1,
'flag1': '',
'hash2': BIN_HASH_2,
'flag2': 'l',
'verylongdir': 255 * 'x',
}
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549 A_DEEPER_MANIFEST = (
'a/b/c/bar.py\0%(hash3)s%(flag1)s\n'
'a/b/c/bar.txt\0%(hash1)s%(flag1)s\n'
'a/b/c/foo.py\0%(hash3)s%(flag1)s\n'
'a/b/c/foo.txt\0%(hash2)s%(flag2)s\n'
'a/b/d/baz.py\0%(hash3)s%(flag1)s\n'
'a/b/d/qux.py\0%(hash1)s%(flag2)s\n'
'a/b/d/ten.txt\0%(hash3)s%(flag2)s\n'
'a/b/dog.py\0%(hash3)s%(flag1)s\n'
'a/b/fish.py\0%(hash2)s%(flag1)s\n'
'a/c/london.py\0%(hash3)s%(flag2)s\n'
'a/c/paper.txt\0%(hash2)s%(flag2)s\n'
'a/c/paris.py\0%(hash2)s%(flag1)s\n'
'a/d/apple.py\0%(hash3)s%(flag1)s\n'
'a/d/pizza.py\0%(hash3)s%(flag2)s\n'
'a/green.py\0%(hash1)s%(flag2)s\n'
'a/purple.py\0%(hash2)s%(flag1)s\n'
'app.py\0%(hash3)s%(flag1)s\n'
'readme.txt\0%(hash2)s%(flag1)s\n'
) % {'hash1': HASH_1,
'flag1': '',
'hash2': HASH_2,
'flag2': 'l',
'hash3': HASH_3,
}
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 HUGE_MANIFEST_ENTRIES = 200001
A_HUGE_MANIFEST = ''.join(sorted(
'file%d\0%s%s\n' % (i, h, f) for i, h, f in
itertools.izip(xrange(200001),
itertools.cycle((HASH_1, HASH_2)),
itertools.cycle(('', 'x', 'l')))))
Drew Gottlieb
test-manifest: make manifesttest a base class that is extended...
r24655 class basemanifesttests(object):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 def parsemanifest(self, text):
Drew Gottlieb
test-manifest: make manifesttest a base class that is extended...
r24655 raise NotImplementedError('parsemanifest not implemented by test case')
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
def assertIn(self, thing, container, msg=None):
# assertIn new in 2.7, use it if available, otherwise polyfill
sup = getattr(unittest.TestCase, 'assertIn', False)
if sup:
return sup(self, thing, container, msg=msg)
if not msg:
msg = 'Expected %r in %r' % (thing, container)
self.assert_(thing in container, msg)
def testEmptyManifest(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(EMTPY_MANIFEST)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.assertEqual(0, len(m))
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 self.assertEqual([], list(m))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
Martin von Zweigbergk
manifestv2: add support for reading new manifest format...
r24572 def testEmptyManifestv2(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(EMTPY_MANIFEST_V2)
Martin von Zweigbergk
manifestv2: add support for reading new manifest format...
r24572 self.assertEqual(0, len(m))
self.assertEqual([], list(m))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 def testManifest(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_SHORT_MANIFEST)
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 self.assertEqual(['bar/baz/qux.py', 'foo'], list(m))
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 self.assertEqual(BIN_HASH_2, m['bar/baz/qux.py'])
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 self.assertEqual('l', m.flags('bar/baz/qux.py'))
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 self.assertEqual(BIN_HASH_1, m['foo'])
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 self.assertEqual('', m.flags('foo'))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.assertRaises(KeyError, lambda : m['wat'])
Martin von Zweigbergk
manifestv2: add support for reading new manifest format...
r24572 def testParseManifestV2(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m1 = self.parsemanifest(A_SHORT_MANIFEST)
m2 = self.parsemanifest(A_SHORT_MANIFEST_V2)
Martin von Zweigbergk
manifestv2: add support for reading new manifest format...
r24572 # Should have same content as A_SHORT_MANIFEST
self.assertEqual(m1.text(), m2.text())
def testParseManifestMetadata(self):
# Metadata is for future-proofing and should be accepted but ignored
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_METADATA_MANIFEST)
Martin von Zweigbergk
manifestv2: add support for reading new manifest format...
r24572 self.assertEqual(A_SHORT_MANIFEST, m.text())
def testParseManifestStemCompression(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_STEM_COMPRESSED_MANIFEST)
Martin von Zweigbergk
manifestv2: add support for reading new manifest format...
r24572 self.assertIn('bar/baz/qux.py', m)
self.assertIn('bar/qux/foo.py', m)
self.assertIn('bar/qux/foz.py', m)
self.assertIn(256 * 'x' + '/x', m)
self.assertIn(256 * 'x' + '/y', m)
Martin von Zweigbergk
manifestv2: add support for writing new manifest format...
r24573 self.assertEqual(A_STEM_COMPRESSED_MANIFEST, m.text(usemanifestv2=True))
def testTextV2(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m1 = self.parsemanifest(A_SHORT_MANIFEST)
Martin von Zweigbergk
manifestv2: add support for writing new manifest format...
r24573 v2text = m1.text(usemanifestv2=True)
self.assertEqual(A_SHORT_MANIFEST_V2, v2text)
Martin von Zweigbergk
manifestv2: add support for reading new manifest format...
r24572
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 def testSetItem(self):
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 want = BIN_HASH_1
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(EMTPY_MANIFEST)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 m['a'] = want
self.assertIn('a', m)
self.assertEqual(want, m['a'])
self.assertEqual('a\0' + HASH_1 + '\n', m.text())
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_SHORT_MANIFEST)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 m['a'] = want
self.assertEqual(want, m['a'])
self.assertEqual('a\0' + HASH_1 + '\n' + A_SHORT_MANIFEST,
m.text())
Martin von Zweigbergk
test-manifest.py: separate out test for double-free after copy()...
r24465
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 def testSetFlag(self):
want = 'x'
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(EMTPY_MANIFEST)
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 # first add a file; a file-less flag makes no sense
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 m['a'] = BIN_HASH_1
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 m.setflag('a', want)
self.assertEqual(want, m.flags('a'))
self.assertEqual('a\0' + HASH_1 + want + '\n', m.text())
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_SHORT_MANIFEST)
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 # first add a file; a file-less flag makes no sense
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 m['a'] = BIN_HASH_1
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 m.setflag('a', want)
self.assertEqual(want, m.flags('a'))
self.assertEqual('a\0' + HASH_1 + want + '\n' + A_SHORT_MANIFEST,
m.text())
Martin von Zweigbergk
test-manifest.py: separate out test for double-free after copy()...
r24465 def testCopy(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_SHORT_MANIFEST)
timeless
cleanup: remove superfluous space after space after equals (python)
r27637 m['a'] = BIN_HASH_1
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 m2 = m.copy()
del m
del m2 # make sure we don't double free() anything
def testCompaction(self):
unhex = binascii.unhexlify
h1, h2 = unhex(HASH_1), unhex(HASH_2)
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_SHORT_MANIFEST)
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 m['alpha'] = h1
m['beta'] = h2
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 del m['foo']
want = 'alpha\0%s\nbar/baz/qux.py\0%sl\nbeta\0%s\n' % (
HASH_1, HASH_2, HASH_2)
self.assertEqual(want, m.text())
self.assertEqual(3, len(m))
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 self.assertEqual(['alpha', 'bar/baz/qux.py', 'beta'], list(m))
self.assertEqual(h1, m['alpha'])
self.assertEqual(h2, m['bar/baz/qux.py'])
self.assertEqual(h2, m['beta'])
self.assertEqual('', m.flags('alpha'))
self.assertEqual('l', m.flags('bar/baz/qux.py'))
self.assertEqual('', m.flags('beta'))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.assertRaises(KeyError, lambda : m['foo'])
def testSetGetNodeSuffix(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 clean = self.parsemanifest(A_SHORT_MANIFEST)
m = self.parsemanifest(A_SHORT_MANIFEST)
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 h = m['foo']
f = m.flags('foo')
want = h + 'a'
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 # Merge code wants to set 21-byte fake hashes at times
m['foo'] = want
self.assertEqual(want, m['foo'])
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 self.assertEqual([('bar/baz/qux.py', BIN_HASH_2),
('foo', BIN_HASH_1 + 'a')],
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 list(m.iteritems()))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 # Sometimes it even tries a 22-byte fake hash, but we can
# return 21 and it'll work out
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 m['foo'] = want + '+'
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.assertEqual(want, m['foo'])
# make sure the suffix survives a copy
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 match = matchmod.match('', '', ['re:foo'])
m2 = m.matches(match)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.assertEqual(want, m2['foo'])
self.assertEqual(1, len(m2))
m2 = m.copy()
self.assertEqual(want, m2['foo'])
# suffix with iteration
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 self.assertEqual([('bar/baz/qux.py', BIN_HASH_2),
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 ('foo', want)],
list(m.iteritems()))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 # shows up in diff
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 self.assertEqual({'foo': ((want, f), (h, ''))}, m.diff(clean))
self.assertEqual({'foo': ((h, ''), (want, f))}, clean.diff(m))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 def testMatchException(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_SHORT_MANIFEST)
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 match = matchmod.match('', '', ['re:.*'])
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 def filt(path):
if path == 'foo':
assert False
return True
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 match.matchfn = filt
self.assertRaises(AssertionError, m.matches, match)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
def testRemoveItem(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_SHORT_MANIFEST)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 del m['foo']
self.assertRaises(KeyError, lambda : m['foo'])
self.assertEqual(1, len(m))
self.assertEqual(1, len(list(m)))
Augie Fackler
lazymanifest: use a binary search to do an insertion...
r24228 # now restore and make sure everything works right
Martin von Zweigbergk
test-manifest.py: rewrite tests in terms of manifestdict...
r24466 m['foo'] = 'a' * 20
Augie Fackler
lazymanifest: use a binary search to do an insertion...
r24228 self.assertEqual(2, len(m))
self.assertEqual(2, len(list(m)))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
def testManifestDiff(self):
MISSING = (None, '')
addl = 'z-only-in-left\0' + HASH_1 + '\n'
addr = 'z-only-in-right\0' + HASH_2 + 'x\n'
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 left = self.parsemanifest(
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 A_SHORT_MANIFEST.replace(HASH_1, HASH_3 + 'x') + addl)
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 right = self.parsemanifest(A_SHORT_MANIFEST + addr)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 want = {
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 'foo': ((BIN_HASH_3, 'x'),
(BIN_HASH_1, '')),
'z-only-in-left': ((BIN_HASH_1, ''), MISSING),
'z-only-in-right': (MISSING, (BIN_HASH_2, 'x')),
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 }
self.assertEqual(want, left.diff(right))
want = {
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 'bar/baz/qux.py': (MISSING, (BIN_HASH_2, 'l')),
'foo': (MISSING, (BIN_HASH_3, 'x')),
'z-only-in-left': (MISSING, (BIN_HASH_1, '')),
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 }
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 self.assertEqual(want, self.parsemanifest(EMTPY_MANIFEST).diff(left))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
want = {
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 'bar/baz/qux.py': ((BIN_HASH_2, 'l'), MISSING),
'foo': ((BIN_HASH_3, 'x'), MISSING),
'z-only-in-left': ((BIN_HASH_1, ''), MISSING),
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 }
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 self.assertEqual(want, left.diff(self.parsemanifest(EMTPY_MANIFEST)))
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 copy = right.copy()
del copy['z-only-in-right']
del right['foo']
want = {
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 'foo': (MISSING, (BIN_HASH_1, '')),
'z-only-in-right': ((BIN_HASH_2, 'x'), MISSING),
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 }
self.assertEqual(want, right.diff(copy))
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 short = self.parsemanifest(A_SHORT_MANIFEST)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 pruned = short.copy()
del pruned['foo']
want = {
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 'foo': ((BIN_HASH_1, ''), MISSING),
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 }
self.assertEqual(want, short.diff(pruned))
want = {
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 'foo': (MISSING, (BIN_HASH_1, '')),
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 }
self.assertEqual(want, pruned.diff(short))
want = {
'bar/baz/qux.py': None,
Martin von Zweigbergk
test-manifest: extract constants for binary hashes...
r24570 'foo': (MISSING, (BIN_HASH_1, '')),
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 }
self.assertEqual(want, pruned.diff(short, True))
def testReversedLines(self):
backwards = ''.join(
l + '\n' for l in reversed(A_SHORT_MANIFEST.split('\n')) if l)
try:
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 self.parsemanifest(backwards)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.fail('Should have raised ValueError')
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except ValueError as v:
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.assertIn('Manifest lines not in sorted order.', str(v))
def testNoTerminalNewline(self):
try:
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 self.parsemanifest(A_SHORT_MANIFEST + 'wat')
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.fail('Should have raised ValueError')
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except ValueError as v:
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.assertIn('Manifest did not end in a newline.', str(v))
def testNoNewLineAtAll(self):
try:
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 self.parsemanifest('wat')
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.fail('Should have raised ValueError')
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except ValueError as v:
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.assertIn('Manifest did not end in a newline.', str(v))
def testHugeManifest(self):
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_HUGE_MANIFEST)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 self.assertEqual(HUGE_MANIFEST_ENTRIES, len(m))
self.assertEqual(len(m), len(list(m)))
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549 def testMatchesMetadata(self):
'''Tests matches() for a few specific files to make sure that both
the set of files as well as their flags and nodeids are correct in
the resulting manifest.'''
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_HUGE_MANIFEST)
Drew Gottlieb
manifest: make manifest.intersectfiles() internal...
r24495
match = matchmod.match('/', '',
['file1', 'file200', 'file300'], exact=True)
m2 = m.matches(match)
Augie Fackler
manifest: split manifestdict into high-level and low-level logic...
r24225 w = ('file1\0%sx\n'
'file200\0%sl\n'
'file300\0%s\n') % (HASH_2, HASH_1, HASH_1)
self.assertEqual(w, m2.text())
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549 def testMatchesNonexistentFile(self):
'''Tests matches() for a small set of specific files, including one
nonexistent file to make sure in only matches against existing files.
'''
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_DEEPER_MANIFEST)
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549
match = matchmod.match('/', '',
['a/b/c/bar.txt', 'a/b/d/qux.py', 'readme.txt', 'nonexistent'],
exact=True)
m2 = m.matches(match)
self.assertEqual(
['a/b/c/bar.txt', 'a/b/d/qux.py', 'readme.txt'],
m2.keys())
def testMatchesNonexistentDirectory(self):
'''Tests matches() for a relpath match on a directory that doesn't
actually exist.'''
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_DEEPER_MANIFEST)
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549
match = matchmod.match('/', '', ['a/f'], default='relpath')
m2 = m.matches(match)
self.assertEqual([], m2.keys())
def testMatchesExactLarge(self):
'''Tests matches() for files matching a large list of exact files.
'''
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_HUGE_MANIFEST)
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549
flist = m.keys()[80:300]
match = matchmod.match('/', '', flist, exact=True)
m2 = m.matches(match)
self.assertEqual(flist, m2.keys())
def testMatchesFull(self):
'''Tests matches() for what should be a full match.'''
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_DEEPER_MANIFEST)
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549
match = matchmod.match('/', '', [''])
m2 = m.matches(match)
self.assertEqual(m.keys(), m2.keys())
def testMatchesDirectory(self):
'''Tests matches() on a relpath match on a directory, which should
match against all files within said directory.'''
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_DEEPER_MANIFEST)
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549
match = matchmod.match('/', '', ['a/b'], default='relpath')
m2 = m.matches(match)
self.assertEqual([
'a/b/c/bar.py', 'a/b/c/bar.txt', 'a/b/c/foo.py', 'a/b/c/foo.txt',
'a/b/d/baz.py', 'a/b/d/qux.py', 'a/b/d/ten.txt', 'a/b/dog.py',
'a/b/fish.py'], m2.keys())
def testMatchesExactPath(self):
'''Tests matches() on an exact match on a directory, which should
result in an empty manifest because you can't perform an exact match
against a directory.'''
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_DEEPER_MANIFEST)
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549
match = matchmod.match('/', '', ['a/b'], exact=True)
m2 = m.matches(match)
self.assertEqual([], m2.keys())
def testMatchesCwd(self):
'''Tests matches() on a relpath match with the current directory ('.')
when not in the root directory.'''
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_DEEPER_MANIFEST)
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549
match = matchmod.match('/', 'a/b', ['.'], default='relpath')
m2 = m.matches(match)
self.assertEqual([
'a/b/c/bar.py', 'a/b/c/bar.txt', 'a/b/c/foo.py', 'a/b/c/foo.txt',
'a/b/d/baz.py', 'a/b/d/qux.py', 'a/b/d/ten.txt', 'a/b/dog.py',
'a/b/fish.py'], m2.keys())
def testMatchesWithPattern(self):
'''Tests matches() for files matching a pattern that reside
deeper than the specified directory.'''
Drew Gottlieb
test-manifest: move parsemanifest() to be a testmanifest class method...
r24654 m = self.parsemanifest(A_DEEPER_MANIFEST)
Drew Gottlieb
manifest: add some tests for manifest.matches()...
r24549
match = matchmod.match('/', '', ['a/b/*/*.txt'])
m2 = m.matches(match)
self.assertEqual(
['a/b/c/bar.txt', 'a/b/c/foo.txt', 'a/b/d/ten.txt'],
m2.keys())
Drew Gottlieb
test-manifest: make manifesttest a base class that is extended...
r24655 class testmanifestdict(unittest.TestCase, basemanifesttests):
def parsemanifest(self, text):
return manifestmod.manifestdict(text)
Drew Gottlieb
test-manifest: add some test coverage for treemanifest...
r24656 class testtreemanifest(unittest.TestCase, basemanifesttests):
def parsemanifest(self, text):
return manifestmod.treemanifest('', text)
Augie Fackler
manifest.c: new extension code to lazily parse manifests...
r24214 if __name__ == '__main__':
silenttestrunner.main(__name__)