Implement pre-commit init-templatedir

This commit is contained in:
Anthony Sottile 2019-07-20 19:10:50 -07:00
parent 3def940574
commit 9a52eefc99
7 changed files with 114 additions and 9 deletions

View file

@ -0,0 +1,21 @@
import logging
import os.path
from pre_commit.commands.install_uninstall import install
from pre_commit.util import cmd_output
logger = logging.getLogger('pre_commit')
def init_templatedir(config_file, store, directory, hook_type):
install(
config_file, store, overwrite=True, hook_type=hook_type,
skip_on_missing_config=True, git_dir=directory,
)
_, out, _ = cmd_output('git', 'config', 'init.templateDir', retcode=None)
dest = os.path.realpath(directory)
if os.path.realpath(out.strip()) != dest:
logger.warning('`init.templateDir` not set to the target directory')
logger.warning(
'maybe `git config --global init.templateDir {}`?'.format(dest),
)

View file

@ -34,8 +34,9 @@ TEMPLATE_START = '# start templated\n'
TEMPLATE_END = '# end templated\n' TEMPLATE_END = '# end templated\n'
def _hook_paths(hook_type): def _hook_paths(hook_type, git_dir=None):
pth = os.path.join(git.get_git_dir(), 'hooks', hook_type) git_dir = git_dir if git_dir is not None else git.get_git_dir()
pth = os.path.join(git_dir, 'hooks', hook_type)
return pth, '{}.legacy'.format(pth) return pth, '{}.legacy'.format(pth)
@ -69,7 +70,7 @@ def shebang():
def install( def install(
config_file, store, config_file, store,
overwrite=False, hooks=False, hook_type='pre-commit', overwrite=False, hooks=False, hook_type='pre-commit',
skip_on_missing_conf=False, skip_on_missing_config=False, git_dir=None,
): ):
"""Install the pre-commit hooks.""" """Install the pre-commit hooks."""
if cmd_output('git', 'config', 'core.hooksPath', retcode=None)[1].strip(): if cmd_output('git', 'config', 'core.hooksPath', retcode=None)[1].strip():
@ -79,7 +80,7 @@ def install(
) )
return 1 return 1
hook_path, legacy_path = _hook_paths(hook_type) hook_path, legacy_path = _hook_paths(hook_type, git_dir=git_dir)
mkdirp(os.path.dirname(hook_path)) mkdirp(os.path.dirname(hook_path))
@ -100,7 +101,7 @@ def install(
'CONFIG': config_file, 'CONFIG': config_file,
'HOOK_TYPE': hook_type, 'HOOK_TYPE': hook_type,
'INSTALL_PYTHON': sys.executable, 'INSTALL_PYTHON': sys.executable,
'SKIP_ON_MISSING_CONFIG': skip_on_missing_conf, 'SKIP_ON_MISSING_CONFIG': skip_on_missing_config,
} }
with io.open(hook_path, 'w') as hook_file: with io.open(hook_path, 'w') as hook_file:

View file

@ -12,6 +12,7 @@ from pre_commit import git
from pre_commit.commands.autoupdate import autoupdate from pre_commit.commands.autoupdate import autoupdate
from pre_commit.commands.clean import clean from pre_commit.commands.clean import clean
from pre_commit.commands.gc import gc from pre_commit.commands.gc import gc
from pre_commit.commands.init_templatedir import init_templatedir
from pre_commit.commands.install_uninstall import install from pre_commit.commands.install_uninstall import install
from pre_commit.commands.install_uninstall import install_hooks from pre_commit.commands.install_uninstall import install_hooks
from pre_commit.commands.install_uninstall import uninstall from pre_commit.commands.install_uninstall import uninstall
@ -162,6 +163,20 @@ def main(argv=None):
_add_color_option(gc_parser) _add_color_option(gc_parser)
_add_config_option(gc_parser) _add_config_option(gc_parser)
init_templatedir_parser = subparsers.add_parser(
'init-templatedir',
help=(
'Install hook script in a directory intended for use with '
'`git config init.templateDir`.'
),
)
_add_color_option(init_templatedir_parser)
_add_config_option(init_templatedir_parser)
init_templatedir_parser.add_argument(
'directory', help='The directory in which to write the hook script.',
)
_add_hook_type_option(init_templatedir_parser)
install_parser = subparsers.add_parser( install_parser = subparsers.add_parser(
'install', help='Install the pre-commit script.', 'install', help='Install the pre-commit script.',
) )
@ -282,7 +297,12 @@ def main(argv=None):
args.config, store, args.config, store,
overwrite=args.overwrite, hooks=args.install_hooks, overwrite=args.overwrite, hooks=args.install_hooks,
hook_type=args.hook_type, hook_type=args.hook_type,
skip_on_missing_conf=args.allow_missing_config, skip_on_missing_config=args.allow_missing_config,
)
elif args.command == 'init-templatedir':
return init_templatedir(
args.config, store,
args.directory, hook_type=args.hook_type,
) )
elif args.command == 'install-hooks': elif args.command == 'install-hooks':
return install_hooks(args.config, store) return install_hooks(args.config, store)

