-
Notifications
You must be signed in to change notification settings - Fork 1.2k
/
revision.go
123 lines (106 loc) · 4.73 KB
/
revision.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
/*
Copyright 2019 The Knative Authors
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 v1
import (
"context"
"fmt"
"knative.dev/pkg/reconciler"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/util/wait"
"knative.dev/pkg/test/logging"
"knative.dev/serving/pkg/apis/serving"
v1 "knative.dev/serving/pkg/apis/serving/v1"
"knative.dev/serving/test"
)
// WaitForRevisionState polls the status of the Revision called name
// from client every `PollInterval` until `inState` returns `true` indicating it
// is done, returns an error or timeout. desc will be used to name the metric
// that is emitted to track how long it took for name to get into the state checked by inState.
func WaitForRevisionState(client *test.ServingClients, name string, inState func(r *v1.Revision) (bool, error), desc string) error {
span := logging.GetEmitableSpan(context.Background(), fmt.Sprintf("WaitForRevision/%s/%s", name, desc))
defer span.End()
var lastState *v1.Revision
waitErr := wait.PollUntilContextTimeout(context.Background(), test.PollInterval, test.PollTimeout, true, func(context.Context) (bool, error) {
var err error
lastState, err = client.Revisions.Get(context.Background(), name, metav1.GetOptions{})
if err != nil {
return true, err
}
return inState(lastState)
})
if waitErr != nil {
return fmt.Errorf("revision %q is not in desired state, got: %+v: %w", name, lastState, waitErr)
}
return nil
}
// CheckRevisionState verifies the status of the Revision called name from client
// is in a particular state by calling `inState` and expecting `true`.
// This is the non-polling variety of WaitForRevisionState
func CheckRevisionState(client *test.ServingClients, name string, inState func(r *v1.Revision) (bool, error)) error {
r, err := client.Revisions.Get(context.Background(), name, metav1.GetOptions{})
if err != nil {
return err
}
done, err := inState(r)
if err != nil {
return err
}
if !done {
return fmt.Errorf("revision %q is not in desired state, got: %+v", name, r)
}
return nil
}
// IsRevisionReady will check the status conditions of the revision and return true if the revision is
// ready to serve traffic. It will return false if the status indicates a state other than deploying
// or being ready. It will also return false if the type of the condition is unexpected.
func IsRevisionReady(r *v1.Revision) (bool, error) {
return r.IsReady(), nil
}
// IsRevisionFailed will check the status condition sof the revision and return true if the revision is
// marked as failed, otherwise it will return false.
func IsRevisionFailed(r *v1.Revision) (bool, error) {
return r.IsFailed(), nil
}
// IsRevisionRoutingActive will check if the revision is actively routing to a route.
func IsRevisionRoutingActive(r *v1.Revision) (bool, error) {
routingState := r.Labels[serving.RoutingStateLabelKey]
return v1.RoutingState(routingState) == v1.RoutingStateActive, nil
}
// IsRevisionAtExpectedGeneration returns a function that will check if the annotations
// on the revision include an annotation for the generation and that the annotation is
// set to the expected value.
func IsRevisionAtExpectedGeneration(expectedGeneration string) func(r *v1.Revision) (bool, error) {
return func(r *v1.Revision) (bool, error) {
if a, ok := r.Labels[serving.ConfigurationGenerationLabelKey]; ok {
if a != expectedGeneration {
return true, fmt.Errorf("expected Revision %s to be labeled with generation %s but was %s instead", r.Name, expectedGeneration, a)
}
return true, nil
}
return true, fmt.Errorf("expected Revision %s to be labeled with generation %s but there was no label", r.Name, expectedGeneration)
}
}
// GetRevision return a revision by name
func GetRevision(clients *test.Clients, revisionName string) (revision *v1.Revision, err error) {
return revision, reconciler.RetryTestErrors(func(int) (err error) {
revision, err = clients.ServingClient.Revisions.Get(context.Background(), revisionName, metav1.GetOptions{})
return err
})
}
// GetRevisions return all the available revisions
func GetRevisions(clients *test.Clients) (list *v1.RevisionList, err error) {
return list, reconciler.RetryTestErrors(func(int) (err error) {
list, err = clients.ServingClient.Revisions.List(context.Background(), metav1.ListOptions{})
return err
})
}