@@ -21,6 +21,7 @@ import (
21
21
"fmt"
22
22
"reflect"
23
23
24
+ "github.com/prometheus/client_golang/prometheus"
24
25
corev1 "k8s.io/api/core/v1"
25
26
"k8s.io/apimachinery/pkg/types"
26
27
"k8s.io/client-go/tools/record"
@@ -29,6 +30,7 @@ import (
29
30
"sigs.k8s.io/controller-runtime/pkg/builder"
30
31
"sigs.k8s.io/controller-runtime/pkg/client"
31
32
"sigs.k8s.io/controller-runtime/pkg/controller"
33
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
32
34
"sigs.k8s.io/controller-runtime/pkg/event"
33
35
"sigs.k8s.io/controller-runtime/pkg/handler"
34
36
ctrllog "sigs.k8s.io/controller-runtime/pkg/log"
@@ -37,9 +39,11 @@ import (
37
39
"sigs.k8s.io/controller-runtime/pkg/source"
38
40
39
41
networkingv1 "github.com/alibaba/hybridnet/pkg/apis/networking/v1"
42
+ "github.com/alibaba/hybridnet/pkg/constants"
40
43
"github.com/alibaba/hybridnet/pkg/controllers/concurrency"
41
44
"github.com/alibaba/hybridnet/pkg/controllers/utils"
42
45
ipamtypes "github.com/alibaba/hybridnet/pkg/ipam/types"
46
+ "github.com/alibaba/hybridnet/pkg/metrics"
43
47
)
44
48
45
49
const ControllerSubnetStatus = "SubnetStatus"
@@ -78,6 +82,17 @@ func (r *SubnetStatusReconciler) Reconcile(ctx context.Context, req ctrl.Request
78
82
return ctrl.Result {}, wrapError ("unable to fetch Subnet" , client .IgnoreNotFound (err ))
79
83
}
80
84
85
+ if subnet .DeletionTimestamp != nil {
86
+ cleanSubnetMetrics (subnet .Spec .Network , subnet .Name )
87
+ return ctrl.Result {}, wrapError ("unable to remove finalizer" , r .removeFinalizer (ctx , subnet ))
88
+
89
+ }
90
+
91
+ // make sure metrics will be un-registered before deletion
92
+ if err = r .addFinalizer (ctx , subnet ); err != nil {
93
+ return ctrl.Result {}, wrapError ("unable to add finalizer to subnet" , err )
94
+ }
95
+
81
96
// fetch subnet usage from manager
82
97
var usage * ipamtypes.Usage
83
98
if usage , err = r .IPAMManager .GetSubnetUsage (subnet .Spec .Network , subnet .Name ); err != nil {
@@ -99,6 +114,9 @@ func (r *SubnetStatusReconciler) Reconcile(ctx context.Context, req ctrl.Request
99
114
return ctrl.Result {}, nil
100
115
}
101
116
117
+ // update metrics
118
+ updateSubnetUsageMetrics (subnet .Spec .Network , subnet .Name , & subnet .Status )
119
+
102
120
// patch subnet status
103
121
subnetPatch := client .MergeFrom (subnet .DeepCopy ())
104
122
subnet .Status = * subnetStatus
@@ -114,15 +132,93 @@ func (r *SubnetStatusReconciler) Reconcile(ctx context.Context, req ctrl.Request
114
132
return ctrl.Result {}, nil
115
133
}
116
134
135
+ func updateSubnetUsageMetrics (networkName , subnetName string , subnetStatus * networkingv1.SubnetStatus ) {
136
+ if subnetStatus .Total > 0 {
137
+ metrics .SubnetIPUsageGauge .With (
138
+ prometheus.Labels {
139
+ "subnetName" : subnetName ,
140
+ "networkName" : networkName ,
141
+ "usageType" : metrics .IPTotalUsageType ,
142
+ },
143
+ ).Set (float64 (subnetStatus .Total ))
144
+
145
+ metrics .SubnetIPUsageGauge .With (
146
+ prometheus.Labels {
147
+ "subnetName" : subnetName ,
148
+ "networkName" : networkName ,
149
+ "usageType" : metrics .IPUsedUsageType ,
150
+ },
151
+ ).Set (float64 (subnetStatus .Used ))
152
+
153
+ metrics .SubnetIPUsageGauge .With (
154
+ prometheus.Labels {
155
+ "subnetName" : subnetName ,
156
+ "networkName" : networkName ,
157
+ "usageType" : metrics .IPAvailableUsageType ,
158
+ },
159
+ ).Set (float64 (subnetStatus .Available ))
160
+ }
161
+ }
162
+
163
+ func cleanSubnetMetrics (networkName , subnetName string ) {
164
+ _ = metrics .SubnetIPUsageGauge .Delete (
165
+ prometheus.Labels {
166
+ "subnetName" : subnetName ,
167
+ "networkName" : networkName ,
168
+ "usageType" : metrics .IPTotalUsageType ,
169
+ })
170
+
171
+ _ = metrics .SubnetIPUsageGauge .Delete (
172
+ prometheus.Labels {
173
+ "subnetName" : subnetName ,
174
+ "networkName" : networkName ,
175
+ "usageType" : metrics .IPUsedUsageType ,
176
+ })
177
+
178
+ _ = metrics .SubnetIPUsageGauge .Delete (
179
+ prometheus.Labels {
180
+ "subnetName" : subnetName ,
181
+ "networkName" : networkName ,
182
+ "usageType" : metrics .IPAvailableUsageType ,
183
+ })
184
+ }
185
+
186
+ func (r * SubnetStatusReconciler ) addFinalizer (ctx context.Context , subnet * networkingv1.Subnet ) error {
187
+ if controllerutil .ContainsFinalizer (subnet , constants .FinalizerMetricsRegistered ) {
188
+ return nil
189
+ }
190
+
191
+ patch := client .MergeFrom (subnet .DeepCopy ())
192
+ controllerutil .AddFinalizer (subnet , constants .FinalizerMetricsRegistered )
193
+ return retry .RetryOnConflict (retry .DefaultRetry , func () error {
194
+ return r .Patch (ctx , subnet , patch )
195
+ })
196
+ }
197
+
198
+ func (r * SubnetStatusReconciler ) removeFinalizer (ctx context.Context , subnet * networkingv1.Subnet ) error {
199
+ if ! controllerutil .ContainsFinalizer (subnet , constants .FinalizerMetricsRegistered ) {
200
+ return nil
201
+ }
202
+
203
+ patch := client .MergeFrom (subnet .DeepCopy ())
204
+ controllerutil .RemoveFinalizer (subnet , constants .FinalizerMetricsRegistered )
205
+ return retry .RetryOnConflict (retry .DefaultRetry , func () error {
206
+ return r .Patch (ctx , subnet , patch )
207
+ })
208
+ }
209
+
117
210
// SetupWithManager sets up the controller with the Manager.
118
211
func (r * SubnetStatusReconciler ) SetupWithManager (mgr ctrl.Manager ) error {
119
212
return ctrl .NewControllerManagedBy (mgr ).
120
213
Named (ControllerSubnetStatus ).
121
214
For (& networkingv1.Subnet {},
122
215
builder .WithPredicates (
123
- & predicate.GenerationChangedPredicate {},
124
216
& utils.IgnoreDeletePredicate {},
125
- & utils.SubnetSpecChangePredicate {},
217
+ & predicate.ResourceVersionChangedPredicate {},
218
+ predicate .Or (
219
+ & utils.SubnetSpecChangePredicate {},
220
+ & utils.TerminatingPredicate {},
221
+ ),
126
222
)).
127
223
Watches (& source.Kind {Type : & networkingv1.IPInstance {}},
128
224
handler .EnqueueRequestsFromMapFunc (func (object client.Object ) []reconcile.Request {
0 commit comments