-
Notifications
You must be signed in to change notification settings - Fork 200
/
Copy pathcommand.py
307 lines (271 loc) · 11 KB
/
command.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
from __future__ import annotations
import logging
import traceback
import click
from sqlmesh.core.analytics import cli_analytics
from sqlmesh.core.console import set_console, MarkdownConsole
from sqlmesh.integrations.github.cicd.controller import (
GithubCheckConclusion,
GithubCheckStatus,
GithubController,
TestFailure,
)
from sqlmesh.utils.errors import CICDBotError, ConflictingPlanError, PlanError, LinterError
logger = logging.getLogger(__name__)
@click.group(no_args_is_help=True)
@click.option(
"--token",
type=str,
help="The Github Token to be used. Pass in `${{ secrets.GITHUB_TOKEN }}` if you want to use the one created by Github actions",
)
@click.pass_context
def github(ctx: click.Context, token: str) -> None:
"""Github Action CI/CD Bot. See https://sqlmesh.readthedocs.io/en/stable/integrations/github/ for details"""
set_console(MarkdownConsole())
ctx.obj["github"] = GithubController(
paths=ctx.obj["paths"],
token=token,
config=ctx.obj["config"],
)
def _check_required_approvers(controller: GithubController) -> bool:
controller.update_required_approval_check(status=GithubCheckStatus.IN_PROGRESS)
if controller.has_required_approval:
controller.update_required_approval_check(
status=GithubCheckStatus.COMPLETED, conclusion=GithubCheckConclusion.SUCCESS
)
return True
controller.update_required_approval_check(
status=GithubCheckStatus.COMPLETED, conclusion=GithubCheckConclusion.NEUTRAL
)
return False
@github.command()
@click.pass_context
@cli_analytics
def check_required_approvers(ctx: click.Context) -> None:
"""Checks if a required approver has provided approval on the PR."""
if not _check_required_approvers(ctx.obj["github"]):
raise CICDBotError(
"Required approver has not approved the PR. See check status for more information."
)
def _run_tests(controller: GithubController) -> bool:
controller.update_test_check(status=GithubCheckStatus.IN_PROGRESS)
try:
result, output = controller.run_tests()
controller.update_test_check(
status=GithubCheckStatus.COMPLETED,
# Conclusion will be updated with final status based on test results
conclusion=GithubCheckConclusion.NEUTRAL,
result=result,
output=output,
)
return result.wasSuccessful()
except Exception:
controller.update_test_check(
status=GithubCheckStatus.COMPLETED,
conclusion=GithubCheckConclusion.FAILURE,
output=traceback.format_exc(),
)
return False
def _run_linter(controller: GithubController) -> bool:
controller.update_linter_check(status=GithubCheckStatus.IN_PROGRESS)
try:
controller.run_linter()
except LinterError:
controller.update_linter_check(
status=GithubCheckStatus.COMPLETED,
conclusion=GithubCheckConclusion.FAILURE,
)
return False
controller.update_linter_check(
status=GithubCheckStatus.COMPLETED,
conclusion=GithubCheckConclusion.SUCCESS,
)
return True
@github.command()
@click.pass_context
@cli_analytics
def run_tests(ctx: click.Context) -> None:
"""Runs the unit tests"""
if not _run_tests(ctx.obj["github"]):
raise CICDBotError("Failed to run tests. See check status for more information.")
def _update_pr_environment(controller: GithubController) -> bool:
controller.update_pr_environment_check(status=GithubCheckStatus.IN_PROGRESS)
try:
controller.update_pr_environment()
conclusion = controller.update_pr_environment_check(status=GithubCheckStatus.COMPLETED)
return conclusion is not None and conclusion.is_success
except Exception as e:
conclusion = controller.update_pr_environment_check(
status=GithubCheckStatus.COMPLETED, exception=e
)
return (
conclusion is not None
and not conclusion.is_failure
and not conclusion.is_action_required
)
@github.command()
@click.pass_context
@cli_analytics
def update_pr_environment(ctx: click.Context) -> None:
"""Creates or updates the PR environments"""
if not _update_pr_environment(ctx.obj["github"]):
raise CICDBotError(
"Failed to update PR environment. See check status for more information."
)
def _gen_prod_plan(controller: GithubController) -> bool:
controller.update_prod_plan_preview_check(status=GithubCheckStatus.IN_PROGRESS)
try:
plan_summary = controller.get_plan_summary(controller.prod_plan)
controller.update_prod_plan_preview_check(
status=GithubCheckStatus.COMPLETED,
conclusion=GithubCheckConclusion.SUCCESS,
summary=plan_summary,
)
return bool(plan_summary)
except Exception as e:
controller.update_prod_plan_preview_check(
status=GithubCheckStatus.COMPLETED,
conclusion=GithubCheckConclusion.FAILURE,
summary=str(e),
)
return False
@github.command()
@click.pass_context
@cli_analytics
def gen_prod_plan(ctx: click.Context) -> None:
"""Generates the production plan"""
controller = ctx.obj["github"]
controller.update_prod_plan_preview_check(status=GithubCheckStatus.IN_PROGRESS)
if not _gen_prod_plan(controller):
raise CICDBotError(
"Failed to generate production plan. See check status for more information."
)
def _deploy_production(controller: GithubController) -> bool:
controller.update_prod_environment_check(status=GithubCheckStatus.IN_PROGRESS)
try:
controller.deploy_to_prod()
controller.update_prod_environment_check(
status=GithubCheckStatus.COMPLETED, conclusion=GithubCheckConclusion.SUCCESS
)
controller.try_merge_pr()
controller.try_invalidate_pr_environment()
return True
except ConflictingPlanError as e:
controller.update_prod_environment_check(
status=GithubCheckStatus.COMPLETED,
conclusion=GithubCheckConclusion.SKIPPED,
skip_reason=str(e),
)
return False
except PlanError:
controller.update_prod_environment_check(
status=GithubCheckStatus.COMPLETED, conclusion=GithubCheckConclusion.ACTION_REQUIRED
)
return False
except Exception:
controller.update_prod_environment_check(
status=GithubCheckStatus.COMPLETED, conclusion=GithubCheckConclusion.FAILURE
)
return False
@github.command()
@click.pass_context
@cli_analytics
def deploy_production(ctx: click.Context) -> None:
"""Deploys the production environment"""
if not _deploy_production(ctx.obj["github"]):
raise CICDBotError("Failed to deploy to production. See check status for more information.")
def _run_all(controller: GithubController) -> None:
has_required_approval = False
is_auto_deploying_prod = (
controller.deploy_command_enabled or controller.do_required_approval_check
)
if controller.is_comment_added:
if not controller.deploy_command_enabled:
# We aren't using commands so we can just return
return
command = controller.get_command_from_comment()
if command.is_invalid:
# Probably a comment unrelated to SQLMesh so we do nothing
return
elif command.is_deploy_prod:
has_required_approval = True
else:
raise CICDBotError(f"Unsupported command: {command}")
controller.update_linter_check(status=GithubCheckStatus.QUEUED)
controller.update_pr_environment_check(status=GithubCheckStatus.QUEUED)
controller.update_prod_plan_preview_check(status=GithubCheckStatus.QUEUED)
controller.update_test_check(status=GithubCheckStatus.QUEUED)
if is_auto_deploying_prod:
controller.update_prod_environment_check(status=GithubCheckStatus.QUEUED)
linter_passed = _run_linter(controller)
tests_passed = _run_tests(controller)
if controller.do_required_approval_check:
if has_required_approval:
controller.update_required_approval_check(
status=GithubCheckStatus.COMPLETED, conclusion=GithubCheckConclusion.SKIPPED
)
else:
controller.update_required_approval_check(status=GithubCheckStatus.QUEUED)
has_required_approval = _check_required_approvers(controller)
if not tests_passed or not linter_passed:
controller.update_pr_environment_check(
status=GithubCheckStatus.COMPLETED,
exception=LinterError("") if not linter_passed else TestFailure(),
)
controller.update_prod_plan_preview_check(
status=GithubCheckStatus.COMPLETED,
conclusion=GithubCheckConclusion.SKIPPED,
summary="Linter or Unit Test(s) failed so skipping creating prod plan",
)
if is_auto_deploying_prod:
controller.update_prod_environment_check(
status=GithubCheckStatus.COMPLETED,
conclusion=GithubCheckConclusion.SKIPPED,
skip_reason="Linter or Unit Test(s) failed so skipping deploying to production",
)
raise CICDBotError("Linter or Unit Test(s) failed. See check status for more information.")
pr_environment_updated = _update_pr_environment(controller)
prod_plan_generated = False
if pr_environment_updated:
prod_plan_generated = _gen_prod_plan(controller)
else:
controller.update_prod_plan_preview_check(
status=GithubCheckStatus.COMPLETED, conclusion=GithubCheckConclusion.SKIPPED
)
deployed_to_prod = False
if has_required_approval and prod_plan_generated:
deployed_to_prod = _deploy_production(controller)
elif is_auto_deploying_prod:
if not has_required_approval:
skip_reason = (
"Skipped Deploying to Production because a required approver has not approved"
)
elif not pr_environment_updated:
skip_reason = (
"Skipped Deploying to Production because the PR environment was not updated"
)
elif not prod_plan_generated:
skip_reason = (
"Skipped Deploying to Production because the production plan could not be generated"
)
else:
skip_reason = "Skipped Deploying to Production for an unknown reason"
controller.update_prod_environment_check(
status=GithubCheckStatus.COMPLETED,
conclusion=GithubCheckConclusion.SKIPPED,
skip_reason=skip_reason,
)
if (
not pr_environment_updated
or not prod_plan_generated
or (has_required_approval and not deployed_to_prod)
):
raise CICDBotError(
"A step of the run-all check failed. See check status for more information."
)
@github.command()
@click.pass_context
@cli_analytics
def run_all(ctx: click.Context) -> None:
"""Runs all the commands in the correct order."""
return _run_all(ctx.obj["github"])