|
13 | 13 | from pytest_mock.plugin import MockerFixture
|
14 | 14 | from sqlglot import exp
|
15 | 15 |
|
16 |
| -from sqlmesh.core.config import CategorizerConfig |
| 16 | +from sqlmesh.core.config import CategorizerConfig, Config, ModelDefaultsConfig, LinterConfig |
17 | 17 | from sqlmesh.core.engine_adapter.shared import DataObject
|
18 | 18 | from sqlmesh.core.user import User, UserRole
|
19 | 19 | from sqlmesh.integrations.github.cicd import command
|
@@ -52,6 +52,144 @@ def get_columns(
|
52 | 52 | return controller._context.engine_adapter.columns(table)
|
53 | 53 |
|
54 | 54 |
|
| 55 | +@time_machine.travel("2023-01-01 15:00:00 UTC") |
| 56 | +def test_linter( |
| 57 | + github_client, |
| 58 | + make_controller, |
| 59 | + make_mock_check_run, |
| 60 | + make_mock_issue_comment, |
| 61 | + make_pull_request_review, |
| 62 | + tmp_path: pathlib.Path, |
| 63 | + mocker: MockerFixture, |
| 64 | +): |
| 65 | + """ |
| 66 | + PR with a non-breaking change and auto-categorization will be backfilled, merged, and deployed to prod |
| 67 | +
|
| 68 | + Scenario: |
| 69 | + - PR is not merged |
| 70 | + - PR has been approved by a required reviewer |
| 71 | + - Tests passed |
| 72 | + - PR Merge Method defined |
| 73 | + - Delete environment is disabled |
| 74 | + - Changes made in PR with auto-categorization |
| 75 | + """ |
| 76 | + |
| 77 | + mock_repo = github_client.get_repo() |
| 78 | + mock_repo.create_check_run = mocker.MagicMock( |
| 79 | + side_effect=lambda **kwargs: make_mock_check_run(**kwargs) |
| 80 | + ) |
| 81 | + |
| 82 | + created_comments: t.List[MockIssueComment] = [] |
| 83 | + mock_issue = mock_repo.get_issue() |
| 84 | + mock_issue.create_comment = mocker.MagicMock( |
| 85 | + side_effect=lambda comment: make_mock_issue_comment( |
| 86 | + comment=comment, created_comments=created_comments |
| 87 | + ) |
| 88 | + ) |
| 89 | + mock_issue.get_comments = mocker.MagicMock(side_effect=lambda: created_comments) |
| 90 | + |
| 91 | + mock_pull_request = mock_repo.get_pull() |
| 92 | + mock_pull_request.get_reviews = mocker.MagicMock( |
| 93 | + side_effect=lambda: [make_pull_request_review(username="test_github", state="APPROVED")] |
| 94 | + ) |
| 95 | + mock_pull_request.merged = False |
| 96 | + mock_pull_request.merge = mocker.MagicMock() |
| 97 | + |
| 98 | + # Case 1: Test for linter errors |
| 99 | + config = Config( |
| 100 | + model_defaults=ModelDefaultsConfig(dialect="duckdb"), |
| 101 | + linter=LinterConfig(enabled=True, rules="ALL"), |
| 102 | + ) |
| 103 | + |
| 104 | + controller = make_controller( |
| 105 | + "tests/fixtures/github/pull_request_synchronized.json", |
| 106 | + github_client, |
| 107 | + bot_config=GithubCICDBotConfig( |
| 108 | + merge_method=MergeMethod.MERGE, |
| 109 | + invalidate_environment_after_deploy=False, |
| 110 | + auto_categorize_changes=CategorizerConfig.all_full(), |
| 111 | + default_pr_start=None, |
| 112 | + skip_pr_backfill=False, |
| 113 | + ), |
| 114 | + mock_out_context=False, |
| 115 | + config=config, |
| 116 | + ) |
| 117 | + |
| 118 | + github_output_file = tmp_path / "github_output.txt" |
| 119 | + |
| 120 | + with mock.patch.dict(os.environ, {"GITHUB_OUTPUT": str(github_output_file)}): |
| 121 | + with pytest.raises(CICDBotError): |
| 122 | + command._run_all(controller) |
| 123 | + |
| 124 | + assert "SQLMesh - Linter" in controller._check_run_mapping |
| 125 | + linter_checks_runs = controller._check_run_mapping["SQLMesh - Linter"].all_kwargs |
| 126 | + assert "Linter errors for" in linter_checks_runs[-1]["output"]["summary"] |
| 127 | + assert GithubCheckConclusion(linter_checks_runs[-1]["conclusion"]).is_failure |
| 128 | + |
| 129 | + for check in ( |
| 130 | + "SQLMesh - Run Unit Tests", |
| 131 | + "SQLMesh - PR Environment Synced", |
| 132 | + "SQLMesh - Prod Plan Preview", |
| 133 | + ): |
| 134 | + assert check in controller._check_run_mapping |
| 135 | + check_runs = controller._check_run_mapping[check].all_kwargs |
| 136 | + assert GithubCheckConclusion(check_runs[-1]["conclusion"]).is_skipped |
| 137 | + |
| 138 | + with open(github_output_file, "r", encoding="utf-8") as f: |
| 139 | + output = f.read() |
| 140 | + assert ( |
| 141 | + output |
| 142 | + == "linter=failure\npr_environment_name=hello_world_2\npr_environment_synced=skipped\nprod_plan_preview=skipped\nrun_unit_tests=skipped\n" |
| 143 | + ) |
| 144 | + |
| 145 | + # empty github file for next case |
| 146 | + open(github_output_file, "w").close() |
| 147 | + |
| 148 | + # Case 2: Test for linter warnings |
| 149 | + config = Config( |
| 150 | + model_defaults=ModelDefaultsConfig(dialect="duckdb"), |
| 151 | + linter=LinterConfig(enabled=True, warn_rules="ALL"), |
| 152 | + ) |
| 153 | + |
| 154 | + controller = make_controller( |
| 155 | + "tests/fixtures/github/pull_request_synchronized.json", |
| 156 | + github_client, |
| 157 | + bot_config=GithubCICDBotConfig( |
| 158 | + merge_method=MergeMethod.MERGE, |
| 159 | + invalidate_environment_after_deploy=False, |
| 160 | + auto_categorize_changes=CategorizerConfig.all_full(), |
| 161 | + default_pr_start=None, |
| 162 | + skip_pr_backfill=False, |
| 163 | + ), |
| 164 | + mock_out_context=False, |
| 165 | + config=config, |
| 166 | + ) |
| 167 | + |
| 168 | + with mock.patch.dict(os.environ, {"GITHUB_OUTPUT": str(github_output_file)}): |
| 169 | + command._run_all(controller) |
| 170 | + |
| 171 | + assert "SQLMesh - Linter" in controller._check_run_mapping |
| 172 | + linter_checks_runs = controller._check_run_mapping["SQLMesh - Linter"].all_kwargs |
| 173 | + assert "Linter warnings for" in linter_checks_runs[-1]["output"]["summary"] |
| 174 | + assert GithubCheckConclusion(linter_checks_runs[-1]["conclusion"]).is_success |
| 175 | + |
| 176 | + for check in ( |
| 177 | + "SQLMesh - Run Unit Tests", |
| 178 | + "SQLMesh - PR Environment Synced", |
| 179 | + "SQLMesh - Prod Plan Preview", |
| 180 | + ): |
| 181 | + assert check in controller._check_run_mapping |
| 182 | + check_runs = controller._check_run_mapping[check].all_kwargs |
| 183 | + assert GithubCheckConclusion(check_runs[-1]["conclusion"]).is_success |
| 184 | + |
| 185 | + with open(github_output_file, "r", encoding="utf-8") as f: |
| 186 | + output = f.read() |
| 187 | + assert ( |
| 188 | + output |
| 189 | + == "linter=success\nrun_unit_tests=success\ncreated_pr_environment=true\npr_environment_name=hello_world_2\npr_environment_synced=success\nprod_plan_preview=success\n" |
| 190 | + ) |
| 191 | + |
| 192 | + |
55 | 193 | @time_machine.travel("2023-01-01 15:00:00 UTC")
|
56 | 194 | def test_merge_pr_has_non_breaking_change(
|
57 | 195 | github_client,
|
|
0 commit comments