View file

@ -0,0 +1,49 @@
import subprocess
import pre_commit.constants as C
from pre_commit.commands.init_templatedir import init_templatedir
from pre_commit.envcontext import envcontext
from pre_commit.util import cmd_output
from testing.fixtures import git_dir
from testing.fixtures import make_consuming_repo
from testing.util import cmd_output_mocked_pre_commit_home
from testing.util import cwd
from testing.util import git_commit
def test_init_templatedir(tmpdir, tempdir_factory, store, cap_out):
target = str(tmpdir.join('tmpl'))
init_templatedir(C.CONFIG_FILE, store, target, hook_type='pre-commit')
lines = cap_out.get().splitlines()
assert lines[0].startswith('pre-commit installed at ')
assert lines[1] == (
'[WARNING] `init.templateDir` not set to the target directory'
)
assert lines[2].startswith(
'[WARNING] maybe `git config --global init.templateDir',
)
with envcontext([('GIT_TEMPLATE_DIR', target)]):
path = make_consuming_repo(tempdir_factory, 'script_hooks_repo')
with cwd(path):
retcode, output, _ = git_commit(
fn=cmd_output_mocked_pre_commit_home,
tempdir_factory=tempdir_factory,
# git commit puts pre-commit to stderr
stderr=subprocess.STDOUT,
)
assert retcode == 0
assert 'Bash hook....' in output
def test_init_templatedir_already_set(tmpdir, tempdir_factory, store, cap_out):
target = str(tmpdir.join('tmpl'))
tmp_git_dir = git_dir(tempdir_factory)
with cwd(tmp_git_dir):
cmd_output('git', 'config', 'init.templateDir', target)
init_templatedir(C.CONFIG_FILE, store, target, hook_type='pre-commit')
lines = cap_out.get().splitlines()
assert len(lines) == 1
assert lines[0].startswith('pre-commit installed at')

View file

@ -735,7 +735,7 @@ def test_install_disallow_missing_config(tempdir_factory, store):
with cwd(path): with cwd(path):
remove_config_from_repo(path) remove_config_from_repo(path)
ret = install( ret = install(
C.CONFIG_FILE, store, overwrite=True, skip_on_missing_conf=False, C.CONFIG_FILE, store, overwrite=True, skip_on_missing_config=False,
) )
assert ret == 0 assert ret == 0
@ -748,7 +748,7 @@ def test_install_allow_missing_config(tempdir_factory, store):
with cwd(path): with cwd(path):
remove_config_from_repo(path) remove_config_from_repo(path)
ret = install( ret = install(
C.CONFIG_FILE, store, overwrite=True, skip_on_missing_conf=True, C.CONFIG_FILE, store, overwrite=True, skip_on_missing_config=True,
) )
assert ret == 0 assert ret == 0
@ -766,7 +766,7 @@ def test_install_temporarily_allow_mising_config(tempdir_factory, store):
with cwd(path): with cwd(path):
remove_config_from_repo(path) remove_config_from_repo(path)
ret = install( ret = install(
C.CONFIG_FILE, store, overwrite=True, skip_on_missing_conf=False, C.CONFIG_FILE, store, overwrite=True, skip_on_missing_config=False,
) )
assert ret == 0 assert ret == 0

View file

@ -11,6 +11,7 @@ import pytest
import six import six
from pre_commit import output from pre_commit import output
from pre_commit.envcontext import envcontext
from pre_commit.logging_handler import logging_handler from pre_commit.logging_handler import logging_handler
from pre_commit.store import Store from pre_commit.store import Store
from pre_commit.util import cmd_output from pre_commit.util import cmd_output
@ -272,3 +273,10 @@ def fake_log_handler():
logger.addHandler(handler) logger.addHandler(handler)
yield handler yield handler
logger.removeHandler(handler) logger.removeHandler(handler)
@pytest.fixture(scope='session', autouse=True)
def set_git_templatedir(tmpdir_factory):
tdir = str(tmpdir_factory.mktemp('git_template_dir'))
with envcontext([('GIT_TEMPLATE_DIR', tdir)]):
yield

View file

@ -140,6 +140,12 @@ def test_try_repo(mock_store_dir):
assert patch.call_count == 1 assert patch.call_count == 1
def test_init_templatedir(mock_store_dir):
with mock.patch.object(main, 'init_templatedir') as patch:
main.main(('init-templatedir', 'tdir'))
assert patch.call_count == 1
def test_help_cmd_in_empty_directory( def test_help_cmd_in_empty_directory(
in_tmpdir, in_tmpdir,
mock_commands, mock_commands,