Add back in git commit message checking.
Change-Id: If608a117b9584c1138b5fbbdd56599922bdfa49e Reviewed-on: https://review.openstack.org/28219 Reviewed-by: Joe Gordon <jogo@cloudscaling.com> Approved: Sean Dague <sean@dague.net> Reviewed-by: Sean Dague <sean@dague.net> Tested-by: Jenkins
This commit is contained in:
parent
ec4833b206
commit
78cb53f086
121
hacking/core.py
121
hacking/core.py
@ -536,22 +536,32 @@ def hacking_not_in(logical_line):
|
|||||||
yield (logical_line.find('not'), "H902: Use the 'not in' "
|
yield (logical_line.find('not'), "H902: Use the 'not in' "
|
||||||
"operator for collection membership evaluation")
|
"operator for collection membership evaluation")
|
||||||
|
|
||||||
|
_has_run_registry = dict()
|
||||||
|
|
||||||
def once_git_check_commit_title():
|
|
||||||
"""Check git commit messages.
|
|
||||||
|
|
||||||
nova HACKING recommends not referencing a bug or blueprint in first line,
|
class GlobalCheck(object):
|
||||||
it should provide an accurate description of the change
|
"""Base class for checks that should be run only once."""
|
||||||
H801
|
|
||||||
H802 Title limited to 72 chars
|
|
||||||
"""
|
|
||||||
#Get title of most recent commit
|
|
||||||
|
|
||||||
subp = subprocess.Popen(['git', 'log', '--no-merges', '--pretty=%s', '-1'],
|
version = '0.0.1'
|
||||||
stdout=subprocess.PIPE)
|
|
||||||
title = subp.communicate()[0]
|
def run(self):
|
||||||
if subp.returncode:
|
"""Make run a no-op if run() has been called before.
|
||||||
raise Exception("git log failed with code %s" % subp.returncode)
|
|
||||||
|
Store in a global registry the list of checks we've run. If we have
|
||||||
|
run that one before, just skip doing anything the subsequent times.
|
||||||
|
This way, since pep8 is file/line based, we don't wind up re-running
|
||||||
|
a check on a git commit message over and over again.
|
||||||
|
"""
|
||||||
|
global _has_run_registry
|
||||||
|
if self.name not in _has_run_registry.keys():
|
||||||
|
_has_run_registry[self.name] = True
|
||||||
|
ret = self.run_once()
|
||||||
|
if ret is not None:
|
||||||
|
for r in ret:
|
||||||
|
yield ret
|
||||||
|
|
||||||
|
|
||||||
|
class GitCheck(GlobalCheck):
|
||||||
|
|
||||||
#From https://github.com/openstack/openstack-ci-puppet
|
#From https://github.com/openstack/openstack-ci-puppet
|
||||||
# /blob/master/modules/gerrit/manifests/init.pp#L74
|
# /blob/master/modules/gerrit/manifests/init.pp#L74
|
||||||
@ -561,29 +571,66 @@ def once_git_check_commit_title():
|
|||||||
'([Bb]lue[Pp]rint|[Bb][Pp])[\s\#:]*([A-Za-z0-9\\-]+)')
|
'([Bb]lue[Pp]rint|[Bb][Pp])[\s\#:]*([A-Za-z0-9\\-]+)')
|
||||||
GIT_REGEX = re.compile(git_keywords)
|
GIT_REGEX = re.compile(git_keywords)
|
||||||
|
|
||||||
error = False
|
def __init__(self, tree, *args):
|
||||||
#NOTE(jogo) if match regex but over 3 words, acceptable title
|
pass
|
||||||
if GIT_REGEX.search(title) is not None and len(title.split()) <= 3:
|
|
||||||
print ("H801: git commit title ('%s') should provide an accurate "
|
|
||||||
"description of the change, not just a reference to a bug "
|
|
||||||
"or blueprint" % title.strip())
|
|
||||||
error = True
|
|
||||||
# HACKING.rst recommends commit titles 50 chars or less, but enforces
|
|
||||||
# a 72 character limit
|
|
||||||
if len(title.decode('utf-8')) > 72:
|
|
||||||
print ("H802: git commit title ('%s') should be under 50 chars"
|
|
||||||
% title.strip())
|
|
||||||
error = True
|
|
||||||
return error
|
|
||||||
|
|
||||||
@classmethod
|
def _get_commit_title(self):
|
||||||
def add_options(cls, parser):
|
if not os.path.exists('.git'):
|
||||||
parser.add_option('--git-message', action='store_true',
|
return None
|
||||||
default=False, dest='git_message',
|
|
||||||
help="Check for well-formed git commit messages")
|
|
||||||
parser.config_options.append('git-message')
|
|
||||||
|
|
||||||
@classmethod
|
#Get title of most recent commit
|
||||||
def parse_options(cls, options):
|
subp = subprocess.Popen(
|
||||||
if options.git_message:
|
['git', 'log', '--no-merges', '--pretty=%s', '-1'],
|
||||||
once_git_check_commit_title()
|
stdout=subprocess.PIPE)
|
||||||
|
title = subp.communicate()[0]
|
||||||
|
if subp.returncode:
|
||||||
|
raise Exception("git log failed with code %s" % subp.returncode)
|
||||||
|
return title
|
||||||
|
|
||||||
|
|
||||||
|
class OnceGitCheckCommitTitleBug(GitCheck):
|
||||||
|
"""Check git commit messages for bugs.
|
||||||
|
|
||||||
|
nova HACKING recommends not referencing a bug or blueprint in first line,
|
||||||
|
it should provide an accurate description of the change
|
||||||
|
H801
|
||||||
|
"""
|
||||||
|
name = "GitCheckCommitTitleBug"
|
||||||
|
|
||||||
|
def __init__(self, tree, *args):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def run_once(self):
|
||||||
|
title = self._get_commit_title()
|
||||||
|
|
||||||
|
#NOTE(jogo) if match regex but over 3 words, acceptable title
|
||||||
|
if (title and self.GIT_REGEX.search(title) is not None
|
||||||
|
and len(title.split()) <= 3):
|
||||||
|
return(1, 0,
|
||||||
|
"H801: git commit title ('%s') should provide an accurate "
|
||||||
|
"description of the change, not just a reference to a bug "
|
||||||
|
"or blueprint" % title.strip(), self.name)
|
||||||
|
|
||||||
|
|
||||||
|
class OnceGitCheckCommitTitleLength(GitCheck):
|
||||||
|
"""Check git commit message length.
|
||||||
|
|
||||||
|
HACKING.rst recommends commit titles 50 chars or less, but enforces
|
||||||
|
a 72 character limit
|
||||||
|
|
||||||
|
H802 Title limited to 72 chars
|
||||||
|
"""
|
||||||
|
name = "GitCheckCommitTitleLength"
|
||||||
|
|
||||||
|
def __init__(self, tree, *args):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def run_once(self):
|
||||||
|
title = self._get_commit_title()
|
||||||
|
|
||||||
|
if title and len(title.decode('utf-8')) > 72:
|
||||||
|
return(
|
||||||
|
1, 0,
|
||||||
|
"H802: git commit title ('%s') should be under 50 chars"
|
||||||
|
% title.strip(),
|
||||||
|
self.name)
|
||||||
|
@ -37,6 +37,8 @@ flake8.extension =
|
|||||||
H404 = hacking.core:hacking_docstring_multiline_start
|
H404 = hacking.core:hacking_docstring_multiline_start
|
||||||
H601 = hacking.core:hacking_no_cr
|
H601 = hacking.core:hacking_no_cr
|
||||||
H700 = hacking.core:hacking_localization_strings
|
H700 = hacking.core:hacking_localization_strings
|
||||||
|
H801 = hacking.core:OnceGitCheckCommitTitleBug
|
||||||
|
H802 = hacking.core:OnceGitCheckCommitTitleLength
|
||||||
H901 = hacking.core:hacking_is_not
|
H901 = hacking.core:hacking_is_not
|
||||||
H902 = hacking.core:hacking_not_in
|
H902 = hacking.core:hacking_not_in
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user