forked from gnolang/gno
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconfig.go
137 lines (128 loc) · 4.59 KB
/
config.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
package config
import (
"github.com/gnolang/gno/contribs/github-bot/internal/client"
c "github.com/gnolang/gno/contribs/github-bot/internal/conditions"
r "github.com/gnolang/gno/contribs/github-bot/internal/requirements"
"github.com/gnolang/gno/contribs/github-bot/internal/utils"
)
type Teams []string
// Automatic check that will be performed by the bot.
type AutomaticCheck struct {
Description string
If c.Condition // If the condition is met, the rule is displayed and the requirement is executed.
Then r.Requirement // If the requirement is satisfied, the check passes.
}
// Manual check that will be performed by users.
type ManualCheck struct {
Description string
If c.Condition // If the condition is met, a checkbox will be displayed on bot comment.
Teams Teams // Members of these teams can check the checkbox to make the check pass.
}
// This is the description for a persistent rule with a non-standard behavior
// that allow maintainer to force the "success" state of the CI check
const ForceSkipDescription = "**IGNORE** the bot requirements for this PR (force green CI check)"
// This function returns the configuration of the bot consisting of automatic and manual checks
// in which the GitHub client is injected.
func Config(gh *client.GitHub) ([]AutomaticCheck, []ManualCheck) {
auto := []AutomaticCheck{
{
Description: "Maintainers must be able to edit this pull request ([more info](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/working-with-forks/allowing-changes-to-a-pull-request-branch-created-from-a-fork))",
If: c.And(
c.BaseBranch("^master$"),
c.CreatedFromFork(),
),
Then: r.MaintainerCanModify(),
},
{
Description: "Changes to 'docs' folder must be reviewed/authored by at least one devrel and one tech-staff",
If: c.And(
c.BaseBranch("^master$"),
c.FileChanged(gh, "^docs/"),
),
Then: r.And(
r.Or(
r.AuthorInTeam(gh, "tech-staff"),
r.ReviewByTeamMembers(gh, "tech-staff").WithDesiredState(utils.ReviewStateApproved),
),
r.Or(
r.AuthorInTeam(gh, "devrels"),
r.ReviewByTeamMembers(gh, "devrels").WithDesiredState(utils.ReviewStateApproved),
),
),
},
{
Description: "Changes related to gnoweb must be reviewed by its codeowners",
If: c.And(
c.BaseBranch("^master$"),
c.FileChanged(gh, "^gno.land/pkg/gnoweb/"),
),
Then: r.Or(
r.ReviewByUser(gh, "alexiscolin"),
r.ReviewByUser(gh, "gfanton"),
),
},
{
Description: "Must not contain the \"don't merge\" label",
If: c.Label("don't merge"),
Then: r.Never(),
},
{
Description: "Pending initial approval by a review team member, or review from tech-staff",
If: c.And(
c.BaseBranch("^master$"),
c.Not(c.AuthorInTeam(gh, "tech-staff")),
),
Then: r.
If(r.Or(
r.ReviewByOrgMembers(gh).WithDesiredState(utils.ReviewStateApproved),
r.ReviewByTeamMembers(gh, "tech-staff"),
r.Draft(),
)).
// Either there was a first approval from a member, and we
// assert that the label for triage-pending is removed...
Then(r.Not(r.Label(gh, "review/triage-pending", r.LabelRemove))).
// Or there was not, and we apply the triage pending label.
// The requirement should always fail, to mark the PR is not
// ready to be merged.
Else(r.And(r.Label(gh, "review/triage-pending", r.LabelApply), r.Never())),
},
}
manual := []ManualCheck{
{
// WARN: Do not edit this special rule which must remain persistent.
Description: ForceSkipDescription,
If: c.Always(),
},
{
Description: "The pull request description provides enough details",
If: c.And(
c.Not(c.AuthorInTeam(gh, "core-contributors")),
c.Not(c.Author("dependabot[bot]")),
),
Teams: Teams{"core-contributors"},
},
{
Description: "Determine if infra needs to be updated before merging",
If: c.And(
c.BaseBranch("^master$"),
c.Or(
c.FileChanged(gh, `Dockerfile`),
c.FileChanged(gh, `^misc/deployments`),
c.FileChanged(gh, `^misc/docker-`),
c.FileChanged(gh, `^.github/workflows/releaser.*\.yml$`),
c.FileChanged(gh, `^.github/workflows/portal-loop\.yml$`),
),
),
Teams: Teams{"devops"},
},
}
// Check for duplicates in manual rule descriptions (needs to be unique for the bot operations).
unique := make(map[string]struct{})
for _, rule := range manual {
if _, exists := unique[rule.Description]; exists {
gh.Logger.Fatalf("Manual rule descriptions must be unique (duplicate: %s)", rule.Description)
}
unique[rule.Description] = struct{}{}
}
return auto, manual
}