Close files opened by Scanner, Loader

We're seeing noise like the following in doc builds:

  /foo/.tox/docs/lib/python3.9/site-packages/docutils/statemachine.py:707:
  ResourceWarning: unclosed file <_io.BufferedReader name='/foo/.git/objects/pack/pack-dd69481843ca1b7377f2f109b0022221437aca20.pack'>
    if not hasattr(pattern, 'match'):
  ResourceWarning: Enable tracemalloc to get the object allocation traceback

Enable tracemalloc for Sphinx gives us the 'Loader' as the root cause:

  $ python -W all::ResourceWarning -X tracemalloc=100 -m sphinx.cmd.build ...
  ...
  File "/foo/.tox/docs/lib/python3.9/site-packages/reno/sphinxext.py", lineno 114
    ldr = loader.Loader(conf)
  File "/foo/.tox/docs/lib/python3.9/site-packages/reno/loader.py", lineno 63
    self._load_data()
  ...

(you could also use PYTHONTRACEMALLOC and PYTHONWARNINGS env vars)

Following this thread, it appears 'reno.scanner.Scanner' creates an
instance of 'dulwich.repo.Repo', however, it fails to close it [1]. The
'reno.loader.Loader' uses 'Scanner', meaning this also leaves around
open files.

The solution is simple: add a 'close()' method to both the 'Scanner' and
'Loader', and provide the necessary '__enter__' and '__exit__' magic
methods to use it as a context manager, like the 'Repo' object itself
supports.

[1] https://www.dulwich.io/docs/api/dulwich.repo.html#dulwich.repo.Repo.close

Change-Id: I0b9776f431cf902a9ace5d52961eb77caaae8eaa
Signed-off-by: Stephen Finucane <stephenfin@redhat.com>
This commit is contained in:
Stephen Finucane 2021-09-28 12:29:40 +01:00
parent 8a80087d92
commit cd31b907c3
13 changed files with 421 additions and 346 deletions

View File

@ -0,0 +1,31 @@
---
fixes:
- |
Instances of ``reno.scanner.Scanner`` and ``reno.loader.Loader`` will now
corectly close all open files related to the scanned Git repo when the
``close()`` method is called. Alternatively, these classes may be used as
a context manager. Previously, Python would attempt to close these files
itself, resulting in a ``ResourceWarning`` warning being emitted.
features:
- |
The ``reno.scanner.Scanner`` and ``reno.loader.Loader`` classes can now
be used as context managers. For example::
import reno.scannner
with reno.scanner.Scanner(...) as scanner:
pass
This will ensure any open files pertaining to the scanned Git repo are
correctly closed, avoiding ``ResourceWarning`` instances otherwise seen.
A ``close()`` method is also provided for both, allowing use outside of
context managers. For example::
import reno.loader
loader = reno.loader.Loader(...)
try:
pass
finally:
loader.close()

View File

