mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-25 13:13:48 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			341 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			341 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """Unit tests for script/clang_tidy_hash.py module."""
 | |
| 
 | |
| import hashlib
 | |
| from pathlib import Path
 | |
| import sys
 | |
| from unittest.mock import Mock, patch
 | |
| 
 | |
| import pytest
 | |
| 
 | |
| # Add the script directory to Python path so we can import clang_tidy_hash
 | |
| sys.path.insert(0, str(Path(__file__).parent.parent.parent / "script"))
 | |
| 
 | |
| import clang_tidy_hash  # noqa: E402
 | |
| 
 | |
| 
 | |
| @pytest.mark.parametrize(
 | |
|     ("file_content", "expected"),
 | |
|     [
 | |
|         (
 | |
|             "clang-tidy==18.1.5 # via -r requirements_dev.in\n",
 | |
|             "clang-tidy==18.1.5 # via -r requirements_dev.in",
 | |
|         ),
 | |
|         (
 | |
|             "other-package==1.0\nclang-tidy==17.0.0\nmore-packages==2.0\n",
 | |
|             "clang-tidy==17.0.0",
 | |
|         ),
 | |
|         (
 | |
|             "# comment\nclang-tidy==16.0.0  # some comment\n",
 | |
|             "clang-tidy==16.0.0  # some comment",
 | |
|         ),
 | |
|         ("no-clang-tidy-here==1.0\n", "clang-tidy version not found"),
 | |
|     ],
 | |
| )
 | |
| def test_get_clang_tidy_version_from_requirements(
 | |
|     file_content: str, expected: str
 | |
| ) -> None:
 | |
|     """Test extracting clang-tidy version from various file formats."""
 | |
|     # Mock read_file_lines to return our test content
 | |
|     with patch("clang_tidy_hash.read_file_lines") as mock_read:
 | |
|         mock_read.return_value = file_content.splitlines(keepends=True)
 | |
| 
 | |
|         result = clang_tidy_hash.get_clang_tidy_version_from_requirements()
 | |
| 
 | |
|     assert result == expected
 | |
| 
 | |
| 
 | |
| def test_calculate_clang_tidy_hash_with_sdkconfig(tmp_path: Path) -> None:
 | |
|     """Test calculating hash from all configuration sources including sdkconfig.defaults."""
 | |
|     clang_tidy_content = b"Checks: '-*,readability-*'\n"
 | |
|     requirements_version = "clang-tidy==18.1.5"
 | |
|     platformio_content = b"[env:esp32]\nplatform = espressif32\n"
 | |
|     sdkconfig_content = b"CONFIG_AUTOSTART_ARDUINO=y\n"
 | |
|     requirements_content = "clang-tidy==18.1.5\n"
 | |
| 
 | |
|     # Create temporary files
 | |
|     (tmp_path / ".clang-tidy").write_bytes(clang_tidy_content)
 | |
|     (tmp_path / "platformio.ini").write_bytes(platformio_content)
 | |
|     (tmp_path / "sdkconfig.defaults").write_bytes(sdkconfig_content)
 | |
|     (tmp_path / "requirements_dev.txt").write_text(requirements_content)
 | |
| 
 | |
|     # Expected hash calculation
 | |
|     expected_hasher = hashlib.sha256()
 | |
|     expected_hasher.update(clang_tidy_content)
 | |
|     expected_hasher.update(requirements_version.encode())
 | |
|     expected_hasher.update(platformio_content)
 | |
|     expected_hasher.update(sdkconfig_content)
 | |
|     expected_hash = expected_hasher.hexdigest()
 | |
| 
 | |
|     result = clang_tidy_hash.calculate_clang_tidy_hash(repo_root=tmp_path)
 | |
| 
 | |
|     assert result == expected_hash
 | |
| 
 | |
| 
 | |
| def test_calculate_clang_tidy_hash_without_sdkconfig(tmp_path: Path) -> None:
 | |
|     """Test calculating hash without sdkconfig.defaults file."""
 | |
|     clang_tidy_content = b"Checks: '-*,readability-*'\n"
 | |
|     requirements_version = "clang-tidy==18.1.5"
 | |
|     platformio_content = b"[env:esp32]\nplatform = espressif32\n"
 | |
|     requirements_content = "clang-tidy==18.1.5\n"
 | |
| 
 | |
|     # Create temporary files (without sdkconfig.defaults)
 | |
