123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 |
- # test_bisect.py -- Tests for bisect functionality
- # Copyright (C) 2025 Jelmer Vernooij <jelmer@jelmer.uk>
- #
- # SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later
- # Dulwich is dual-licensed under the Apache License, Version 2.0 and the GNU
- # General Public License as public by the Free Software Foundation; version 2.0
- # or (at your option) any later version. You can redistribute it and/or
- # modify it under the terms of either of these two licenses.
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- #
- # You should have received a copy of the licenses; if not, see
- # <http://www.gnu.org/licenses/> for a copy of the GNU General Public License
- # and <http://www.apache.org/licenses/LICENSE-2.0> for a copy of the Apache
- # License, Version 2.0.
- #
- """Tests for bisect functionality."""
- import os
- import shutil
- import tempfile
- from dulwich import porcelain
- from dulwich.bisect import BisectState
- from dulwich.objects import Tree
- from dulwich.tests.utils import make_commit
- from . import TestCase
- class BisectStateTests(TestCase):
- """Tests for BisectState class."""
- def setUp(self):
- self.test_dir = tempfile.mkdtemp()
- self.repo = porcelain.init(self.test_dir)
- def tearDown(self):
- shutil.rmtree(self.test_dir)
- def test_is_active_false(self):
- """Test is_active when no bisect session is active."""
- state = BisectState(self.repo)
- self.assertFalse(state.is_active)
- def test_start_bisect(self):
- """Test starting a bisect session."""
- # Create at least one commit so HEAD exists
- c1 = make_commit(id=b"1" * 40, message=b"initial commit")
- self.repo.object_store.add_object(c1)
- self.repo.refs[b"HEAD"] = c1.id
- self.repo.refs[b"refs/heads/main"] = c1.id
- state = BisectState(self.repo)
- state.start()
- self.assertTrue(state.is_active)
- self.assertTrue(
- os.path.exists(os.path.join(self.repo.controldir(), "BISECT_START"))
- )
- self.assertTrue(
- os.path.exists(os.path.join(self.repo.controldir(), "BISECT_TERMS"))
- )
- self.assertTrue(
- os.path.exists(os.path.join(self.repo.controldir(), "BISECT_NAMES"))
- )
- self.assertTrue(
- os.path.exists(os.path.join(self.repo.controldir(), "BISECT_LOG"))
- )
- def test_start_bisect_no_head(self):
- """Test starting a bisect session when repository has no HEAD."""
- state = BisectState(self.repo)
- with self.assertRaises(ValueError) as cm:
- state.start()
- self.assertIn("Cannot start bisect: repository has no HEAD", str(cm.exception))
- def test_start_bisect_already_active(self):
- """Test starting a bisect session when one is already active."""
- # Create at least one commit so HEAD exists
- c1 = make_commit(id=b"1" * 40, message=b"initial commit")
- self.repo.object_store.add_object(c1)
- self.repo.refs[b"HEAD"] = c1.id
- state = BisectState(self.repo)
- state.start()
- with self.assertRaises(ValueError):
- state.start()
- def test_mark_bad_no_session(self):
- """Test marking bad commit when no session is active."""
- state = BisectState(self.repo)
- with self.assertRaises(ValueError):
- state.mark_bad()
- def test_mark_good_no_session(self):
- """Test marking good commit when no session is active."""
- state = BisectState(self.repo)
- with self.assertRaises(ValueError):
- state.mark_good()
- def test_reset_no_session(self):
- """Test resetting when no session is active."""
- state = BisectState(self.repo)
- with self.assertRaises(ValueError):
- state.reset()
- def test_bisect_workflow(self):
- """Test a complete bisect workflow."""
- # Create some commits
- c1 = make_commit(id=b"1" * 40, message=b"good commit 1")
- c2 = make_commit(id=b"2" * 40, message=b"good commit 2", parents=[b"1" * 40])
- c3 = make_commit(id=b"3" * 40, message=b"bad commit", parents=[b"2" * 40])
- c4 = make_commit(id=b"4" * 40, message=b"bad commit 2", parents=[b"3" * 40])
- # Add commits to object store
- for commit in [c1, c2, c3, c4]:
- self.repo.object_store.add_object(commit)
- # Set HEAD to latest commit
- self.repo.refs[b"HEAD"] = c4.id
- # Start bisect
- state = BisectState(self.repo)
- state.start()
- # Mark bad and good
- state.mark_bad(c4.id)
- state.mark_good(c1.id)
- # Check that refs were created
- self.assertTrue(
- os.path.exists(
- os.path.join(self.repo.controldir(), "refs", "bisect", "bad")
- )
- )
- self.assertTrue(
- os.path.exists(
- os.path.join(
- self.repo.controldir(),
- "refs",
- "bisect",
- f"good-{c1.id.decode('ascii')}",
- )
- )
- )
- # Reset
- state.reset()
- self.assertFalse(state.is_active)
- class BisectPorcelainTests(TestCase):
- """Tests for porcelain bisect functions."""
- def setUp(self):
- self.test_dir = tempfile.mkdtemp()
- self.repo = porcelain.init(self.test_dir)
- # Create tree objects
- tree = Tree()
- self.repo.object_store.add_object(tree)
- # Create some commits with proper trees
- self.c1 = make_commit(id=b"1" * 40, message=b"initial commit", tree=tree.id)
- self.c2 = make_commit(
- id=b"2" * 40, message=b"second commit", parents=[b"1" * 40], tree=tree.id
- )
- self.c3 = make_commit(
- id=b"3" * 40, message=b"third commit", parents=[b"2" * 40], tree=tree.id
- )
- self.c4 = make_commit(
- id=b"4" * 40, message=b"fourth commit", parents=[b"3" * 40], tree=tree.id
- )
- # Add commits to object store
- for commit in [self.c1, self.c2, self.c3, self.c4]:
- self.repo.object_store.add_object(commit)
- # Set HEAD to latest commit
- self.repo.refs[b"HEAD"] = self.c4.id
- self.repo.refs[b"refs/heads/master"] = self.c4.id
- def tearDown(self):
- shutil.rmtree(self.test_dir)
- def test_bisect_start(self):
- """Test bisect_start porcelain function."""
- porcelain.bisect_start(self.test_dir)
- # Check that bisect state files exist
- self.assertTrue(
- os.path.exists(os.path.join(self.repo.controldir(), "BISECT_START"))
- )
- def test_bisect_bad_good(self):
- """Test marking commits as bad and good."""
- porcelain.bisect_start(self.test_dir)
- porcelain.bisect_bad(self.test_dir, self.c4.id.decode("ascii"))
- porcelain.bisect_good(self.test_dir, self.c1.id.decode("ascii"))
- # Check that refs were created
- self.assertTrue(
- os.path.exists(
- os.path.join(self.repo.controldir(), "refs", "bisect", "bad")
- )
- )
- self.assertTrue(
- os.path.exists(
- os.path.join(
- self.repo.controldir(),
- "refs",
- "bisect",
- f"good-{self.c1.id.decode('ascii')}",
- )
- )
- )
- def test_bisect_log(self):
- """Test getting bisect log."""
- porcelain.bisect_start(self.test_dir)
- porcelain.bisect_bad(self.test_dir, self.c4.id.decode("ascii"))
- porcelain.bisect_good(self.test_dir, self.c1.id.decode("ascii"))
- log = porcelain.bisect_log(self.test_dir)
- self.assertIn("git bisect start", log)
- self.assertIn("git bisect bad", log)
- self.assertIn("git bisect good", log)
- def test_bisect_reset(self):
- """Test resetting bisect state."""
- porcelain.bisect_start(self.test_dir)
- porcelain.bisect_bad(self.test_dir)
- porcelain.bisect_good(self.test_dir, self.c1.id.decode("ascii"))
- porcelain.bisect_reset(self.test_dir)
- # Check that bisect state files are removed
- self.assertFalse(
- os.path.exists(os.path.join(self.repo.controldir(), "BISECT_START"))
- )
- self.assertFalse(
- os.path.exists(os.path.join(self.repo.controldir(), "refs", "bisect"))
- )
|