mirror of
https://github.com/pre-commit/pre-commit.git
synced 2026-02-19 17:14:43 +04:00
Merge pull request #64 from pre-commit/color_63
Wire in color for pre-commit
This commit is contained in:
commit
749615118e
3 changed files with 102 additions and 36 deletions
38
pre_commit/color.py
Normal file
38
pre_commit/color.py
Normal file
|
|
@ -0,0 +1,38 @@
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
|
RED = '\033[41m'
|
||||||
|
GREEN = '\033[42m'
|
||||||
|
NORMAL = '\033[0m'
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidColorSetting(ValueError): pass
|
||||||
|
|
||||||
|
|
||||||
|
def format_color(text, color, use_color):
|
||||||
|
"""Format text with color.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
text - Text to be formatted with color if `use_color`
|
||||||
|
color - The color start string
|
||||||
|
use_color - Whether or not to color
|
||||||
|
"""
|
||||||
|
if not use_color:
|
||||||
|
return text
|
||||||
|
else:
|
||||||
|
return u'{0}{1}{2}'.format(color, text, NORMAL)
|
||||||
|
|
||||||
|
|
||||||
|
def use_color(setting):
|
||||||
|
"""Choose whether to use color based on the command argument.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
setting - Either `auto`, `always`, or `never`
|
||||||
|
"""
|
||||||
|
if setting not in ('auto', 'always', 'never'):
|
||||||
|
raise InvalidColorSetting(setting)
|
||||||
|
|
||||||
|
return (
|
||||||
|
setting == 'always' or
|
||||||
|
(setting == 'auto' and sys.stdout.isatty())
|
||||||
|
)
|
||||||
|
|
@ -5,22 +5,20 @@ import argparse
|
||||||
import subprocess
|
import subprocess
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
|
from pre_commit import color
|
||||||
from pre_commit import commands
|
from pre_commit import commands
|
||||||
from pre_commit import git
|
from pre_commit import git
|
||||||
from pre_commit.runner import Runner
|
from pre_commit.runner import Runner
|
||||||
from pre_commit.util import entry
|
from pre_commit.util import entry
|
||||||
|
|
||||||
|
|
||||||
RED = '\033[41m'
|
|
||||||
GREEN = '\033[42m'
|
|
||||||
NORMAL = '\033[0m'
|
|
||||||
COLS = int(subprocess.Popen(['tput', 'cols'], stdout=subprocess.PIPE).communicate()[0])
|
COLS = int(subprocess.Popen(['tput', 'cols'], stdout=subprocess.PIPE).communicate()[0])
|
||||||
|
|
||||||
PASS_FAIL_LENGTH = 6
|
PASS_FAIL_LENGTH = 6
|
||||||
|
|
||||||
|
|
||||||
def _run_single_hook(runner, repository, hook_id, all_files=False, verbose=False):
|
def _run_single_hook(runner, repository, hook_id, args):
|
||||||
if all_files:
|
if args.all_files:
|
||||||
get_filenames = git.get_all_files_matching
|
get_filenames = git.get_all_files_matching
|
||||||
else:
|
else:
|
||||||
get_filenames = git.get_staged_files_matching
|
get_filenames = git.get_staged_files_matching
|
||||||
|
|
@ -46,54 +44,49 @@ def _run_single_hook(runner, repository, hook_id, all_files=False, verbose=False
|
||||||
output = '\n'.join([stdout, stderr]).strip()
|
output = '\n'.join([stdout, stderr]).strip()
|
||||||
if retcode != repository.hooks[hook_id]['expected_return_value']:
|
if retcode != repository.hooks[hook_id]['expected_return_value']:
|
||||||
retcode = 1
|
retcode = 1
|
||||||
color = RED
|
print_color = color.RED
|
||||||
pass_fail = 'Failed'
|
pass_fail = 'Failed'
|
||||||
else:
|
else:
|
||||||
retcode = 0
|
retcode = 0
|
||||||
color = GREEN
|
print_color = color.GREEN
|
||||||
pass_fail = 'Passed'
|
pass_fail = 'Passed'
|
||||||
|
|
||||||
|
|
||||||
print('{0}{1}{2}'.format(color, pass_fail, NORMAL))
|
print(color.format_color(pass_fail, print_color, args.color))
|
||||||
|
|
||||||
if output and (retcode or verbose):
|
if output and (retcode or args.verbose):
|
||||||
print('\n' + output)
|
print('\n' + output)
|
||||||
|
|
||||||
return retcode
|
return retcode
|
||||||
|
|
||||||
|
|
||||||
def run_hooks(runner, all_files=False, verbose=False):
|
def run_hooks(runner, args):
|
||||||
"""Actually run the hooks."""
|
"""Actually run the hooks."""
|
||||||
retval = 0
|
retval = 0
|
||||||
|
|
||||||
for repo in runner.repositories:
|
for repo in runner.repositories:
|
||||||
for hook_id in repo.hooks:
|
for hook_id in repo.hooks:
|
||||||
retval |= _run_single_hook(
|
retval |= _run_single_hook(runner, repo, hook_id, args)
|
||||||
runner,
|
|
||||||
repo,
|
|
||||||
hook_id,
|
|
||||||
all_files=all_files,
|
|
||||||
verbose=verbose,
|
|
||||||
)
|
|
||||||
|
|
||||||
return retval
|
return retval
|
||||||
|
|
||||||
|
|
||||||
def run_single_hook(runner, hook_id, all_files=False, verbose=False):
|
def run_single_hook(runner, hook_id, args):
|
||||||
for repo in runner.repositories:
|
for repo in runner.repositories:
|
||||||
if hook_id in repo.hooks:
|
if hook_id in repo.hooks:
|
||||||
return _run_single_hook(
|
return _run_single_hook(runner, repo, hook_id, args)
|
||||||
runner,
|
|
||||||
repo,
|
|
||||||
hook_id,
|
|
||||||
all_files=all_files,
|
|
||||||
verbose=verbose,
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
print('No hook with id `{0}`'.format(hook_id))
|
print('No hook with id `{0}`'.format(hook_id))
|
||||||
return 1
|
return 1
|
||||||
|
|
||||||
|
|
||||||
|
def _run(runner, args):
|
||||||
|
if args.hook:
|
||||||
|
return run_single_hook(runner, args.hook, args)
|
||||||
|
else:
|
||||||
|
return run_hooks(runner, args)
|
||||||
|
|
||||||
|
|
||||||
@entry
|
@entry
|
||||||
def run(argv):
|
def run(argv):
|
||||||
parser = argparse.ArgumentParser()
|
parser = argparse.ArgumentParser()
|
||||||
|
|
@ -115,6 +108,10 @@ def run(argv):
|
||||||
help='Run on all the files in the repo.',
|
help='Run on all the files in the repo.',
|
||||||
)
|
)
|
||||||
run.add_argument('--verbose', '-v', action='store_true', default=False)
|
run.add_argument('--verbose', '-v', action='store_true', default=False)
|
||||||
|
run.add_argument(
|
||||||
|
'--color', default='auto', type=color.use_color,
|
||||||
|
help='Whether to use color in output. Defaults to `auto`',
|
||||||
|
)
|
||||||
|
|
||||||
help = subparsers.add_parser('help', help='Show help for a specific command.')
|
help = subparsers.add_parser('help', help='Show help for a specific command.')
|
||||||
help.add_argument('help_cmd', nargs='?', help='Command to show help for.')
|
help.add_argument('help_cmd', nargs='?', help='Command to show help for.')
|
||||||
|
|
@ -135,17 +132,7 @@ def run(argv):
|
||||||
elif args.command == 'autoupdate':
|
elif args.command == 'autoupdate':
|
||||||
return commands.autoupdate(runner)
|
return commands.autoupdate(runner)
|
||||||
elif args.command == 'run':
|
elif args.command == 'run':
|
||||||
if args.hook:
|
return _run(runner, args)
|
||||||
return run_single_hook(
|
|
||||||
runner,
|
|
||||||
args.hook,
|
|
||||||
all_files=args.all_files,
|
|
||||||
verbose=args.verbose,
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
return run_hooks(
|
|
||||||
runner, all_files=args.all_files, verbose=args.verbose,
|
|
||||||
)
|
|
||||||
elif args.command == 'help':
|
elif args.command == 'help':
|
||||||
if args.help_cmd:
|
if args.help_cmd:
|
||||||
parser.parse_args([args.help_cmd, '--help'])
|
parser.parse_args([args.help_cmd, '--help'])
|
||||||
|
|
|
||||||
41
tests/color_test.py
Normal file
41
tests/color_test.py
Normal file
|
|
@ -0,0 +1,41 @@
|
||||||
|
|
||||||
|
import mock
|
||||||
|
import pytest
|
||||||
|
import sys
|
||||||
|
|
||||||
|
from pre_commit.color import format_color
|
||||||
|
from pre_commit.color import GREEN
|
||||||
|
from pre_commit.color import InvalidColorSetting
|
||||||
|
from pre_commit.color import use_color
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(('in_text', 'in_color', 'in_use_color', 'expected'), (
|
||||||
|
('foo', GREEN, True, '{0}foo\033[0m'.format(GREEN)),
|
||||||
|
('foo', GREEN, False, 'foo'),
|
||||||
|
))
|
||||||
|
def test_format_color(in_text, in_color, in_use_color, expected):
|
||||||
|
ret = format_color(in_text, in_color, in_use_color)
|
||||||
|
assert ret == expected
|
||||||
|
|
||||||
|
|
||||||
|
def test_use_color_never():
|
||||||
|
assert use_color('never') is False
|
||||||
|
|
||||||
|
|
||||||
|
def test_use_color_always():
|
||||||
|
assert use_color('always') is True
|
||||||
|
|
||||||
|
|
||||||
|
def test_use_color_no_tty():
|
||||||
|
with mock.patch.object(sys.stdout, 'isatty', return_value=False):
|
||||||
|
assert use_color('auto') is False
|
||||||
|
|
||||||
|
|
||||||
|
def test_use_color_tty():
|
||||||
|
with mock.patch.object(sys.stdout, 'isatty', return_value=True):
|
||||||
|
assert use_color('auto') is True
|
||||||
|
|
||||||
|
|
||||||
|
def test_use_color_raises_if_given_shenanigans():
|
||||||
|
with pytest.raises(InvalidColorSetting):
|
||||||
|
use_color('herpaderp')
|
||||||
Loading…
Add table
Add a link
Reference in a new issue