Completed
Pull Request — master (#778)
by Michael
01:17
created

test_clone_handles_branch_typo()   B

Complexity

Conditions 3

Size

Total Lines 26

Duplication

Lines 0
Ratio 0 %

Importance

Changes 4
Bugs 0 Features 0
Metric Value
cc 3
dl 0
loc 26
rs 8.8571
c 4
b 0
f 0
1
# -*- coding: utf-8 -*-
2
import os
3
import subprocess
4
5
import pytest
6
7
from cookiecutter import exceptions, vcs
8
9
10
@pytest.fixture
11
def clone_dir(tmpdir):
12
    """Simulates creation of a directory called `clone_dir` inside of `tmpdir`.
13
    Returns a str to said directory.
14
    """
15
    return str(tmpdir.mkdir('clone_dir'))
16
17
18
def test_clone_should_raise_if_vcs_not_installed(mocker, clone_dir):
19
    """In `clone()`, a `VCSNotInstalled` exception should be raised if no VCS
20
    is installed.
21
    """
22
    mocker.patch(
23
        'cookiecutter.vcs.is_vcs_installed',
24
        autospec=True,
25
        return_value=False
26
    )
27
28
    repo_url = 'https://github.com/pytest-dev/cookiecutter-pytest-plugin.git'
29
30
    with pytest.raises(exceptions.VCSNotInstalled):
31
        vcs.clone(repo_url, clone_to_dir=clone_dir)
32
33
34
@pytest.mark.parametrize('which_return, result', [
35
    ('', False),
36
    (None, False),
37
    (False, False),
38
    ('/usr/local/bin/git', True),
39
])
40
def test_is_vcs_installed(mocker, which_return, result):
41
    mocker.patch(
42
        'cookiecutter.vcs.which',
43
        autospec=True,
44
        return_value=which_return
45
    )
46
    assert vcs.is_vcs_installed('git') == result
47
48
49
@pytest.mark.parametrize('repo_type, repo_url, repo_name', [
50
    ('git', 'https://github.com/hello/world.git', 'world'),
51
    ('hg', 'https://bitbucket.org/foo/bar', 'bar'),
52
])
53
def test_clone_should_invoke_git(
54
        mocker, clone_dir, repo_type, repo_url, repo_name):
55
    """When `clone()` is called with a git/hg repo, the corresponding VCS
56
    command should be run via `subprocess.check_output()`.
57
58
    This should take place:
59
    * In the correct dir
60
    * With the correct args.
61
    """
62
    mocker.patch(
63
        'cookiecutter.vcs.is_vcs_installed',
64
        autospec=True,
65
        return_value=True
66
    )
67
68
    mock_subprocess = mocker.patch(
69
        'cookiecutter.vcs.subprocess.check_output',
70
        autospec=True,
71
    )
72
    expected_repo_dir = os.path.normpath(os.path.join(clone_dir, repo_name))
73
74
    branch = 'foobar'
75
76
    repo_dir = vcs.clone(
77
        repo_url,
78
        checkout=branch,
79
        clone_to_dir=clone_dir,
80
        no_input=True
81
    )
82
83
    assert repo_dir == expected_repo_dir
84
85
    mock_subprocess.assert_any_call(
86
        [repo_type, 'clone', repo_url],
87
        cwd=clone_dir,
88
        stderr=subprocess.STDOUT
89
    )
90
    mock_subprocess.assert_any_call(
91
        [repo_type, 'checkout', branch],
92
        cwd=expected_repo_dir,
93
        stderr=subprocess.STDOUT
94
    )
95
96
97
def test_clone_should_abort_if_user_does_not_want_to_reclone(mocker, tmpdir):
98
    """In `clone()`, if user doesn't want to reclone, Cookiecutter should exit
99
    without cloning anything.
100
    """
101
    mocker.patch(
102
        'cookiecutter.vcs.is_vcs_installed',
103
        autospec=True,
104
        return_value=True
105
    )
106
    mocker.patch(
107
        'cookiecutter.vcs.prompt_and_delete_repo',
108
        side_effect=SystemExit,
109
        autospec=True
110
    )
111
    mock_subprocess = mocker.patch(
112
        'cookiecutter.vcs.subprocess.check_output',
113
        autospec=True,
114
    )
115
116
    clone_to_dir = tmpdir.mkdir('clone')
117
118
    # Create repo_dir to trigger prompt_and_delete_repo
119
    clone_to_dir.mkdir('cookiecutter-pytest-plugin')
120
121
    repo_url = 'https://github.com/pytest-dev/cookiecutter-pytest-plugin.git'
122
123
    with pytest.raises(SystemExit):
124
        vcs.clone(repo_url, clone_to_dir=str(clone_to_dir))
125
    assert not mock_subprocess.called
126
127
128
def test_clone_should_rstrip_trailing_slash_in_repo_url(mocker, clone_dir):
129
    """In `clone()`, repo URL's trailing slash should be stripped if one is
130
    present.
131
    """
132
    mocker.patch(
133
        'cookiecutter.vcs.is_vcs_installed',
134
        autospec=True,
135
        return_value=True
136
    )
137
138
    mock_subprocess = mocker.patch(
139
        'cookiecutter.vcs.subprocess.check_output',
140
        autospec=True,
141
    )
142
143
    vcs.clone(
144
        'https://github.com/foo/bar/',
145
        clone_to_dir=clone_dir,
146
        no_input=True
147
    )
148
149
    mock_subprocess.assert_called_once_with(
150
        ['git', 'clone', 'https://github.com/foo/bar'],
151
        cwd=clone_dir,
152
        stderr=subprocess.STDOUT
153
    )
154
155
156
@pytest.mark.parametrize('error_message', [
157
    "fatal: repository 'https://github.com/hackebro/cookiedozer' not found",
158
    'hg: abort: HTTP Error 404: Not Found',
159
])
160
def test_clone_handles_repo_typo(mocker, clone_dir, error_message):
161
    """In `clone()`, repository not found errors should raise an
162
    appropriate exception.
163
    """
164
    mocker.patch(
165
        'cookiecutter.vcs.subprocess.check_output',
166
        autospec=True,
167
        side_effect=[subprocess.CalledProcessError(
168
            -1, 'cmd', output=error_message
169
        )]
170
    )
171
172
    repository_url = 'https://github.com/hackebro/cookiedozer'
173
    with pytest.raises(exceptions.RepositoryNotFound) as err:
174
        vcs.clone(
175
            repository_url,
176
            clone_to_dir=clone_dir,
177
            no_input=True
178
        )
179
180
    assert str(err.value) == (
181
        'The repository {} could not be found, have you made a typo?'
182
    ).format(repository_url)
183
184
185
@pytest.mark.parametrize('error_message', [
186
    "error: pathspec 'unknown_branch' did not match any file(s) known to git.",
187
    "hg: abort: unknown revision 'unknown_branch'!",
188
])
189
def test_clone_handles_branch_typo(mocker, clone_dir, error_message):
190
    """In `clone()`, branch not found errors should raise an
191
    appropriate exception.
192
    """
193
    mocker.patch(
194
        'cookiecutter.vcs.subprocess.check_output',
195
        autospec=True,
196
        side_effect=[subprocess.CalledProcessError(
197
            -1, 'cmd', output=error_message
198
        )]
199
    )
200
201
    repository_url = 'https://github.com/pytest-dev/cookiecutter-pytest-plugin'
202
    with pytest.raises(exceptions.RepositoryCloneFailed) as err:
203
        vcs.clone(
204
            repository_url,
205
            clone_to_dir=clone_dir,
206
            checkout='unknown_branch',
207
            no_input=True
208
        )
209
210
    assert str(err.value) == (
211
        'The unknown_branch branch of repository '
212
        '{} could not found, have you made a typo?'
213
    ).format(repository_url)
214
215
216
def test_clone_unknown_subprocess_error(mocker, clone_dir):
217
    """In `clone()`, unknown subprocess errors should be raised."""
218
    mocker.patch(
219
        'cookiecutter.vcs.subprocess.check_output',
220
        autospec=True,
221
        side_effect=[subprocess.CalledProcessError(
222
            -1, 'cmd', output='Something went wrong'
223
        )]
224
    )
225
226
    with pytest.raises(subprocess.CalledProcessError):
227
        vcs.clone(
228
            'https://github.com/pytest-dev/cookiecutter-pytest-plugin',
229
            clone_to_dir=clone_dir,
230
            no_input=True
231
        )
232