|     (tmp_path / ".clang-tidy").write_bytes(clang_tidy_content)
 | |
|     (tmp_path / "platformio.ini").write_bytes(platformio_content)
 | |
|     (tmp_path / "requirements_dev.txt").write_text(requirements_content)
 | |
| 
 | |
|     # Expected hash calculation (no sdkconfig)
 | |
|     expected_hasher = hashlib.sha256()
 | |
|     expected_hasher.update(clang_tidy_content)
 | |
|     expected_hasher.update(requirements_version.encode())
 | |
|     expected_hasher.update(platformio_content)
 | |
|     expected_hash = expected_hasher.hexdigest()
 | |
| 
 | |
|     result = clang_tidy_hash.calculate_clang_tidy_hash(repo_root=tmp_path)
 | |
| 
 | |
|     assert result == expected_hash
 | |
| 
 | |
| 
 | |
| def test_read_stored_hash_exists(tmp_path: Path) -> None:
 | |
|     """Test reading hash when file exists."""
 | |
|     stored_hash = "abc123def456"
 | |
|     hash_file = tmp_path / ".clang-tidy.hash"
 | |
|     hash_file.write_text(f"{stored_hash}\n")
 | |
| 
 | |
|     result = clang_tidy_hash.read_stored_hash(repo_root=tmp_path)
 | |
| 
 | |
|     assert result == stored_hash
 | |
| 
 | |
| 
 | |
| def test_read_stored_hash_not_exists(tmp_path: Path) -> None:
 | |
|     """Test reading hash when file doesn't exist."""
 | |
|     result = clang_tidy_hash.read_stored_hash(repo_root=tmp_path)
 | |
| 
 | |
|     assert result is None
 | |
| 
 | |
| 
 | |
| def test_write_hash(tmp_path: Path) -> None:
 | |
|     """Test writing hash to file."""
 | |
|     hash_value = "abc123def456"
 | |
|     hash_file = tmp_path / ".clang-tidy.hash"
 | |
| 
 | |
|     clang_tidy_hash.write_hash(hash_value, repo_root=tmp_path)
 | |
| 
 | |
|     assert hash_file.exists()
 | |
|     assert hash_file.read_text() == hash_value.strip() + "\n"
 | |
| 
 | |
| 
 | |
| @pytest.mark.parametrize(
 | |
|     ("args", "current_hash", "stored_hash", "hash_file_in_changed", "expected_exit"),
 | |
|     [
 | |
|         (["--check"], "abc123", "abc123", False, 1),  # Hashes match, no scan needed
 | |
|         (["--check"], "abc123", "def456", False, 0),  # Hashes differ, scan needed
 | |
|         (["--check"], "abc123", None, False, 0),  # No stored hash, scan needed
 | |
|         (
 | |
|             ["--check"],
 | |
|             "abc123",
 | |
|             "abc123",
 | |
|             True,
 | |
|             0,
 | |
|         ),  # Hash file updated in PR, scan needed
 | |
|     ],
 | |
| )
 | |
| def test_main_check_mode(
 | |
|     args: list[str],
 | |
|     current_hash: str,
 | |
|     stored_hash: str | None,
 | |
|     hash_file_in_changed: bool,
 | |
|     expected_exit: int,
 | |
| ) -> None:
 | |
|     """Test main function in check mode."""
 | |
|     changed = [".clang-tidy.hash"] if hash_file_in_changed else []
 | |
| 
 | |
|     # Create a mock module that can be imported
 | |
|     mock_helpers = Mock()
 | |
|     mock_helpers.changed_files = Mock(return_value=changed)
 | |
| 
 | |
|     with (
 | |
|         patch("sys.argv", ["clang_tidy_hash.py"] + args),
 | |
|         patch("clang_tidy_hash.calculate_clang_tidy_hash", return_value=current_hash),
 | |
|         patch("clang_tidy_hash.read_stored_hash", return_value=stored_hash),
 | |
|         patch.dict("sys.modules", {"helpers": mock_helpers}),
 | |
|         pytest.raises(SystemExit) as exc_info,
 | |
|     ):
 | |
|         clang_tidy_hash.main()
 | |
| 
 | |
|     assert exc_info.value.code == expected_exit
 | |
| 
 | |
| 
 | |
| def test_main_update_mode(capsys: pytest.CaptureFixture[str]) -> None:
 | |