@ -21,50 +21,48 @@ from reno import scanner
def build_cache_db(conf, versions_to_include): def build_cache_db(conf, versions_to_include):
s = scanner.Scanner(conf) with scanner.Scanner(conf) as s:
branches = [conf.branch]
if not conf.branch: # if no branch requested, scan all
branches += s.get_series_branches()
branches = [conf.branch] notes = collections.OrderedDict()
if not conf.branch: # if no branch requested, scan all for branch in branches:
branches += s.get_series_branches() notes.update(s.get_notes_by_version(branch))
notes = collections.OrderedDict() # Default to including all versions returned by the scanner.
for branch in branches: if not versions_to_include:
notes.update(s.get_notes_by_version(branch)) versions_to_include = list(notes.keys())
# Default to including all versions returned by the scanner. # Build a cache data structure including the file contents as well
if not versions_to_include: # as the basic data returned by the scanner.
versions_to_include = list(notes.keys()) file_contents = {}
for version in versions_to_include:
for filename, sha in notes[version]:
body = s.get_file_at_commit(filename, sha)
# We want to save the contents of the file, which is YAML,
# inside another YAML file. That looks terribly ugly with
# all of the escapes needed to format it properly as
# embedded YAML, so parse the input and convert it to a
# data structure that can be serialized cleanly.
y = yaml.safe_load(body)
file_contents[filename] = y
# Build a cache data structure including the file contents as well cache = {
# as the basic data returned by the scanner. 'notes': [
file_contents = {} {'version': k, 'files': v}
for version in versions_to_include: for k, v in notes.items()
for filename, sha in notes[version]: ],
body = s.get_file_at_commit(filename, sha) 'dates': [
# We want to save the contents of the file, which is YAML, {'version': k, 'date': v}
# inside another YAML file. That looks terribly ugly with for k, v in s.get_version_dates().items()
# all of the escapes needed to format it properly as ],
# embedded YAML, so parse the input and convert it to a 'file-contents': file_contents,
# data structure that can be serialized cleanly. }
y = yaml.safe_load(body) return cache
file_contents[filename] = y
cache = {
'notes': [
{'version': k, 'files': v}
for k, v in notes.items()
],
'dates': [
{'version': k, 'date': v}
for k, v in s.get_version_dates().items()
],
'file-contents': file_contents,
}
return cache
def write_cache_db(conf, versions_to_include, def write_cache_db(conf, versions_to_include, outfilename=None):
outfilename=None):
"""Create a cache database file for the release notes data. """Create a cache database file for the release notes data.
Build the cache database from scanning the project history and Build the cache database from scanning the project history and

View File

@ -27,22 +27,24 @@ def lint_cmd(args, conf):
notes = glob.glob(os.path.join(notesdir, '*.yaml')) notes = glob.glob(os.path.join(notesdir, '*.yaml'))
error = 0 error = 0
load = loader.Loader(conf, ignore_cache=True)
allowed_section_names = [conf.prelude_section_name] + \ allowed_section_names = [conf.prelude_section_name] + \
[s[0] for s in conf.sections] [s[0] for s in conf.sections]
uids = {} uids = {}
for f in notes: with loader.Loader(conf, ignore_cache=True) as ldr:
LOG.debug('examining %s', f) for f in notes:
uid = scanner._get_unique_id(f) LOG.debug('examining %s', f)
uids.setdefault(uid, []).append(f) uid = scanner._get_unique_id(f)
uids.setdefault(uid, []).append(f)
content = load.parse_note_file(f, None) content = ldr.parse_note_file(f, None)
for section_name in content.keys(): for section_name in content.keys():
if section_name not in allowed_section_names: if section_name not in allowed_section_names:
LOG.warning('unrecognized section name %s in %s', LOG.warning(
section_name, f) 'unrecognized section name %s in %s',
error = 1 section_name, f,
)
error = 1
for uid, names in sorted(uids.items()): for uid, names in sorted(uids.items()):
if len(names) > 1: if len(names) > 1:

View File

@ -21,16 +21,16 @@ def list_cmd(args, conf):
"List notes files based on query arguments" "List notes files based on query arguments"
LOG.debug('starting list') LOG.debug('starting list')
reporoot = conf.reporoot reporoot = conf.reporoot
ldr = loader.Loader(conf) with loader.Loader(conf) as ldr:
if args.version: if args.version:
versions = args.version versions = args.version
else: else:
versions = ldr.versions versions = ldr.versions
for version in versions: for version in versions:
notefiles = ldr[version] notefiles = ldr[version]
print(version) print(version)
for n, sha in notefiles: for n, sha in notefiles:
if n.startswith(reporoot): if n.startswith(reporoot):
n = n[len(reporoot):] n = n[len(reporoot):]
print('\t%s (%s)' % (n, sha)) print('\t%s (%s)' % (n, sha))
return return

View File

@ -30,8 +30,7 @@ def get_cache_filename(conf):
class Loader(object): class Loader(object):
"Load the release notes for a given repository." "Load the release notes for a given repository."
def __init__(self, conf, def __init__(self, conf, ignore_cache=False):
ignore_cache=False):
"""Initialize a Loader. """Initialize a Loader.
The versions are presented in reverse chronological order. The versions are presented in reverse chronological order.
@ -70,24 +69,37 @@ class Loader(object):
if (not self._ignore_cache) and cache_file_exists: if (not self._ignore_cache) and cache_file_exists:
LOG.debug('loading cache file %s', self._cache_filename) LOG.debug('loading cache file %s', self._cache_filename)
with open(self._cache_filename, 'r', encoding=self._encoding) as f: with open(self._cache_filename, 'r', encoding=self._encoding) as f:
self._cache = yaml.safe_load(f.read()) self._cache = yaml.safe_load(f.read())
# Save the cached scanner output to the same attribute
# it would be in if we had loaded it "live". This # Save the cached scanner output to the same attribute
# simplifies some of the logic in the other methods. # it would be in if we had loaded it "live". This
self._scanner_output = collections.OrderedDict( # simplifies some of the logic in the other methods.
(n['version'], n['files']) self._scanner_output = collections.OrderedDict(
for n in self._cache['notes'] (n['version'], n['files'])
) for n in self._cache['notes']
self._tags_to_dates = collections.OrderedDict( )
(n['version'], n['date']) self._tags_to_dates = collections.OrderedDict(
for n in self._cache['dates'] (n['version'], n['date'])
) for n in self._cache['dates']
)
else: else:
self._scanner = scanner.Scanner(self._config) self._scanner = scanner.Scanner(self._config)
self._scanner_output = self._scanner.get_notes_by_version() self._scanner_output = self._scanner.get_notes_by_version()
self._tags_to_dates = self._scanner.get_version_dates() self._tags_to_dates = self._scanner.get_version_dates()
def close(self):
"""Close any files opened by this loader."""
if self._scanner is not None:
self._scanner.close()
def __enter__(self):
return self
def __exit__(self, exc_type, exc_val, exc_tb):
self.close()
@property @property
def versions(self): def versions(self):
"A list of all of the versions found." "A list of all of the versions found."

View File

@ -16,20 +16,22 @@ from reno import loader
def report_cmd(args, conf): def report_cmd(args, conf):
"Generates a release notes report" "Generates a release notes report"
ldr = loader.Loader(conf)
encoding = conf.options['encoding'] encoding = conf.options['encoding']
if args.version:
versions = args.version with loader.Loader(conf) as ldr:
else: if args.version:
versions = ldr.versions versions = args.version
text = formatter.format_report( else:
ldr, versions = ldr.versions
conf, text = formatter.format_report(
versions, ldr,
title=args.title, conf,
show_source=args.show_source, versions,
branch=args.branch, title=args.title,
) show_source=args.show_source,
branch=args.branch,
)
if args.output: if args.output:
with open(args.output, 'w', encoding=encoding) as f: with open(args.output, 'w', encoding=encoding) as f:
f.write(text) f.write(text)

View File

@ -533,6 +533,16 @@ class Scanner(object):
) )
self._encoding = conf.options['encoding'] self._encoding = conf.options['encoding']
def close(self):
"""Close any files opened by this scanner."""
self._repo.close()
def __enter__(self):
return self
def __exit__(self, exc_type, exc_val, exc_tb):
self.close()
def _get_ref(self, name): def _get_ref(self, name):
if name: if name:
candidates = [ candidates = [

View File

@ -22,76 +22,85 @@ LOG = logging.getLogger(__name__)
def compute_next_version(conf): def compute_next_version(conf):
"Compute the next semantic version based on the available release notes." "Compute the next semantic version based on the available release notes."
LOG.debug('starting semver-next') LOG.debug('starting semver-next')
ldr = loader.Loader(conf, ignore_cache=True) with loader.Loader(conf, ignore_cache=True) as ldr:
LOG.debug('known versions: %s', ldr.versions) LOG.debug('known versions: %s', ldr.versions)
# We want to include any notes in the local working directory or # We want to include any notes in the local working directory or
# in any commits that came after the last tag. We should never end # in any commits that came after the last tag. We should never end
# up with more than 2 entries in to_include. # up with more than 2 entries in to_include.
to_include = [] to_include = []
for to_consider in ldr.versions: for to_consider in ldr.versions:
if to_consider == '*working-copy*': if to_consider == '*working-copy*':
to_include.append(to_consider) to_include.append(to_consider)
continue continue
# This check relies on PEP 440 versioning
parsed = version.Version(to_consider)
if parsed.post:
to_include.append(to_consider)
continue
break
# If we found no commits then we're sitting on a real tag and # This check relies on PEP 440 versioning
# there is nothing to do to update the version. parsed = version.Version(to_consider)
if not to_include: if parsed.post:
LOG.debug('found no staged notes and no post-release commits') to_include.append(to_consider)
return ldr.versions[0] continue
LOG.debug('including notes from %s', to_include) break
candidate_bases = to_include[:] # If we found no commits then we're sitting on a real tag and
if candidate_bases[0] == '*working-copy*': # there is nothing to do to update the version.
candidate_bases = candidate_bases[1:] if not to_include:
LOG.debug('found no staged notes and no post-release commits')
return ldr.versions[0]
if not candidate_bases: LOG.debug('including notes from %s', to_include)
# We have a real tag and some locally modified files. Use the
# real tag as the basis of the next version.
base_version = version.Version(ldr.versions[1])
else:
base_version = version.Version(candidate_bases[0])
LOG.debug('base version %s', base_version) candidate_bases = to_include[:]
if candidate_bases[0] == '*working-copy*':
candidate_bases = candidate_bases[1:]
inc_minor = False if not candidate_bases:
inc_patch = False # We have a real tag and some locally modified files. Use the
for ver in to_include: # real tag as the basis of the next version.
for filename, sha in ldr[ver]: base_version = version.Version(ldr.versions[1])
notes = ldr.parse_note_file(filename, sha) else:
for section in conf.semver_major: base_version = version.Version(candidate_bases[0])
if notes.get(section, []):
LOG.debug('found breaking change in %r section of %s', LOG.debug('base version %s', base_version)
section, filename)
return '{}.0.0'.format(base_version.major + 1) inc_minor = False
for section in conf.semver_minor: inc_patch = False
if notes.get(section, []):
LOG.debug('found feature in %r section of %s', for ver in to_include:
section, filename) for filename, sha in ldr[ver]:
inc_minor = True notes = ldr.parse_note_file(filename, sha)
break
for section in conf.semver_patch: for section in conf.semver_major:
if notes.get(section, []): if notes.get(section, []):
LOG.debug('found bugfix in %r section of %s', LOG.debug('found breaking change in %r section of %s',
section, filename) section, filename)
inc_patch = True return '{}.0.0'.format(base_version.major + 1)
break
for section in conf.semver_minor:
if notes.get(section, []):
LOG.debug('found feature in %r section of %s',
section, filename)
inc_minor = True
break
for section in conf.semver_patch:
if notes.get(section, []):
LOG.debug('found bugfix in %r section of %s',
section, filename)
inc_patch = True
break
major = base_version.major major = base_version.major
minor = base_version.minor minor = base_version.minor
patch = base_version.micro patch = base_version.micro
if inc_patch: if inc_patch:
patch += 1 patch += 1
if inc_minor: if inc_minor:
minor += 1 minor += 1
patch = 0 patch = 0
return '{}.{}.{}'.format(major, minor, patch) return '{}.{}.{}'.format(major, minor, patch)

View File

@ -127,13 +127,15 @@ class BuildReno(cmd.Command):
) )
log.info('wrote cache file to %s', cache_filename) log.info('wrote cache file to %s', cache_filename)
ldr = loader.Loader(conf) with loader.Loader(conf) as ldr:
text = formatter.format_report( text = formatter.format_report(
ldr, ldr,
conf, conf,
ldr.versions, ldr.versions,
title=self.distribution.metadata.name, title=self.distribution.metadata.name,
) )
with open(self.output_file, 'w') as f: with open(self.output_file, 'w') as f:
f.write(text) f.write(text)
log.info('wrote release notes to %s', self.output_file) log.info('wrote release notes to %s', self.output_file)

