2023-12-14 13:51:59 +00:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
|
|
|
|
//go:build !plan9
|
|
|
|
|
|
|
|
package kube
|
|
|
|
|
|
|
|
import (
|
|
|
|
"slices"
|
2024-02-13 05:27:54 +00:00
|
|
|
"time"
|
2023-12-14 13:51:59 +00:00
|
|
|
|
|
|
|
"go.uber.org/zap"
|
|
|
|
xslices "golang.org/x/exp/slices"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
tsapi "tailscale.com/k8s-operator/apis/v1alpha1"
|
|
|
|
"tailscale.com/tstime"
|
|
|
|
)
|
|
|
|
|
|
|
|
// SetConnectorCondition ensures that Connector status has a condition with the
|
|
|
|
// given attributes. LastTransitionTime gets set every time condition's status
|
2024-02-13 05:27:54 +00:00
|
|
|
// changes.
|
2023-12-14 13:51:59 +00:00
|
|
|
func SetConnectorCondition(cn *tsapi.Connector, conditionType tsapi.ConnectorConditionType, status metav1.ConditionStatus, reason, message string, gen int64, clock tstime.Clock, logger *zap.SugaredLogger) {
|
2024-02-13 05:27:54 +00:00
|
|
|
conds := updateCondition(cn.Status.Conditions, conditionType, status, reason, message, gen, clock, logger)
|
|
|
|
cn.Status.Conditions = conds
|
|
|
|
}
|
|
|
|
|
2024-04-30 20:18:23 +01:00
|
|
|
// RemoveConnectorCondition will remove condition of the given type if it exists.
|
2024-02-13 05:27:54 +00:00
|
|
|
func RemoveConnectorCondition(conn *tsapi.Connector, conditionType tsapi.ConnectorConditionType) {
|
|
|
|
conn.Status.Conditions = slices.DeleteFunc(conn.Status.Conditions, func(cond tsapi.ConnectorCondition) bool {
|
|
|
|
return cond.Type == conditionType
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetProxyClassCondition ensures that ProxyClass status has a condition with the
|
|
|
|
// given attributes. LastTransitionTime gets set every time condition's status
|
|
|
|
// changes.
|
|
|
|
func SetProxyClassCondition(pc *tsapi.ProxyClass, conditionType tsapi.ConnectorConditionType, status metav1.ConditionStatus, reason, message string, gen int64, clock tstime.Clock, logger *zap.SugaredLogger) {
|
|
|
|
conds := updateCondition(pc.Status.Conditions, conditionType, status, reason, message, gen, clock, logger)
|
|
|
|
pc.Status.Conditions = conds
|
|
|
|
}
|
|
|
|
|
2024-04-30 20:18:23 +01:00
|
|
|
// SetDNSConfigCondition ensures that DNSConfig status has a condition with the
|
|
|
|
// given attributes. LastTransitionTime gets set every time condition's status
|
|
|
|
// changes
|
|
|
|
func SetDNSConfigCondition(dnsCfg *tsapi.DNSConfig, conditionType tsapi.ConnectorConditionType, status metav1.ConditionStatus, reason, message string, gen int64, clock tstime.Clock, logger *zap.SugaredLogger) {
|
|
|
|
conds := updateCondition(dnsCfg.Status.Conditions, conditionType, status, reason, message, gen, clock, logger)
|
|
|
|
dnsCfg.Status.Conditions = conds
|
|
|
|
}
|
|
|
|
|
2024-02-13 05:27:54 +00:00
|
|
|
func updateCondition(conds []tsapi.ConnectorCondition, conditionType tsapi.ConnectorConditionType, status metav1.ConditionStatus, reason, message string, gen int64, clock tstime.Clock, logger *zap.SugaredLogger) []tsapi.ConnectorCondition {
|
2023-12-14 13:51:59 +00:00
|
|
|
newCondition := tsapi.ConnectorCondition{
|
|
|
|
Type: conditionType,
|
|
|
|
Status: status,
|
|
|
|
Reason: reason,
|
|
|
|
Message: message,
|
|
|
|
ObservedGeneration: gen,
|
|
|
|
}
|
|
|
|
|
2024-02-13 05:27:54 +00:00
|
|
|
nowTime := metav1.NewTime(clock.Now().Truncate(time.Second))
|
2023-12-14 13:51:59 +00:00
|
|
|
newCondition.LastTransitionTime = &nowTime
|
|
|
|
|
2024-02-13 05:27:54 +00:00
|
|
|
idx := xslices.IndexFunc(conds, func(cond tsapi.ConnectorCondition) bool {
|
2023-12-14 13:51:59 +00:00
|
|
|
return cond.Type == conditionType
|
|
|
|
})
|
|
|
|
|
|
|
|
if idx == -1 {
|
2024-02-13 05:27:54 +00:00
|
|
|
conds = append(conds, newCondition)
|
|
|
|
return conds
|
2023-12-14 13:51:59 +00:00
|
|
|
}
|
|
|
|
|
2024-02-13 05:27:54 +00:00
|
|
|
cond := conds[idx] // update the existing condition
|
2024-04-30 20:18:23 +01:00
|
|
|
|
|
|
|
// If this update doesn't contain a state transition, don't update last
|
|
|
|
// transition time.
|
2023-12-14 13:51:59 +00:00
|
|
|
if cond.Status == status {
|
|
|
|
newCondition.LastTransitionTime = cond.LastTransitionTime
|
|
|
|
} else {
|
2024-02-13 05:27:54 +00:00
|
|
|
logger.Infof("Status change for condition %s from %s to %s", conditionType, cond.Status, status)
|
2023-12-14 13:51:59 +00:00
|
|
|
}
|
2024-02-13 05:27:54 +00:00
|
|
|
conds[idx] = newCondition
|
|
|
|
return conds
|
2023-12-14 13:51:59 +00:00
|
|
|
}
|
|
|
|
|
2024-02-13 05:27:54 +00:00
|
|
|
func ProxyClassIsReady(pc *tsapi.ProxyClass) bool {
|
|
|
|
idx := xslices.IndexFunc(pc.Status.Conditions, func(cond tsapi.ConnectorCondition) bool {
|
|
|
|
return cond.Type == tsapi.ProxyClassready
|
2023-12-14 13:51:59 +00:00
|
|
|
})
|
2024-02-13 05:27:54 +00:00
|
|
|
if idx == -1 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
cond := pc.Status.Conditions[idx]
|
|
|
|
return cond.Status == metav1.ConditionTrue && cond.ObservedGeneration == pc.Generation
|
2023-12-14 13:51:59 +00:00
|
|
|
}
|
2024-04-30 20:18:23 +01:00
|
|
|
|
|
|
|
func DNSCfgIsReady(cfg *tsapi.DNSConfig) bool {
|
|
|
|
idx := xslices.IndexFunc(cfg.Status.Conditions, func(cond tsapi.ConnectorCondition) bool {
|
|
|
|
return cond.Type == tsapi.NameserverReady
|
|
|
|
})
|
|
|
|
if idx == -1 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
cond := cfg.Status.Conditions[idx]
|
|
|
|
return cond.Status == metav1.ConditionTrue && cond.ObservedGeneration == cfg.Generation
|
|
|
|
}
|