|     """Test main function in update mode."""
 | |
|     current_hash = "abc123"
 | |
| 
 | |
|     with (
 | |
|         patch("sys.argv", ["clang_tidy_hash.py", "--update"]),
 | |
|         patch("clang_tidy_hash.calculate_clang_tidy_hash", return_value=current_hash),
 | |
|         patch("clang_tidy_hash.write_hash") as mock_write,
 | |
|     ):
 | |
|         clang_tidy_hash.main()
 | |
| 
 | |
|     mock_write.assert_called_once_with(current_hash)
 | |
|     captured = capsys.readouterr()
 | |
|     assert f"Hash updated: {current_hash}" in captured.out
 | |
| 
 | |
| 
 | |
| @pytest.mark.parametrize(
 | |
|     ("current_hash", "stored_hash"),
 | |
|     [
 | |
|         ("abc123", "def456"),  # Hash changed, should update
 | |
|         ("abc123", None),  # No stored hash, should update
 | |
|     ],
 | |
| )
 | |
| def test_main_update_if_changed_mode_update(
 | |
|     current_hash: str, stored_hash: str | None, capsys: pytest.CaptureFixture[str]
 | |
| ) -> None:
 | |
|     """Test main function in update-if-changed mode when update is needed."""
 | |
|     with (
 | |
|         patch("sys.argv", ["clang_tidy_hash.py", "--update-if-changed"]),
 | |
|         patch("clang_tidy_hash.calculate_clang_tidy_hash", return_value=current_hash),
 | |
|         patch("clang_tidy_hash.read_stored_hash", return_value=stored_hash),
 | |
|         patch("clang_tidy_hash.write_hash") as mock_write,
 | |
|         pytest.raises(SystemExit) as exc_info,
 | |
|     ):
 | |
|         clang_tidy_hash.main()
 | |
| 
 | |
|     assert exc_info.value.code == 0
 | |
|     mock_write.assert_called_once_with(current_hash)
 | |
|     captured = capsys.readouterr()
 | |
|     assert "Clang-tidy hash updated" in captured.out
 | |
| 
 | |
| 
 | |
| def test_main_update_if_changed_mode_no_update(
 | |
|     capsys: pytest.CaptureFixture[str],
 | |
| ) -> None:
 | |
|     """Test main function in update-if-changed mode when no update is needed."""
 | |
|     current_hash = "abc123"
 | |
|     stored_hash = "abc123"
 | |
| 
 | |
|     with (
 | |
|         patch("sys.argv", ["clang_tidy_hash.py", "--update-if-changed"]),
 | |
|         patch("clang_tidy_hash.calculate_clang_tidy_hash", return_value=current_hash),
 | |
|         patch("clang_tidy_hash.read_stored_hash", return_value=stored_hash),
 | |
|         patch("clang_tidy_hash.write_hash") as mock_write,
 | |
|         pytest.raises(SystemExit) as exc_info,
 | |
|     ):
 | |
|         clang_tidy_hash.main()
 | |
| 
 | |
|     assert exc_info.value.code == 0
 | |
|     mock_write.assert_not_called()
 | |
|     captured = capsys.readouterr()
 | |
|     assert "Clang-tidy hash unchanged" in captured.out
 | |
| 
 | |
| 
 | |
| def test_main_verify_mode_success(capsys: pytest.CaptureFixture[str]) -> None:
 | |
|     """Test main function in verify mode when verification passes."""
 | |
|     current_hash = "abc123"
 | |
|     stored_hash = "abc123"
 | |
| 
 | |
|     with (
 | |
|         patch("sys.argv", ["clang_tidy_hash.py", "--verify"]),
 | |
|         patch("clang_tidy_hash.calculate_clang_tidy_hash", return_value=current_hash),
 | |
|         patch("clang_tidy_hash.read_stored_hash", return_value=stored_hash),
 | |
|     ):
 | |
|         clang_tidy_hash.main()
 | |
|         captured = capsys.readouterr()
 | |
|         assert "Hash verification passed" in captured.out
 | |
| 
 | |
| 
 | |
| @pytest.mark.parametrize(
 | |
|     ("current_hash", "stored_hash"),
 | |
|     [
 | |
|         ("abc123", "def456"),  # Hashes differ, verification fails
 | |
|         ("abc123", None),  # No stored hash, verification fails
 | |
|     ],
 | |
| )
 | |