View File

@ -111,22 +111,23 @@ class ReleaseNotesDirective(rst.Directive):
os.path.join(conf.reporoot, notesdir), os.path.join(conf.reporoot, notesdir),
branch or 'current branch')) branch or 'current branch'))
ldr = loader.Loader(conf) with loader.Loader(conf) as ldr:
if version_opt is not None: if version_opt is not None:
versions = [ versions = [
v.strip() v.strip()
for v in version_opt.split(',') for v in version_opt.split(',')
] ]
else: else:
versions = ldr.versions versions = ldr.versions
LOG.info('got versions %s' % (versions,)) LOG.info('got versions %s' % (versions,))
text = formatter.format_report( text = formatter.format_report(
ldr, ldr,
conf, conf,
versions, versions,
title=title, title=title,
branch=branch, branch=branch,
) )
source_name = '<%s %s>' % (__name__, branch or 'current branch') source_name = '<%s %s>' % (__name__, branch or 'current branch')
result = statemachine.ViewList() result = statemachine.ViewList()
for line_num, line in enumerate(text.splitlines(), 1): for line_num, line in enumerate(text.splitlines(), 1):

View File

@ -49,6 +49,13 @@ class TestFormatterBase(base.TestCase):
ignore_cache=False, ignore_cache=False,
) )
def tearDown(self):
# we don't need to worry about closing this after since we're not
# actually using a real Git repo here (see the mock above), but we'll
# do so to enforce the contract
self.ldr.close()
super().tearDown()
class TestFormatter(TestFormatterBase): class TestFormatter(TestFormatterBase):

View File

@ -65,8 +65,8 @@ class TestValidate(base.TestCase):
This is a single string. It should be converted to a list. This is a single string. It should be converted to a list.
""")) """))
self.assertIsInstance(note_bodies['issues'], str) self.assertIsInstance(note_bodies['issues'], str)
ldr = self._make_loader(note_bodies) with self._make_loader(note_bodies) as ldr:
parse_results = ldr.parse_note_file('note1', None) parse_results = ldr.parse_note_file('note1', None)
self.assertIsInstance(parse_results['issues'], list) self.assertIsInstance(parse_results['issues'], list)
def test_invalid_note_with_prelude_as_list(self): def test_invalid_note_with_prelude_as_list(self):
@ -75,8 +75,8 @@ class TestValidate(base.TestCase):
- The prelude should not be a list. - The prelude should not be a list.
''')) '''))
self.assertIsInstance(note_bodies['prelude'], list) self.assertIsInstance(note_bodies['prelude'], list)
ldr = self._make_loader(note_bodies) with self._make_loader(note_bodies) as ldr:
ldr.parse_note_file('note1', None) ldr.parse_note_file('note1', None)
self.assertIn('does not parse as a single string', self.logger.output) self.assertIn('does not parse as a single string', self.logger.output)
def test_invalid_note_with_colon_as_dict(self): def test_invalid_note_with_colon_as_dict(self):
@ -86,8 +86,8 @@ class TestValidate(base.TestCase):
- dict: But this is parsed as a mapping (dictionary), which is bad. - dict: But this is parsed as a mapping (dictionary), which is bad.
''')) '''))
self.assertIsInstance(note_bodies['issues'][-1], dict) self.assertIsInstance(note_bodies['issues'][-1], dict)
ldr = self._make_loader(note_bodies) with self._make_loader(note_bodies) as ldr:
ldr.parse_note_file('note1', None) ldr.parse_note_file('note1', None)
self.assertIn('instead of a string', self.logger.output) self.assertIn('instead of a string', self.logger.output)
def test_invalid_note_with_unrecognized_key(self): def test_invalid_note_with_unrecognized_key(self):
@ -98,8 +98,8 @@ class TestValidate(base.TestCase):
This is an issue but we're using an unrecognized section key. This is an issue but we're using an unrecognized section key.
''')) '''))
self.assertIsInstance(note_bodies, dict) self.assertIsInstance(note_bodies, dict)
ldr = self._make_loader(note_bodies) with self._make_loader(note_bodies) as ldr:
ldr.parse_note_file('note1', None) ldr.parse_note_file('note1', None)
self.assertIn( self.assertIn(
'The foobar section of note1 is not a recognized section.', 'The foobar section of note1 is not a recognized section.',
self.logger.output) self.logger.output)
@ -114,8 +114,8 @@ class TestValidate(base.TestCase):
This is an issue but we're missing the top-level 'issues' key. This is an issue but we're missing the top-level 'issues' key.
''')) '''))
self.assertIsInstance(note_bodies, list) self.assertIsInstance(note_bodies, list)
ldr = self._make_loader(note_bodies) with self._make_loader(note_bodies) as ldr:
self.assertRaises(ValueError, ldr.parse_note_file, 'note1', None) self.assertRaises(ValueError, ldr.parse_note_file, 'note1', None)
self.assertIn( self.assertIn(
'does not appear to be structured as a YAML mapping', 'does not appear to be structured as a YAML mapping',
self.logger.output) self.logger.output)

