mirror of
https://github.com/pre-commit/pre-commit.git
synced 2026-01-06 09:00:02 -06:00
Refactored how the installer works
This commit is contained in:
@@ -1,7 +1,7 @@
|
|||||||
|
|
||||||
CONFIG_FILE = '.pre-commit-config.yaml'
|
CONFIG_FILE = '.pre-commit-config.yaml'
|
||||||
|
|
||||||
PRE_COMMIT_DIR = '.pre-commit-files'
|
HOOKS_WORKSPACE = '.pre-commit-files'
|
||||||
|
|
||||||
MANIFEST_FILE = 'manifest.yaml'
|
MANIFEST_FILE = 'manifest.yaml'
|
||||||
|
|
||||||
|
|||||||
@@ -1,23 +1,18 @@
|
|||||||
import os
|
import os
|
||||||
import pkg_resources
|
import pkg_resources
|
||||||
|
|
||||||
import pre_commit.constants as C
|
|
||||||
from plumbum import local
|
from plumbum import local
|
||||||
|
|
||||||
|
|
||||||
|
# TODO: optimization: memoize based on local.cwd.getpath()
|
||||||
def get_root():
|
def get_root():
|
||||||
return local['git']['rev-parse', '--show-toplevel']().strip()
|
return local['git']['rev-parse', '--show-toplevel']().strip()
|
||||||
|
|
||||||
|
|
||||||
def get_pre_commit_path():
|
def get_pre_commit_path():
|
||||||
return os.path.join(get_root(), '.git/hooks/pre-commit')
|
return os.path.join(get_root(), '.git/hooks/pre-commit')
|
||||||
|
|
||||||
|
|
||||||
def get_pre_commit_dir_path():
|
|
||||||
return os.path.join(get_root(), C.PRE_COMMIT_DIR)
|
|
||||||
|
|
||||||
def create_pre_commit_package_dir():
|
|
||||||
local.path(get_pre_commit_dir_path()).mkdir()
|
|
||||||
|
|
||||||
def create_pre_commit():
|
def create_pre_commit():
|
||||||
path = get_pre_commit_path()
|
path = get_pre_commit_path()
|
||||||
pre_commit_file = pkg_resources.resource_filename('pre_commit', 'resources/pre-commit.sh')
|
pre_commit_file = pkg_resources.resource_filename('pre_commit', 'resources/pre-commit.sh')
|
||||||
@@ -28,4 +23,6 @@ def remove_pre_commit():
|
|||||||
local.path(get_pre_commit_path()).delete()
|
local.path(get_pre_commit_path()).delete()
|
||||||
|
|
||||||
|
|
||||||
|
def get_head_sha(git_repo_path):
|
||||||
|
with local.cwd(git_repo_path):
|
||||||
|
return (local['git']['rev-parse', 'HEAD'])().strip()
|
||||||
20
pre_commit/hooks_workspace.py
Normal file
20
pre_commit/hooks_workspace.py
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
|
||||||
|
import contextlib
|
||||||
|
import os.path
|
||||||
|
from plumbum import local
|
||||||
|
|
||||||
|
import pre_commit.constants as C
|
||||||
|
from pre_commit import git
|
||||||
|
|
||||||
|
|
||||||
|
def get_pre_commit_dir_path():
|
||||||
|
return os.path.join(git.get_root(), C.HOOKS_WORKSPACE)
|
||||||
|
|
||||||
|
@contextlib.contextmanager
|
||||||
|
def in_hooks_workspace():
|
||||||
|
"""Change into the hooks workspace. If it does not exist create it."""
|
||||||
|
if not os.path.exists(get_pre_commit_dir_path()):
|
||||||
|
local.path(get_pre_commit_dir_path()).mkdir()
|
||||||
|
|
||||||
|
with local.cwd(get_pre_commit_dir_path()):
|
||||||
|
yield
|
||||||
@@ -1,46 +1,44 @@
|
|||||||
import contextlib
|
|
||||||
|
|
||||||
|
import contextlib
|
||||||
from plumbum import local
|
from plumbum import local
|
||||||
from pre_commit import git
|
|
||||||
|
from pre_commit.hooks_workspace import in_hooks_workspace
|
||||||
|
|
||||||
|
|
||||||
class RepoInstaller(object):
|
class RepoInstaller(object):
|
||||||
def __init__(self, git_repo_path, sha):
|
def __init__(self, repo_config):
|
||||||
self.git_repo_path = git_repo_path
|
self.repo_config = repo_config
|
||||||
self.sha = sha
|
|
||||||
|
@property
|
||||||
|
def repo_url(self):
|
||||||
|
return self.repo_config['repo']
|
||||||
|
|
||||||
|
@property
|
||||||
|
def sha(self):
|
||||||
|
return self.repo_config['sha']
|
||||||
|
|
||||||
@contextlib.contextmanager
|
@contextlib.contextmanager
|
||||||
def in_checkout(self):
|
def in_checkout(self):
|
||||||
with local.cwd(git.get_pre_commit_dir_path()):
|
with in_hooks_workspace():
|
||||||
with local.cwd(self.sha):
|
with local.cwd(self.sha):
|
||||||
yield
|
yield
|
||||||
|
|
||||||
def create(self):
|
def create(self):
|
||||||
git.create_pre_commit_package_dir()
|
with in_hooks_workspace():
|
||||||
|
|
||||||
with local.cwd(git.get_pre_commit_dir_path()):
|
|
||||||
if local.path(self.sha).exists():
|
if local.path(self.sha).exists():
|
||||||
# Project already exists, no reason to re-create it
|
# Project already exists, no reason to re-create it
|
||||||
return
|
return
|
||||||
|
|
||||||
local['git']['clone', self.git_repo_path, self.sha]()
|
local['git']['clone', self.repo_url, self.sha]()
|
||||||
with self.in_checkout():
|
with self.in_checkout():
|
||||||
local['git']['checkout', self.sha]()
|
local['git']['checkout', self.sha]()
|
||||||
|
|
||||||
def install(self):
|
def install(self):
|
||||||
|
# Create if we have not already
|
||||||
|
self.create()
|
||||||
# TODO: need to take in the config here and determine if we actually
|
# TODO: need to take in the config here and determine if we actually
|
||||||
# need to run any installers (and what languages to install)
|
# need to run any installers (and what languages to install)
|
||||||
with self.in_checkout():
|
with self.in_checkout():
|
||||||
if local.path('setup.py').exists():
|
if local.path('setup.py').exists():
|
||||||
local['virtualenv']['py_env']()
|
local['virtualenv']['py_env']()
|
||||||
local['bash']['-c', 'source py_env/bin/activate && pip install .']()
|
local['bash']['-c', 'source py_env/bin/activate && pip install .']()
|
||||||
|
|
||||||
|
|
||||||
def create_repo_in_env(git_repo_path, sha):
|
|
||||||
project = RepoInstaller(git_repo_path, sha)
|
|
||||||
project.create()
|
|
||||||
|
|
||||||
def install_pre_commit(git_repo_path, sha):
|
|
||||||
project = RepoInstaller(git_repo_path, sha)
|
|
||||||
project.create()
|
|
||||||
project.install()
|
|
||||||
@@ -1,8 +1,14 @@
|
|||||||
|
from __future__ import absolute_import
|
||||||
|
|
||||||
|
import jsonschema
|
||||||
import pytest
|
import pytest
|
||||||
import pre_commit.constants as C
|
import time
|
||||||
from plumbum import local
|
from plumbum import local
|
||||||
|
|
||||||
|
import pre_commit.constants as C
|
||||||
|
from pre_commit import git
|
||||||
|
from pre_commit.clientlib.validate_config import CONFIG_JSON_SCHEMA
|
||||||
|
|
||||||
|
|
||||||
@pytest.yield_fixture
|
@pytest.yield_fixture
|
||||||
def empty_git_dir(tmpdir):
|
def empty_git_dir(tmpdir):
|
||||||
@@ -11,10 +17,9 @@ def empty_git_dir(tmpdir):
|
|||||||
yield tmpdir.strpath
|
yield tmpdir.strpath
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def add_and_commit():
|
def add_and_commit():
|
||||||
local['git']['add', '.']()
|
local['git']['add', '.']()
|
||||||
local['git']['commit', '-m', 'random commit']()
|
local['git']['commit', '-m', 'random commit {0}'.format(time.time())]()
|
||||||
|
|
||||||
|
|
||||||
@pytest.yield_fixture
|
@pytest.yield_fixture
|
||||||
@@ -42,8 +47,7 @@ hooks:
|
|||||||
|
|
||||||
@pytest.yield_fixture
|
@pytest.yield_fixture
|
||||||
def python_pre_commit_git_repo(dummy_pre_commit_hooks_git_repo):
|
def python_pre_commit_git_repo(dummy_pre_commit_hooks_git_repo):
|
||||||
local.path('setup.py').write(
|
local.path('setup.py').write("""
|
||||||
"""
|
|
||||||
from setuptools import find_packages
|
from setuptools import find_packages
|
||||||
from setuptools import setup
|
from setuptools import setup
|
||||||
|
|
||||||
@@ -53,7 +57,7 @@ setup(
|
|||||||
packages=find_packages('.'),
|
packages=find_packages('.'),
|
||||||
entry_points={
|
entry_points={
|
||||||
'console_scripts': [
|
'console_scripts': [
|
||||||
'entry = foo.main:func'
|
'foo = foo.main:func'
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
@@ -66,15 +70,28 @@ setup(
|
|||||||
|
|
||||||
with local.cwd(foo_module):
|
with local.cwd(foo_module):
|
||||||
local.path('__init__.py').write('')
|
local.path('__init__.py').write('')
|
||||||
local.path('main.py').write(
|
local.path('main.py').write("""
|
||||||
"""
|
|
||||||
|
|
||||||
def func():
|
def func():
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
|
|
||||||
add_and_commit()
|
add_and_commit()
|
||||||
|
|
||||||
yield dummy_pre_commit_hooks_git_repo
|
yield dummy_pre_commit_hooks_git_repo
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def config_for_python_pre_commit_git_repo(python_pre_commit_git_repo):
|
||||||
|
config = {
|
||||||
|
'repo': python_pre_commit_git_repo,
|
||||||
|
'sha': git.get_head_sha(python_pre_commit_git_repo),
|
||||||
|
'hooks': [{
|
||||||
|
'id': 'foo',
|
||||||
|
'files': '*.py',
|
||||||
|
}],
|
||||||
|
}
|
||||||
|
|
||||||
|
jsonschema.validate([config], CONFIG_JSON_SCHEMA)
|
||||||
|
|
||||||
|
return config
|
||||||
@@ -16,7 +16,9 @@ def test_get_root(empty_git_dir):
|
|||||||
|
|
||||||
|
|
||||||
def test_get_pre_commit_path(empty_git_dir):
|
def test_get_pre_commit_path(empty_git_dir):
|
||||||
assert git.get_pre_commit_path() == '{0}/.git/hooks/pre-commit'.format(empty_git_dir)
|
assert git.get_pre_commit_path() == '{0}/.git/hooks/pre-commit'.format(
|
||||||
|
empty_git_dir,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def test_create_pre_commit(empty_git_dir):
|
def test_create_pre_commit(empty_git_dir):
|
||||||
|
|||||||
0
tests/hooks_workspace_test.py
Normal file
0
tests/hooks_workspace_test.py
Normal file
@@ -2,31 +2,46 @@ import os
|
|||||||
|
|
||||||
import jsonschema
|
import jsonschema
|
||||||
import pytest
|
import pytest
|
||||||
from plumbum import local
|
from pre_commit import git
|
||||||
|
|
||||||
import pre_commit.constants as C
|
import pre_commit.constants as C
|
||||||
from pre_commit.clientlib.validate_config import CONFIG_JSON_SCHEMA
|
from pre_commit.clientlib.validate_config import CONFIG_JSON_SCHEMA
|
||||||
from pre_commit.repo_installer import create_repo_in_env
|
from pre_commit.repo_installer import RepoInstaller
|
||||||
from pre_commit.repo_installer import install_pre_commit
|
|
||||||
|
|
||||||
|
|
||||||
def get_sha(git_repo):
|
@pytest.fixture
|
||||||
with local.cwd(git_repo):
|
def dummy_repo_config(dummy_git_repo):
|
||||||
return (local['git']['log', '--format="%H"'] | local['head']['-n1'])().strip('"\n')
|
# This is not a valid config, but it is pretty close
|
||||||
|
return {
|
||||||
|
'repo': dummy_git_repo,
|
||||||
|
'sha': git.get_head_sha(dummy_git_repo),
|
||||||
|
'hooks': [],
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.integration
|
@pytest.mark.integration
|
||||||
def test_create_repo_in_env(empty_git_dir, dummy_git_repo):
|
def test_create_repo_in_env(dummy_repo_config, dummy_git_repo):
|
||||||
sha = get_sha(dummy_git_repo)
|
repo_installer = RepoInstaller(dummy_repo_config)
|
||||||
create_repo_in_env(dummy_git_repo, sha)
|
repo_installer.create()
|
||||||
|
|
||||||
assert os.path.exists(os.path.join(dummy_git_repo, C.PRE_COMMIT_DIR, sha))
|
assert os.path.exists(
|
||||||
|
os.path.join(dummy_git_repo, C.HOOKS_WORKSPACE, repo_installer.sha),
|
||||||
|
)
|
||||||
|
|
||||||
@pytest.mark.integration
|
@pytest.mark.integration
|
||||||
def test_install_python_repo_in_env(empty_git_dir, python_pre_commit_git_repo):
|
def test_install_python_repo_in_env(python_pre_commit_git_repo, config_for_python_pre_commit_git_repo):
|
||||||
sha = get_sha(python_pre_commit_git_repo)
|
repo_installer = RepoInstaller(config_for_python_pre_commit_git_repo)
|
||||||
install_pre_commit(python_pre_commit_git_repo, sha)
|
# TODO: do we need create here?
|
||||||
|
repo_installer.install()
|
||||||
|
|
||||||
assert os.path.exists(os.path.join(python_pre_commit_git_repo, C.PRE_COMMIT_DIR, sha, 'py_env'))
|
assert os.path.exists(
|
||||||
|
os.path.join(
|
||||||
|
python_pre_commit_git_repo,
|
||||||
|
C.HOOKS_WORKSPACE,
|
||||||
|
repo_installer.sha,
|
||||||
|
'py_env',
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
@@ -34,28 +49,14 @@ def simple_config(python_pre_commit_git_repo):
|
|||||||
config = [
|
config = [
|
||||||
{
|
{
|
||||||
'repo': python_pre_commit_git_repo,
|
'repo': python_pre_commit_git_repo,
|
||||||
'sha': get_sha(python_pre_commit_git_repo),
|
'sha': git.get_head_sha(python_pre_commit_git_repo),
|
||||||
'hooks': [
|
'hooks': [
|
||||||
{
|
{
|
||||||
'id': 'foo',
|
'id': 'foo',
|
||||||
'files': '*.py',
|
'files': '*.py',
|
||||||
}
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
jsonschema.validate(config, CONFIG_JSON_SCHEMA)
|
jsonschema.validate(config, CONFIG_JSON_SCHEMA)
|
||||||
return config
|
return config
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.integration
|
|
||||||
def test_install_config(empty_git_dir, python_pre_commit_git_repo, simple_config):
|
|
||||||
for repo in simple_config:
|
|
||||||
install_pre_commit(repo['repo'], repo['sha'])
|
|
||||||
|
|
||||||
assert os.path.exists(
|
|
||||||
os.path.join(
|
|
||||||
python_pre_commit_git_repo,
|
|
||||||
C.PRE_COMMIT_DIR, simple_config[0]['sha'],
|
|
||||||
'py_env',
|
|
||||||
),
|
|
||||||
)
|
|
||||||
Reference in New Issue
Block a user