Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 1 | = Gerrit Code Review - Prolog Submit Rules Cookbook |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 2 | |
Edwin Kempin | 1781adb | 2014-04-22 10:02:40 +0200 | [diff] [blame] | 3 | [[SubmitRule]] |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 4 | == Submit Rule |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 5 | A _Submit Rule_ in Gerrit is logic that defines when a change is submittable. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 6 | By default, a change is submittable when it gets at least one |
| 7 | highest vote in each voting category and has no lowest vote (aka veto vote) in |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 8 | any category. Typically, this means that a change needs `Code-Review+2`, |
| 9 | `Verified+1` and has neither `Code-Review-2` nor `Verified-1` to become |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 10 | submittable. |
| 11 | |
| 12 | While this rule is a good default, there are projects which need more |
| 13 | flexibility for defining when a change is submittable. In Gerrit, it is |
| 14 | possible to use Prolog based rules to provide project specific submit rules and |
| 15 | replace the default submit rules. Using Prolog based rules, project owners can |
| 16 | define a set of criteria which must be fulfilled for a change to become |
| 17 | submittable. For a change that is not submittable, the set of needed criteria |
| 18 | is displayed in the Gerrit UI. |
| 19 | |
| 20 | NOTE: Loading and executing Prolog submit rules may be disabled by setting |
Jonathan Nieder | f0e6dc0 | 2016-02-29 13:57:23 -0800 | [diff] [blame^] | 21 | `rules.enable=false` in the Gerrit config file (see |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 22 | link:config-gerrit.html#_a_id_rules_a_section_rules[rules section]) |
| 23 | |
| 24 | link:https://groups.google.com/d/topic/repo-discuss/wJxTGhlHZMM/discussion[This |
| 25 | discussion thread] explains why Prolog was chosen for the purpose of writing |
| 26 | project specific submit rules. |
| 27 | link:http://gerrit-documentation.googlecode.com/svn/ReleaseNotes/ReleaseNotes-2.2.2.html[Gerrit |
| 28 | 2.2.2 ReleaseNotes] introduces Prolog support in Gerrit. |
| 29 | |
Edwin Kempin | 1781adb | 2014-04-22 10:02:40 +0200 | [diff] [blame] | 30 | [[SubmitType]] |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 31 | == Submit Type |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 32 | A _Submit Type_ is a strategy that is used on submit to integrate the |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 33 | change into the destination branch. Supported submit types are: |
| 34 | |
| 35 | * `Fast Forward Only` |
| 36 | * `Merge If Necessary` |
| 37 | * `Merge Always` |
| 38 | * `Cherry Pick` |
| 39 | * `Rebase If Necessary` |
| 40 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 41 | _Submit Type_ is a project global setting. This means that the same submit type |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 42 | is used for all changes of one project. |
| 43 | |
| 44 | Projects which need more flexibility in choosing, or enforcing, a submit type |
| 45 | can use Prolog based submit type which replaces the project's default submit |
| 46 | type. |
| 47 | |
| 48 | Prolog based submit type computes a submit type for each change. The computed |
| 49 | submit type is shown on the change screen for each change. |
| 50 | |
Dave Borowitz | 0761fd5 | 2015-12-13 21:59:04 -0500 | [diff] [blame] | 51 | When submitting changes in a batch using "Submit including ancestors" or "Submit |
| 52 | whole topic", submit type rules may not be used to mix submit types on a single |
| 53 | branch, and trying to submit such a batch will fail. This avoids potentially |
| 54 | confusing behavior and spurious submit failures. It is recommended to only use |
| 55 | submit type rules to change submit types for an entire branch, which avoids this |
| 56 | situation. |
| 57 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 58 | == Prolog Language |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 59 | This document is not a complete Prolog tutorial. |
| 60 | link:http://en.wikipedia.org/wiki/Prolog[This Wikipedia page on Prolog] is a |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 61 | good starting point for learning the Prolog language. This document will only |
| 62 | explain some elements of Prolog that are necessary to understand the provided |
| 63 | examples. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 64 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 65 | == Prolog in Gerrit |
Shawn Pearce | 792b2c4 | 2015-06-12 17:25:40 -0700 | [diff] [blame] | 66 | Gerrit uses its own link:https://gerrit.googlesource.com/prolog-cafe/[fork] of the |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 67 | original link:http://kaminari.istc.kobe-u.ac.jp/PrologCafe/[prolog-cafe] |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 68 | project. Gerrit embeds the prolog-cafe library and can interpret Prolog programs |
| 69 | at runtime. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 70 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 71 | == Interactive Prolog Cafe Shell |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 72 | For interactive testing and playing with Prolog, Gerrit provides the |
| 73 | link:pgm-prolog-shell.html[prolog-shell] program which opens an interactive |
| 74 | Prolog interpreter shell. |
| 75 | |
Johan Björk | 2119f05 | 2012-10-24 12:10:45 -0400 | [diff] [blame] | 76 | NOTE: The interactive shell is just a prolog shell, it does not load |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 77 | a gerrit server environment and thus is not intended for |
| 78 | xref:TestingSubmitRules[testing submit rules]. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 79 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 80 | == SWI-Prolog |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 81 | Instead of using the link:pgm-prolog-shell.html[prolog-shell] program one can |
| 82 | also use the link:http://www.swi-prolog.org/[SWI-Prolog] environment. It |
| 83 | provides a better shell interface and a graphical source-level debugger. |
| 84 | |
Edwin Kempin | 1781adb | 2014-04-22 10:02:40 +0200 | [diff] [blame] | 85 | [[RulesFile]] |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 86 | == The rules.pl file |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 87 | This section explains how to create and edit project specific submit rules. How |
| 88 | to actually write the submit rules is explained in the next section. |
| 89 | |
| 90 | Project specific submit rules are stored in the `rules.pl` file in the |
| 91 | `refs/meta/config` branch of that project. Therefore, we need to fetch and |
| 92 | checkout the `refs/meta/config` branch in order to create or edit the `rules.pl` |
| 93 | file: |
| 94 | |
| 95 | ==== |
| 96 | $ git fetch origin refs/meta/config:config |
| 97 | $ git checkout config |
| 98 | ... edit or create the rules.pl file |
| 99 | $ git add rules.pl |
| 100 | $ git commit -m "My submit rules" |
| 101 | $ git push origin HEAD:refs/meta/config |
| 102 | ==== |
| 103 | |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 104 | [[HowToWriteSubmitRules]] |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 105 | == How to write submit rules |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 106 | Whenever Gerrit needs to evaluate submit rules for a change `C` from project `P` |
| 107 | it will first initialize the embedded Prolog interpreter by: |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 108 | |
| 109 | * consulting a set of facts about the change `C` |
| 110 | * consulting the `rules.pl` from the project `P` |
| 111 | |
| 112 | Conceptually we can imagine that Gerrit adds a set of facts about the change |
| 113 | `C` on top of the `rules.pl` file and then consults it. The set of facts about |
| 114 | the change `C` will look like: |
| 115 | |
| 116 | ==== |
| 117 | :- package gerrit. <1> |
| 118 | |
| 119 | commit_author(user(1000000), 'John Doe', 'john.doe@example.com'). <2> |
| 120 | commit_committer(user(1000000), 'John Doe', 'john.doe@example.com'). <3> |
| 121 | commit_message('Add plugin support to Gerrit'). <4> |
| 122 | ... |
| 123 | ==== |
| 124 | |
| 125 | <1> Gerrit will provide its facts in a package named `gerrit`. This means we |
| 126 | have to use qualified names when writing our code and referencing these facts. |
| 127 | For example: `gerrit:commit_author(ID, N, M)` |
| 128 | <2> user ID, full name and email address of the commit author |
| 129 | <3> user ID, full name and email address of the commit committer |
| 130 | <4> commit message |
| 131 | |
| 132 | A complete set of facts which Gerrit provides about the change is listed in the |
| 133 | link:prolog-change-facts.html[Prolog Facts for Gerrit Change]. |
| 134 | |
| 135 | By default, Gerrit will search for a `submit_rule/1` predicate in the `rules.pl` |
| 136 | file, evaluate the `submit_rule(X)` and then inspect the value of `X` in order |
| 137 | to decide whether the change is submittable or not and also to find the set of |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 138 | needed criteria for the change to become submittable. This means that Gerrit has |
| 139 | an expectation on the format and value of the result of the `submit_rule` |
| 140 | predicate which is expected to be a `submit` term of the following format: |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 141 | |
| 142 | ==== |
| 143 | submit(label(label-name, status) [, label(label-name, status)]*) |
| 144 | ==== |
| 145 | |
| 146 | where `label-name` is usually `'Code-Review'` or `'Verified'` but could also |
| 147 | be any other string (see examples below). The `status` is one of: |
| 148 | |
| 149 | * `ok(user(ID))` or just `ok(_)` if user info is not important. This status is |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 150 | used to tell that this label/category has been met. |
David Pursehouse | 3f91890 | 2015-05-19 09:41:02 +0900 | [diff] [blame] | 151 | * `need(_)` is used to tell that this label/category is needed for the change to |
| 152 | become submittable. |
| 153 | * `reject(user(ID))` or just `reject(_)`. This status is used to tell that this |
| 154 | label/category is blocking submission of the change. |
| 155 | * `impossible(_)` is used when the logic knows that the change cannot be submitted |
| 156 | as-is. This is meant for cases where the logic requires members of a specific |
| 157 | group to apply a specific label on a change, but no users are in that group. |
| 158 | This is usually caused by misconfiguration of permissions. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 159 | * `may(_)` allows expression of approval categories that are optional, i.e. |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 160 | could either be set or unset without ever influencing whether the change |
| 161 | could be submitted. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 162 | |
| 163 | NOTE: For a change to be submittable all `label` terms contained in the returned |
| 164 | `submit` term must have either `ok` or `may` status. |
| 165 | |
| 166 | IMPORTANT: Gerrit will let the Prolog engine continue searching for solutions of |
| 167 | the `submit_rule(X)` query until it finds the first one where all labels in the |
| 168 | return result have either status `ok` or `may` or there are no more solutions. |
| 169 | If a solution where all labels have status `ok` is found then all previously |
| 170 | found solutions are ignored. Otherwise, all labels names with status `need` |
| 171 | from all solutions will be displayed in the UI indicating the set of conditions |
| 172 | needed for the change to become submittable. |
| 173 | |
| 174 | Here some examples of possible return values from the `submit_rule` predicate: |
| 175 | |
| 176 | ==== |
| 177 | submit(label('Code-Review', ok(_))) <1> |
| 178 | submit(label('Code-Review', ok(_)), label('Verified', reject(_))) <2> |
| 179 | submit(label('Author-is-John-Doe', need(_)) <3> |
| 180 | ==== |
| 181 | |
| 182 | <1> label `'Code-Review'` is met. As there are no other labels in the |
| 183 | return result, the change is submittable. |
| 184 | <2> label `'Verified'` is rejected. Change is not submittable. |
| 185 | <3> label `'Author-is-John-Doe'` is needed for the change to become submittable. |
| 186 | Note that this tells nothing about how this criteria will be met. It is up |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 187 | to the implementer of the `submit_rule` to return |
| 188 | `label('Author-is-John-Doe', ok(_))` when this criteria is met. Most |
| 189 | likely, it will have to match against `gerrit:commit_author` in order to |
| 190 | check if this criteria is met. This will become clear through the examples |
| 191 | below. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 192 | |
| 193 | Of course, when implementing the `submit_rule` we will use the facts about the |
| 194 | change that are already provided by Gerrit. |
| 195 | |
| 196 | Another aspect of the return result from the `submit_rule` predicate is that |
| 197 | Gerrit uses it to decide which set of labels to display on the change review |
| 198 | screen for voting. If the return result contains label `'ABC'` and if the label |
Dave Borowitz | 01c1b1f | 2013-02-27 13:49:04 -0800 | [diff] [blame] | 199 | `'ABC'` is link:config-labels.html[defined for the project] then voting for the |
| 200 | label `'ABC'` will be displayed. Otherwise, it is not displayed. Note that the |
| 201 | project doesn't need a defined label for each label contained in the result of |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 202 | `submit_rule` predicate. For example, the decision whether |
| 203 | `'Author-is-John-Doe'` label is met will probably not be made by explicit voting |
| 204 | but, instead, by inspecting the facts about the change. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 205 | |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 206 | [[SubmitFilter]] |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 207 | == Submit Filter |
Sasa Zivkov | 3d4f0aa | 2012-08-07 15:11:32 +0200 | [diff] [blame] | 208 | Another mechanism of changing the default submit rules is to implement the |
| 209 | `submit_filter/2` predicate. While Gerrit will search for the `submit_rule` only |
| 210 | in the `rules.pl` file of the current project, the `submit_filter` will be |
| 211 | searched for in the `rules.pl` of all parent projects of the current project, |
| 212 | but not in the `rules.pl` of the current project. The search will start from the |
| 213 | immediate parent of the current project, then in the parent project of that |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 214 | project and so on until, and including, the `'All-Projects'` project. |
Sasa Zivkov | 3d4f0aa | 2012-08-07 15:11:32 +0200 | [diff] [blame] | 215 | |
| 216 | The purpose of the submit filter is, as its name says, to filter the results |
| 217 | of the `submit_rule`. Therefore, the `submit_filter` predicate has two |
| 218 | parameters: |
| 219 | |
| 220 | ==== |
| 221 | submit_filter(In, Out) :- ... |
| 222 | ==== |
| 223 | |
| 224 | Gerrit will invoke `submit_filter` with the `In` parameter containing a `submit` |
| 225 | structure produced by the `submit_rule` and will take the value of the `Out` |
| 226 | parameter as the result. |
| 227 | |
| 228 | The `Out` value of a `submit_filter` will become the `In` value for the |
| 229 | next `submit_filter` in the parent line. The value of the `Out` parameter |
| 230 | of the top-most `submit_filter` is the final result of the submit rule that |
| 231 | is used to decide whether a change is submittable or not. |
| 232 | |
| 233 | IMPORTANT: `submit_filter` is a mechanism for Gerrit administrators to implement |
| 234 | and enforce submit rules that would apply to all projects while `submit_rule` is |
| 235 | a mechanism for project owners to implement project specific submit rules. |
| 236 | However, project owners who own several projects could also make use of |
| 237 | `submit_filter` by using a common parent project for all their projects and |
| 238 | implementing the `submit_filter` in this common parent project. This way they |
| 239 | can avoid implementing the same `submit_rule` in all their projects. |
| 240 | |
| 241 | The following "drawing" illustrates the order of the invocation and the chaining |
| 242 | of the results of the `submit_rule` and `submit_filter` predicates. |
Sasa Zivkov | 3d4f0aa | 2012-08-07 15:11:32 +0200 | [diff] [blame] | 243 | ==== |
| 244 | All-Projects |
| 245 | ^ submit_filter(B, S) :- ... <4> |
| 246 | | |
| 247 | Parent-3 |
| 248 | ^ <no submit filter here> |
| 249 | | |
| 250 | Parent-2 |
| 251 | ^ submit_filter(A, B) :- ... <3> |
| 252 | | |
| 253 | Parent-1 |
| 254 | ^ submit_filter(X, A) :- ... <2> |
| 255 | | |
| 256 | MyProject |
| 257 | submit_rule(X) :- ... <1> |
| 258 | ==== |
| 259 | |
| 260 | <1> The `submit_rule` of `MyProject` is invoked first. |
| 261 | <2> The result `X` is filtered through the `submit_filter` from the `Parent-1` |
| 262 | project. |
| 263 | <3> The result of `submit_filter` from `Parent-1` project is filtered by the |
| 264 | `submit_filter` in the `Parent-2` project. Since `Parent-3` project doesn't have |
| 265 | a `submit_filter` it is skipped. |
| 266 | <4> The result of `submit_filter` from `Parent-2` project is filtered by the |
| 267 | `submit_filter` in the `All-Projects` project. The value in `S` is the final |
| 268 | value of the submit rule evaluation. |
| 269 | |
| 270 | NOTE: If `MyProject` doesn't define its own `submit_rule` Gerrit will invoke the |
| 271 | default implementation of submit rule that is named `gerrit:default_submit` and |
| 272 | its result will be filtered as described above. |
| 273 | |
Edwin Kempin | ca24f5c | 2013-03-26 13:23:20 +0100 | [diff] [blame] | 274 | [[HowToWriteSubmitType]] |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 275 | == How to write submit type |
Chris Lee | 849cbd8 | 2014-10-20 16:23:21 -0700 | [diff] [blame] | 276 | Writing custom submit type logic in Prolog is similar to |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 277 | xref:HowToWriteSubmitRules[writing submit rules]. The only difference is that |
| 278 | one has to implement a `submit_type` predicate (instead of the `submit_rule`) |
| 279 | and that the return result of the `submit_type` has to be an atom that |
| 280 | represents one of the supported submit types: |
| 281 | |
| 282 | * `fast_forward_only` |
| 283 | * `merge_if_necessary` |
| 284 | * `merge_always` |
| 285 | * `cherry_pick` |
| 286 | * `rebase_if_necessary` |
| 287 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 288 | == Submit Type Filter |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 289 | Submit type filter works the same way as the xref:SubmitFilter[Submit Filter] |
| 290 | where the name of the filter predicate is `submit_type_filter`. |
| 291 | |
| 292 | ==== |
| 293 | submit_type_filter(In, Out). |
| 294 | ==== |
| 295 | |
| 296 | Gerrit will invoke `submit_type_filter` with the `In` parameter containing a |
| 297 | result of the `submit_type` and will take the value of the `Out` parameter as |
| 298 | the result. |
| 299 | |
Johan Björk | 2119f05 | 2012-10-24 12:10:45 -0400 | [diff] [blame] | 300 | [[TestingSubmitRules]] |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 301 | == Testing submit rules |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 302 | The prolog environment running the `submit_rule` is loaded with state describing |
| 303 | the change that is being evaluated. The easiest way to load this state is to |
| 304 | test your `submit_rule` against a real change on a running gerrit instance. The |
| 305 | command link:cmd-test-submit-rule.html[test-submit rule] loads a specific change |
| 306 | and executes the `submit_rule`. It optionally reads the rule from from `stdin` |
| 307 | to facilitate easy testing. |
Johan Björk | 2119f05 | 2012-10-24 12:10:45 -0400 | [diff] [blame] | 308 | |
| 309 | ==== |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 310 | $ cat rules.pl | ssh gerrit_srv gerrit test-submit rule I45e080b105a50a625cc8e1fb5b357c0bfabe6d68 -s |
Johan Björk | 2119f05 | 2012-10-24 12:10:45 -0400 | [diff] [blame] | 311 | ==== |
Sasa Zivkov | 3d4f0aa | 2012-08-07 15:11:32 +0200 | [diff] [blame] | 312 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 313 | == Prolog vs Gerrit plugin for project specific submit rules |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 314 | Since version 2.5 Gerrit supports plugins and extension points. A plugin or an |
| 315 | extension point could also be used as another means to provide custom submit |
| 316 | rules. One could ask for a guideline when to use Prolog based submit rules and |
| 317 | when to go for writing a new plugin. Writing a Prolog program is usually much |
| 318 | faster than writing a Gerrit plugin. Prolog based submit rules can be pushed |
| 319 | to a project by project owners while Gerrit plugins could only be installed by |
| 320 | Gerrit administrators. In addition, Prolog based submit rules can be pushed |
| 321 | for review by pushing to `refs/for/refs/meta/config` branch. |
| 322 | |
| 323 | On the other hand, Prolog based submit rules get a limited amount of facts about |
| 324 | the change exposed to them. Gerrit plugins get full access to Gerrit internals |
| 325 | and can potentially check more things than Prolog based rules. |
| 326 | |
Sasa Zivkov | d0e5526 | 2013-01-16 14:26:06 +0100 | [diff] [blame] | 327 | From version 2.6 Gerrit plugins can contribute Prolog predicates. This way, we |
| 328 | can make use of the plugin provided predicates when writing Prolog based rules. |
| 329 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 330 | == Examples - Submit Rule |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 331 | The following examples should serve as a cookbook for developing own submit |
| 332 | rules. Some of them are too trivial to be used in production and their only |
| 333 | purpose is to provide step by step introduction and understanding. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 334 | |
Sasa Zivkov | 3d4f0aa | 2012-08-07 15:11:32 +0200 | [diff] [blame] | 335 | Some of the examples will implement the `submit_rule` and some will implement |
| 336 | the `submit_filter` just to show both possibilities. Remember that |
| 337 | `submit_rule` is only invoked from the current project and `submit_filter` is |
| 338 | invoked from all parent projects. This is the most important fact in deciding |
| 339 | whether to implement `submit_rule` or `submit_filter`. |
| 340 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 341 | === Example 1: Make every change submittable |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 342 | Let's start with a most trivial example where we would make every change |
| 343 | submittable regardless of the votes it has: |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 344 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 345 | `rules.pl` |
| 346 | [source,prolog] |
| 347 | ---- |
| 348 | submit_rule(submit(W)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 349 | W = label('Any-Label-Name', ok(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 350 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 351 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 352 | In this case we make no use of facts about the change. We don't need it as we |
| 353 | are simply making every change submittable. Note that, in this case, the Gerrit |
| 354 | UI will not show the UI for voting for the standard `'Code-Review'` and |
| 355 | `'Verified'` categories as labels with these names are not part of the return |
| 356 | result. The `'Any-Label-Name'` could really be any string. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 357 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 358 | === Example 2: Every change submittable and voting in the standard categories possible |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 359 | This is continuation of the previous example where, in addition, to making |
| 360 | every change submittable we want to enable voting in the standard |
| 361 | `'Code-Review'` and `'Verified'` categories. |
| 362 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 363 | `rules.pl` |
| 364 | [source,prolog] |
| 365 | ---- |
| 366 | submit_rule(submit(CR, V)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 367 | CR = label('Code-Review', ok(_)), |
| 368 | V = label('Verified', ok(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 369 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 370 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 371 | Since for every change all label statuses are `'ok'` every change will be |
| 372 | submittable. Voting in the standard labels will be shown in the UI as the |
| 373 | standard label names are included in the return result. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 374 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 375 | === Example 3: Nothing is submittable |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 376 | This example shows how to make all changes non-submittable regardless of the |
| 377 | votes they have. |
| 378 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 379 | `rules.pl` |
| 380 | [source,prolog] |
| 381 | ---- |
| 382 | submit_rule(submit(R)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 383 | R = label('Any-Label-Name', reject(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 384 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 385 | |
| 386 | Since for any change we return only one label with status `reject`, no change |
| 387 | will be submittable. The UI will, however, not indicate what is needed for a |
| 388 | change to become submittable as we return no labels with status `need`. |
| 389 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 390 | === Example 4: Nothing is submittable but UI shows several 'Need ...' criteria |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 391 | In this example no change is submittable but here we show how to present 'Need |
| 392 | <label>' information to the user in the UI. |
| 393 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 394 | `rules.pl` |
| 395 | [source,prolog] |
| 396 | ---- |
| 397 | % In the UI this will show: Need Any-Label-Name |
| 398 | submit_rule(submit(N)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 399 | N = label('Any-Label-Name', need(_)). |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 400 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 401 | % We could define more "need" labels by adding more rules |
| 402 | submit_rule(submit(N)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 403 | N = label('Another-Label-Name', need(_)). |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 404 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 405 | % or by providing more than one need label in the same rule |
| 406 | submit_rule(submit(NX, NY)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 407 | NX = label('X-Label-Name', need(_)), |
| 408 | NY = label('Y-Label-Name', need(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 409 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 410 | |
| 411 | In the UI this will show: |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 412 | |
| 413 | * `Need Any-Label-Name` |
| 414 | * `Need Another-Label-Name` |
| 415 | * `Need X-Label-Name` |
| 416 | * `Need Y-Label-Name` |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 417 | |
| 418 | From the example above we can see a few more things: |
| 419 | |
| 420 | * comment in Prolog starts with the `%` character |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 421 | * there could be multiple `submit_rule` predicates. Since Prolog, by default, |
| 422 | tries to find all solutions for a query, the result will be union of all |
| 423 | solutions. Therefore, we see all 4 `need` labels in the UI. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 424 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 425 | === Example 5: The 'Need ...' labels not shown when change is submittable |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 426 | This example shows that, when there is a solution for `submit_rule(X)` where all |
| 427 | labels have status `ok` then Gerrit will not show any labels with the `need` |
| 428 | status from any of the previous `submit_rule(X)` solutions. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 429 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 430 | `rules.pl` |
| 431 | [source,prolog] |
| 432 | ---- |
| 433 | submit_rule(submit(N)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 434 | N = label('Some-Condition', need(_)). |
| 435 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 436 | submit_rule(submit(OK)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 437 | OK = label('Another-Condition', ok(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 438 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 439 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 440 | The `'Need Some-Condition'` will not be shown in the UI because of the result of |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 441 | the second rule. |
| 442 | |
| 443 | The same is valid if the two rules are swapped: |
| 444 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 445 | `rules.pl` |
| 446 | [source,prolog] |
| 447 | ---- |
| 448 | submit_rule(submit(OK)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 449 | OK = label('Another-Condition', ok(_)). |
| 450 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 451 | submit_rule(submit(N)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 452 | N = label('Some-Condition', need(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 453 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 454 | |
| 455 | The result of the first rule will stop search for any further solutions. |
| 456 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 457 | === Example 6: Make change submittable if commit author is "John Doe" |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 458 | This is the first example where we will use the Prolog facts about a change that |
| 459 | are automatically exposed by Gerrit. Our goal is to make any change submittable |
| 460 | when the commit author is named `'John Doe'`. In the very first |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 461 | step let's make sure Gerrit UI shows `'Need Author-is-John-Doe'` in |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 462 | the UI to clearly indicate to the user what is needed for a change to become |
| 463 | submittable: |
| 464 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 465 | `rules.pl` |
| 466 | [source,prolog] |
| 467 | ---- |
| 468 | submit_rule(submit(Author)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 469 | Author = label('Author-is-John-Doe', need(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 470 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 471 | |
| 472 | This will show: |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 473 | |
| 474 | * `Need Author-is-John-Doe` |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 475 | |
| 476 | in the UI but no change will be submittable yet. Let's add another rule: |
| 477 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 478 | `rules.pl` |
| 479 | [source,prolog] |
| 480 | ---- |
| 481 | submit_rule(submit(Author)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 482 | Author = label('Author-is-John-Doe', need(_)). |
| 483 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 484 | submit_rule(submit(Author)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 485 | gerrit:commit_author(_, 'John Doe', _), |
| 486 | Author = label('Author-is-John-Doe', ok(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 487 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 488 | |
| 489 | In the second rule we return `ok` status for the `'Author-is-John-Doe'` label |
| 490 | if there is a `commit_author` fact where the full name is `'John Doe'`. If |
| 491 | author of a change is `'John Doe'` then the second rule will return a solution |
| 492 | where all labels have `ok` status and the change will become submittable. If |
| 493 | author of a change is not `'John Doe'` then only the first rule will produce a |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 494 | solution. The UI will show `'Need Author-is-John-Doe'` but, as expected, the |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 495 | change will not be submittable. |
| 496 | |
| 497 | Instead of checking by full name we could also check by the email address: |
| 498 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 499 | `rules.pl` |
| 500 | [source,prolog] |
| 501 | ---- |
| 502 | submit_rule(submit(Author)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 503 | Author = label('Author-is-John-Doe', need(_)). |
| 504 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 505 | submit_rule(submit(Author)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 506 | gerrit:commit_author(_, _, 'john.doe@example.com'), |
| 507 | Author = label('Author-is-John-Doe', ok(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 508 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 509 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 510 | or by user id (assuming it is `1000000`): |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 511 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 512 | `rules.pl` |
| 513 | [source,prolog] |
| 514 | ---- |
| 515 | submit_rule(submit(Author)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 516 | Author = label('Author-is-John-Doe', need(_)). |
| 517 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 518 | submit_rule(submit(Author)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 519 | gerrit:commit_author(user(1000000), _, _), |
| 520 | Author = label('Author-is-John-Doe', ok(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 521 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 522 | |
| 523 | or by a combination of these 3 attributes: |
| 524 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 525 | `rules.pl` |
| 526 | [source,prolog] |
| 527 | ---- |
| 528 | submit_rule(submit(Author)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 529 | Author = label('Author-is-John-Doe', need(_)). |
| 530 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 531 | submit_rule(submit(Author)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 532 | gerrit:commit_author(_, 'John Doe', 'john.doe@example.com'), |
| 533 | Author = label('Author-is-John-Doe', ok(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 534 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 535 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 536 | === Example 7: Make change submittable if commit message starts with "Fix " |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 537 | Besides showing how to make use of the commit message text the purpose of this |
| 538 | example is also to show how to match only a part of a string symbol. Similarly |
| 539 | like commit author the commit message is provided as a string symbol which is |
| 540 | an atom in Prolog terms. When working with an atom we could only match against |
| 541 | the whole value. To match only part of a string symbol we have, at least, two |
| 542 | options: |
| 543 | |
| 544 | * convert the string symbol into a list of characters and then perform |
| 545 | the "classical" list matching |
| 546 | * use the `regex_matches/2` or, even more convenient, the |
| 547 | `gerrit:commit_message_matches/1` predicate |
| 548 | |
| 549 | Let's implement both options: |
| 550 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 551 | `rules.pl` |
| 552 | [source,prolog] |
| 553 | ---- |
| 554 | submit_rule(submit(Fix)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 555 | Fix = label('Commit-Message-starts-with-Fix', need(_)). |
| 556 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 557 | submit_rule(submit(Fix)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 558 | gerrit:commit_message(M), name(M, L), starts_with(L, "Fix "), |
| 559 | Fix = label('Commit-Message-starts-with-Fix', ok(_)). |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 560 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 561 | starts_with(L, []). |
| 562 | starts_with([H|T1], [H|T2]) :- starts_with(T1, T2). |
| 563 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 564 | |
| 565 | NOTE: The `name/2` embedded predicate is used to convert a string symbol into a |
| 566 | list of characters. A string `abc` is converted into a list of characters `[97, |
| 567 | 98, 99]`. A double quoted string in Prolog is just a shortcut for creating a |
| 568 | list of characters. `"abc"` is a shortcut for `[97, 98, 99]`. This is why we use |
| 569 | double quotes for the `"Trivial Fix"` in the example above. |
| 570 | |
| 571 | The `starts_with` predicate is self explaining. |
| 572 | |
| 573 | Using the `gerrit:commit_message_matches` predicate is probably more efficient: |
| 574 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 575 | `rules.pl` |
| 576 | [source,prolog] |
| 577 | ---- |
| 578 | submit_rule(submit(Fix)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 579 | Fix = label('Commit-Message-starts-with-Fix', need(_)). |
| 580 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 581 | submit_rule(submit(Fix)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 582 | gerrit:commit_message_matches('^Fix '), |
| 583 | Fix = label('Commit-Message-starts-with-Fix', ok(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 584 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 585 | |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 586 | The previous example could also be written so that it first checks if the commit |
| 587 | message starts with 'Fix '. If true then it sets OK for that category and stops |
| 588 | further backtracking by using the cut `!` operator: |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 589 | |
| 590 | `rules.pl` |
| 591 | [source,prolog] |
| 592 | ---- |
| 593 | submit_rule(submit(Fix)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 594 | gerrit:commit_message_matches('^Fix '), |
| 595 | Fix = label('Commit-Message-starts-with-Fix', ok(_)), |
| 596 | !. |
| 597 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 598 | % Message does not start with 'Fix ' so Fix is needed to submit |
| 599 | submit_rule(submit(Fix)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 600 | Fix = label('Commit-Message-starts-with-Fix', need(_)). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 601 | ---- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 602 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 603 | == The default submit policy |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 604 | All examples until now concentrate on one particular aspect of change data. |
| 605 | However, in real-life scenarios we would rather want to reuse Gerrit's default |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 606 | submit policy and extend/change it for our specific purpose. This could be |
| 607 | done in one of the following ways: |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 608 | |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 609 | * understand how the default submit policy is implemented and use that as a |
| 610 | template for implementing custom submit rules, |
| 611 | * invoke the default submit rule implementation and then perform further |
| 612 | actions on its return result. |
| 613 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 614 | === Default submit rule implementation |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 615 | The default submit rule with the two default categories, `Code-Review` and |
| 616 | `Verified`, can be implemented as: |
| 617 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 618 | `rules.pl` |
| 619 | [source,prolog] |
| 620 | ---- |
| 621 | submit_rule(submit(V, CR)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 622 | gerrit:max_with_block(-2, 2, 'Code-Review', CR), |
| 623 | gerrit:max_with_block(-1, 1, 'Verified', V). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 624 | ---- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 625 | |
| 626 | Once this implementation is understood it can be customized to implement |
| 627 | project specific submit rules. Note, that this implementation hardcodes |
| 628 | the two default categories. Introducing a new category in the database would |
| 629 | require introducing the same category here or a `submit_filter` in a parent |
| 630 | project would have to care about including the new category in the result of |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 631 | this `submit_rule`. On the other side, this example is easy to read and |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 632 | understand. |
| 633 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 634 | === Reusing the default submit policy |
David Pursehouse | 9246356 | 2013-06-24 10:16:28 +0900 | [diff] [blame] | 635 | To get results of Gerrit's default submit policy we use the |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 636 | `gerrit:default_submit` predicate. The `gerrit:default_submit(X)` includes all |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 637 | categories from the database. This means that if we write a submit rule like |
| 638 | this: |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 639 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 640 | `rules.pl` |
| 641 | [source,prolog] |
| 642 | ---- |
| 643 | submit_rule(X) :- gerrit:default_submit(X). |
| 644 | ---- |
| 645 | |
| 646 | it is equivalent to not using `rules.pl` at all. We just delegate to |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 647 | default logic. However, once we invoke the `gerrit:default_submit(X)` we can |
| 648 | perform further actions on the return result `X` and apply our specific |
| 649 | logic. The following pattern illustrates this technique: |
| 650 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 651 | `rules.pl` |
| 652 | [source,prolog] |
| 653 | ---- |
| 654 | submit_rule(S) :- gerrit:default_submit(R), project_specific_policy(R, S). |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 655 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 656 | project_specific_policy(R, S) :- ... |
| 657 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 658 | |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 659 | In the following examples both styles will be shown. |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 660 | |
Edwin Kempin | 1781adb | 2014-04-22 10:02:40 +0200 | [diff] [blame] | 661 | [[NonAuthorCodeReview]] |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 662 | === Example 8: Make change submittable only if `Code-Review+2` is given by a non author |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 663 | In this example we introduce a new label `Non-Author-Code-Review` and make it |
| 664 | satisfied if there is at least one `Code-Review+2` from a non author. All other |
| 665 | default policies like the `Verified` category and vetoing changes still apply. |
| 666 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 667 | ==== Reusing the `gerrit:default_submit` |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 668 | First, we invoke `gerrit:default_submit` to compute the result for the default |
| 669 | submit policy and then add the `Non-Author-Code-Review` label to it. The |
| 670 | `Non-Author-Code-Review` label is added with status `ok` if such an approval |
| 671 | exists or with status `need` if it doesn't exist. |
| 672 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 673 | `rules.pl` |
| 674 | [source,prolog] |
| 675 | ---- |
| 676 | submit_rule(S) :- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 677 | gerrit:default_submit(X), |
| 678 | X =.. [submit | Ls], |
| 679 | add_non_author_approval(Ls, R), |
| 680 | S =.. [submit | R]. |
| 681 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 682 | add_non_author_approval(S1, S2) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 683 | gerrit:commit_author(A), |
| 684 | gerrit:commit_label(label('Code-Review', 2), R), |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 685 | R \= A, !, |
| 686 | S2 = [label('Non-Author-Code-Review', ok(R)) | S1]. |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 687 | add_non_author_approval(S1, [label('Non-Author-Code-Review', need(_)) | S1]). |
| 688 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 689 | |
| 690 | This example uses the `univ` operator `=..` to "unpack" the result of the |
| 691 | default_submit, which is a structure of the form `submit(label('Code-Review', |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 692 | ok(_)), label('Verified', need(_)), ...)` into a list like `[submit, |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 693 | label('Code-Review', ok(_)), label('Verified', need(_)), ...]`. Then we |
| 694 | process the tail of the list (the list of labels) as a Prolog list, which is |
| 695 | much easier than processing a structure. In the end we use the same `univ` |
| 696 | operator to convert the resulting list of labels back into a `submit` structure |
| 697 | which is expected as a return result. The `univ` operator works both ways. |
| 698 | |
| 699 | In `add_non_author_approval` we use the `cut` operator `!` to prevent Prolog |
| 700 | from searching for more solutions once the `cut` point is reached. This is |
| 701 | important because in the second `add_non_author_approval` rule we just add the |
| 702 | `label('Non-Author-Code-Review', need(_))` without first checking that there |
| 703 | is no non author `Code-Review+2`. The second rule will only be reached |
| 704 | if the `cut` in the first rule is not reached and it only happens if a |
| 705 | predicate before the `cut` fails. |
| 706 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 707 | ==== Don't use `gerrit:default_submit` |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 708 | Let's implement the same submit rule the other way, without reusing the |
| 709 | `gerrit:default_submit`: |
| 710 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 711 | `rules.pl` |
| 712 | [source,prolog] |
| 713 | ---- |
| 714 | submit_rule(submit(CR, V)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 715 | base(CR, V), |
| 716 | CR = label(_, ok(Reviewer)), |
| 717 | gerrit:commit_author(Author), |
| 718 | Author \= Reviewer, |
| 719 | !. |
| 720 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 721 | submit_rule(submit(CR, V, N)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 722 | base(CR, V), |
| 723 | N = label('Non-Author-Code-Review', need(_)). |
| 724 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 725 | base(CR, V) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 726 | gerrit:max_with_block(-2, 2, 'Code-Review', CR), |
| 727 | gerrit:max_with_block(-1, 1, 'Verified', V). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 728 | ---- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 729 | |
| 730 | The latter implementation is probably easier to understand and the code looks |
| 731 | cleaner. Note, however, that the latter implementation will always return the |
| 732 | two standard categories only (`Code-Review` and `Verified`) even if a new |
David Pursehouse | 9246356 | 2013-06-24 10:16:28 +0900 | [diff] [blame] | 733 | category has been inserted into the database. To include the new category |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 734 | the `rules.pl` would need to be modified or a `submit_filter` in a parent |
| 735 | project would have to care about including the new category in the result |
| 736 | of this `submit_rule`. |
| 737 | |
| 738 | The former example, however, would include any newly added category as it |
| 739 | invokes the `gerrit:default_submit` and then modifies its result. |
| 740 | |
| 741 | Which of these two behaviors is desired will always depend on how a particular |
| 742 | Gerrit server is managed. |
| 743 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 744 | ==== Example 9: Remove the `Verified` category |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 745 | A project has no build and test. It consists of only text files and needs only |
| 746 | code review. We want to remove the `Verified` category from this project so |
| 747 | that `Code-Review+2` is the only criteria for a change to become submittable. |
| 748 | We also want the UI to not show the `Verified` category in the table with |
| 749 | votes and on the voting screen. |
| 750 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 751 | This is quite simple without reusing the `gerrit:default_submit`: |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 752 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 753 | `rules.pl` |
| 754 | [source,prolog] |
| 755 | ---- |
| 756 | submit_rule(submit(CR)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 757 | gerrit:max_with_block(-2, 2, 'Code-Review', CR). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 758 | ---- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 759 | |
| 760 | Implementing the same rule by reusing `gerrit:default_submit` is a bit more complex: |
| 761 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 762 | `rules.pl` |
| 763 | [source,prolog] |
| 764 | ---- |
| 765 | submit_rule(S) :- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 766 | gerrit:default_submit(X), |
| 767 | X =.. [submit | Ls], |
| 768 | remove_verified_category(Ls, R), |
| 769 | S =.. [submit | R]. |
| 770 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 771 | remove_verified_category([], []). |
| 772 | remove_verified_category([label('Verified', _) | T], R) :- remove_verified_category(T, R), !. |
| 773 | remove_verified_category([H|T], [H|R]) :- remove_verified_category(T, R). |
| 774 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 775 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 776 | === Example 10: Combine examples 8 and 9 |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 777 | In this example we want to both remove the verified and have the four eyes |
| 778 | principle. This means we want a combination of examples 7 and 8. |
| 779 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 780 | `rules.pl` |
| 781 | [source,prolog] |
| 782 | ---- |
| 783 | submit_rule(S) :- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 784 | gerrit:default_submit(X), |
| 785 | X =.. [submit | Ls], |
| 786 | remove_verified_category(Ls, R1), |
| 787 | add_non_author_approval(R1, R), |
| 788 | S =.. [submit | R]. |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 789 | ---- |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 790 | |
| 791 | The `remove_verified_category` and `add_non_author_approval` predicates are the |
| 792 | same as defined in the previous two examples. |
| 793 | |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 794 | Without reusing the `gerrit:default_submit` the same example may be implemented |
| 795 | as: |
| 796 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 797 | `rules.pl` |
| 798 | [source,prolog] |
| 799 | ---- |
| 800 | submit_rule(submit(CR)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 801 | base(CR), |
| 802 | CR = label(_, ok(Reviewer)), |
| 803 | gerrit:commit_author(Author), |
| 804 | Author \= Reviewer, |
| 805 | !. |
| 806 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 807 | submit_rule(submit(CR, N)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 808 | base(CR), |
| 809 | N = label('Non-Author-Code-Review', need(_)). |
| 810 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 811 | base(CR) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 812 | gerrit:max_with_block(-2, 2, 'Code-Review', CR), |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 813 | ---- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 814 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 815 | === Example 11: Remove the `Verified` category from all projects |
Sasa Zivkov | 3d4f0aa | 2012-08-07 15:11:32 +0200 | [diff] [blame] | 816 | Example 9, implements `submit_rule` that removes the `Verified` category from |
| 817 | one project. In this example we do the same but we want to remove the `Verified` |
| 818 | category from all projects. This means we have to implement `submit_filter` and |
| 819 | we have to do that in the `rules.pl` of the `All-Projects` project. |
| 820 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 821 | `rules.pl` |
| 822 | [source,prolog] |
| 823 | ---- |
| 824 | submit_filter(In, Out) :- |
Sasa Zivkov | 3d4f0aa | 2012-08-07 15:11:32 +0200 | [diff] [blame] | 825 | In =.. [submit | Ls], |
| 826 | remove_verified_category(Ls, R), |
| 827 | Out =.. [submit | R]. |
| 828 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 829 | remove_verified_category([], []). |
| 830 | remove_verified_category([label('Verified', _) | T], R) :- remove_verified_category(T, R), !. |
| 831 | remove_verified_category([H|T], [H|R]) :- remove_verified_category(T, R). |
| 832 | ---- |
Sasa Zivkov | 3d4f0aa | 2012-08-07 15:11:32 +0200 | [diff] [blame] | 833 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 834 | === Example 12: On release branches require DrNo in addition to project rules |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 835 | A new category 'DrNo' is added to the database and is required for release |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 836 | branches. To mark a branch as a release branch we use |
| 837 | `drno('refs/heads/branch')`. |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 838 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 839 | `rules.pl` |
| 840 | [source,prolog] |
| 841 | ---- |
| 842 | drno('refs/heads/master'). |
| 843 | drno('refs/heads/stable-2.3'). |
| 844 | drno('refs/heads/stable-2.4'). |
| 845 | drno('refs/heads/stable-2.5'). |
| 846 | drno('refs/heads/stable-2.5'). |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 847 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 848 | submit_filter(In, Out) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 849 | gerrit:change_branch(Branch), |
| 850 | drno(Branch), |
| 851 | !, |
| 852 | In =.. [submit | I], |
| 853 | gerrit:max_with_block(-1, 1, 'DrNo', DrNo), |
| 854 | Out =.. [submit, DrNo | I]. |
| 855 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 856 | submit_filter(In, Out) :- In = Out. |
| 857 | ---- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 858 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 859 | === Example 13: 1+1=2 Code-Review |
Johan Björk | ef02854 | 2012-10-24 12:06:33 -0400 | [diff] [blame] | 860 | In this example we introduce accumulative voting to determine if a change is |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 861 | submittable or not. We modify the standard `Code-Review` to be accumulative, and |
| 862 | make the change submittable if the total score is `2` or higher. |
Johan Björk | ef02854 | 2012-10-24 12:06:33 -0400 | [diff] [blame] | 863 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 864 | The code in this example is very similar to Example 8, with the addition of |
| 865 | `findall/3` and `gerrit:remove_label`. |
Johan Björk | ef02854 | 2012-10-24 12:06:33 -0400 | [diff] [blame] | 866 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 867 | The `findall/3` embedded predicate is used to form a list of all objects that |
| 868 | satisfy a specified Goal. In this example it is used to get a list of all the |
| 869 | `Code-Review` scores. `gerrit:remove_label` is a built-in helper that is |
| 870 | implemented similarly to the `remove_verified_category` as seen in the previous |
| 871 | example. |
Johan Björk | ef02854 | 2012-10-24 12:06:33 -0400 | [diff] [blame] | 872 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 873 | `rules.pl` |
| 874 | [source,prolog] |
| 875 | ---- |
| 876 | sum_list([], 0). |
| 877 | sum_list([H | Rest], Sum) :- sum_list(Rest,Tmp), Sum is H + Tmp. |
| 878 | |
| 879 | add_category_min_score(In, Category, Min, P) :- |
Johan Björk | ef02854 | 2012-10-24 12:06:33 -0400 | [diff] [blame] | 880 | findall(X, gerrit:commit_label(label(Category,X),R),Z), |
| 881 | sum_list(Z, Sum), |
| 882 | Sum >= Min, !, |
| 883 | P = [label(Category,ok(R)) | In]. |
| 884 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 885 | add_category_min_score(In, Category,Min,P) :- |
Johan Björk | ef02854 | 2012-10-24 12:06:33 -0400 | [diff] [blame] | 886 | P = [label(Category,need(Min)) | In]. |
| 887 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 888 | submit_rule(S) :- |
Johan Björk | ef02854 | 2012-10-24 12:06:33 -0400 | [diff] [blame] | 889 | gerrit:default_submit(X), |
| 890 | X =.. [submit | Ls], |
| 891 | gerrit:remove_label(Ls,label('Code-Review',_),NoCR), |
| 892 | add_category_min_score(NoCR,'Code-Review', 2, Labels), |
| 893 | S =.. [submit | Labels]. |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 894 | ---- |
Johan Björk | ef02854 | 2012-10-24 12:06:33 -0400 | [diff] [blame] | 895 | |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 896 | Implementing the same example without using `gerrit:default_submit`: |
| 897 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 898 | `rules.pl` |
| 899 | [source,prolog] |
| 900 | ---- |
| 901 | submit_rule(submit(CR, V)) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 902 | sum(2, 'Code-Review', CR), |
| 903 | gerrit:max_with_block(-1, 1, 'Verified', V). |
| 904 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 905 | % Sum the votes in a category. Uses a helper function score/2 |
| 906 | % to select out only the score values the given category. |
| 907 | sum(VotesNeeded, Category, label(Category, ok(_))) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 908 | findall(Score, score(Category, Score), All), |
| 909 | sum_list(All, Sum), |
| 910 | Sum >= VotesNeeded, |
| 911 | !. |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 912 | sum(VotesNeeded, Category, label(Category, need(VotesNeeded))). |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 913 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 914 | score(Category, Score) :- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 915 | gerrit:commit_label(label(Category, Score), User). |
| 916 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 917 | % Simple Prolog routine to sum a list of integers. |
| 918 | sum_list(List, Sum) :- sum_list(List, 0, Sum). |
| 919 | sum_list([X|T], Y, S) :- Z is X + Y, sum_list(T, Z, S). |
| 920 | sum_list([], S, S). |
| 921 | ---- |
Sasa Zivkov | 79b08cc | 2013-01-16 17:00:54 +0100 | [diff] [blame] | 922 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 923 | === Example 14: Master and apprentice |
Johan Björk | 87927a9 | 2012-10-25 15:00:36 -0400 | [diff] [blame] | 924 | The master and apprentice example allow you to specify a user (the `master`) |
| 925 | that must approve all changes done by another user (the `apprentice`). |
| 926 | |
| 927 | The code first checks if the commit author is in the apprentice database. |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 928 | If the commit is done by an `apprentice`, it will check if there is a `+2` |
Johan Björk | 87927a9 | 2012-10-25 15:00:36 -0400 | [diff] [blame] | 929 | review by the associated `master`. |
| 930 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 931 | `rules.pl` |
| 932 | [source,prolog] |
| 933 | ---- |
| 934 | % master_apprentice(Master, Apprentice). |
| 935 | % Extend this with appropriate user-id for your master/apprentice setup. |
| 936 | master_apprentice(user(1000064), user(1000000)). |
Johan Björk | 87927a9 | 2012-10-25 15:00:36 -0400 | [diff] [blame] | 937 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 938 | submit_rule(S) :- |
Johan Björk | 87927a9 | 2012-10-25 15:00:36 -0400 | [diff] [blame] | 939 | gerrit:default_submit(In), |
| 940 | In =.. [submit | Ls], |
| 941 | add_apprentice_master(Ls, R), |
| 942 | S =.. [submit | R]. |
| 943 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 944 | check_master_approval(S1, S2, Master) :- |
Johan Björk | 87927a9 | 2012-10-25 15:00:36 -0400 | [diff] [blame] | 945 | gerrit:commit_label(label('Code-Review', 2), R), |
| 946 | R = Master, !, |
| 947 | S2 = [label('Master-Approval', ok(R)) | S1]. |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 948 | check_master_approval(S1, [label('Master-Approval', need(_)) | S1], _). |
Johan Björk | 87927a9 | 2012-10-25 15:00:36 -0400 | [diff] [blame] | 949 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 950 | add_apprentice_master(S1, S2) :- |
Johan Björk | 87927a9 | 2012-10-25 15:00:36 -0400 | [diff] [blame] | 951 | gerrit:commit_author(Id), |
| 952 | master_apprentice(Master, Id), |
| 953 | !, |
| 954 | check_master_approval(S1, S2, Master). |
| 955 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 956 | add_apprentice_master(S, S). |
| 957 | ---- |
Johan Björk | 87927a9 | 2012-10-25 15:00:36 -0400 | [diff] [blame] | 958 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 959 | === Example 15: Only allow Author to submit change |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 960 | This example adds a new needed category `Only-Author-Can-Submit` for any user |
| 961 | that is not the author of the patch. This effectively blocks all users except |
| 962 | the author from submitting the change. This could result in an impossible |
| 963 | situation if the author does not have permissions for submitting the change. |
Johan Björk | 0f13254 | 2012-10-26 10:16:07 -0400 | [diff] [blame] | 964 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 965 | `rules.pl` |
| 966 | [source,prolog] |
| 967 | ---- |
| 968 | submit_rule(S) :- |
Johan Björk | 0f13254 | 2012-10-26 10:16:07 -0400 | [diff] [blame] | 969 | gerrit:default_submit(In), |
| 970 | In =.. [submit | Ls], |
| 971 | only_allow_author_to_submit(Ls, R), |
| 972 | S =.. [submit | R]. |
| 973 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 974 | only_allow_author_to_submit(S, S) :- |
Johan Björk | 0f13254 | 2012-10-26 10:16:07 -0400 | [diff] [blame] | 975 | gerrit:commit_author(Id), |
| 976 | gerrit:current_user(Id), |
| 977 | !. |
| 978 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 979 | only_allow_author_to_submit(S1, [label('Only-Author-Can-Submit', need(_)) | S1]). |
| 980 | ---- |
Johan Björk | 0f13254 | 2012-10-26 10:16:07 -0400 | [diff] [blame] | 981 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 982 | == Examples - Submit Type |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 983 | The following examples show how to implement own submit type rules. |
| 984 | |
Yuxuan 'fishy' Wang | 61698b1 | 2013-12-20 12:55:51 -0800 | [diff] [blame] | 985 | === Example 1: Set a `Cherry Pick` submit type for all changes |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 986 | This example sets the `Cherry Pick` submit type for all changes. It overrides |
| 987 | whatever is set as project default submit type. |
| 988 | |
| 989 | rules.pl |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 990 | [source,prolog] |
| 991 | ---- |
| 992 | submit_type(cherry_pick). |
| 993 | ---- |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 994 | |
Edwin Kempin | 1781adb | 2014-04-22 10:02:40 +0200 | [diff] [blame] | 995 | [[SubmitTypePerBranch]] |
Jonathan Nieder | 5758f18 | 2015-03-30 11:28:55 -0700 | [diff] [blame] | 996 | === Example 2: `Fast Forward Only` for all `+refs/heads/stable*+` branches |
| 997 | For all `+refs/heads/stable*+` branches we would like to enforce the `Fast |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 998 | Forward Only` submit type. A reason for this decision may be a need to never |
| 999 | break the build in the stable branches. For all other branches, those not |
Jonathan Nieder | 5758f18 | 2015-03-30 11:28:55 -0700 | [diff] [blame] | 1000 | matching the `+refs/heads/stable*+` pattern, we would like to use the project's |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 1001 | default submit type as defined on the project settings page. |
| 1002 | |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 1003 | `rules.pl` |
| 1004 | [source,prolog] |
| 1005 | ---- |
| 1006 | submit_type(fast_forward_only) :- |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 1007 | gerrit:change_branch(B), regex_matches('refs/heads/stable.*', B), |
| 1008 | !. |
Dave Borowitz | 6ba7976 | 2015-12-16 15:41:52 -0500 | [diff] [blame] | 1009 | submit_type(T) :- gerrit:project_default_submit_type(T). |
David Myllykangas | c5f591d | 2015-01-23 11:39:01 +0100 | [diff] [blame] | 1010 | ---- |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 1011 | |
| 1012 | The first `submit_type` predicate defines the `Fast Forward Only` submit type |
Jonathan Nieder | 5758f18 | 2015-03-30 11:28:55 -0700 | [diff] [blame] | 1013 | for `+refs/heads/stable.*+` branches. The second `submit_type` predicate returns |
Sasa Zivkov | b91296d | 2012-11-08 14:19:12 +0100 | [diff] [blame] | 1014 | the project's default submit type. |
| 1015 | |
Sasa Zivkov | ae50f71 | 2012-07-24 14:51:44 +0200 | [diff] [blame] | 1016 | GERRIT |
| 1017 | ------ |
| 1018 | Part of link:index.html[Gerrit Code Review] |
Yuxuan 'fishy' Wang | 99cb68d | 2013-10-31 17:26:00 -0700 | [diff] [blame] | 1019 | |
| 1020 | SEARCHBOX |
| 1021 | --------- |