mirror of
https://github.com/pre-commit/pre-commit.git
synced 2026-01-13 12:30:08 -06:00
285 lines
9.1 KiB
Python
285 lines
9.1 KiB
Python
from __future__ import unicode_literals
|
|
|
|
import io
|
|
import json
|
|
import logging
|
|
import os
|
|
import pipes
|
|
import shutil
|
|
import sys
|
|
|
|
import pkg_resources
|
|
from cached_property import cached_property
|
|
from cfgv import apply_defaults
|
|
from cfgv import validate
|
|
|
|
import pre_commit.constants as C
|
|
from pre_commit import five
|
|
from pre_commit import git
|
|
from pre_commit.clientlib import is_local_repo
|
|
from pre_commit.clientlib import is_meta_repo
|
|
from pre_commit.clientlib import load_manifest
|
|
from pre_commit.clientlib import MANIFEST_HOOK_DICT
|
|
from pre_commit.languages.all import languages
|
|
from pre_commit.languages.helpers import environment_dir
|
|
from pre_commit.prefix import Prefix
|
|
|
|
|
|
logger = logging.getLogger('pre_commit')
|
|
|
|
|
|
def _state(additional_deps):
|
|
return {'additional_dependencies': sorted(additional_deps)}
|
|
|
|
|
|
def _state_filename(prefix, venv):
|
|
return prefix.path(
|
|
venv, '.install_state_v' + C.INSTALLED_STATE_VERSION,
|
|
)
|
|
|
|
|
|
def _read_state(prefix, venv):
|
|
filename = _state_filename(prefix, venv)
|
|
if not os.path.exists(filename):
|
|
return None
|
|
else:
|
|
return json.loads(io.open(filename).read())
|
|
|
|
|
|
def _write_state(prefix, venv, state):
|
|
state_filename = _state_filename(prefix, venv)
|
|
staging = state_filename + 'staging'
|
|
with io.open(staging, 'w') as state_file:
|
|
state_file.write(five.to_text(json.dumps(state)))
|
|
# Move the file into place atomically to indicate we've installed
|
|
os.rename(staging, state_filename)
|
|
|
|
|
|
def _installed(prefix, language_name, language_version, additional_deps):
|
|
language = languages[language_name]
|
|
venv = environment_dir(language.ENVIRONMENT_DIR, language_version)
|
|
return (
|
|
venv is None or (
|
|
_read_state(prefix, venv) == _state(additional_deps) and
|
|
language.healthy(prefix, language_version)
|
|
)
|
|
)
|
|
|
|
|
|
def _install_all(venvs, repo_url, store):
|
|
"""Tuple of (prefix, language, version, deps)"""
|
|
def _need_installed():
|
|
return tuple(
|
|
(prefix, language_name, version, deps)
|
|
for prefix, language_name, version, deps in venvs
|
|
if not _installed(prefix, language_name, version, deps)
|
|
)
|
|
|
|
if not _need_installed():
|
|
return
|
|
with store.exclusive_lock():
|
|
# Another process may have already completed this work
|
|
need_installed = _need_installed()
|
|
if not need_installed: # pragma: no cover (race)
|
|
return
|
|
|
|
logger.info(
|
|
'Installing environment for {}.'.format(repo_url),
|
|
)
|
|
logger.info('Once installed this environment will be reused.')
|
|
logger.info('This may take a few minutes...')
|
|
|
|
for prefix, language_name, version, deps in need_installed:
|
|
language = languages[language_name]
|
|
venv = environment_dir(language.ENVIRONMENT_DIR, version)
|
|
|
|
# There's potentially incomplete cleanup from previous runs
|
|
# Clean it up!
|
|
if prefix.exists(venv):
|
|
shutil.rmtree(prefix.path(venv))
|
|
|
|
language.install_environment(prefix, version, deps)
|
|
# Write our state to indicate we're installed
|
|
state = _state(deps)
|
|
_write_state(prefix, venv, state)
|
|
|
|
|
|
def _hook(*hook_dicts):
|
|
ret, rest = dict(hook_dicts[0]), hook_dicts[1:]
|
|
for dct in rest:
|
|
ret.update(dct)
|
|
|
|
version = pkg_resources.parse_version(ret['minimum_pre_commit_version'])
|
|
if version > C.VERSION_PARSED:
|
|
logger.error(
|
|
'The hook `{}` requires pre-commit version {} but version {} '
|
|
'is installed. '
|
|
'Perhaps run `pip install --upgrade pre-commit`.'.format(
|
|
ret['id'], version, C.VERSION_PARSED,
|
|
),
|
|
)
|
|
exit(1)
|
|
|
|
if ret['language_version'] == 'default':
|
|
language = languages[ret['language']]
|
|
ret['language_version'] = language.get_default_version()
|
|
|
|
return ret
|
|
|
|
|
|
def _hook_from_manifest_dct(dct):
|
|
dct = validate(apply_defaults(dct, MANIFEST_HOOK_DICT), MANIFEST_HOOK_DICT)
|
|
dct = _hook(dct)
|
|
return dct
|
|
|
|
|
|
class Repository(object):
|
|
def __init__(self, repo_config, store):
|
|
self.repo_config = repo_config
|
|
self.store = store
|
|
self.__installed = False
|
|
|
|
@classmethod
|
|
def create(cls, config, store):
|
|
if is_local_repo(config):
|
|
return LocalRepository(config, store)
|
|
elif is_meta_repo(config):
|
|
return MetaRepository(config, store)
|
|
else:
|
|
return cls(config, store)
|
|
|
|
@cached_property
|
|
def manifest_hooks(self):
|
|
repo, rev = self.repo_config['repo'], self.repo_config['rev']
|
|
repo_path = self.store.clone(repo, rev)
|
|
manifest_path = os.path.join(repo_path, C.MANIFEST_FILE)
|
|
return {hook['id']: hook for hook in load_manifest(manifest_path)}
|
|
|
|
@cached_property
|
|
def hooks(self):
|
|
for hook in self.repo_config['hooks']:
|
|
if hook['id'] not in self.manifest_hooks:
|
|
logger.error(
|
|
'`{}` is not present in repository {}. '
|
|
'Typo? Perhaps it is introduced in a newer version? '
|
|
'Often `pre-commit autoupdate` fixes this.'.format(
|
|
hook['id'], self.repo_config['repo'],
|
|
),
|
|
)
|
|
exit(1)
|
|
|
|
return tuple(
|
|
(hook['id'], _hook(self.manifest_hooks[hook['id']], hook))
|
|
for hook in self.repo_config['hooks']
|
|
)
|
|
|
|
def _prefix_from_deps(self, language_name, deps):
|
|
repo, rev = self.repo_config['repo'], self.repo_config['rev']
|
|
return Prefix(self.store.clone(repo, rev, deps))
|
|
|
|
def _venvs(self):
|
|
ret = []
|
|
for _, hook in self.hooks:
|
|
language = hook['language']
|
|
version = hook['language_version']
|
|
deps = hook['additional_dependencies']
|
|
ret.append((
|
|
self._prefix_from_deps(language, deps),
|
|
language, version, deps,
|
|
))
|
|
return tuple(ret)
|
|
|
|
def require_installed(self):
|
|
if not self.__installed:
|
|
_install_all(self._venvs(), self.repo_config['repo'], self.store)
|
|
self.__installed = True
|
|
|
|
def run_hook(self, hook, file_args):
|
|
"""Run a hook.
|
|
|
|
:param dict hook:
|
|
:param tuple file_args: all the files to run the hook on
|
|
"""
|
|
self.require_installed()
|
|
language_name = hook['language']
|
|
deps = hook['additional_dependencies']
|
|
prefix = self._prefix_from_deps(language_name, deps)
|
|
return languages[language_name].run_hook(prefix, hook, file_args)
|
|
|
|
|
|
class LocalRepository(Repository):
|
|
def _prefix_from_deps(self, language_name, deps):
|
|
"""local repositories have a prefix per hook"""
|
|
language = languages[language_name]
|
|
# pcre / pygrep / script / system / docker_image do not have
|
|
# environments so they work out of the current directory
|
|
if language.ENVIRONMENT_DIR is None:
|
|
return Prefix(git.get_root())
|
|
else:
|
|
return Prefix(self.store.make_local(deps))
|
|
|
|
@cached_property
|
|
def manifest(self):
|
|
raise NotImplementedError
|
|
|
|
@cached_property
|
|
def hooks(self):
|
|
return tuple(
|
|
(hook['id'], _hook_from_manifest_dct(hook))
|
|
for hook in self.repo_config['hooks']
|
|
)
|
|
|
|
|
|
class MetaRepository(LocalRepository):
|
|
@cached_property
|
|
def manifest_hooks(self):
|
|
# The hooks are imported here to prevent circular imports.
|
|
from pre_commit.meta_hooks import check_hooks_apply
|
|
from pre_commit.meta_hooks import check_useless_excludes
|
|
|
|
def _make_entry(mod):
|
|
"""the hook `entry` is passed through `shlex.split()` by the
|
|
command runner, so to prevent issues with spaces and backslashes
|
|
(on Windows) it must be quoted here.
|
|
"""
|
|
return '{} -m {}'.format(pipes.quote(sys.executable), mod.__name__)
|
|
|
|
meta_hooks = [
|
|
{
|
|
'id': 'check-hooks-apply',
|
|
'name': 'Check hooks apply to the repository',
|
|
'files': C.CONFIG_FILE,
|
|
'language': 'system',
|
|
'entry': _make_entry(check_hooks_apply),
|
|
},
|
|
{
|
|
'id': 'check-useless-excludes',
|
|
'name': 'Check for useless excludes',
|
|
'files': C.CONFIG_FILE,
|
|
'language': 'system',
|
|
'entry': _make_entry(check_useless_excludes),
|
|
},
|
|
]
|
|
|
|
return {
|
|
hook['id']: _hook_from_manifest_dct(hook)
|
|
for hook in meta_hooks
|
|
}
|
|
|
|
@cached_property
|
|
def hooks(self):
|
|
for hook in self.repo_config['hooks']:
|
|
if hook['id'] not in self.manifest_hooks:
|
|
logger.error(
|
|
'`{}` is not a valid meta hook. '
|
|
'Typo? Perhaps it is introduced in a newer version? '
|
|
'Often `pip install --upgrade pre-commit` fixes this.'
|
|
.format(hook['id']),
|
|
)
|
|
exit(1)
|
|
|
|
return tuple(
|
|
(hook['id'], _hook(self.manifest_hooks[hook['id']], hook))
|
|
for hook in self.repo_config['hooks']
|
|
)
|