summaryrefslogtreecommitdiff
path: root/doc/source/discussion/github-checks-api.rst
blob: 73788a5d2ecd6c54d78e93c4c157dbc0b9b2e1ea (plain)
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
:title: Github Checks API

Github Checks API
=================

Using the `Github Checks API`_ to report job results back to a PR provides
some additional features compared to the status API like file comments and
custom actions. The latter one could be used to e.g. cancel a running
build.

Design decisions
-----------------

The github checks API consists mainly of two entities: `Check Suites`_ and
`Check Runs`_. Check suites are a collection of check runs for a specific
commit and summarize their status and conclusion.

Following this description, one might think that the check suite is a
perfect mapping for a pipeline execution in zuul and a check run could map
to a single job execution that is part of the pipeline run. Unfortunately,
there are a few restrictions that don't allow this kind of mapping.

First of all, check suites are completely managed by Github. Apart from
creating a check suite for a commit SHA, we can't do anything with it.
The current status, duration and the conclusion are all calculated and
set by Github automatically whenever an included check run is updated.

There can only be one check suite per commit sha per app. Thus, even if
we could update the check suite, we wouldn't be able to create one check
suite for each pipeline, e.g. check and gate.

When configuring the branch protection in Github, only a check run can
be selected as required status check. Having each job as a dedicated
check run would result in a huge list of status checks one would have to
enable to make the branch protection work. Additionally, we would then
loose some of Zuul's features like non-voting jobs and it would break
Zuul's gating capabilities as they are working on a pipeline level, not on
a job level.

Zuul can only report the whole buildset, but no individual jobs. With
that we wouldn't be able to update individual check runs on a job level.

Having said the above, the only possible integration of the checks API is
on a pipeline level, so each pipeline execution maps to a check run in
Github.

Behaviour in Zuul
-----------------

Reporting
~~~~~~~~~

The Github reporter is able to report both a status
:attr:`pipeline.<reporter>.<github source>.status` or a check
:attr:`pipeline.<reporter>.<github source>.check`. While it's possible to
configure a Github reporter to report both, it's recommended to use only one.
Reporting both might result in duplicated status check entries in the Github
PR (the section below the comments).

Trigger
~~~~~~~

The Github driver is able to trigger on a reported check
(:value:`pipeline.trigger.<github source>.event.check_run`) similar to a
reported status (:value:`pipeline.trigger.<github source>.action.status`).

Requirements
~~~~~~~~~~~~

While trigger and reporter differentiates between status and check, the Github
driver does not differentiate between them when it comes to pipeline
requirements. This is mainly because Github also doesn't differentiate between
both in terms of branch protection and `status checks`_.

Actions / Events
----------------

Github provides a set of default actions for check suites and check runs.
Those actions are available as buttons in the Github UI. Clicking on those
buttons will emit webhook events which will be handled by Zuul.

These actions are only available on failed check runs / check suites. So
far, a running or successful check suite / check run does not provide any
action from Github side.

Available actions are:

Re-run all checks
  Github emits a webhook event with type ``check_suite`` and action
  ``rerequested`` that is meant to re-run all check-runs contained in this
  check suite. Github does not provide the list of check-runs in that case,
  so it's up to the Github app what should run.

Re-run failed checks
  Github emits a webhook event with type ``check_run`` and action
  ``rerequested`` for each failed check run contained in this suite.

Re-run
  Github emits a webhook event with type ``check_run`` and action
  ``rerequested`` for the specific check run.

Zuul will handle all events except for the `Re-run all checks` event as
this is not suitable for the Zuul workflow as it doesn't make sense to
trigger all pipelines to run simultaniously.

The drawback here is, that we are not able to customize those events in Github.
Github will always say "You have successfully requested ..." although we aren't
listening to the event at all. Therefore, it might be a solution to handle the
`Re-run all checks` event in Zuul similar to `Re-run failed checks` just to
not do anything while Github makes the user believe an action was really
triggered.


File comments (annotations)
---------------------------

Check runs can be used to post file comments directly in the files of the PR.
Those are similar to user comments, but must provide some more information.

Zuul jobs can already return file comments via ``zuul_return``
(see: :ref:`return_values`). We can simply use this return value, build the
necessary annotations (how Github calls it) from it and attach them to the
check run.


Custom actions
~~~~~~~~~~~~~~

Check runs can provide some custom actions which will result in additional
buttons being available in the Github UI for this specific check run.
Clicking on such a button will emit a webhook event with type ``check_run``
and action ``requested_action`` and will additionally contain the id/name of
the requested action which we can define when creating the action on the
check run.

We could use these custom actions to provide some "Re-run" action on a
running check run (which might otherwise be stuck in case a check run update
fails) or to abort a check run directly from the Github UI.


Restrictions and Recommendations
--------------------------------

Although both the checks API and the status API can be activated for a
Github reporter at the same time, it's not recommmended to do so as this might
result in multiple status checks to be reported to the PR for the same pipeline
execution (which would result in duplicated entries in the status section below
the comments of a PR).

In case the update on a check run fails (e.g. request timeout when reporting
success or failure to Github), the check run will stay in status "in_progess"
and there will be no way to re-run the check run via the Github UI as the
predefined actions are only available on failed check runs.
Thus, it's recommended to configure a
:value:`pipeline.trigger.<github source>.action.comment` trigger on the
pipeline to still be able to trigger re-run of the stuck check run via e.g.
"recheck".

The check suite will only list check runs that were reported by Zuul. If
the requirements for a certain pipeline are not met and it is not run, the
check run for this pipeline won't be listed in the check suite. However,
this does not affect the required status checks. If the check run is enabled
as required, Github will still show it in the list of required status checks
- even if it didn't run yet - just not in the check suite.


.. _Github Checks API: https://developer.github.com/v3/checks/
.. _Check Suites: https://developer.github.com/v3/checks/suites/
.. _Check Runs: https://developer.github.com/v3/checks/runs/
.. _status checks: https://help.github.com/en/github/collaborating-with-issues-and-pull-requests/about-status-checks#types-of-status-checks-on-github