View File

@ -217,8 +217,8 @@ class BasicTest(Base):
def test_non_python_no_tags(self): def test_non_python_no_tags(self):
filename = self._add_notes_file() filename = self._add_notes_file()
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -231,8 +231,8 @@ class BasicTest(Base):
def test_python_no_tags(self): def test_python_no_tags(self):
self._make_python_package() self._make_python_package()
filename = self._add_notes_file() filename = self._add_notes_file()
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -246,8 +246,8 @@ class BasicTest(Base):
filename = self._add_notes_file() filename = self._add_notes_file()
self.repo.add_file('not-a-release-note.txt') self.repo.add_file('not-a-release-note.txt')
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -260,8 +260,8 @@ class BasicTest(Base):
def test_note_commit_tagged(self): def test_note_commit_tagged(self):
filename = self._add_notes_file() filename = self._add_notes_file()
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -274,8 +274,8 @@ class BasicTest(Base):
def test_tag_with_v_prefix(self): def test_tag_with_v_prefix(self):
filename = self._add_notes_file() filename = self._add_notes_file()
self.repo.git('tag', '-s', '-m', 'tag with v prefix', 'v1.0.0') self.repo.git('tag', '-s', '-m', 'tag with v prefix', 'v1.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -289,8 +289,8 @@ class BasicTest(Base):
self._make_python_package() self._make_python_package()
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0')
filename = self._add_notes_file() filename = self._add_notes_file()
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -305,8 +305,8 @@ class BasicTest(Base):
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0rc1') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0rc1')
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0')
filename = self._add_notes_file() filename = self._add_notes_file()
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -321,8 +321,8 @@ class BasicTest(Base):
self.repo.add_file('ignore-1.txt') self.repo.add_file('ignore-1.txt')
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0')
self.repo.add_file('ignore-2.txt') self.repo.add_file('ignore-2.txt')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -337,8 +337,8 @@ class BasicTest(Base):
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0')
f1 = self._add_notes_file() f1 = self._add_notes_file()
f2 = self._add_notes_file() f2 = self._add_notes_file()
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -353,8 +353,8 @@ class BasicTest(Base):
f1 = self._add_notes_file(commit=False) f1 = self._add_notes_file(commit=False)
f2 = self._add_notes_file() f2 = self._add_notes_file()
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -370,8 +370,8 @@ class BasicTest(Base):
f1 = self._add_notes_file() f1 = self._add_notes_file()
self.repo.git('tag', '-s', '-m', 'first tag', '2.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '2.0.0')
f2 = self._add_notes_file() f2 = self._add_notes_file()
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -391,8 +391,8 @@ class BasicTest(Base):
f2 = f1.replace('slug1', 'slug2') f2 = f1.replace('slug1', 'slug2')
self.repo.git('mv', f1, f2) self.repo.git('mv', f1, f2)
self.repo.commit('rename note file') self.repo.commit('rename note file')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -411,8 +411,8 @@ class BasicTest(Base):
f2 = f1.replace('slug1', 'slug0') f2 = f1.replace('slug1', 'slug0')
self.repo.git('mv', f1, f2) self.repo.git('mv', f1, f2)
self.repo.commit('rename note file') self.repo.commit('rename note file')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -431,8 +431,8 @@ class BasicTest(Base):
with open(os.path.join(self.reporoot, f1), 'w') as f: with open(os.path.join(self.reporoot, f1), 'w') as f:
f.write('---\npreamble: new contents for file') f.write('---\npreamble: new contents for file')
self.repo.commit('edit note file') self.repo.commit('edit note file')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -451,8 +451,8 @@ class BasicTest(Base):
f2 = f1.replace('slug1', 'slug2') f2 = f1.replace('slug1', 'slug2')
self.repo.git('mv', f1, f2) self.repo.git('mv', f1, f2)
self.repo.commit('rename note file') self.repo.commit('rename note file')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -474,8 +474,8 @@ class BasicTest(Base):
'slug1-0000000000000001') 'slug1-0000000000000001')
self.repo.git('mv', f1, f2) self.repo.git('mv', f1, f2)
self.repo.commit('rename note file') self.repo.commit('rename note file')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -497,8 +497,8 @@ class BasicTest(Base):
self.c.override( self.c.override(
earliest_version='2.0.0', earliest_version='2.0.0',
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -518,8 +518,8 @@ class BasicTest(Base):
self.repo.git('rm', f1) self.repo.git('rm', f1)
self.repo.commit('remove note file') self.repo.commit('remove note file')
self.repo.git('tag', '-s', '-m', 'first tag', '2.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '2.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -546,8 +546,8 @@ class BasicTest(Base):
'--pretty=%H %d', '--pretty=%H %d',
'--name-only') '--name-only')
self.addDetail('git log', text_content(log_results)) self.addDetail('git log', text_content(log_results))
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -573,8 +573,8 @@ class BasicTest(Base):
status_results = self.repo.git('status') status_results = self.repo.git('status')
self.addDetail('git status', text_content(status_results)) self.addDetail('git status', text_content(status_results))
# Now run the scanner # Now run the scanner
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
self.assertEqual( self.assertEqual(
{'*working-copy*': [ {'*working-copy*': [
(os.path.join('releasenotes', 'notes', basename), (os.path.join('releasenotes', 'notes', basename),
@ -598,8 +598,8 @@ class BasicTest(Base):
status_results = self.repo.git('status') status_results = self.repo.git('status')
self.addDetail('git status', text_content(status_results)) self.addDetail('git status', text_content(status_results))
# Now run the scanner # Now run the scanner
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
# Take the staged version of the file, but associate it with # Take the staged version of the file, but associate it with
# tagged version 1.0.0 because the file was added before that # tagged version 1.0.0 because the file was added before that
# version. # version.
@ -623,8 +623,8 @@ class BasicTest(Base):
status_results = self.repo.git('status') status_results = self.repo.git('status')
self.addDetail('git status', text_content(status_results)) self.addDetail('git status', text_content(status_results))
# Now run the scanner # Now run the scanner
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
# Take the staged version of the file, but associate it with # Take the staged version of the file, but associate it with
# tagged version 1.0.0 because the file was added before that # tagged version 1.0.0 because the file was added before that
# version. # version.
@ -647,8 +647,8 @@ class BasicTest(Base):
self.c.override( self.c.override(
earliest_version=None, earliest_version=None,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -672,8 +672,8 @@ class BasicTest(Base):
self.c.override( self.c.override(
earliest_version=None, earliest_version=None,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -698,8 +698,8 @@ class IgnoreTest(Base):
self.c.override( self.c.override(
ignore_notes=[f1], ignore_notes=[f1],
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -717,8 +717,8 @@ class IgnoreTest(Base):
self.c.override( self.c.override(
ignore_notes=[os.path.basename(f1)], ignore_notes=[os.path.basename(f1)],
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -736,8 +736,8 @@ class IgnoreTest(Base):
self.c.override( self.c.override(
ignore_notes=[scanner._get_unique_id(f1)], ignore_notes=[scanner._get_unique_id(f1)],
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -758,8 +758,8 @@ class IgnoreTest(Base):
scanner._get_unique_id(f2), scanner._get_unique_id(f2),
], ],
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -775,8 +775,8 @@ class FileContentsTest(Base):
def test_basic_file(self): def test_basic_file(self):
# Prove that we can get a file we have committed. # Prove that we can get a file we have committed.
f1 = self._add_notes_file(contents='well-known-contents') f1 = self._add_notes_file(contents='well-known-contents')
r = scanner.RenoRepo(self.reporoot) with scanner.RenoRepo(self.reporoot) as r:
contents = r.get_file_at_commit(f1, 'HEAD') contents = r.get_file_at_commit(f1, 'HEAD')
self.assertEqual( self.assertEqual(
b'well-known-contents', b'well-known-contents',
contents, contents,
@ -786,8 +786,8 @@ class FileContentsTest(Base):
# Returns None when the file does not exist at all. # Returns None when the file does not exist at all.
# (we have to commit something, otherwise there is no HEAD) # (we have to commit something, otherwise there is no HEAD)
self._add_notes_file(contents='well-known-contents') self._add_notes_file(contents='well-known-contents')
r = scanner.RenoRepo(self.reporoot) with scanner.RenoRepo(self.reporoot) as r:
contents = r.get_file_at_commit('no-such-dir/no-such-file', 'HEAD') contents = r.get_file_at_commit('no-such-dir/no-such-file', 'HEAD')
self.assertEqual( self.assertEqual(
None, None,
contents, contents,
@ -799,8 +799,8 @@ class FileContentsTest(Base):
with open(os.path.join(self.reporoot, f1), 'w') as f: with open(os.path.join(self.reporoot, f1), 'w') as f:
f.write('new contents for file') f.write('new contents for file')
self.repo.commit('edit note file') self.repo.commit('edit note file')
r = scanner.RenoRepo(self.reporoot) with scanner.RenoRepo(self.reporoot) as r:
contents = r.get_file_at_commit(f1, 'HEAD') contents = r.get_file_at_commit(f1, 'HEAD')
self.assertEqual( self.assertEqual(
b'new contents for file', b'new contents for file',
contents, contents,
@ -813,12 +813,11 @@ class FileContentsTest(Base):
with open(os.path.join(self.reporoot, f1), 'w') as f: with open(os.path.join(self.reporoot, f1), 'w') as f:
f.write('new contents for file') f.write('new contents for file')
self.repo.commit('edit note file') self.repo.commit('edit note file')
self.scanner = scanner.Scanner(self.c) with scanner.RenoRepo(self.reporoot) as r:
r = scanner.RenoRepo(self.reporoot) head = r.head()
head = r.head() parent = r.get_parents(head)[0]
parent = r.get_parents(head)[0] parent = parent.decode('ascii')
parent = parent.decode('ascii') contents = r.get_file_at_commit(f1, parent)
contents = r.get_file_at_commit(f1, parent)
self.assertEqual( self.assertEqual(
b'initial-contents', b'initial-contents',
contents, contents,
@ -830,8 +829,8 @@ class FileContentsTest(Base):
f1 = self._add_notes_file(contents='initial-contents') f1 = self._add_notes_file(contents='initial-contents')
with open(os.path.join(self.reporoot, f1), 'w') as f: with open(os.path.join(self.reporoot, f1), 'w') as f:
f.write('new contents for file') f.write('new contents for file')
r = scanner.RenoRepo(self.reporoot) with scanner.RenoRepo(self.reporoot) as r:
contents = r.get_file_at_commit(f1, 'HEAD') contents = r.get_file_at_commit(f1, 'HEAD')
self.assertEqual( self.assertEqual(
b'initial-contents', b'initial-contents',
contents, contents,
@ -843,8 +842,8 @@ class FileContentsTest(Base):
f1 = self._add_notes_file(contents='initial-contents') f1 = self._add_notes_file(contents='initial-contents')
with open(os.path.join(self.reporoot, f1), 'w') as f: with open(os.path.join(self.reporoot, f1), 'w') as f:
f.write('new contents for file') f.write('new contents for file')
r = scanner.RenoRepo(self.reporoot) with scanner.RenoRepo(self.reporoot) as r:
contents = r.get_file_at_commit(f1, None) contents = r.get_file_at_commit(f1, None)
self.assertEqual( self.assertEqual(
'new contents for file', 'new contents for file',
contents, contents,
@ -858,8 +857,8 @@ class PreReleaseTest(Base):
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0a1') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0a1')
f1 = self._add_notes_file('slug1') f1 = self._add_notes_file('slug1')
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0a2') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0a2')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -875,8 +874,8 @@ class PreReleaseTest(Base):
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0b1') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0b1')
f1 = self._add_notes_file('slug1') f1 = self._add_notes_file('slug1')
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0b2') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0b2')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -892,8 +891,8 @@ class PreReleaseTest(Base):
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0rc1') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0rc1')
f1 = self._add_notes_file('slug1') f1 = self._add_notes_file('slug1')
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0rc2') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0.0rc2')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -909,8 +908,8 @@ class PreReleaseTest(Base):
self.repo.git('tag', '-s', '-m', 'first tag', 'v1.0.0.0a1') self.repo.git('tag', '-s', '-m', 'first tag', 'v1.0.0.0a1')
f1 = self._add_notes_file('slug1') f1 = self._add_notes_file('slug1')
self.repo.git('tag', '-s', '-m', 'first tag', 'v1.0.0.0a2') self.repo.git('tag', '-s', '-m', 'first tag', 'v1.0.0.0a2')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -935,8 +934,8 @@ class PreReleaseTest(Base):
self.c.override( self.c.override(
collapse_pre_releases=True, collapse_pre_releases=True,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -958,8 +957,8 @@ class PreReleaseTest(Base):
self.c.override( self.c.override(
collapse_pre_releases=True, collapse_pre_releases=True,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -984,8 +983,8 @@ class PreReleaseTest(Base):
self.c.override( self.c.override(
collapse_pre_releases=True, collapse_pre_releases=True,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1015,8 +1014,8 @@ class MergeCommitTest(Base):
time.sleep(0.1) # force a delay between commits time.sleep(0.1) # force a delay between commits
self.repo.add_file('ignore-2.txt') self.repo.add_file('ignore-2.txt')
self.repo.git('tag', '-s', '-m', 'second tag', '2.0.0') self.repo.git('tag', '-s', '-m', 'second tag', '2.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1047,8 +1046,8 @@ class MergeCommitTest(Base):
self.repo.git('show') self.repo.git('show')
self.repo.add_file('ignore-2.txt') self.repo.add_file('ignore-2.txt')
self.repo.git('tag', '-s', '-m', 'second tag', '2.0.0') self.repo.git('tag', '-s', '-m', 'second tag', '2.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1081,8 +1080,8 @@ class MergeCommitTest(Base):
self.repo.add_file('ignore-2.txt') self.repo.add_file('ignore-2.txt')
self.repo.git('tag', '-s', '-m', 'third tag', '2.0.0') self.repo.git('tag', '-s', '-m', 'third tag', '2.0.0')
self.repo.add_file('ignore-3.txt') self.repo.add_file('ignore-3.txt')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1119,8 +1118,8 @@ class MergeCommitTest(Base):
self.repo.add_file('ignore-2.txt') self.repo.add_file('ignore-2.txt')
self.repo.git('tag', '-s', '-m', 'third tag', '2.0.0') self.repo.git('tag', '-s', '-m', 'third tag', '2.0.0')
self.repo.add_file('ignore-3.txt') self.repo.add_file('ignore-3.txt')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1181,8 +1180,8 @@ class NullMergeTest(Base):
# The scanner should skip over the null-merge and include the # The scanner should skip over the null-merge and include the
# notes that come before the version being merged in, up to # notes that come before the version being merged in, up to
# the base of the previous branch. # the base of the previous branch.
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1200,8 +1199,8 @@ class NullMergeTest(Base):
self.c.override( self.c.override(
ignore_null_merges=False, ignore_null_merges=False,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1245,6 +1244,10 @@ class BranchBaseTest(Base):
self.repo.git('checkout', 'master') self.repo.git('checkout', 'master')
self.scanner = scanner.Scanner(self.c) self.scanner = scanner.Scanner(self.c)
def tearDown(self):
self.scanner.close()
super().tearDown()
def test_current_branch_no_extra_commits(self): def test_current_branch_no_extra_commits(self):
# checkout the branch and then ask for its base # checkout the branch and then ask for its base
self.repo.git('checkout', 'not-master') self.repo.git('checkout', 'not-master')
@ -1317,8 +1320,8 @@ class BranchTest(Base):
f21 = self._add_notes_file('slug21') f21 = self._add_notes_file('slug21')
log_text = self.repo.git('log', '--decorate') log_text = self.repo.git('log', '--decorate')
self.addDetail('git log', text_content(log_text)) self.addDetail('git log', text_content(log_text))
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1342,8 +1345,8 @@ class BranchTest(Base):
self.c.override( self.c.override(
branch='stable/2', branch='stable/2',
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1370,8 +1373,8 @@ class BranchTest(Base):
self.c.override( self.c.override(
stop_at_branch_base=False, stop_at_branch_base=False,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1407,8 +1410,8 @@ class BranchTest(Base):
branch='stable/4', branch='stable/4',
collapse_pre_releases=False, collapse_pre_releases=False,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1444,8 +1447,8 @@ class BranchTest(Base):
branch='stable/4', branch='stable/4',
collapse_pre_releases=True, collapse_pre_releases=True,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1482,8 +1485,8 @@ class BranchTest(Base):
branch='stable/4', branch='stable/4',
collapse_pre_releases=True, collapse_pre_releases=True,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1516,8 +1519,8 @@ class BranchTest(Base):
self.c.override( self.c.override(
branch='stable/4', branch='stable/4',
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1550,8 +1553,8 @@ class BranchTest(Base):
self.c.override( self.c.override(
branch='stable/4', branch='stable/4',
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1582,8 +1585,8 @@ class BranchTest(Base):
self.c.override( self.c.override(
branch='stable/4', branch='stable/4',
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1599,8 +1602,8 @@ class BranchTest(Base):
self.repo.git('checkout', '2.0.0') self.repo.git('checkout', '2.0.0')
self.repo.git('checkout', '-b', 'stable/2') self.repo.git('checkout', '-b', 'stable/2')
self.repo.git('checkout', 'master') self.repo.git('checkout', 'master')
scanner1 = scanner.Scanner(self.c) with scanner.Scanner(self.c) as scanner1:
head1 = scanner1._get_ref('stable/2') head1 = scanner1._get_ref('stable/2')
self.assertIsNotNone(head1) self.assertIsNotNone(head1)
print('head1', head1) print('head1', head1)
# Create a second repository by cloning the first. # Create a second repository by cloning the first.
@ -1626,8 +1629,8 @@ class BranchTest(Base):
cwd=reporoot2, cwd=reporoot2,
)) ))
c2 = config.Config(reporoot2) c2 = config.Config(reporoot2)
scanner2 = scanner.Scanner(c2) with scanner.Scanner(c2) as scanner2:
head2 = scanner2._get_ref('origin/stable/2') head2 = scanner2._get_ref('origin/stable/2')
self.assertIsNotNone(head2) self.assertIsNotNone(head2)
self.assertEqual(head1, head2) self.assertEqual(head1, head2)
@ -1635,8 +1638,8 @@ class BranchTest(Base):
self.repo.git('checkout', '2.0.0') self.repo.git('checkout', '2.0.0')
self.repo.git('checkout', '-b', 'stable/2') self.repo.git('checkout', '-b', 'stable/2')
self.repo.git('checkout', 'master') self.repo.git('checkout', 'master')
scanner1 = scanner.Scanner(self.c) with scanner.Scanner(self.c) as scanner1:
head1 = scanner1._get_ref('stable/2') head1 = scanner1._get_ref('stable/2')
self.assertIsNotNone(head1) self.assertIsNotNone(head1)
print('head1', head1) print('head1', head1)
# Create a second repository by cloning the first. # Create a second repository by cloning the first.
@ -1662,8 +1665,8 @@ class BranchTest(Base):
cwd=reporoot2, cwd=reporoot2,
)) ))
c2 = config.Config(reporoot2) c2 = config.Config(reporoot2)
scanner2 = scanner.Scanner(c2) with scanner.Scanner(c2) as scanner2:
head2 = scanner2._get_ref('stable/2') head2 = scanner2._get_ref('stable/2')
self.assertIsNotNone(head2) self.assertIsNotNone(head2)
self.assertEqual(head1, head2) self.assertEqual(head1, head2)
@ -1680,8 +1683,8 @@ class BranchTest(Base):
self.c.override( self.c.override(
earliest_version=None, earliest_version=None,
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
raw_results = self.scanner.get_notes_by_version() raw_results = s.get_notes_by_version()
results = { results = {
k: [f for (f, n) in v] k: [f for (f, n) in v]
for (k, v) in raw_results.items() for (k, v) in raw_results.items()
@ -1720,6 +1723,10 @@ class ScanStopPointPrereleaseVersionsTest(Base):
self.repo.git('tag', '-s', '-m', 'third tag', '2.0.0') self.repo.git('tag', '-s', '-m', 'third tag', '2.0.0')
self.repo.git('checkout', 'master') self.repo.git('checkout', 'master')
def tearDown(self):
self.scanner.close()
super().tearDown()
def test_beta_collapse(self): def test_beta_collapse(self):
self.assertEqual( self.assertEqual(
'1.0.0.0rc1', '1.0.0.0rc1',
@ -1808,6 +1815,10 @@ class ScanStopPointRegularVersionsTest(Base):
self.repo.git('tag', '-s', '-m', 'third tag', '2.0.2') self.repo.git('tag', '-s', '-m', 'third tag', '2.0.2')
self.repo.git('checkout', 'master') self.repo.git('checkout', 'master')
def tearDown(self):
self.scanner.close()
super().tearDown()
def test_invalid_earliest_version(self): def test_invalid_earliest_version(self):
self.assertIsNone( self.assertIsNone(
self.scanner._find_scan_stop_point( self.scanner._find_scan_stop_point(
@ -1850,45 +1861,43 @@ class GetRefTest(Base):
self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '1.0.0')
self.repo.git('branch', 'stable/foo') self.repo.git('branch', 'stable/foo')
self.repo.git('tag', 'bar-eol') self.repo.git('tag', 'bar-eol')
self.scanner = scanner.Scanner(self.c)
def tearDown(self):
self.scanner.close()
super().tearDown()
def test_signed_tag(self): def test_signed_tag(self):
self.scanner = scanner.Scanner(self.c)
ref = self.scanner._get_ref('1.0.0') ref = self.scanner._get_ref('1.0.0')
expected = self.scanner._repo.head() expected = self.scanner._repo.head()
self.assertEqual(expected, ref) self.assertEqual(expected, ref)
def test_unsigned_tag(self): def test_unsigned_tag(self):
self.scanner = scanner.Scanner(self.c)
ref = self.scanner._get_ref('bar-eol') ref = self.scanner._get_ref('bar-eol')
expected = self.scanner._repo.head() expected = self.scanner._repo.head()
self.assertEqual(expected, ref) self.assertEqual(expected, ref)
def test_eol_tag_from_branch(self): def test_eol_tag_from_branch(self):
self.scanner = scanner.Scanner(self.c)
ref = self.scanner._get_ref('stable/bar') ref = self.scanner._get_ref('stable/bar')
expected = self.scanner._repo.head() expected = self.scanner._repo.head()
self.assertEqual(expected, ref) self.assertEqual(expected, ref)
def test_head(self): def test_head(self):
self.scanner = scanner.Scanner(self.c)
ref = self.scanner._get_ref(None) ref = self.scanner._get_ref(None)
expected = self.scanner._repo.head() expected = self.scanner._repo.head()
self.assertEqual(expected, ref) self.assertEqual(expected, ref)
def test_stable_branch(self): def test_stable_branch(self):
self.scanner = scanner.Scanner(self.c)
ref = self.scanner._get_ref('stable/foo') ref = self.scanner._get_ref('stable/foo')
expected = self.scanner._repo.head() expected = self.scanner._repo.head()
self.assertEqual(expected, ref) self.assertEqual(expected, ref)
def test_stable_branch_with_origin_prefix(self): def test_stable_branch_with_origin_prefix(self):
self.scanner = scanner.Scanner(self.c)
ref = self.scanner._get_ref('origin/stable/foo') ref = self.scanner._get_ref('origin/stable/foo')
expected = self.scanner._repo.head() expected = self.scanner._repo.head()
self.assertEqual(expected, ref) self.assertEqual(expected, ref)
def test_no_such_value(self): def test_no_such_value(self):
self.scanner = scanner.Scanner(self.c)
self.assertRaises( self.assertRaises(
ValueError, ValueError,
self.scanner._get_ref, self.scanner._get_ref,
@ -1909,17 +1918,17 @@ class TagsTest(Base):
self.repo.git('tag', '-s', '-m', 'first tag', '3.0.0') self.repo.git('tag', '-s', '-m', 'first tag', '3.0.0')
def test_master(self): def test_master(self):
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
results = self.scanner._get_tags_on_branch(None) results = s._get_tags_on_branch(None)
self.assertEqual( self.assertEqual(
['3.0.0', '2.0.0', '1.0.0'], ['3.0.0', '2.0.0', '1.0.0'],
results, results,
) )
def test_get_ref(self): def test_get_ref(self):
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
ref = self.scanner._get_ref('3.0.0') ref = s._get_ref('3.0.0')
expected = self.scanner._repo.head() expected = s._repo.head()
self.assertEqual(expected, ref) self.assertEqual(expected, ref)
def test_not_master(self): def test_not_master(self):
@ -1928,8 +1937,8 @@ class TagsTest(Base):
self._add_notes_file('slug4') self._add_notes_file('slug4')
self.repo.git('tag', '-s', '-m', 'not on master', '2.0.1') self.repo.git('tag', '-s', '-m', 'not on master', '2.0.1')
self.repo.git('checkout', 'master') self.repo.git('checkout', 'master')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
results = self.scanner._get_tags_on_branch('not-master') results = s._get_tags_on_branch('not-master')
self.assertEqual( self.assertEqual(
['2.0.1', '2.0.0', '1.0.0'], ['2.0.1', '2.0.0', '1.0.0'],
results, results,
@ -1938,8 +1947,8 @@ class TagsTest(Base):
def test_unsigned(self): def test_unsigned(self):
self._add_notes_file('slug4') self._add_notes_file('slug4')
self.repo.git('tag', '-m', 'first tag', '4.0.0') self.repo.git('tag', '-m', 'first tag', '4.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
results = self.scanner._get_tags_on_branch(None) results = s._get_tags_on_branch(None)
self.assertEqual( self.assertEqual(
['4.0.0', '3.0.0', '2.0.0', '1.0.0'], ['4.0.0', '3.0.0', '2.0.0', '1.0.0'],
results, results,
@ -1948,8 +1957,8 @@ class TagsTest(Base):
def test_tagged_tag_annotated(self): def test_tagged_tag_annotated(self):
time.sleep(1) time.sleep(1)
self.repo.git('tag', '-s', '-m', 'fourth tag', '4.0.0', '3.0.0') self.repo.git('tag', '-s', '-m', 'fourth tag', '4.0.0', '3.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
results = self.scanner._get_tags_on_branch(None) results = s._get_tags_on_branch(None)
self.assertEqual( self.assertEqual(
['3.0.0', '4.0.0', '2.0.0', '1.0.0'], ['3.0.0', '4.0.0', '2.0.0', '1.0.0'],
results, results,
@ -1958,8 +1967,8 @@ class TagsTest(Base):
def test_tagged_tag_lightweight(self): def test_tagged_tag_lightweight(self):
time.sleep(1) time.sleep(1)
self.repo.git('tag', '-m', 'fourth tag', '4.0.0', '3.0.0') self.repo.git('tag', '-m', 'fourth tag', '4.0.0', '3.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
results = self.scanner._get_tags_on_branch(None) results = s._get_tags_on_branch(None)
self.assertEqual( self.assertEqual(
['3.0.0', '4.0.0', '2.0.0', '1.0.0'], ['3.0.0', '4.0.0', '2.0.0', '1.0.0'],
results, results,
@ -1979,8 +1988,8 @@ class VersionTest(Base):
self.repo.git('tag', '-s', '-m', 'third tag', '3.0.0') self.repo.git('tag', '-s', '-m', 'third tag', '3.0.0')
def test_tagged_head(self): def test_tagged_head(self):
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
results = self.scanner._get_current_version(None) results = s._get_current_version(None)
self.assertEqual( self.assertEqual(
'3.0.0', '3.0.0',
results, results,
@ -1988,8 +1997,8 @@ class VersionTest(Base):
def test_head_after_tag(self): def test_head_after_tag(self):
self._add_notes_file('slug4') self._add_notes_file('slug4')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
results = self.scanner._get_current_version(None) results = s._get_current_version(None)
self.assertEqual( self.assertEqual(
'3.0.0-1', '3.0.0-1',
results, results,
@ -2002,8 +2011,8 @@ class VersionTest(Base):
# single second (certainly not a person). # single second (certainly not a person).
time.sleep(1) time.sleep(1)
self.repo.git('tag', '-s', '-m', 'fourth tag', '4.0.0') self.repo.git('tag', '-s', '-m', 'fourth tag', '4.0.0')
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
results = self.scanner._get_current_version(None) results = s._get_current_version(None)
self.assertEqual( self.assertEqual(
'4.0.0', '4.0.0',
results, results,
@ -2452,11 +2461,9 @@ class GetSeriesBranchesTest(Base):
self.repo.add_file('test.txt') self.repo.add_file('test.txt')
def test_none(self): def test_none(self):
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
self.assertEqual( branches = s.get_series_branches()
[], self.assertEqual([], branches)
self.scanner.get_series_branches(),
)
def test_real_branches_sorted_names(self): def test_real_branches_sorted_names(self):
self.repo.git( self.repo.git(
@ -2465,21 +2472,17 @@ class GetSeriesBranchesTest(Base):
self.repo.git( self.repo.git(
'checkout', '-b', 'stable/b', 'checkout', '-b', 'stable/b',
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
self.assertEqual( branches = s.get_series_branches()
['stable/a', 'stable/b'], self.assertEqual(['stable/a', 'stable/b'], branches)
self.scanner.get_series_branches(),
)
def test_eol_tag(self): def test_eol_tag(self):
self.repo.git( self.repo.git(
'tag', '-s', '-m', 'closed branch', 'a-eol', 'tag', '-s', '-m', 'closed branch', 'a-eol',
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
self.assertEqual( branches = s.get_series_branches()
['stable/a'], self.assertEqual(['stable/a'], branches)
self.scanner.get_series_branches(),
)
def test_mix_tag_and_branch(self): def test_mix_tag_and_branch(self):
self.repo.git( self.repo.git(
@ -2488,8 +2491,6 @@ class GetSeriesBranchesTest(Base):
self.repo.git( self.repo.git(
'checkout', '-b', 'stable/b', 'checkout', '-b', 'stable/b',
) )
self.scanner = scanner.Scanner(self.c) with scanner.Scanner(self.c) as s:
self.assertEqual( branches = s.get_series_branches()
['stable/a', 'stable/b'], self.assertEqual(['stable/a', 'stable/b'], branches)
self.scanner.get_series_branches(),
)