forked from kubernetes-retired/contrib
/
stale-pending-ci.go
124 lines (100 loc) · 3.52 KB
/
stale-pending-ci.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
/*
Copyright 2016 The Kubernetes Authors All rights reserved.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package mungers
import (
"fmt"
"time"
"k8s.io/contrib/mungegithub/features"
"k8s.io/contrib/mungegithub/github"
"github.com/golang/glog"
githubapi "github.com/google/go-github/github"
"github.com/spf13/cobra"
)
const (
stalePendingCIHours = 24
pendingMsgFormat = `@k8s-bot test this issue: #IGNORE
Tests have been pending for %d hours`
)
var (
pendingMsgBody = fmt.Sprintf(pendingMsgFormat, stalePendingCIHours)
)
// StalePendingCI will ask the k8s-bot to test any PR with a LGTM that has
// been pending for more than 24 hours. This can happen when the jenkins VM
// is restarted.
//
// The real fix would be for the jenkins VM restart to not move every single
// PR to pending without actually testing...
//
// But this is our world and so we should really do this for all PRs which
// aren't likely to get another push (everything that is mergeable). Since that
// can be a lot of PRs, I'm just doing it for the LGTM PRs automatically...
//
// With minor modification this can be run easily by hand. Remove the LGTM check
// godep go build
// ./mungegithub --token-file=/PATH/TO/YOUR/TOKEN --pr-mungers=stale-pending-ci --once (--dry-run)
type StalePendingCI struct{}
func init() {
s := StalePendingCI{}
RegisterMungerOrDie(s)
registerShouldDeleteCommentFunc(s.isStaleComment)
}
// Name is the name usable in --pr-mungers
func (StalePendingCI) Name() string { return "stale-pending-ci" }
// RequiredFeatures is a slice of 'features' that must be provided
func (StalePendingCI) RequiredFeatures() []string { return []string{} }
// Initialize will initialize the munger
func (StalePendingCI) Initialize(config *github.Config, features *features.Features) error {
return nil
}
// EachLoop is called at the start of every munge loop
func (StalePendingCI) EachLoop() error { return nil }
// AddFlags will add any request flags to the cobra `cmd`
func (StalePendingCI) AddFlags(cmd *cobra.Command, config *github.Config) {}
// Munge is the workhorse the will actually make updates to the PR
func (StalePendingCI) Munge(obj *github.MungeObject) {
requiredContexts := []string{jenkinsUnitContext, jenkinsE2EContext}
if !obj.IsPR() {
return
}
if !obj.HasLabel(lgtmLabel) {
return
}
if mergeable, err := obj.IsMergeable(); !mergeable || err != nil {
return
}
status := obj.GetStatusState(requiredContexts)
if status != "pending" {
return
}
for _, context := range requiredContexts {
statusTime := obj.GetStatusTime(context)
if statusTime == nil {
glog.Errorf("%d: unable to determine time %q context was set", *obj.Issue.Number, context)
return
}
if time.Since(*statusTime) > stalePendingCIHours*time.Hour {
obj.WriteComment(pendingMsgBody)
return
}
}
}
func (StalePendingCI) isStaleComment(obj *github.MungeObject, comment *githubapi.IssueComment) bool {
if *comment.Body != pendingMsgBody {
return false
}
stale := commentBeforeLastCI(obj, comment)
if stale {
glog.V(6).Infof("Found stale StalePendingCI comment")
}
return stale
}