| def test_main_verify_mode_failure(
 | |
|     current_hash: str, stored_hash: str | None, capsys: pytest.CaptureFixture[str]
 | |
| ) -> None:
 | |
|     """Test main function in verify mode when verification fails."""
 | |
|     with (
 | |
|         patch("sys.argv", ["clang_tidy_hash.py", "--verify"]),
 | |
|         patch("clang_tidy_hash.calculate_clang_tidy_hash", return_value=current_hash),
 | |
|         patch("clang_tidy_hash.read_stored_hash", return_value=stored_hash),
 | |
|         pytest.raises(SystemExit) as exc_info,
 | |
|     ):
 | |
|         clang_tidy_hash.main()
 | |
| 
 | |
|     assert exc_info.value.code == 1
 | |
|     captured = capsys.readouterr()
 | |
|     assert "ERROR: Clang-tidy configuration has changed" in captured.out
 | |
| 
 | |
| 
 | |
| def test_main_default_mode(capsys: pytest.CaptureFixture[str]) -> None:
 | |
|     """Test main function in default mode (no arguments)."""
 | |
|     current_hash = "abc123"
 | |
|     stored_hash = "def456"
 | |
| 
 | |
|     with (
 | |
|         patch("sys.argv", ["clang_tidy_hash.py"]),
 | |
|         patch("clang_tidy_hash.calculate_clang_tidy_hash", return_value=current_hash),
 | |
|         patch("clang_tidy_hash.read_stored_hash", return_value=stored_hash),
 | |
|     ):
 | |
|         clang_tidy_hash.main()
 | |
| 
 | |
|     captured = capsys.readouterr()
 | |
|     assert f"Current hash: {current_hash}" in captured.out
 | |
|     assert f"Stored hash: {stored_hash}" in captured.out
 | |
|     assert "Match: False" in captured.out
 | |
| 
 | |
| 
 | |
| def test_read_file_lines(tmp_path: Path) -> None:
 | |
|     """Test read_file_lines helper function."""
 | |
|     test_file = tmp_path / "test.txt"
 | |
|     test_content = "line1\nline2\nline3\n"
 | |
|     test_file.write_text(test_content)
 | |
| 
 | |
|     result = clang_tidy_hash.read_file_lines(test_file)
 | |
| 
 | |
|     assert result == ["line1\n", "line2\n", "line3\n"]
 | |
| 
 | |
| 
 | |
| def test_read_file_bytes(tmp_path: Path) -> None:
 | |
|     """Test read_file_bytes helper function."""
 | |
|     test_file = tmp_path / "test.bin"
 | |
|     test_content = b"binary content\x00\xff"
 | |
|     test_file.write_bytes(test_content)
 | |
| 
 | |
|     result = clang_tidy_hash.read_file_bytes(test_file)
 | |
| 
 | |
|     assert result == test_content
 | |
| 
 | |
| 
 | |
| def test_write_file_content(tmp_path: Path) -> None:
 | |
|     """Test write_file_content helper function."""
 | |
|     test_file = tmp_path / "test.txt"
 | |
|     test_content = "test content"
 | |
| 
 | |
|     clang_tidy_hash.write_file_content(test_file, test_content)
 | |
| 
 | |
|     assert test_file.read_text() == test_content
 | |
| 
 | |
| 
 | |
| @pytest.mark.parametrize(
 | |
|     ("line", "expected"),
 | |
|     [
 | |
|         ("clang-tidy==18.1.5", ("clang-tidy", "clang-tidy==18.1.5")),
 | |
|         (
 | |
|             "clang-tidy==18.1.5  # comment",
 | |
|             ("clang-tidy", "clang-tidy==18.1.5  # comment"),
 | |
|         ),
 | |
|         ("some-package>=1.0,<2.0", ("some-package", "some-package>=1.0,<2.0")),
 | |
|         ("pkg_with-dashes==1.0", ("pkg_with-dashes", "pkg_with-dashes==1.0")),
 | |
|         ("# just a comment", None),
 | |
|         ("", None),
 | |
|         ("   ", None),
 | |
|         ("invalid line without version", None),
 | |
|     ],
 | |
| )
 | |
| def test_parse_requirement_line(line: str, expected: tuple[str, str] | None) -> None:
 | |
|     """Test parsing individual requirement lines."""
 | |
|     result = clang_tidy_hash.parse_requirement_line(line)
 | |
|     assert result == expected
 |