blob: 3c243020c66f198cb4a8981e9e40a547f75f8c63 [file] [log] [blame] [edit]
#
# Copyright (C) 2024 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# 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.
#
"""Tests for git_utils."""
import os
import unittest
from contextlib import ExitStack
from pathlib import Path
from subprocess import CalledProcessError
from tempfile import TemporaryDirectory
import git_utils
from .gitrepo import GitRepo
class GitRepoTestCase(unittest.TestCase):
"""Common base for tests that operate on a git repo."""
def setUp(self) -> None:
# Local test runs will probably pass without this since the caller
# almost certainly has git configured, but the bots that run the tests
# may not. **Do not** use `git config --global` for this, since that
# will modify the caller's config during local testing.
self._original_env = os.environ.copy()
os.environ["GIT_AUTHOR_NAME"] = "Testy McTestFace"
os.environ["GIT_AUTHOR_EMAIL"] = "test@example.com"
os.environ["GIT_COMMITTER_NAME"] = os.environ["GIT_AUTHOR_NAME"]
os.environ["GIT_COMMITTER_EMAIL"] = os.environ["GIT_AUTHOR_EMAIL"]
with ExitStack() as stack:
temp_dir = TemporaryDirectory() # pylint: disable=consider-using-with
stack.enter_context(temp_dir)
self.addCleanup(stack.pop_all().close)
self.repo = GitRepo(Path(temp_dir.name) / "repo")
def tearDown(self) -> None:
# This isn't trivially `os.environ = self._original_env` because
# os.environ isn't actually a dict, it's an os._Environ, and there isn't
# a good way to construct a new one of those.
os.environ.clear()
os.environ.update(self._original_env)
class IsAncestorTest(GitRepoTestCase):
"""Tests for git_utils.is_ancestor."""
def test_if_commit_is_its_own_ancestor(self) -> None:
"""Tests that False is returned when both commits are the same."""
self.repo.init()
self.repo.commit("Initial commit.", allow_empty=True)
initial_commit = self.repo.head()
assert not git_utils.is_ancestor(self.repo.path, initial_commit, initial_commit)
def test_is_ancestor(self) -> None:
"""Tests that True is returned when the ref is an ancestor."""
self.repo.init()
self.repo.commit("Initial commit.", allow_empty=True)
initial_commit = self.repo.head()
self.repo.commit("Second commit.", allow_empty=True)
second_commit = self.repo.head()
git_utils.is_ancestor(self.repo.path, initial_commit, second_commit)
def test_is_not_ancestor(self) -> None:
"""Tests that False is returned when the ref is not an ancestor."""
self.repo.init()
self.repo.commit("Initial commit.", allow_empty=True)
initial_commit = self.repo.head()
self.repo.commit("Second commit.", allow_empty=True)
second_commit = self.repo.head()
assert not git_utils.is_ancestor(self.repo.path, second_commit, initial_commit)
def test_error(self) -> None:
"""Tests that an error is raised when git encounters an error."""
self.repo.init()
with self.assertRaises(CalledProcessError):
git_utils.is_ancestor(self.repo.path, "not-a-ref", "not-a-ref")
class GetMostRecentTagTest(GitRepoTestCase):
"""Tests for git_utils.get_most_recent_tag."""
def test_find_tag_on_correct_branch(self) -> None:
"""Tests that only tags on the given branch are found."""
self.repo.init("main")
self.repo.commit("Initial commit.", allow_empty=True)
self.repo.tag("v1.0.0")
self.repo.switch_to_new_branch("release-2.0")
self.repo.commit("Second commit.", allow_empty=True)
self.repo.tag("v2.0.0")
self.assertEqual(
git_utils.get_most_recent_tag(self.repo.path, "main"), "v1.0.0"
)
def test_no_tags(self) -> None:
"""Tests that None is returned when the repo has no tags."""
self.repo.init("main")
self.repo.commit("Initial commit.", allow_empty=True)
self.assertIsNone(git_utils.get_most_recent_tag(self.repo.path, "main"))
def test_no_describing_tags(self) -> None:
"""Tests that None is returned when no tags describe the ref."""
self.repo.init("main")
self.repo.commit("Initial commit.", allow_empty=True)
self.repo.switch_to_new_branch("release-2.0")
self.repo.commit("Second commit.", allow_empty=True)
self.repo.tag("v2.0.0")
self.assertIsNone(git_utils.get_most_recent_tag(self.repo.path, "main"))
class DiffTest(GitRepoTestCase):
"""Tests for git_utils.diff."""
def test_git_diff_added_filter(self) -> None:
self.repo.init("main")
self.repo.commit(
"Add README.md", update_files={"README.md": "Hello, world!"}
)
first_commit = self.repo.head()
self.repo.commit(
"Add OWNERS", update_files={"OWNERS": "nobody"}
)
diff = git_utils.diff(self.repo.path, 'A', first_commit)
self.assertIn('OWNERS', diff)
if __name__ == "__main__":
unittest.main(verbosity=2)