
Move to use git_remote_push() instead of doing the steps ourselves. We also change to accept a list of refspecs instead of just the one refspec for the push method. As part of this, we no longer error out if the server rejected any updates, as this is a different concern from whether the push itself failed or not. We do still error out if we attempt to push non-ff updates.
302 lines
11 KiB
Python
302 lines
11 KiB
Python
# -*- coding: UTF-8 -*-
|
|
#
|
|
# Copyright 2010-2014 The pygit2 contributors
|
|
#
|
|
# This file is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License, version 2,
|
|
# as published by the Free Software Foundation.
|
|
#
|
|
# In addition to the permissions in the GNU General Public License,
|
|
# the authors give you unlimited permission to link the compiled
|
|
# version of this file into combinations with other programs,
|
|
# and to distribute those combinations without any restriction
|
|
# coming from the use of this file. (The General Public License
|
|
# restrictions do apply in other respects; for example, they cover
|
|
# modification of the file, and distribution when not linked into
|
|
# a combined executable.)
|
|
#
|
|
# This file is distributed in the hope that it will be useful, but
|
|
# WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
# General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; see the file COPYING. If not, write to
|
|
# the Free Software Foundation, 51 Franklin Street, Fifth Floor,
|
|
# Boston, MA 02110-1301, USA.
|
|
|
|
"""Tests for Remote objects."""
|
|
|
|
|
|
import unittest
|
|
import pygit2
|
|
import sys
|
|
from pygit2 import Oid
|
|
from . import utils
|
|
|
|
try:
|
|
import __pypy__
|
|
except ImportError:
|
|
__pypy__ = None
|
|
|
|
REMOTE_NAME = 'origin'
|
|
REMOTE_URL = 'git://github.com/libgit2/pygit2.git'
|
|
REMOTE_FETCHSPEC_SRC = 'refs/heads/*'
|
|
REMOTE_FETCHSPEC_DST = 'refs/remotes/origin/*'
|
|
REMOTE_REPO_OBJECTS = 30
|
|
REMOTE_REPO_BYTES = 2758
|
|
|
|
ORIGIN_REFSPEC = '+refs/heads/*:refs/remotes/origin/*'
|
|
|
|
class RepositoryTest(utils.RepoTestCase):
|
|
def test_remote_create(self):
|
|
name = 'upstream'
|
|
url = 'git://github.com/libgit2/pygit2.git'
|
|
|
|
remote = self.repo.create_remote(name, url)
|
|
|
|
self.assertEqual(type(remote), pygit2.Remote)
|
|
self.assertEqual(name, remote.name)
|
|
self.assertEqual(url, remote.url)
|
|
self.assertEqual(None, remote.push_url)
|
|
|
|
self.assertRaises(ValueError, self.repo.create_remote, *(name, url))
|
|
|
|
def test_remote_delete(self):
|
|
name = 'upstream'
|
|
url = 'git://github.com/libgit2/pygit2.git'
|
|
|
|
self.repo.create_remote(name, url)
|
|
self.assertEqual(2, len(self.repo.remotes))
|
|
remote = self.repo.remotes[1]
|
|
|
|
self.assertEqual(name, remote.name)
|
|
self.repo.remotes.delete(remote.name)
|
|
self.assertEqual(1, len(self.repo.remotes))
|
|
|
|
def test_remote_rename(self):
|
|
remote = self.repo.remotes[0]
|
|
|
|
self.assertEqual(REMOTE_NAME, remote.name)
|
|
problems = self.repo.remotes.rename(remote.name, "new")
|
|
self.assertEqual([], problems)
|
|
self.assertNotEqual('new', remote.name)
|
|
|
|
self.assertRaises(ValueError, self.repo.remotes.rename, '', '')
|
|
self.assertRaises(ValueError, self.repo.remotes.rename, None, None)
|
|
|
|
|
|
def test_remote_set_url(self):
|
|
remote = self.repo.remotes[0]
|
|
|
|
self.assertEqual(REMOTE_URL, remote.url)
|
|
new_url = 'git://github.com/cholin/pygit2.git'
|
|
remote.url = new_url
|
|
self.assertEqual(new_url, remote.url)
|
|
|
|
self.assertRaisesAssign(ValueError, remote, 'url', '')
|
|
|
|
remote.push_url = new_url
|
|
self.assertEqual(new_url, remote.push_url)
|
|
self.assertRaisesAssign(ValueError, remote, 'push_url', '')
|
|
|
|
|
|
def test_refspec(self):
|
|
remote = self.repo.remotes[0]
|
|
|
|
self.assertEqual(remote.refspec_count, 1)
|
|
refspec = remote.get_refspec(0)
|
|
self.assertEqual(refspec.src, REMOTE_FETCHSPEC_SRC)
|
|
self.assertEqual(refspec.dst, REMOTE_FETCHSPEC_DST)
|
|
self.assertEqual(True, refspec.force)
|
|
self.assertEqual(ORIGIN_REFSPEC, refspec.string)
|
|
|
|
self.assertEqual(list, type(remote.fetch_refspecs))
|
|
self.assertEqual(1, len(remote.fetch_refspecs))
|
|
self.assertEqual(ORIGIN_REFSPEC, remote.fetch_refspecs[0])
|
|
|
|
self.assertTrue(refspec.src_matches('refs/heads/master'))
|
|
self.assertTrue(refspec.dst_matches('refs/remotes/origin/master'))
|
|
self.assertEqual('refs/remotes/origin/master', refspec.transform('refs/heads/master'))
|
|
self.assertEqual('refs/heads/master', refspec.rtransform('refs/remotes/origin/master'))
|
|
|
|
self.assertEqual(list, type(remote.push_refspecs))
|
|
self.assertEqual(0, len(remote.push_refspecs))
|
|
|
|
push_specs = remote.push_refspecs
|
|
self.assertEqual(list, type(push_specs))
|
|
self.assertEqual(0, len(push_specs))
|
|
|
|
remote.fetch_refspecs = ['+refs/*:refs/remotes/*']
|
|
self.assertEqual('+refs/*:refs/remotes/*', remote.fetch_refspecs[0])
|
|
|
|
fetch_specs = remote.fetch_refspecs
|
|
self.assertEqual(list, type(fetch_specs))
|
|
self.assertEqual(1, len(fetch_specs))
|
|
self.assertEqual('+refs/*:refs/remotes/*', fetch_specs[0])
|
|
|
|
remote.fetch_refspecs = ['+refs/*:refs/remotes/*',
|
|
'+refs/test/*:refs/test/remotes/*']
|
|
self.assertEqual('+refs/*:refs/remotes/*', remote.fetch_refspecs[0])
|
|
self.assertEqual('+refs/test/*:refs/test/remotes/*',
|
|
remote.fetch_refspecs[1])
|
|
|
|
remote.push_refspecs = ['+refs/*:refs/remotes/*',
|
|
'+refs/test/*:refs/test/remotes/*']
|
|
|
|
self.assertRaises(TypeError, setattr, remote, 'push_refspecs',
|
|
'+refs/*:refs/*')
|
|
self.assertRaises(TypeError, setattr, remote, 'fetch_refspecs',
|
|
'+refs/*:refs/*')
|
|
self.assertRaises(TypeError, setattr, remote, 'fetch_refspecs',
|
|
['+refs/*:refs/*', 5])
|
|
|
|
self.assertEqual('+refs/*:refs/remotes/*', remote.push_refspecs[0])
|
|
self.assertEqual('+refs/test/*:refs/test/remotes/*',
|
|
remote.push_refspecs[1])
|
|
|
|
|
|
def test_remote_list(self):
|
|
self.assertEqual(1, len(self.repo.remotes))
|
|
remote = self.repo.remotes[0]
|
|
self.assertEqual(REMOTE_NAME, remote.name)
|
|
self.assertEqual(REMOTE_URL, remote.url)
|
|
|
|
name = 'upstream'
|
|
url = 'git://github.com/libgit2/pygit2.git'
|
|
remote = self.repo.create_remote(name, url)
|
|
self.assertTrue(remote.name in [x.name for x in self.repo.remotes])
|
|
|
|
def test_remote_collection(self):
|
|
remote = self.repo.remotes['origin']
|
|
self.assertEqual(REMOTE_NAME, remote.name)
|
|
self.assertEqual(REMOTE_URL, remote.url)
|
|
|
|
with self.assertRaises(KeyError):
|
|
self.repo.remotes['upstream']
|
|
|
|
name = 'upstream'
|
|
url = 'git://github.com/libgit2/pygit2.git'
|
|
remote = self.repo.remotes.create(name, url)
|
|
self.assertTrue(remote.name in [x.name for x in self.repo.remotes])
|
|
|
|
|
|
def test_remote_save(self):
|
|
remote = self.repo.remotes[0]
|
|
remote.url = 'http://example.com/test.git'
|
|
remote.save()
|
|
|
|
self.assertEqual('http://example.com/test.git',
|
|
self.repo.remotes[0].url)
|
|
|
|
@unittest.skipIf(__pypy__ is not None, "skip refcounts checks in pypy")
|
|
def test_remote_refcount(self):
|
|
start = sys.getrefcount(self.repo)
|
|
remote = self.repo.remotes[0]
|
|
del remote
|
|
end = sys.getrefcount(self.repo)
|
|
self.assertEqual(start, end)
|
|
|
|
def test_add_refspec(self):
|
|
remote = self.repo.create_remote('test_add_refspec', REMOTE_URL)
|
|
remote.add_push('refs/heads/*:refs/heads/test_refspec/*')
|
|
self.assertEqual('refs/heads/*:refs/heads/test_refspec/*',
|
|
remote.push_refspecs[0])
|
|
remote.add_fetch('+refs/heads/*:refs/remotes/test_refspec/*')
|
|
self.assertEqual('+refs/heads/*:refs/remotes/test_refspec/*',
|
|
remote.fetch_refspecs[1])
|
|
|
|
def test_remote_callback_typecheck(self):
|
|
remote = self.repo.remotes[0]
|
|
remote.progress = 5
|
|
self.assertRaises(TypeError, remote, 'fetch')
|
|
|
|
remote = self.repo.remotes[0]
|
|
remote.transfer_progress = 5
|
|
self.assertRaises(TypeError, remote, 'fetch')
|
|
|
|
remote = self.repo.remotes[0]
|
|
remote.update_tips = 5
|
|
self.assertRaises(TypeError, remote, 'fetch')
|
|
|
|
|
|
|
|
class EmptyRepositoryTest(utils.EmptyRepoTestCase):
|
|
def test_fetch(self):
|
|
remote = self.repo.remotes[0]
|
|
stats = remote.fetch()
|
|
self.assertEqual(stats.received_bytes, REMOTE_REPO_BYTES)
|
|
self.assertEqual(stats.indexed_objects, REMOTE_REPO_OBJECTS)
|
|
self.assertEqual(stats.received_objects, REMOTE_REPO_OBJECTS)
|
|
|
|
def test_transfer_progress(self):
|
|
self.tp = None
|
|
def tp_cb(stats):
|
|
self.tp = stats
|
|
|
|
remote = self.repo.remotes[0]
|
|
remote.transfer_progress = tp_cb
|
|
stats = remote.fetch()
|
|
self.assertEqual(stats.received_bytes, self.tp.received_bytes)
|
|
self.assertEqual(stats.indexed_objects, self.tp.indexed_objects)
|
|
self.assertEqual(stats.received_objects, self.tp.received_objects)
|
|
|
|
def test_update_tips(self):
|
|
remote = self.repo.remotes[0]
|
|
self.i = 0
|
|
self.tips = [('refs/remotes/origin/master', Oid(hex='0'*40),
|
|
Oid(hex='784855caf26449a1914d2cf62d12b9374d76ae78')),
|
|
('refs/tags/root', Oid(hex='0'*40),
|
|
Oid(hex='3d2962987c695a29f1f80b6c3aa4ec046ef44369'))]
|
|
|
|
def ut_cb(name, old, new):
|
|
self.assertEqual(self.tips[self.i], (name, old, new))
|
|
self.i += 1
|
|
|
|
remote.update_tips = ut_cb
|
|
remote.fetch()
|
|
|
|
class PushTestCase(unittest.TestCase):
|
|
def setUp(self):
|
|
self.origin_ctxtmgr = utils.TemporaryRepository(('git', 'testrepo.git'))
|
|
self.clone_ctxtmgr = utils.TemporaryRepository(('git', 'testrepo.git'))
|
|
self.origin = pygit2.Repository(self.origin_ctxtmgr.__enter__())
|
|
self.clone = pygit2.Repository(self.clone_ctxtmgr.__enter__())
|
|
self.remote = self.clone.create_remote('origin', self.origin.path)
|
|
|
|
def tearDown(self):
|
|
self.origin_ctxtmgr.__exit__(None, None, None)
|
|
self.clone_ctxtmgr.__exit__(None, None, None)
|
|
|
|
def test_push_fast_forward_commits_to_remote_succeeds(self):
|
|
tip = self.clone[self.clone.head.target]
|
|
oid = self.clone.create_commit(
|
|
'refs/heads/master', tip.author, tip.author, 'empty commit',
|
|
tip.tree.id, [tip.id]
|
|
)
|
|
self.remote.push(['refs/heads/master'])
|
|
self.assertEqual(self.origin[self.origin.head.target].id, oid)
|
|
|
|
def test_push_when_up_to_date_succeeds(self):
|
|
self.remote.push(['refs/heads/master'])
|
|
origin_tip = self.origin[self.origin.head.target].id
|
|
clone_tip = self.clone[self.clone.head.target].id
|
|
self.assertEqual(origin_tip, clone_tip)
|
|
|
|
def test_push_non_fast_forward_commits_to_remote_fails(self):
|
|
tip = self.origin[self.origin.head.target]
|
|
oid = self.origin.create_commit(
|
|
'refs/heads/master', tip.author, tip.author, 'some commit',
|
|
tip.tree.id, [tip.id]
|
|
)
|
|
tip = self.clone[self.clone.head.target]
|
|
oid = self.clone.create_commit(
|
|
'refs/heads/master', tip.author, tip.author, 'other commit',
|
|
tip.tree.id, [tip.id]
|
|
)
|
|
|
|
self.assertRaises(pygit2.GitError, self.remote.push, ['refs/heads/master'])
|
|
|
|
if __name__ == '__main__':
|
|
unittest.main()
|