blob: d31c401931a06c88f78bb7cfd665ca1856afe372 [file] [log] [blame]
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -08001= Gerrit Code Review - Prolog Submit Rules Cookbook
Sasa Zivkovae50f712012-07-24 14:51:44 +02002
Edwin Kempin1781adb2014-04-22 10:02:40 +02003[[SubmitRule]]
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -08004== Submit Rule
David Myllykangasc5f591d2015-01-23 11:39:01 +01005A _Submit Rule_ in Gerrit is logic that defines when a change is submittable.
Sasa Zivkovae50f712012-07-24 14:51:44 +02006By default, a change is submittable when it gets at least one
7highest vote in each voting category and has no lowest vote (aka veto vote) in
David Myllykangasc5f591d2015-01-23 11:39:01 +01008any 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 Zivkovae50f712012-07-24 14:51:44 +020010submittable.
11
12While this rule is a good default, there are projects which need more
13flexibility for defining when a change is submittable. In Gerrit, it is
14possible to use Prolog based rules to provide project specific submit rules and
15replace the default submit rules. Using Prolog based rules, project owners can
16define a set of criteria which must be fulfilled for a change to become
17submittable. For a change that is not submittable, the set of needed criteria
18is displayed in the Gerrit UI.
19
20NOTE: Loading and executing Prolog submit rules may be disabled by setting
Jonathan Niederf0e6dc02016-02-29 13:57:23 -080021`rules.enable=false` in the Gerrit config file (see
Sasa Zivkovae50f712012-07-24 14:51:44 +020022link:config-gerrit.html#_a_id_rules_a_section_rules[rules section])
23
24link:https://groups.google.com/d/topic/repo-discuss/wJxTGhlHZMM/discussion[This
25discussion thread] explains why Prolog was chosen for the purpose of writing
26project specific submit rules.
27link:http://gerrit-documentation.googlecode.com/svn/ReleaseNotes/ReleaseNotes-2.2.2.html[Gerrit
282.2.2 ReleaseNotes] introduces Prolog support in Gerrit.
29
Edwin Kempin1781adb2014-04-22 10:02:40 +020030[[SubmitType]]
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -080031== Submit Type
David Myllykangasc5f591d2015-01-23 11:39:01 +010032A _Submit Type_ is a strategy that is used on submit to integrate the
Sasa Zivkovb91296d2012-11-08 14:19:12 +010033change 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 Myllykangasc5f591d2015-01-23 11:39:01 +010041_Submit Type_ is a project global setting. This means that the same submit type
Sasa Zivkovb91296d2012-11-08 14:19:12 +010042is used for all changes of one project.
43
44Projects which need more flexibility in choosing, or enforcing, a submit type
45can use Prolog based submit type which replaces the project's default submit
46type.
47
48Prolog based submit type computes a submit type for each change. The computed
49submit type is shown on the change screen for each change.
50
Dave Borowitz0761fd52015-12-13 21:59:04 -050051When submitting changes in a batch using "Submit including ancestors" or "Submit
52whole topic", submit type rules may not be used to mix submit types on a single
53branch, and trying to submit such a batch will fail. This avoids potentially
54confusing behavior and spurious submit failures. It is recommended to only use
55submit type rules to change submit types for an entire branch, which avoids this
56situation.
57
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -080058== Prolog Language
Sasa Zivkovae50f712012-07-24 14:51:44 +020059This document is not a complete Prolog tutorial.
60link:http://en.wikipedia.org/wiki/Prolog[This Wikipedia page on Prolog] is a
David Myllykangasc5f591d2015-01-23 11:39:01 +010061good starting point for learning the Prolog language. This document will only
62explain some elements of Prolog that are necessary to understand the provided
63examples.
Sasa Zivkovae50f712012-07-24 14:51:44 +020064
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -080065== Prolog in Gerrit
Shawn Pearce792b2c42015-06-12 17:25:40 -070066Gerrit uses its own link:https://gerrit.googlesource.com/prolog-cafe/[fork] of the
Sasa Zivkovae50f712012-07-24 14:51:44 +020067original link:http://kaminari.istc.kobe-u.ac.jp/PrologCafe/[prolog-cafe]
David Myllykangasc5f591d2015-01-23 11:39:01 +010068project. Gerrit embeds the prolog-cafe library and can interpret Prolog programs
69at runtime.
Sasa Zivkovae50f712012-07-24 14:51:44 +020070
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -080071== Interactive Prolog Cafe Shell
Sasa Zivkovae50f712012-07-24 14:51:44 +020072For interactive testing and playing with Prolog, Gerrit provides the
73link:pgm-prolog-shell.html[prolog-shell] program which opens an interactive
74Prolog interpreter shell.
75
Johan Björk2119f052012-10-24 12:10:45 -040076NOTE: The interactive shell is just a prolog shell, it does not load
David Myllykangasc5f591d2015-01-23 11:39:01 +010077a gerrit server environment and thus is not intended for
78xref:TestingSubmitRules[testing submit rules].
Sasa Zivkovae50f712012-07-24 14:51:44 +020079
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -080080== SWI-Prolog
Sasa Zivkovae50f712012-07-24 14:51:44 +020081Instead of using the link:pgm-prolog-shell.html[prolog-shell] program one can
82also use the link:http://www.swi-prolog.org/[SWI-Prolog] environment. It
83provides a better shell interface and a graphical source-level debugger.
84
Edwin Kempin1781adb2014-04-22 10:02:40 +020085[[RulesFile]]
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -080086== The rules.pl file
Sasa Zivkovae50f712012-07-24 14:51:44 +020087This section explains how to create and edit project specific submit rules. How
88to actually write the submit rules is explained in the next section.
89
90Project 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
92checkout the `refs/meta/config` branch in order to create or edit the `rules.pl`
93file:
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 Zivkovb91296d2012-11-08 14:19:12 +0100104[[HowToWriteSubmitRules]]
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800105== How to write submit rules
David Myllykangasc5f591d2015-01-23 11:39:01 +0100106Whenever Gerrit needs to evaluate submit rules for a change `C` from project `P`
107it will first initialize the embedded Prolog interpreter by:
Sasa Zivkovae50f712012-07-24 14:51:44 +0200108
109* consulting a set of facts about the change `C`
110* consulting the `rules.pl` from the project `P`
111
112Conceptually 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
114the 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
126have to use qualified names when writing our code and referencing these facts.
127For 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
132A complete set of facts which Gerrit provides about the change is listed in the
133link:prolog-change-facts.html[Prolog Facts for Gerrit Change].
134
135By default, Gerrit will search for a `submit_rule/1` predicate in the `rules.pl`
136file, evaluate the `submit_rule(X)` and then inspect the value of `X` in order
137to decide whether the change is submittable or not and also to find the set of
David Myllykangasc5f591d2015-01-23 11:39:01 +0100138needed criteria for the change to become submittable. This means that Gerrit has
139an expectation on the format and value of the result of the `submit_rule`
140predicate which is expected to be a `submit` term of the following format:
Sasa Zivkovae50f712012-07-24 14:51:44 +0200141
142====
143 submit(label(label-name, status) [, label(label-name, status)]*)
144====
145
146where `label-name` is usually `'Code-Review'` or `'Verified'` but could also
147be 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 Myllykangasc5f591d2015-01-23 11:39:01 +0100150 used to tell that this label/category has been met.
David Pursehouse3f918902015-05-19 09:41:02 +0900151* `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 Zivkovae50f712012-07-24 14:51:44 +0200159* `may(_)` allows expression of approval categories that are optional, i.e.
David Myllykangasc5f591d2015-01-23 11:39:01 +0100160 could either be set or unset without ever influencing whether the change
161 could be submitted.
Sasa Zivkovae50f712012-07-24 14:51:44 +0200162
163NOTE: For a change to be submittable all `label` terms contained in the returned
164`submit` term must have either `ok` or `may` status.
165
166IMPORTANT: Gerrit will let the Prolog engine continue searching for solutions of
167the `submit_rule(X)` query until it finds the first one where all labels in the
168return result have either status `ok` or `may` or there are no more solutions.
169If a solution where all labels have status `ok` is found then all previously
170found solutions are ignored. Otherwise, all labels names with status `need`
171from all solutions will be displayed in the UI indicating the set of conditions
172needed for the change to become submittable.
173
174Here 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 Myllykangasc5f591d2015-01-23 11:39:01 +0100187 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 Zivkovae50f712012-07-24 14:51:44 +0200192
193Of course, when implementing the `submit_rule` we will use the facts about the
194change that are already provided by Gerrit.
195
196Another aspect of the return result from the `submit_rule` predicate is that
197Gerrit uses it to decide which set of labels to display on the change review
198screen for voting. If the return result contains label `'ABC'` and if the label
Dave Borowitz01c1b1f2013-02-27 13:49:04 -0800199`'ABC'` is link:config-labels.html[defined for the project] then voting for the
200label `'ABC'` will be displayed. Otherwise, it is not displayed. Note that the
201project doesn't need a defined label for each label contained in the result of
David Myllykangasc5f591d2015-01-23 11:39:01 +0100202`submit_rule` predicate. For example, the decision whether
203`'Author-is-John-Doe'` label is met will probably not be made by explicit voting
204but, instead, by inspecting the facts about the change.
Sasa Zivkovae50f712012-07-24 14:51:44 +0200205
Sasa Zivkovb91296d2012-11-08 14:19:12 +0100206[[SubmitFilter]]
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800207== Submit Filter
Sasa Zivkov3d4f0aa2012-08-07 15:11:32 +0200208Another 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
210in the `rules.pl` file of the current project, the `submit_filter` will be
211searched for in the `rules.pl` of all parent projects of the current project,
212but not in the `rules.pl` of the current project. The search will start from the
213immediate parent of the current project, then in the parent project of that
David Myllykangasc5f591d2015-01-23 11:39:01 +0100214project and so on until, and including, the `'All-Projects'` project.
Sasa Zivkov3d4f0aa2012-08-07 15:11:32 +0200215
216The purpose of the submit filter is, as its name says, to filter the results
217of the `submit_rule`. Therefore, the `submit_filter` predicate has two
218parameters:
219
220====
221 submit_filter(In, Out) :- ...
222====
223
224Gerrit will invoke `submit_filter` with the `In` parameter containing a `submit`
225structure produced by the `submit_rule` and will take the value of the `Out`
226parameter as the result.
227
228The `Out` value of a `submit_filter` will become the `In` value for the
229next `submit_filter` in the parent line. The value of the `Out` parameter
230of the top-most `submit_filter` is the final result of the submit rule that
231is used to decide whether a change is submittable or not.
232
233IMPORTANT: `submit_filter` is a mechanism for Gerrit administrators to implement
234and enforce submit rules that would apply to all projects while `submit_rule` is
235a mechanism for project owners to implement project specific submit rules.
236However, 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
238implementing the `submit_filter` in this common parent project. This way they
239can avoid implementing the same `submit_rule` in all their projects.
240
241The following "drawing" illustrates the order of the invocation and the chaining
242of the results of the `submit_rule` and `submit_filter` predicates.
Sasa Zivkov3d4f0aa2012-08-07 15:11:32 +0200243====
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`
262project.
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
265a `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
268value of the submit rule evaluation.
269
270NOTE: If `MyProject` doesn't define its own `submit_rule` Gerrit will invoke the
271default implementation of submit rule that is named `gerrit:default_submit` and
272its result will be filtered as described above.
273
Edwin Kempinca24f5c2013-03-26 13:23:20 +0100274[[HowToWriteSubmitType]]
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800275== How to write submit type
Chris Lee849cbd82014-10-20 16:23:21 -0700276Writing custom submit type logic in Prolog is similar to
Sasa Zivkovb91296d2012-11-08 14:19:12 +0100277xref:HowToWriteSubmitRules[writing submit rules]. The only difference is that
278one has to implement a `submit_type` predicate (instead of the `submit_rule`)
279and that the return result of the `submit_type` has to be an atom that
280represents 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' Wang61698b12013-12-20 12:55:51 -0800288== Submit Type Filter
Sasa Zivkovb91296d2012-11-08 14:19:12 +0100289Submit type filter works the same way as the xref:SubmitFilter[Submit Filter]
290where the name of the filter predicate is `submit_type_filter`.
291
292====
293 submit_type_filter(In, Out).
294====
295
296Gerrit will invoke `submit_type_filter` with the `In` parameter containing a
297result of the `submit_type` and will take the value of the `Out` parameter as
298the result.
299
Johan Björk2119f052012-10-24 12:10:45 -0400300[[TestingSubmitRules]]
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800301== Testing submit rules
David Myllykangasc5f591d2015-01-23 11:39:01 +0100302The prolog environment running the `submit_rule` is loaded with state describing
303the change that is being evaluated. The easiest way to load this state is to
304test your `submit_rule` against a real change on a running gerrit instance. The
305command link:cmd-test-submit-rule.html[test-submit rule] loads a specific change
306and executes the `submit_rule`. It optionally reads the rule from from `stdin`
307to facilitate easy testing.
Johan Björk2119f052012-10-24 12:10:45 -0400308
309====
David Myllykangasc5f591d2015-01-23 11:39:01 +0100310 $ cat rules.pl | ssh gerrit_srv gerrit test-submit rule I45e080b105a50a625cc8e1fb5b357c0bfabe6d68 -s
Johan Björk2119f052012-10-24 12:10:45 -0400311====
Sasa Zivkov3d4f0aa2012-08-07 15:11:32 +0200312
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800313== Prolog vs Gerrit plugin for project specific submit rules
Sasa Zivkovae50f712012-07-24 14:51:44 +0200314Since version 2.5 Gerrit supports plugins and extension points. A plugin or an
315extension point could also be used as another means to provide custom submit
316rules. One could ask for a guideline when to use Prolog based submit rules and
317when to go for writing a new plugin. Writing a Prolog program is usually much
318faster than writing a Gerrit plugin. Prolog based submit rules can be pushed
319to a project by project owners while Gerrit plugins could only be installed by
320Gerrit administrators. In addition, Prolog based submit rules can be pushed
321for review by pushing to `refs/for/refs/meta/config` branch.
322
323On the other hand, Prolog based submit rules get a limited amount of facts about
324the change exposed to them. Gerrit plugins get full access to Gerrit internals
325and can potentially check more things than Prolog based rules.
326
Sasa Zivkovd0e55262013-01-16 14:26:06 +0100327From version 2.6 Gerrit plugins can contribute Prolog predicates. This way, we
328can make use of the plugin provided predicates when writing Prolog based rules.
329
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800330== Examples - Submit Rule
David Myllykangasc5f591d2015-01-23 11:39:01 +0100331The following examples should serve as a cookbook for developing own submit
332rules. Some of them are too trivial to be used in production and their only
333purpose is to provide step by step introduction and understanding.
Sasa Zivkovae50f712012-07-24 14:51:44 +0200334
Sasa Zivkov3d4f0aa2012-08-07 15:11:32 +0200335Some of the examples will implement the `submit_rule` and some will implement
336the `submit_filter` just to show both possibilities. Remember that
337`submit_rule` is only invoked from the current project and `submit_filter` is
338invoked from all parent projects. This is the most important fact in deciding
339whether to implement `submit_rule` or `submit_filter`.
340
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800341=== Example 1: Make every change submittable
David Myllykangasc5f591d2015-01-23 11:39:01 +0100342Let's start with a most trivial example where we would make every change
343submittable regardless of the votes it has:
Sasa Zivkovae50f712012-07-24 14:51:44 +0200344
David Myllykangasc5f591d2015-01-23 11:39:01 +0100345`rules.pl`
346[source,prolog]
347----
348submit_rule(submit(W)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100349 W = label('Any-Label-Name', ok(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100350----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200351
David Myllykangasc5f591d2015-01-23 11:39:01 +0100352In this case we make no use of facts about the change. We don't need it as we
353are simply making every change submittable. Note that, in this case, the Gerrit
354UI 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
356result. The `'Any-Label-Name'` could really be any string.
Sasa Zivkovae50f712012-07-24 14:51:44 +0200357
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800358=== Example 2: Every change submittable and voting in the standard categories possible
Sasa Zivkovae50f712012-07-24 14:51:44 +0200359This is continuation of the previous example where, in addition, to making
360every change submittable we want to enable voting in the standard
361`'Code-Review'` and `'Verified'` categories.
362
David Myllykangasc5f591d2015-01-23 11:39:01 +0100363`rules.pl`
364[source,prolog]
365----
366submit_rule(submit(CR, V)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100367 CR = label('Code-Review', ok(_)),
368 V = label('Verified', ok(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100369----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200370
David Myllykangasc5f591d2015-01-23 11:39:01 +0100371Since for every change all label statuses are `'ok'` every change will be
372submittable. Voting in the standard labels will be shown in the UI as the
373standard label names are included in the return result.
Sasa Zivkovae50f712012-07-24 14:51:44 +0200374
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800375=== Example 3: Nothing is submittable
Sasa Zivkovae50f712012-07-24 14:51:44 +0200376This example shows how to make all changes non-submittable regardless of the
377votes they have.
378
David Myllykangasc5f591d2015-01-23 11:39:01 +0100379`rules.pl`
380[source,prolog]
381----
382submit_rule(submit(R)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100383 R = label('Any-Label-Name', reject(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100384----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200385
386Since for any change we return only one label with status `reject`, no change
387will be submittable. The UI will, however, not indicate what is needed for a
388change to become submittable as we return no labels with status `need`.
389
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800390=== Example 4: Nothing is submittable but UI shows several 'Need ...' criteria
Sasa Zivkovae50f712012-07-24 14:51:44 +0200391In 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 Myllykangasc5f591d2015-01-23 11:39:01 +0100394`rules.pl`
395[source,prolog]
396----
397% In the UI this will show: Need Any-Label-Name
398submit_rule(submit(N)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100399 N = label('Any-Label-Name', need(_)).
Sasa Zivkovae50f712012-07-24 14:51:44 +0200400
David Myllykangasc5f591d2015-01-23 11:39:01 +0100401% We could define more "need" labels by adding more rules
402submit_rule(submit(N)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100403 N = label('Another-Label-Name', need(_)).
Sasa Zivkovae50f712012-07-24 14:51:44 +0200404
David Myllykangasc5f591d2015-01-23 11:39:01 +0100405% or by providing more than one need label in the same rule
406submit_rule(submit(NX, NY)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100407 NX = label('X-Label-Name', need(_)),
408 NY = label('Y-Label-Name', need(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100409----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200410
411In the UI this will show:
David Myllykangasc5f591d2015-01-23 11:39:01 +0100412
413* `Need Any-Label-Name`
414* `Need Another-Label-Name`
415* `Need X-Label-Name`
416* `Need Y-Label-Name`
Sasa Zivkovae50f712012-07-24 14:51:44 +0200417
418From the example above we can see a few more things:
419
420* comment in Prolog starts with the `%` character
David Myllykangasc5f591d2015-01-23 11:39:01 +0100421* 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 Zivkovae50f712012-07-24 14:51:44 +0200424
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800425=== Example 5: The 'Need ...' labels not shown when change is submittable
David Myllykangasc5f591d2015-01-23 11:39:01 +0100426This example shows that, when there is a solution for `submit_rule(X)` where all
427labels have status `ok` then Gerrit will not show any labels with the `need`
428status from any of the previous `submit_rule(X)` solutions.
Sasa Zivkovae50f712012-07-24 14:51:44 +0200429
David Myllykangasc5f591d2015-01-23 11:39:01 +0100430`rules.pl`
431[source,prolog]
432----
433submit_rule(submit(N)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100434 N = label('Some-Condition', need(_)).
435
David Myllykangasc5f591d2015-01-23 11:39:01 +0100436submit_rule(submit(OK)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100437 OK = label('Another-Condition', ok(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100438----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200439
David Myllykangasc5f591d2015-01-23 11:39:01 +0100440The `'Need Some-Condition'` will not be shown in the UI because of the result of
Sasa Zivkovae50f712012-07-24 14:51:44 +0200441the second rule.
442
443The same is valid if the two rules are swapped:
444
David Myllykangasc5f591d2015-01-23 11:39:01 +0100445`rules.pl`
446[source,prolog]
447----
448submit_rule(submit(OK)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100449 OK = label('Another-Condition', ok(_)).
450
David Myllykangasc5f591d2015-01-23 11:39:01 +0100451submit_rule(submit(N)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100452 N = label('Some-Condition', need(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100453----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200454
455The result of the first rule will stop search for any further solutions.
456
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800457=== Example 6: Make change submittable if commit author is "John Doe"
Sasa Zivkovae50f712012-07-24 14:51:44 +0200458This is the first example where we will use the Prolog facts about a change that
459are automatically exposed by Gerrit. Our goal is to make any change submittable
460when the commit author is named `'John Doe'`. In the very first
David Myllykangasc5f591d2015-01-23 11:39:01 +0100461step let's make sure Gerrit UI shows `'Need Author-is-John-Doe'` in
Sasa Zivkovae50f712012-07-24 14:51:44 +0200462the UI to clearly indicate to the user what is needed for a change to become
463submittable:
464
David Myllykangasc5f591d2015-01-23 11:39:01 +0100465`rules.pl`
466[source,prolog]
467----
468submit_rule(submit(Author)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100469 Author = label('Author-is-John-Doe', need(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100470----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200471
472This will show:
David Myllykangasc5f591d2015-01-23 11:39:01 +0100473
474* `Need Author-is-John-Doe`
Sasa Zivkovae50f712012-07-24 14:51:44 +0200475
476in the UI but no change will be submittable yet. Let's add another rule:
477
David Myllykangasc5f591d2015-01-23 11:39:01 +0100478`rules.pl`
479[source,prolog]
480----
481submit_rule(submit(Author)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100482 Author = label('Author-is-John-Doe', need(_)).
483
David Myllykangasc5f591d2015-01-23 11:39:01 +0100484submit_rule(submit(Author)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100485 gerrit:commit_author(_, 'John Doe', _),
486 Author = label('Author-is-John-Doe', ok(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100487----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200488
489In the second rule we return `ok` status for the `'Author-is-John-Doe'` label
490if there is a `commit_author` fact where the full name is `'John Doe'`. If
491author of a change is `'John Doe'` then the second rule will return a solution
492where all labels have `ok` status and the change will become submittable. If
493author of a change is not `'John Doe'` then only the first rule will produce a
David Myllykangasc5f591d2015-01-23 11:39:01 +0100494solution. The UI will show `'Need Author-is-John-Doe'` but, as expected, the
Sasa Zivkovae50f712012-07-24 14:51:44 +0200495change will not be submittable.
496
497Instead of checking by full name we could also check by the email address:
498
David Myllykangasc5f591d2015-01-23 11:39:01 +0100499`rules.pl`
500[source,prolog]
501----
502submit_rule(submit(Author)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100503 Author = label('Author-is-John-Doe', need(_)).
504
David Myllykangasc5f591d2015-01-23 11:39:01 +0100505submit_rule(submit(Author)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100506 gerrit:commit_author(_, _, 'john.doe@example.com'),
507 Author = label('Author-is-John-Doe', ok(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100508----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200509
David Myllykangasc5f591d2015-01-23 11:39:01 +0100510or by user id (assuming it is `1000000`):
Sasa Zivkovae50f712012-07-24 14:51:44 +0200511
David Myllykangasc5f591d2015-01-23 11:39:01 +0100512`rules.pl`
513[source,prolog]
514----
515submit_rule(submit(Author)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100516 Author = label('Author-is-John-Doe', need(_)).
517
David Myllykangasc5f591d2015-01-23 11:39:01 +0100518submit_rule(submit(Author)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100519 gerrit:commit_author(user(1000000), _, _),
520 Author = label('Author-is-John-Doe', ok(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100521----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200522
523or by a combination of these 3 attributes:
524
David Myllykangasc5f591d2015-01-23 11:39:01 +0100525`rules.pl`
526[source,prolog]
527----
528submit_rule(submit(Author)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100529 Author = label('Author-is-John-Doe', need(_)).
530
David Myllykangasc5f591d2015-01-23 11:39:01 +0100531submit_rule(submit(Author)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100532 gerrit:commit_author(_, 'John Doe', 'john.doe@example.com'),
533 Author = label('Author-is-John-Doe', ok(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100534----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200535
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800536=== Example 7: Make change submittable if commit message starts with "Fix "
Sasa Zivkovae50f712012-07-24 14:51:44 +0200537Besides showing how to make use of the commit message text the purpose of this
538example is also to show how to match only a part of a string symbol. Similarly
539like commit author the commit message is provided as a string symbol which is
540an atom in Prolog terms. When working with an atom we could only match against
541the whole value. To match only part of a string symbol we have, at least, two
542options:
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
549Let's implement both options:
550
David Myllykangasc5f591d2015-01-23 11:39:01 +0100551`rules.pl`
552[source,prolog]
553----
554submit_rule(submit(Fix)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100555 Fix = label('Commit-Message-starts-with-Fix', need(_)).
556
David Myllykangasc5f591d2015-01-23 11:39:01 +0100557submit_rule(submit(Fix)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100558 gerrit:commit_message(M), name(M, L), starts_with(L, "Fix "),
559 Fix = label('Commit-Message-starts-with-Fix', ok(_)).
Sasa Zivkovae50f712012-07-24 14:51:44 +0200560
David Myllykangasc5f591d2015-01-23 11:39:01 +0100561starts_with(L, []).
562starts_with([H|T1], [H|T2]) :- starts_with(T1, T2).
563----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200564
565NOTE: The `name/2` embedded predicate is used to convert a string symbol into a
566list of characters. A string `abc` is converted into a list of characters `[97,
56798, 99]`. A double quoted string in Prolog is just a shortcut for creating a
568list of characters. `"abc"` is a shortcut for `[97, 98, 99]`. This is why we use
569double quotes for the `"Trivial Fix"` in the example above.
570
571The `starts_with` predicate is self explaining.
572
573Using the `gerrit:commit_message_matches` predicate is probably more efficient:
574
David Myllykangasc5f591d2015-01-23 11:39:01 +0100575`rules.pl`
576[source,prolog]
577----
578submit_rule(submit(Fix)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100579 Fix = label('Commit-Message-starts-with-Fix', need(_)).
580
David Myllykangasc5f591d2015-01-23 11:39:01 +0100581submit_rule(submit(Fix)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100582 gerrit:commit_message_matches('^Fix '),
583 Fix = label('Commit-Message-starts-with-Fix', ok(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100584----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200585
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100586The previous example could also be written so that it first checks if the commit
587message starts with 'Fix '. If true then it sets OK for that category and stops
588further backtracking by using the cut `!` operator:
David Myllykangasc5f591d2015-01-23 11:39:01 +0100589
590`rules.pl`
591[source,prolog]
592----
593submit_rule(submit(Fix)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100594 gerrit:commit_message_matches('^Fix '),
595 Fix = label('Commit-Message-starts-with-Fix', ok(_)),
596 !.
597
David Myllykangasc5f591d2015-01-23 11:39:01 +0100598% Message does not start with 'Fix ' so Fix is needed to submit
599submit_rule(submit(Fix)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100600 Fix = label('Commit-Message-starts-with-Fix', need(_)).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100601----
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100602
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800603== The default submit policy
Sasa Zivkovae50f712012-07-24 14:51:44 +0200604All examples until now concentrate on one particular aspect of change data.
605However, in real-life scenarios we would rather want to reuse Gerrit's default
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100606submit policy and extend/change it for our specific purpose. This could be
607done in one of the following ways:
Sasa Zivkovae50f712012-07-24 14:51:44 +0200608
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100609* 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' Wang61698b12013-12-20 12:55:51 -0800614=== Default submit rule implementation
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100615The default submit rule with the two default categories, `Code-Review` and
616`Verified`, can be implemented as:
617
David Myllykangasc5f591d2015-01-23 11:39:01 +0100618`rules.pl`
619[source,prolog]
620----
621submit_rule(submit(V, CR)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100622 gerrit:max_with_block(-2, 2, 'Code-Review', CR),
623 gerrit:max_with_block(-1, 1, 'Verified', V).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100624----
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100625
626Once this implementation is understood it can be customized to implement
627project specific submit rules. Note, that this implementation hardcodes
628the two default categories. Introducing a new category in the database would
629require introducing the same category here or a `submit_filter` in a parent
630project would have to care about including the new category in the result of
David Myllykangasc5f591d2015-01-23 11:39:01 +0100631this `submit_rule`. On the other side, this example is easy to read and
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100632understand.
633
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800634=== Reusing the default submit policy
David Pursehouse92463562013-06-24 10:16:28 +0900635To get results of Gerrit's default submit policy we use the
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100636`gerrit:default_submit` predicate. The `gerrit:default_submit(X)` includes all
David Myllykangasc5f591d2015-01-23 11:39:01 +0100637categories from the database. This means that if we write a submit rule like
638this:
Sasa Zivkovae50f712012-07-24 14:51:44 +0200639
David Myllykangasc5f591d2015-01-23 11:39:01 +0100640`rules.pl`
641[source,prolog]
642----
643submit_rule(X) :- gerrit:default_submit(X).
644----
645
646it is equivalent to not using `rules.pl` at all. We just delegate to
Sasa Zivkovae50f712012-07-24 14:51:44 +0200647default logic. However, once we invoke the `gerrit:default_submit(X)` we can
648perform further actions on the return result `X` and apply our specific
649logic. The following pattern illustrates this technique:
650
David Myllykangasc5f591d2015-01-23 11:39:01 +0100651`rules.pl`
652[source,prolog]
653----
654submit_rule(S) :- gerrit:default_submit(R), project_specific_policy(R, S).
Sasa Zivkovae50f712012-07-24 14:51:44 +0200655
David Myllykangasc5f591d2015-01-23 11:39:01 +0100656project_specific_policy(R, S) :- ...
657----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200658
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100659In the following examples both styles will be shown.
Sasa Zivkovae50f712012-07-24 14:51:44 +0200660
Edwin Kempin1781adb2014-04-22 10:02:40 +0200661[[NonAuthorCodeReview]]
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800662=== Example 8: Make change submittable only if `Code-Review+2` is given by a non author
Sasa Zivkovae50f712012-07-24 14:51:44 +0200663In this example we introduce a new label `Non-Author-Code-Review` and make it
664satisfied if there is at least one `Code-Review+2` from a non author. All other
665default policies like the `Verified` category and vetoing changes still apply.
666
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800667==== Reusing the `gerrit:default_submit`
Sasa Zivkovae50f712012-07-24 14:51:44 +0200668First, we invoke `gerrit:default_submit` to compute the result for the default
669submit 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
671exists or with status `need` if it doesn't exist.
672
David Myllykangasc5f591d2015-01-23 11:39:01 +0100673`rules.pl`
674[source,prolog]
675----
676submit_rule(S) :-
Sasa Zivkovae50f712012-07-24 14:51:44 +0200677 gerrit:default_submit(X),
678 X =.. [submit | Ls],
679 add_non_author_approval(Ls, R),
680 S =.. [submit | R].
681
David Myllykangasc5f591d2015-01-23 11:39:01 +0100682add_non_author_approval(S1, S2) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100683 gerrit:commit_author(A),
684 gerrit:commit_label(label('Code-Review', 2), R),
Sasa Zivkovae50f712012-07-24 14:51:44 +0200685 R \= A, !,
686 S2 = [label('Non-Author-Code-Review', ok(R)) | S1].
David Myllykangasc5f591d2015-01-23 11:39:01 +0100687add_non_author_approval(S1, [label('Non-Author-Code-Review', need(_)) | S1]).
688----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200689
690This example uses the `univ` operator `=..` to "unpack" the result of the
691default_submit, which is a structure of the form `submit(label('Code-Review',
David Myllykangasc5f591d2015-01-23 11:39:01 +0100692ok(_)), label('Verified', need(_)), ...)` into a list like `[submit,
Sasa Zivkovae50f712012-07-24 14:51:44 +0200693label('Code-Review', ok(_)), label('Verified', need(_)), ...]`. Then we
694process the tail of the list (the list of labels) as a Prolog list, which is
695much easier than processing a structure. In the end we use the same `univ`
696operator to convert the resulting list of labels back into a `submit` structure
697which is expected as a return result. The `univ` operator works both ways.
698
699In `add_non_author_approval` we use the `cut` operator `!` to prevent Prolog
700from searching for more solutions once the `cut` point is reached. This is
701important 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
703is no non author `Code-Review+2`. The second rule will only be reached
704if the `cut` in the first rule is not reached and it only happens if a
705predicate before the `cut` fails.
706
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800707==== Don't use `gerrit:default_submit`
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100708Let's implement the same submit rule the other way, without reusing the
709`gerrit:default_submit`:
710
David Myllykangasc5f591d2015-01-23 11:39:01 +0100711`rules.pl`
712[source,prolog]
713----
714submit_rule(submit(CR, V)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100715 base(CR, V),
716 CR = label(_, ok(Reviewer)),
717 gerrit:commit_author(Author),
718 Author \= Reviewer,
719 !.
720
David Myllykangasc5f591d2015-01-23 11:39:01 +0100721submit_rule(submit(CR, V, N)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100722 base(CR, V),
723 N = label('Non-Author-Code-Review', need(_)).
724
David Myllykangasc5f591d2015-01-23 11:39:01 +0100725base(CR, V) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100726 gerrit:max_with_block(-2, 2, 'Code-Review', CR),
727 gerrit:max_with_block(-1, 1, 'Verified', V).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100728----
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100729
730The latter implementation is probably easier to understand and the code looks
731cleaner. Note, however, that the latter implementation will always return the
732two standard categories only (`Code-Review` and `Verified`) even if a new
David Pursehouse92463562013-06-24 10:16:28 +0900733category has been inserted into the database. To include the new category
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100734the `rules.pl` would need to be modified or a `submit_filter` in a parent
735project would have to care about including the new category in the result
736of this `submit_rule`.
737
738The former example, however, would include any newly added category as it
739invokes the `gerrit:default_submit` and then modifies its result.
740
741Which of these two behaviors is desired will always depend on how a particular
742Gerrit server is managed.
743
David Myllykangasc5f591d2015-01-23 11:39:01 +0100744==== Example 9: Remove the `Verified` category
Sasa Zivkovae50f712012-07-24 14:51:44 +0200745A project has no build and test. It consists of only text files and needs only
746code review. We want to remove the `Verified` category from this project so
747that `Code-Review+2` is the only criteria for a change to become submittable.
748We also want the UI to not show the `Verified` category in the table with
749votes and on the voting screen.
750
David Myllykangasc5f591d2015-01-23 11:39:01 +0100751This is quite simple without reusing the `gerrit:default_submit`:
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100752
David Myllykangasc5f591d2015-01-23 11:39:01 +0100753`rules.pl`
754[source,prolog]
755----
756submit_rule(submit(CR)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100757 gerrit:max_with_block(-2, 2, 'Code-Review', CR).
David Myllykangasc5f591d2015-01-23 11:39:01 +0100758----
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100759
760Implementing the same rule by reusing `gerrit:default_submit` is a bit more complex:
761
David Myllykangasc5f591d2015-01-23 11:39:01 +0100762`rules.pl`
763[source,prolog]
764----
765submit_rule(S) :-
Sasa Zivkovae50f712012-07-24 14:51:44 +0200766 gerrit:default_submit(X),
767 X =.. [submit | Ls],
768 remove_verified_category(Ls, R),
769 S =.. [submit | R].
770
David Myllykangasc5f591d2015-01-23 11:39:01 +0100771remove_verified_category([], []).
772remove_verified_category([label('Verified', _) | T], R) :- remove_verified_category(T, R), !.
773remove_verified_category([H|T], [H|R]) :- remove_verified_category(T, R).
774----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200775
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800776=== Example 10: Combine examples 8 and 9
Sasa Zivkovae50f712012-07-24 14:51:44 +0200777In this example we want to both remove the verified and have the four eyes
778principle. This means we want a combination of examples 7 and 8.
779
David Myllykangasc5f591d2015-01-23 11:39:01 +0100780`rules.pl`
781[source,prolog]
782----
783submit_rule(S) :-
Sasa Zivkovae50f712012-07-24 14:51:44 +0200784 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 Myllykangasc5f591d2015-01-23 11:39:01 +0100789----
Sasa Zivkovae50f712012-07-24 14:51:44 +0200790
791The `remove_verified_category` and `add_non_author_approval` predicates are the
792same as defined in the previous two examples.
793
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100794Without reusing the `gerrit:default_submit` the same example may be implemented
795as:
796
David Myllykangasc5f591d2015-01-23 11:39:01 +0100797`rules.pl`
798[source,prolog]
799----
800submit_rule(submit(CR)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100801 base(CR),
802 CR = label(_, ok(Reviewer)),
803 gerrit:commit_author(Author),
804 Author \= Reviewer,
805 !.
806
David Myllykangasc5f591d2015-01-23 11:39:01 +0100807submit_rule(submit(CR, N)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100808 base(CR),
809 N = label('Non-Author-Code-Review', need(_)).
810
David Myllykangasc5f591d2015-01-23 11:39:01 +0100811base(CR) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100812 gerrit:max_with_block(-2, 2, 'Code-Review', CR),
David Myllykangasc5f591d2015-01-23 11:39:01 +0100813----
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100814
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800815=== Example 11: Remove the `Verified` category from all projects
Sasa Zivkov3d4f0aa2012-08-07 15:11:32 +0200816Example 9, implements `submit_rule` that removes the `Verified` category from
817one project. In this example we do the same but we want to remove the `Verified`
818category from all projects. This means we have to implement `submit_filter` and
819we have to do that in the `rules.pl` of the `All-Projects` project.
820
David Myllykangasc5f591d2015-01-23 11:39:01 +0100821`rules.pl`
822[source,prolog]
823----
824submit_filter(In, Out) :-
Sasa Zivkov3d4f0aa2012-08-07 15:11:32 +0200825 In =.. [submit | Ls],
826 remove_verified_category(Ls, R),
827 Out =.. [submit | R].
828
David Myllykangasc5f591d2015-01-23 11:39:01 +0100829remove_verified_category([], []).
830remove_verified_category([label('Verified', _) | T], R) :- remove_verified_category(T, R), !.
831remove_verified_category([H|T], [H|R]) :- remove_verified_category(T, R).
832----
Sasa Zivkov3d4f0aa2012-08-07 15:11:32 +0200833
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800834=== Example 12: On release branches require DrNo in addition to project rules
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100835A new category 'DrNo' is added to the database and is required for release
David Myllykangasc5f591d2015-01-23 11:39:01 +0100836branches. To mark a branch as a release branch we use
837`drno('refs/heads/branch')`.
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100838
David Myllykangasc5f591d2015-01-23 11:39:01 +0100839`rules.pl`
840[source,prolog]
841----
842drno('refs/heads/master').
843drno('refs/heads/stable-2.3').
844drno('refs/heads/stable-2.4').
845drno('refs/heads/stable-2.5').
846drno('refs/heads/stable-2.5').
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100847
David Myllykangasc5f591d2015-01-23 11:39:01 +0100848submit_filter(In, Out) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100849 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 Myllykangasc5f591d2015-01-23 11:39:01 +0100856submit_filter(In, Out) :- In = Out.
857----
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100858
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800859=== Example 13: 1+1=2 Code-Review
Johan Björkef028542012-10-24 12:06:33 -0400860In this example we introduce accumulative voting to determine if a change is
David Myllykangasc5f591d2015-01-23 11:39:01 +0100861submittable or not. We modify the standard `Code-Review` to be accumulative, and
862make the change submittable if the total score is `2` or higher.
Johan Björkef028542012-10-24 12:06:33 -0400863
David Myllykangasc5f591d2015-01-23 11:39:01 +0100864The code in this example is very similar to Example 8, with the addition of
865`findall/3` and `gerrit:remove_label`.
Johan Björkef028542012-10-24 12:06:33 -0400866
David Myllykangasc5f591d2015-01-23 11:39:01 +0100867The `findall/3` embedded predicate is used to form a list of all objects that
868satisfy 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
870implemented similarly to the `remove_verified_category` as seen in the previous
871example.
Johan Björkef028542012-10-24 12:06:33 -0400872
David Myllykangasc5f591d2015-01-23 11:39:01 +0100873`rules.pl`
874[source,prolog]
875----
876sum_list([], 0).
877sum_list([H | Rest], Sum) :- sum_list(Rest,Tmp), Sum is H + Tmp.
878
879add_category_min_score(In, Category, Min, P) :-
Johan Björkef028542012-10-24 12:06:33 -0400880 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 Myllykangasc5f591d2015-01-23 11:39:01 +0100885add_category_min_score(In, Category,Min,P) :-
Johan Björkef028542012-10-24 12:06:33 -0400886 P = [label(Category,need(Min)) | In].
887
David Myllykangasc5f591d2015-01-23 11:39:01 +0100888submit_rule(S) :-
Johan Björkef028542012-10-24 12:06:33 -0400889 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 Myllykangasc5f591d2015-01-23 11:39:01 +0100894----
Johan Björkef028542012-10-24 12:06:33 -0400895
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100896Implementing the same example without using `gerrit:default_submit`:
897
David Myllykangasc5f591d2015-01-23 11:39:01 +0100898`rules.pl`
899[source,prolog]
900----
901submit_rule(submit(CR, V)) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100902 sum(2, 'Code-Review', CR),
903 gerrit:max_with_block(-1, 1, 'Verified', V).
904
David Myllykangasc5f591d2015-01-23 11:39:01 +0100905% Sum the votes in a category. Uses a helper function score/2
906% to select out only the score values the given category.
907sum(VotesNeeded, Category, label(Category, ok(_))) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100908 findall(Score, score(Category, Score), All),
909 sum_list(All, Sum),
910 Sum >= VotesNeeded,
911 !.
David Myllykangasc5f591d2015-01-23 11:39:01 +0100912sum(VotesNeeded, Category, label(Category, need(VotesNeeded))).
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100913
David Myllykangasc5f591d2015-01-23 11:39:01 +0100914score(Category, Score) :-
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100915 gerrit:commit_label(label(Category, Score), User).
916
David Myllykangasc5f591d2015-01-23 11:39:01 +0100917% Simple Prolog routine to sum a list of integers.
918sum_list(List, Sum) :- sum_list(List, 0, Sum).
919sum_list([X|T], Y, S) :- Z is X + Y, sum_list(T, Z, S).
920sum_list([], S, S).
921----
Sasa Zivkov79b08cc2013-01-16 17:00:54 +0100922
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800923=== Example 14: Master and apprentice
Johan Björk87927a92012-10-25 15:00:36 -0400924The master and apprentice example allow you to specify a user (the `master`)
925that must approve all changes done by another user (the `apprentice`).
926
927The code first checks if the commit author is in the apprentice database.
David Myllykangasc5f591d2015-01-23 11:39:01 +0100928If the commit is done by an `apprentice`, it will check if there is a `+2`
Johan Björk87927a92012-10-25 15:00:36 -0400929review by the associated `master`.
930
David Myllykangasc5f591d2015-01-23 11:39:01 +0100931`rules.pl`
932[source,prolog]
933----
934% master_apprentice(Master, Apprentice).
935% Extend this with appropriate user-id for your master/apprentice setup.
936master_apprentice(user(1000064), user(1000000)).
Johan Björk87927a92012-10-25 15:00:36 -0400937
David Myllykangasc5f591d2015-01-23 11:39:01 +0100938submit_rule(S) :-
Johan Björk87927a92012-10-25 15:00:36 -0400939 gerrit:default_submit(In),
940 In =.. [submit | Ls],
941 add_apprentice_master(Ls, R),
942 S =.. [submit | R].
943
David Myllykangasc5f591d2015-01-23 11:39:01 +0100944check_master_approval(S1, S2, Master) :-
Johan Björk87927a92012-10-25 15:00:36 -0400945 gerrit:commit_label(label('Code-Review', 2), R),
946 R = Master, !,
947 S2 = [label('Master-Approval', ok(R)) | S1].
David Myllykangasc5f591d2015-01-23 11:39:01 +0100948check_master_approval(S1, [label('Master-Approval', need(_)) | S1], _).
Johan Björk87927a92012-10-25 15:00:36 -0400949
David Myllykangasc5f591d2015-01-23 11:39:01 +0100950add_apprentice_master(S1, S2) :-
Johan Björk87927a92012-10-25 15:00:36 -0400951 gerrit:commit_author(Id),
952 master_apprentice(Master, Id),
953 !,
954 check_master_approval(S1, S2, Master).
955
David Myllykangasc5f591d2015-01-23 11:39:01 +0100956add_apprentice_master(S, S).
957----
Johan Björk87927a92012-10-25 15:00:36 -0400958
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800959=== Example 15: Only allow Author to submit change
David Myllykangasc5f591d2015-01-23 11:39:01 +0100960This example adds a new needed category `Only-Author-Can-Submit` for any user
961that is not the author of the patch. This effectively blocks all users except
962the author from submitting the change. This could result in an impossible
963situation if the author does not have permissions for submitting the change.
Johan Björk0f132542012-10-26 10:16:07 -0400964
David Myllykangasc5f591d2015-01-23 11:39:01 +0100965`rules.pl`
966[source,prolog]
967----
968submit_rule(S) :-
Johan Björk0f132542012-10-26 10:16:07 -0400969 gerrit:default_submit(In),
970 In =.. [submit | Ls],
971 only_allow_author_to_submit(Ls, R),
972 S =.. [submit | R].
973
David Myllykangasc5f591d2015-01-23 11:39:01 +0100974only_allow_author_to_submit(S, S) :-
Johan Björk0f132542012-10-26 10:16:07 -0400975 gerrit:commit_author(Id),
976 gerrit:current_user(Id),
977 !.
978
David Myllykangasc5f591d2015-01-23 11:39:01 +0100979only_allow_author_to_submit(S1, [label('Only-Author-Can-Submit', need(_)) | S1]).
980----
Johan Björk0f132542012-10-26 10:16:07 -0400981
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800982== Examples - Submit Type
Sasa Zivkovb91296d2012-11-08 14:19:12 +0100983The following examples show how to implement own submit type rules.
984
Yuxuan 'fishy' Wang61698b12013-12-20 12:55:51 -0800985=== Example 1: Set a `Cherry Pick` submit type for all changes
Sasa Zivkovb91296d2012-11-08 14:19:12 +0100986This example sets the `Cherry Pick` submit type for all changes. It overrides
987whatever is set as project default submit type.
988
989rules.pl
David Myllykangasc5f591d2015-01-23 11:39:01 +0100990[source,prolog]
991----
992submit_type(cherry_pick).
993----
Sasa Zivkovb91296d2012-11-08 14:19:12 +0100994
Edwin Kempin1781adb2014-04-22 10:02:40 +0200995[[SubmitTypePerBranch]]
Jonathan Nieder5758f182015-03-30 11:28:55 -0700996=== Example 2: `Fast Forward Only` for all `+refs/heads/stable*+` branches
997For all `+refs/heads/stable*+` branches we would like to enforce the `Fast
Sasa Zivkovb91296d2012-11-08 14:19:12 +0100998Forward Only` submit type. A reason for this decision may be a need to never
999break the build in the stable branches. For all other branches, those not
Jonathan Nieder5758f182015-03-30 11:28:55 -07001000matching the `+refs/heads/stable*+` pattern, we would like to use the project's
Sasa Zivkovb91296d2012-11-08 14:19:12 +01001001default submit type as defined on the project settings page.
1002
David Myllykangasc5f591d2015-01-23 11:39:01 +01001003`rules.pl`
1004[source,prolog]
1005----
1006submit_type(fast_forward_only) :-
Sasa Zivkovb91296d2012-11-08 14:19:12 +01001007 gerrit:change_branch(B), regex_matches('refs/heads/stable.*', B),
1008 !.
Dave Borowitz6ba79762015-12-16 15:41:52 -05001009submit_type(T) :- gerrit:project_default_submit_type(T).
David Myllykangasc5f591d2015-01-23 11:39:01 +01001010----
Sasa Zivkovb91296d2012-11-08 14:19:12 +01001011
1012The first `submit_type` predicate defines the `Fast Forward Only` submit type
Jonathan Nieder5758f182015-03-30 11:28:55 -07001013for `+refs/heads/stable.*+` branches. The second `submit_type` predicate returns
Sasa Zivkovb91296d2012-11-08 14:19:12 +01001014the project's default submit type.
1015
Sasa Zivkovae50f712012-07-24 14:51:44 +02001016GERRIT
1017------
1018Part of link:index.html[Gerrit Code Review]
Yuxuan 'fishy' Wang99cb68d2013-10-31 17:26:00 -07001019
1020SEARCHBOX
1021---------