from __future__ import absolute_import from __future__ import unicode_literals import io import logging import os import os.path import mock import pytest from pre_commit import five from pre_commit.prefixed_command_runner import PrefixedCommandRunner from pre_commit.runner import Runner from pre_commit.store import Store from pre_commit.util import cmd_output from pre_commit.util import cwd from testing.fixtures import git_dir from testing.fixtures import make_consuming_repo @pytest.yield_fixture def tempdir_factory(tmpdir): class TmpdirFactory(object): def __init__(self): self.tmpdir_count = 0 def get(self): path = tmpdir.join(five.text(self.tmpdir_count)).strpath self.tmpdir_count += 1 os.mkdir(path) return path yield TmpdirFactory() @pytest.yield_fixture def in_tmpdir(tempdir_factory): path = tempdir_factory.get() with cwd(path): yield path def _make_conflict(): cmd_output('git', 'checkout', 'origin/master', '-b', 'foo') with io.open('conflict_file', 'w') as conflict_file: conflict_file.write('herp\nderp\n') cmd_output('git', 'add', 'conflict_file') with io.open('foo_only_file', 'w') as foo_only_file: foo_only_file.write('foo') cmd_output('git', 'add', 'foo_only_file') cmd_output('git', 'commit', '-m', 'conflict_file') cmd_output('git', 'checkout', 'origin/master', '-b', 'bar') with io.open('conflict_file', 'w') as conflict_file: conflict_file.write('harp\nddrp\n') cmd_output('git', 'add', 'conflict_file') with io.open('bar_only_file', 'w') as bar_only_file: bar_only_file.write('bar') cmd_output('git', 'add', 'bar_only_file') cmd_output('git', 'commit', '-m', 'conflict_file') cmd_output('git', 'merge', 'foo', retcode=None) @pytest.yield_fixture def in_merge_conflict(tempdir_factory): path = make_consuming_repo(tempdir_factory, 'script_hooks_repo') with cwd(path): cmd_output('touch', 'dummy') cmd_output('git', 'add', 'dummy') cmd_output('git', 'commit', '-m', 'Add config.') conflict_path = tempdir_factory.get() cmd_output('git', 'clone', path, conflict_path) with cwd(conflict_path): _make_conflict() yield os.path.join(conflict_path) @pytest.yield_fixture def in_conflicting_submodule(tempdir_factory): git_dir_1 = git_dir(tempdir_factory) git_dir_2 = git_dir(tempdir_factory) with cwd(git_dir_2): cmd_output('git', 'commit', '--allow-empty', '-m', 'init!') with cwd(git_dir_1): cmd_output('git', 'submodule', 'add', git_dir_2, 'sub') with cwd(os.path.join(git_dir_1, 'sub')): _make_conflict() yield @pytest.yield_fixture(scope='session', autouse=True) def dont_write_to_home_directory(): """pre_commit.store.Store will by default write to the home directory We'll mock out `Store.get_default_directory` to raise invariantly so we don't construct a `Store` object that writes to our home directory. """ class YouForgotToExplicitlyChooseAStoreDirectory(AssertionError): pass with mock.patch.object( Store, 'get_default_directory', side_effect=YouForgotToExplicitlyChooseAStoreDirectory, ): yield @pytest.yield_fixture def mock_out_store_directory(tempdir_factory): tmpdir = tempdir_factory.get() with mock.patch.object( Store, 'get_default_directory', return_value=tmpdir, ): yield tmpdir @pytest.yield_fixture def store(tempdir_factory): yield Store(os.path.join(tempdir_factory.get(), '.pre-commit')) @pytest.yield_fixture def cmd_runner(tempdir_factory): yield PrefixedCommandRunner(tempdir_factory.get()) @pytest.yield_fixture def runner_with_mocked_store(mock_out_store_directory): yield Runner('/') @pytest.yield_fixture def log_info_mock(): with mock.patch.object(logging.getLogger('pre_commit'), 'info') as mck: yield mck