Skip to content

Commit 4c24d25

Browse files
committed
Add proxy option to clone_repository
Fixes: #1328
1 parent c4b9c2e commit 4c24d25

File tree

5 files changed

+69
-43
lines changed

5 files changed

+69
-43
lines changed

docs/repository.rst

+1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ Functions
2929
>>> repo_path = '/path/to/create/repository'
3030
>>> repo = clone_repository(repo_url, repo_path) # Clones a non-bare repository
3131
>>> repo = clone_repository(repo_url, repo_path, bare=True) # Clones a bare repository
32+
>>> repo = clone_repository(repo_url, repo_path, proxy=True) # Enable automatic proxy detection
3233

3334

3435
.. autofunction:: pygit2.discover_repository

pygit2/__init__.py

+20-12
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,7 @@
4040
from .blame import Blame, BlameHunk
4141
from .blob import BlobIO
4242
from .callbacks import Payload, RemoteCallbacks, CheckoutCallbacks, StashApplyCallbacks
43-
from .callbacks import git_clone_options, git_fetch_options, get_credentials
43+
from .callbacks import git_clone_options, git_fetch_options, git_proxy_options, get_credentials
4444
from .config import Config
4545
from .credentials import *
4646
from .errors import check_error, Passthrough
@@ -146,14 +146,15 @@ def init_repository(
146146

147147

148148
def clone_repository(
149-
url,
150-
path,
151-
bare=False,
152-
repository=None,
153-
remote=None,
154-
checkout_branch=None,
155-
callbacks=None,
156-
depth=0,
149+
url: str,
150+
path: str,
151+
bare: bool = False,
152+
repository: typing.Callable | None = None,
153+
remote: typing.Callable | None = None,
154+
checkout_branch: str | None = None,
155+
callbacks: RemoteCallbacks | None = None,
156+
depth: int = 0,
157+
proxy: None | bool | str = None,
157158
):
158159
"""
159160
Clones a new Git repository from *url* in the given *path*.
@@ -194,6 +195,12 @@ def clone_repository(
194195
If greater than 0, creates a shallow clone with a history truncated to
195196
the specified number of commits.
196197
The default is 0 (full commit history).
198+
proxy : None or True or str
199+
Proxy configuration. Can be one of:
200+
201+
* `None` (the default) to disable proxy usage
202+
* `True` to enable automatic proxy detection
203+
* an url to a proxy (`http://proxy.example.org:3128/`)
197204
"""
198205

199206
if callbacks is None:
@@ -214,9 +221,10 @@ def clone_repository(
214221
opts.checkout_branch = checkout_branch_ref
215222

216223
with git_fetch_options(payload, opts=opts.fetch_opts):
217-
crepo = ffi.new('git_repository **')
218-
err = C.git_clone(crepo, to_bytes(url), to_bytes(path), opts)
219-
payload.check_error(err)
224+
with git_proxy_options(payload, opts=opts.fetch_opts.proxy_opts, proxy=proxy):
225+
crepo = ffi.new('git_repository **')
226+
err = C.git_clone(crepo, to_bytes(url), to_bytes(path), opts)
227+
payload.check_error(err)
220228

221229
# Ok
222230
return Repository._from_c(crepo[0], owned=True)

pygit2/callbacks.py

+22
Original file line numberDiff line numberDiff line change
@@ -369,6 +369,28 @@ def git_fetch_options(payload, opts=None):
369369
payload._stored_exception = None
370370
yield payload
371371

372+
@contextmanager
373+
def git_proxy_options(
374+
payload,
375+
opts,
376+
proxy: None | bool | str = None,
377+
):
378+
if opts is None:
379+
opts = ffi.new('git_proxy_options *')
380+
C.git_proxy_options_init(opts, C.GIT_PROXY_OPTIONS_VERSION)
381+
if proxy is None:
382+
opts.type = C.GIT_PROXY_NONE
383+
elif proxy is True:
384+
opts.type = C.GIT_PROXY_AUTO
385+
elif type(proxy) is str:
386+
opts.type = C.GIT_PROXY_SPECIFIED
387+
# Keep url in memory, otherwise memory is freed and bad things happen
388+
payload.__proxy_url = ffi.new('char[]', to_bytes(proxy))
389+
opts.url = payload.__proxy_url
390+
else:
391+
raise TypeError('Proxy must be None, True, or a string')
392+
yield opts
393+
372394

373395
@contextmanager
374396
def git_push_options(payload, opts=None):

pygit2/remotes.py

+16-31
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@
2828

2929
# Import from pygit2
3030
from ._pygit2 import Oid
31-
from .callbacks import git_fetch_options, git_push_options, git_remote_callbacks
31+
from .callbacks import git_fetch_options, git_push_options, git_proxy_options, git_remote_callbacks
3232
from .enums import FetchPrune
3333
from .errors import check_error
3434
from .ffi import ffi, C
@@ -107,14 +107,12 @@ def connect(self, callbacks=None, direction=C.GIT_DIRECTION_FETCH, proxy=None):
107107
* `True` to enable automatic proxy detection
108108
* an url to a proxy (`http://proxy.example.org:3128/`)
109109
"""
110-
proxy_opts = ffi.new('git_proxy_options *')
111-
C.git_proxy_options_init(proxy_opts, C.GIT_PROXY_OPTIONS_VERSION)
112-
self.__set_proxy(proxy_opts, proxy)
113-
with git_remote_callbacks(callbacks) as payload:
114-
err = C.git_remote_connect(
115-
self._remote, direction, payload.remote_callbacks, proxy_opts, ffi.NULL
116-
)
117-
payload.check_error(err)
110+
with git_proxy_options(self, proxy=proxy) as proxy_opts:
111+
with git_remote_callbacks(callbacks) as payload:
112+
err = C.git_remote_connect(
113+
self._remote, direction, payload.remote_callbacks, proxy_opts, ffi.NULL
114+
)
115+
payload.check_error(err)
118116

119117
def fetch(
120118
self,
@@ -154,10 +152,10 @@ def fetch(
154152
opts = payload.fetch_options
155153
opts.prune = prune
156154
opts.depth = depth
157-
self.__set_proxy(opts.proxy_opts, proxy)
158-
with StrArray(refspecs) as arr:
159-
err = C.git_remote_fetch(self._remote, arr.ptr, opts, to_bytes(message))
160-
payload.check_error(err)
155+
with git_proxy_options(self, payload.fetch_options.proxy_opts, proxy):
156+
with StrArray(refspecs) as arr:
157+
err = C.git_remote_fetch(self._remote, arr.ptr, opts, to_bytes(message))
158+
payload.check_error(err)
161159

162160
return TransferProgress(C.git_remote_stats(self._remote))
163161

@@ -276,24 +274,11 @@ def push(self, specs, callbacks=None, proxy=None, push_options=None, threads=1):
276274
with git_push_options(callbacks) as payload:
277275
opts = payload.push_options
278276
opts.pb_parallelism = threads
279-
self.__set_proxy(opts.proxy_opts, proxy)
280-
with StrArray(specs) as refspecs, StrArray(push_options) as pushopts:
281-
pushopts.assign_to(opts.remote_push_options)
282-
err = C.git_remote_push(self._remote, refspecs.ptr, opts)
283-
payload.check_error(err)
284-
285-
def __set_proxy(self, proxy_opts, proxy):
286-
if proxy is None:
287-
proxy_opts.type = C.GIT_PROXY_NONE
288-
elif proxy is True:
289-
proxy_opts.type = C.GIT_PROXY_AUTO
290-
elif type(proxy) is str:
291-
proxy_opts.type = C.GIT_PROXY_SPECIFIED
292-
# Keep url in memory, otherwise memory is freed and bad things happen
293-
self.__url = ffi.new('char[]', to_bytes(proxy))
294-
proxy_opts.url = self.__url
295-
else:
296-
raise TypeError('Proxy must be None, True, or a string')
277+
with git_proxy_options(self, payload.push_options.proxy_opts, proxy):
278+
with StrArray(specs) as refspecs, StrArray(push_options) as pushopts:
279+
pushopts.assign_to(opts.remote_push_options)
280+
err = C.git_remote_push(self._remote, refspecs.ptr, opts)
281+
payload.check_error(err)
297282

298283

299284
class RemoteCollection:

test/test_repository.py

+10
Original file line numberDiff line numberDiff line change
@@ -736,6 +736,16 @@ def test_clone_with_checkout_branch(barerepo, tmp_path):
736736
assert repo.lookup_reference('HEAD').target == 'refs/heads/test'
737737

738738

739+
@utils.requires_proxy
740+
@utils.requires_network
741+
def test_clone_with_proxy(tmp_path):
742+
url = 'https://github.com/libgit2/TestGitRepository'
743+
repo = clone_repository(
744+
url, tmp_path / 'testrepo-orig.git', proxy=True,
745+
)
746+
assert not repo.is_empty
747+
748+
739749
# FIXME The tests below are commented because they are broken:
740750
#
741751
# - test_clone_push_url: Passes, but does nothing useful.

0 commit comments

Comments
 (0)