autoscaler/cluster-autoscaler/clusterstate/utils/status.go

143 lines
5.2 KiB
Go

/*
Copyright 2017 The Kubernetes 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 utils
import (
"errors"
"fmt"
"time"
kube_errors "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/client-go/tools/record"
apiv1 "k8s.io/kubernetes/pkg/api/v1"
kube_client "k8s.io/kubernetes/pkg/client/clientset_generated/clientset"
"github.com/golang/glog"
)
const (
// StatusConfigMapNamespace is the namespace where ConfigMap with status is stored.
StatusConfigMapNamespace = "kube-system"
// StatusConfigMapName is the name of ConfigMap with status.
StatusConfigMapName = "cluster-autoscaler-status"
// ConfigMapLastUpdatedKey is the name of annotation informing about status ConfigMap last update.
ConfigMapLastUpdatedKey = "cluster-autoscaler.kubernetes.io/last-updated"
)
// LogEventRecorder records events on some top-level object, to give user (without access to logs) a view of most important CA actions.
type LogEventRecorder struct {
recorder record.EventRecorder
statusObject runtime.Object
active bool
}
// Event record an event on underlying object. This does nothing if the underlying object is not set.
func (ler *LogEventRecorder) Event(eventtype, reason, message string) {
if ler.active && ler.statusObject != nil {
ler.recorder.Event(ler.statusObject, eventtype, reason, message)
}
}
// Eventf record an event on underlying object. This does nothing if the underlying object is not set.
func (ler *LogEventRecorder) Eventf(eventtype, reason, message string, args ...interface{}) {
if ler.active && ler.statusObject != nil {
ler.recorder.Eventf(ler.statusObject, eventtype, reason, message, args...)
}
}
// NewStatusMapRecorder creates a LogEventRecorder creating events on status configmap.
// If the configmap doesn't exist it will be created (with 'Initializing' status).
// If active == false the map will not be created and no events will be recorded.
func NewStatusMapRecorder(kubeClient kube_client.Interface, recorder record.EventRecorder, active bool) (*LogEventRecorder, error) {
var mapObj runtime.Object
var err error
if active {
mapObj, err = WriteStatusConfigMap(kubeClient, "Initializing", nil)
if err != nil {
return nil, errors.New("Failed to init status ConfigMap")
}
}
return &LogEventRecorder{
recorder: recorder,
statusObject: mapObj,
active: active,
}, nil
}
// WriteStatusConfigMap writes updates status ConfigMap with a given message or creates a new
// ConfigMap if it doesn't exist. If logRecorder is passed and configmap update is successfull
// logRecorder's internal reference will be updated.
func WriteStatusConfigMap(kubeClient kube_client.Interface, msg string, logRecorder *LogEventRecorder) (*apiv1.ConfigMap, error) {
statusUpdateTime := time.Now()
statusMsg := fmt.Sprintf("Cluster-autoscaler status at %v:\n%v", statusUpdateTime, msg)
var configMap *apiv1.ConfigMap
var getStatusError, writeStatusError error
var errMsg string
maps := kubeClient.CoreV1().ConfigMaps(StatusConfigMapNamespace)
configMap, getStatusError = maps.Get(StatusConfigMapName, metav1.GetOptions{})
if getStatusError == nil {
configMap.Data["status"] = statusMsg
if configMap.ObjectMeta.Annotations == nil {
configMap.ObjectMeta.Annotations = make(map[string]string)
}
configMap.ObjectMeta.Annotations[ConfigMapLastUpdatedKey] = fmt.Sprintf("%v", statusUpdateTime)
configMap, writeStatusError = maps.Update(configMap)
} else if kube_errors.IsNotFound(getStatusError) {
configMap = &apiv1.ConfigMap{
ObjectMeta: metav1.ObjectMeta{
Namespace: StatusConfigMapNamespace,
Name: StatusConfigMapName,
Annotations: map[string]string{
ConfigMapLastUpdatedKey: fmt.Sprintf("%v", statusUpdateTime),
},
},
Data: map[string]string{
"status": statusMsg,
},
}
configMap, writeStatusError = maps.Create(configMap)
} else {
errMsg = fmt.Sprintf("Failed to retrieve status configmap for update: %v", getStatusError)
}
if writeStatusError != nil {
errMsg = fmt.Sprintf("Failed to write status configmap: %v", writeStatusError)
}
if errMsg != "" {
glog.Error(errMsg)
return nil, errors.New(errMsg)
}
glog.V(8).Infof("Succesfully wrote status configmap with body \"%v\"", statusMsg)
// Having this as a side-effect is somewhat ugly
// But it makes error handling easier, as we get a free retry each loop
if logRecorder != nil {
logRecorder.statusObject = configMap
}
return configMap, nil
}
// DeleteStatusConfigMap deletes status configmap
func DeleteStatusConfigMap(kubeClient kube_client.Interface) error {
maps := kubeClient.CoreV1().ConfigMaps(StatusConfigMapNamespace)
err := maps.Delete(StatusConfigMapName, &metav1.DeleteOptions{})
if err != nil {
glog.Error("Failed to delete status configmap")
}
return err
}