mirror of
https://github.com/pre-commit/pre-commit.git
synced 2026-02-17 08:14:42 +04:00
Merge pull request #84 from pre-commit/merge_conflict_files_only_31
Merge conflict files only 31
This commit is contained in:
commit
9c35a113ee
7 changed files with 130 additions and 37 deletions
|
|
@ -147,6 +147,8 @@ def clean(runner):
|
||||||
def _run_single_hook(runner, repository, hook_id, args, write):
|
def _run_single_hook(runner, repository, hook_id, args, write):
|
||||||
if args.all_files:
|
if args.all_files:
|
||||||
get_filenames = git.get_all_files_matching
|
get_filenames = git.get_all_files_matching
|
||||||
|
elif git.is_in_merge_conflict():
|
||||||
|
get_filenames = git.get_conflicted_files_matching
|
||||||
else:
|
else:
|
||||||
get_filenames = git.get_staged_files_matching
|
get_filenames = git.get_staged_files_matching
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
import functools
|
import functools
|
||||||
|
import logging
|
||||||
import os
|
import os
|
||||||
import os.path
|
import os.path
|
||||||
import re
|
import re
|
||||||
|
|
@ -7,7 +8,11 @@ from plumbum import local
|
||||||
from pre_commit.util import memoize_by_cwd
|
from pre_commit.util import memoize_by_cwd
|
||||||
|
|
||||||
|
|
||||||
def _get_root_new():
|
logger = logging.getLogger('pre_commit')
|
||||||
|
|
||||||
|
|
||||||
|
@memoize_by_cwd
|
||||||
|
def get_root():
|
||||||
path = os.getcwd()
|
path = os.getcwd()
|
||||||
while len(path) > 1:
|
while len(path) > 1:
|
||||||
if os.path.exists(os.path.join(path, '.git')):
|
if os.path.exists(os.path.join(path, '.git')):
|
||||||
|
|
@ -17,14 +22,33 @@ def _get_root_new():
|
||||||
raise AssertionError('called from outside of the gits')
|
raise AssertionError('called from outside of the gits')
|
||||||
|
|
||||||
|
|
||||||
|
def is_in_merge_conflict():
|
||||||
|
return os.path.exists(os.path.join('.git', 'MERGE_MSG'))
|
||||||
|
|
||||||
|
|
||||||
|
def parse_merge_msg_for_conflicts(merge_msg):
|
||||||
|
# Conflicted files start with tabs
|
||||||
|
return [
|
||||||
|
line.strip() for line in merge_msg.splitlines() if line.startswith('\t')
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
@memoize_by_cwd
|
@memoize_by_cwd
|
||||||
def get_root():
|
def get_conflicted_files():
|
||||||
return _get_root_new()
|
logger.info('Checking merge-conflict files only.')
|
||||||
|
# Need to get the conflicted files from the MERGE_MSG because they could
|
||||||
|
# have resolved the conflict by choosing one side or the other
|
||||||
|
merge_msg = open(os.path.join('.git', 'MERGE_MSG')).read()
|
||||||
|
merge_conflict_filenames = parse_merge_msg_for_conflicts(merge_msg)
|
||||||
|
|
||||||
|
# This will get the rest of the changes made after the merge.
|
||||||
def get_head_sha(git_repo_path):
|
# If they resolved the merge conflict by choosing a mesh of both sides
|
||||||
with local.cwd(git_repo_path):
|
# this will also include the conflicted files
|
||||||
return local['git']['rev-parse', 'HEAD']().strip()
|
tree_hash = local['git']['write-tree']().strip()
|
||||||
|
merge_diff_filenames = local['git'][
|
||||||
|
'diff', '-m', tree_hash, 'HEAD', 'MERGE_HEAD', '--name-only',
|
||||||
|
]().splitlines()
|
||||||
|
return set(merge_conflict_filenames) | set(merge_diff_filenames)
|
||||||
|
|
||||||
|
|
||||||
@memoize_by_cwd
|
@memoize_by_cwd
|
||||||
|
|
@ -57,3 +81,4 @@ def get_files_matching(all_file_list_strategy):
|
||||||
|
|
||||||
get_staged_files_matching = get_files_matching(get_staged_files)
|
get_staged_files_matching = get_files_matching(get_staged_files)
|
||||||
get_all_files_matching = get_files_matching(get_all_files)
|
get_all_files_matching = get_files_matching(get_all_files)
|
||||||
|
get_conflicted_files_matching = get_files_matching(get_conflicted_files)
|
||||||
|
|
|
||||||
|
|
@ -2,6 +2,7 @@ import jsonschema
|
||||||
import os
|
import os
|
||||||
import os.path
|
import os.path
|
||||||
import shutil
|
import shutil
|
||||||
|
from plumbum import local
|
||||||
|
|
||||||
|
|
||||||
TESTING_DIR = os.path.abspath(os.path.dirname(__file__))
|
TESTING_DIR = os.path.abspath(os.path.dirname(__file__))
|
||||||
|
|
@ -29,6 +30,11 @@ def copy_tree_to_path(src_dir, dest_dir):
|
||||||
shutil.copy(srcname, destname)
|
shutil.copy(srcname, destname)
|
||||||
|
|
||||||
|
|
||||||
|
def get_head_sha(dir):
|
||||||
|
with local.cwd(dir):
|
||||||
|
return local['git']['rev-parse', 'HEAD']().strip()
|
||||||
|
|
||||||
|
|
||||||
def is_valid_according_to_schema(obj, schema):
|
def is_valid_according_to_schema(obj, schema):
|
||||||
try:
|
try:
|
||||||
jsonschema.validate(obj, schema)
|
jsonschema.validate(obj, schema)
|
||||||
|
|
|
||||||
|
|
@ -9,15 +9,14 @@ from asottile.ordereddict import OrderedDict
|
||||||
from asottile.yaml import ordered_dump
|
from asottile.yaml import ordered_dump
|
||||||
from plumbum import local
|
from plumbum import local
|
||||||
|
|
||||||
|
|
||||||
import pre_commit.constants as C
|
import pre_commit.constants as C
|
||||||
from pre_commit import commands
|
from pre_commit import commands
|
||||||
from pre_commit import git
|
|
||||||
from pre_commit.clientlib.validate_config import CONFIG_JSON_SCHEMA
|
from pre_commit.clientlib.validate_config import CONFIG_JSON_SCHEMA
|
||||||
from pre_commit.clientlib.validate_config import validate_config_extra
|
from pre_commit.clientlib.validate_config import validate_config_extra
|
||||||
from pre_commit.jsonschema_extensions import apply_defaults
|
from pre_commit.jsonschema_extensions import apply_defaults
|
||||||
from pre_commit.runner import Runner
|
from pre_commit.runner import Runner
|
||||||
from testing.auto_namedtuple import auto_namedtuple
|
from testing.auto_namedtuple import auto_namedtuple
|
||||||
|
from testing.util import get_head_sha
|
||||||
from testing.util import get_resource_path
|
from testing.util import get_resource_path
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -53,7 +52,7 @@ def test_uninstall(empty_git_dir):
|
||||||
def up_to_date_repo(python_hooks_repo):
|
def up_to_date_repo(python_hooks_repo):
|
||||||
config = OrderedDict((
|
config = OrderedDict((
|
||||||
('repo', python_hooks_repo),
|
('repo', python_hooks_repo),
|
||||||
('sha', git.get_head_sha(python_hooks_repo)),
|
('sha', get_head_sha(python_hooks_repo)),
|
||||||
('hooks', [OrderedDict((('id', 'foo'), ('files', '')))]),
|
('hooks', [OrderedDict((('id', 'foo'), ('files', '')))]),
|
||||||
))
|
))
|
||||||
wrapped_config = apply_defaults([config], CONFIG_JSON_SCHEMA)
|
wrapped_config = apply_defaults([config], CONFIG_JSON_SCHEMA)
|
||||||
|
|
@ -90,14 +89,14 @@ def test_autoupdate_up_to_date_repo(up_to_date_repo):
|
||||||
def out_of_date_repo(python_hooks_repo):
|
def out_of_date_repo(python_hooks_repo):
|
||||||
config = OrderedDict((
|
config = OrderedDict((
|
||||||
('repo', python_hooks_repo),
|
('repo', python_hooks_repo),
|
||||||
('sha', git.get_head_sha(python_hooks_repo)),
|
('sha', get_head_sha(python_hooks_repo)),
|
||||||
('hooks', [OrderedDict((('id', 'foo'), ('files', '')))]),
|
('hooks', [OrderedDict((('id', 'foo'), ('files', '')))]),
|
||||||
))
|
))
|
||||||
config_wrapped = apply_defaults([config], CONFIG_JSON_SCHEMA)
|
config_wrapped = apply_defaults([config], CONFIG_JSON_SCHEMA)
|
||||||
validate_config_extra(config_wrapped)
|
validate_config_extra(config_wrapped)
|
||||||
config = config_wrapped[0]
|
config = config_wrapped[0]
|
||||||
local['git']['commit', '--allow-empty', '-m', 'foo']()
|
local['git']['commit', '--allow-empty', '-m', 'foo']()
|
||||||
head_sha = git.get_head_sha(python_hooks_repo)
|
head_sha = get_head_sha(python_hooks_repo)
|
||||||
|
|
||||||
with open(os.path.join(python_hooks_repo, C.CONFIG_FILE), 'w') as file_obj:
|
with open(os.path.join(python_hooks_repo, C.CONFIG_FILE), 'w') as file_obj:
|
||||||
file_obj.write(
|
file_obj.write(
|
||||||
|
|
@ -136,7 +135,7 @@ def test_autoupdate_out_of_date_repo(out_of_date_repo):
|
||||||
def hook_disappearing_repo(python_hooks_repo):
|
def hook_disappearing_repo(python_hooks_repo):
|
||||||
config = OrderedDict((
|
config = OrderedDict((
|
||||||
('repo', python_hooks_repo),
|
('repo', python_hooks_repo),
|
||||||
('sha', git.get_head_sha(python_hooks_repo)),
|
('sha', get_head_sha(python_hooks_repo)),
|
||||||
('hooks', [OrderedDict((('id', 'foo'), ('files', '')))]),
|
('hooks', [OrderedDict((('id', 'foo'), ('files', '')))]),
|
||||||
))
|
))
|
||||||
config_wrapped = apply_defaults([config], CONFIG_JSON_SCHEMA)
|
config_wrapped = apply_defaults([config], CONFIG_JSON_SCHEMA)
|
||||||
|
|
@ -284,26 +283,6 @@ def test_has_unmerged_paths(output, expected):
|
||||||
assert commands._has_unmerged_paths(mock_runner) is expected
|
assert commands._has_unmerged_paths(mock_runner) is expected
|
||||||
|
|
||||||
|
|
||||||
@pytest.yield_fixture
|
|
||||||
def in_merge_conflict(repo_with_passing_hook):
|
|
||||||
local['git']['add', C.CONFIG_FILE]()
|
|
||||||
local['git']['commit', '-m' 'add hooks file']()
|
|
||||||
local['git']['clone', '.', 'foo']()
|
|
||||||
with local.cwd('foo'):
|
|
||||||
local['git']['checkout', 'origin/master', '-b', 'foo']()
|
|
||||||
with open('conflict_file', 'w') as conflict_file:
|
|
||||||
conflict_file.write('herp\nderp\n')
|
|
||||||
local['git']['add', 'conflict_file']()
|
|
||||||
local['git']['commit', '-m', 'conflict_file']()
|
|
||||||
local['git']['checkout', 'origin/master', '-b', 'bar']()
|
|
||||||
with open('conflict_file', 'w') as conflict_file:
|
|
||||||
conflict_file.write('harp\nddrp\n')
|
|
||||||
local['git']['add', 'conflict_file']()
|
|
||||||
local['git']['commit', '-m', 'conflict_file']()
|
|
||||||
local['git']['merge', 'foo'](retcode=None)
|
|
||||||
yield os.path.join(repo_with_passing_hook, 'foo')
|
|
||||||
|
|
||||||
|
|
||||||
def test_merge_conflict(in_merge_conflict):
|
def test_merge_conflict(in_merge_conflict):
|
||||||
ret, printed = _do_run(in_merge_conflict, _get_opts())
|
ret, printed = _do_run(in_merge_conflict, _get_opts())
|
||||||
assert ret == 1
|
assert ret == 1
|
||||||
|
|
|
||||||
|
|
@ -1,16 +1,18 @@
|
||||||
from __future__ import absolute_import
|
from __future__ import absolute_import
|
||||||
|
|
||||||
|
import os
|
||||||
|
import os.path
|
||||||
import pytest
|
import pytest
|
||||||
import time
|
import time
|
||||||
import yaml
|
import yaml
|
||||||
from plumbum import local
|
from plumbum import local
|
||||||
|
|
||||||
import pre_commit.constants as C
|
import pre_commit.constants as C
|
||||||
from pre_commit import git
|
|
||||||
from pre_commit.clientlib.validate_config import CONFIG_JSON_SCHEMA
|
from pre_commit.clientlib.validate_config import CONFIG_JSON_SCHEMA
|
||||||
from pre_commit.clientlib.validate_config import validate_config_extra
|
from pre_commit.clientlib.validate_config import validate_config_extra
|
||||||
from pre_commit.jsonschema_extensions import apply_defaults
|
from pre_commit.jsonschema_extensions import apply_defaults
|
||||||
from testing.util import copy_tree_to_path
|
from testing.util import copy_tree_to_path
|
||||||
|
from testing.util import get_head_sha
|
||||||
from testing.util import get_resource_path
|
from testing.util import get_resource_path
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -78,7 +80,7 @@ def failing_hook_repo(dummy_git_repo):
|
||||||
def _make_config(path, hook_id, file_regex):
|
def _make_config(path, hook_id, file_regex):
|
||||||
config = {
|
config = {
|
||||||
'repo': path,
|
'repo': path,
|
||||||
'sha': git.get_head_sha(path),
|
'sha': get_head_sha(path),
|
||||||
'hooks': [{'id': hook_id, 'files': file_regex}],
|
'hooks': [{'id': hook_id, 'files': file_regex}],
|
||||||
}
|
}
|
||||||
config_wrapped = apply_defaults([config], CONFIG_JSON_SCHEMA)
|
config_wrapped = apply_defaults([config], CONFIG_JSON_SCHEMA)
|
||||||
|
|
@ -126,3 +128,29 @@ def repo_with_passing_hook(config_for_script_hooks_repo, empty_git_dir):
|
||||||
def repo_with_failing_hook(failing_hook_repo, empty_git_dir):
|
def repo_with_failing_hook(failing_hook_repo, empty_git_dir):
|
||||||
_make_repo_from_configs(_make_config(failing_hook_repo, 'failing_hook', ''))
|
_make_repo_from_configs(_make_config(failing_hook_repo, 'failing_hook', ''))
|
||||||
yield empty_git_dir
|
yield empty_git_dir
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.yield_fixture
|
||||||
|
def in_merge_conflict(repo_with_passing_hook):
|
||||||
|
local['git']['add', C.CONFIG_FILE]()
|
||||||
|
local['git']['commit', '-m' 'add hooks file']()
|
||||||
|
local['git']['clone', '.', 'foo']()
|
||||||
|
with local.cwd('foo'):
|
||||||
|
local['git']['checkout', 'origin/master', '-b', 'foo']()
|
||||||
|
with open('conflict_file', 'w') as conflict_file:
|
||||||
|
conflict_file.write('herp\nderp\n')
|
||||||
|
local['git']['add', 'conflict_file']()
|
||||||
|
with open('foo_only_file', 'w') as foo_only_file:
|
||||||
|
foo_only_file.write('foo')
|
||||||
|
local['git']['add', 'foo_only_file']()
|
||||||
|
local['git']['commit', '-m', 'conflict_file']()
|
||||||
|
local['git']['checkout', 'origin/master', '-b', 'bar']()
|
||||||
|
with open('conflict_file', 'w') as conflict_file:
|
||||||
|
conflict_file.write('harp\nddrp\n')
|
||||||
|
local['git']['add', 'conflict_file']()
|
||||||
|
with open('bar_only_file', 'w') as bar_only_file:
|
||||||
|
bar_only_file.write('bar')
|
||||||
|
local['git']['add', 'bar_only_file']()
|
||||||
|
local['git']['commit', '-m', 'conflict_file']()
|
||||||
|
local['git']['merge', 'foo'](retcode=None)
|
||||||
|
yield os.path.join(repo_with_passing_hook, 'foo')
|
||||||
|
|
|
||||||
|
|
@ -14,6 +14,14 @@ def test_get_root(empty_git_dir):
|
||||||
assert git.get_root() == empty_git_dir
|
assert git.get_root() == empty_git_dir
|
||||||
|
|
||||||
|
|
||||||
|
def test_is_in_merge_conflict(empty_git_dir):
|
||||||
|
assert git.is_in_merge_conflict() is False
|
||||||
|
|
||||||
|
|
||||||
|
def test_is_not_in_merge_conflict(in_merge_conflict):
|
||||||
|
assert git.is_in_merge_conflict() is True
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def get_files_matching_func():
|
def get_files_matching_func():
|
||||||
def get_filenames():
|
def get_filenames():
|
||||||
|
|
@ -57,3 +65,48 @@ def test_does_not_include_deleted_fileS(get_files_matching_func):
|
||||||
def test_exclude_removes_files(get_files_matching_func):
|
def test_exclude_removes_files(get_files_matching_func):
|
||||||
ret = get_files_matching_func('', '\\.py$')
|
ret = get_files_matching_func('', '\\.py$')
|
||||||
assert ret == set(['hooks.yaml'])
|
assert ret == set(['hooks.yaml'])
|
||||||
|
|
||||||
|
|
||||||
|
def resolve_conflict():
|
||||||
|
with open('conflict_file', 'w') as conflicted_file:
|
||||||
|
conflicted_file.write('herp\nderp\n')
|
||||||
|
local['git']['add', 'conflict_file']()
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_conflicted_files(in_merge_conflict):
|
||||||
|
resolve_conflict()
|
||||||
|
with open('other_file', 'w') as other_file:
|
||||||
|
other_file.write('oh hai')
|
||||||
|
local['git']['add', 'other_file']()
|
||||||
|
|
||||||
|
ret = set(git.get_conflicted_files())
|
||||||
|
assert ret == set(('conflict_file', 'other_file'))
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_conflicted_files_unstaged_files(in_merge_conflict):
|
||||||
|
# If they for whatever reason did pre-commit run --no-stash during a
|
||||||
|
# conflict
|
||||||
|
resolve_conflict()
|
||||||
|
|
||||||
|
# Make unstaged file.
|
||||||
|
with open('bar_only_file', 'w') as bar_only_file:
|
||||||
|
bar_only_file.write('new contents!\n')
|
||||||
|
|
||||||
|
ret = set(git.get_conflicted_files())
|
||||||
|
assert ret == set(('conflict_file',))
|
||||||
|
|
||||||
|
|
||||||
|
MERGE_MSG = "Merge branch 'foo' into bar\n\nConflicts:\n\tconflict_file\n"
|
||||||
|
OTHER_MERGE_MSG = MERGE_MSG + '\tother_conflict_file\n'
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
('input', 'expected_output'),
|
||||||
|
(
|
||||||
|
(MERGE_MSG, ['conflict_file']),
|
||||||
|
(OTHER_MERGE_MSG, ['conflict_file', 'other_conflict_file']),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
def test_parse_merge_msg_for_conflicts(input, expected_output):
|
||||||
|
ret = git.parse_merge_msg_for_conflicts(input)
|
||||||
|
assert ret == expected_output
|
||||||
|
|
|
||||||
|
|
@ -3,13 +3,13 @@ import os
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
import pre_commit.constants as C
|
import pre_commit.constants as C
|
||||||
from pre_commit import git
|
|
||||||
from pre_commit import repository
|
from pre_commit import repository
|
||||||
from pre_commit.clientlib.validate_config import CONFIG_JSON_SCHEMA
|
from pre_commit.clientlib.validate_config import CONFIG_JSON_SCHEMA
|
||||||
from pre_commit.clientlib.validate_config import validate_config_extra
|
from pre_commit.clientlib.validate_config import validate_config_extra
|
||||||
from pre_commit.jsonschema_extensions import apply_defaults
|
from pre_commit.jsonschema_extensions import apply_defaults
|
||||||
from pre_commit.prefixed_command_runner import PrefixedCommandRunner
|
from pre_commit.prefixed_command_runner import PrefixedCommandRunner
|
||||||
from pre_commit.repository import Repository
|
from pre_commit.repository import Repository
|
||||||
|
from testing.util import get_head_sha
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
|
|
@ -17,7 +17,7 @@ def dummy_repo_config(dummy_git_repo):
|
||||||
# This is not a valid config, but it is pretty close
|
# This is not a valid config, but it is pretty close
|
||||||
return {
|
return {
|
||||||
'repo': dummy_git_repo,
|
'repo': dummy_git_repo,
|
||||||
'sha': git.get_head_sha(dummy_git_repo),
|
'sha': get_head_sha(dummy_git_repo),
|
||||||
'hooks': [],
|
'hooks': [],
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue