2018-03-06 09:54:12 +00:00
|
|
|
//
|
|
|
|
// DISCLAIMER
|
|
|
|
//
|
2024-02-08 14:25:48 +00:00
|
|
|
// Copyright 2016-2024 ArangoDB GmbH, Cologne, Germany
|
2018-03-06 09:54:12 +00:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
// Copyright holder is ArangoDB GmbH, Cologne, Germany
|
|
|
|
//
|
|
|
|
|
|
|
|
package storage
|
|
|
|
|
|
|
|
import (
|
2021-03-23 15:47:28 +00:00
|
|
|
"context"
|
2018-03-29 14:44:23 +00:00
|
|
|
"time"
|
|
|
|
|
2022-06-30 18:39:07 +00:00
|
|
|
core "k8s.io/api/core/v1"
|
2023-06-06 22:53:56 +00:00
|
|
|
apiErrors "k8s.io/apimachinery/pkg/api/errors"
|
2022-06-30 18:39:07 +00:00
|
|
|
meta "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2023-06-06 22:53:56 +00:00
|
|
|
"k8s.io/apimachinery/pkg/types"
|
2022-07-11 11:49:47 +00:00
|
|
|
|
2023-06-06 22:53:56 +00:00
|
|
|
"github.com/arangodb/kube-arangodb/pkg/deployment/features"
|
|
|
|
"github.com/arangodb/kube-arangodb/pkg/deployment/patch"
|
2022-07-11 11:49:47 +00:00
|
|
|
"github.com/arangodb/kube-arangodb/pkg/util/errors"
|
2024-10-31 12:05:01 +00:00
|
|
|
"github.com/arangodb/kube-arangodb/pkg/util/k8sutil/list"
|
2018-03-06 09:54:12 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// inspectPVs queries all PersistentVolume's and triggers a cleanup for
|
|
|
|
// released volumes.
|
|
|
|
// Returns the number of available PV's.
|
|
|
|
func (ls *LocalStorage) inspectPVs() (int, error) {
|
2023-06-06 22:53:56 +00:00
|
|
|
var volumes []*core.PersistentVolume
|
|
|
|
|
2024-10-31 12:05:01 +00:00
|
|
|
volumes, err := list.APIList[*core.PersistentVolumeList](context.Background(), ls.deps.Client.Kubernetes().CoreV1().PersistentVolumes(), meta.ListOptions{}, func(result *core.PersistentVolumeList) []*core.PersistentVolume {
|
|
|
|
q := make([]*core.PersistentVolume, len(result.Items))
|
|
|
|
|
|
|
|
for id, e := range result.Items {
|
|
|
|
q[id] = e.DeepCopy()
|
2023-06-06 22:53:56 +00:00
|
|
|
}
|
|
|
|
|
2024-10-31 12:05:01 +00:00
|
|
|
return q
|
|
|
|
})
|
|
|
|
if err != nil {
|
2023-06-06 22:53:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, errors.WithStack(err)
|
|
|
|
}
|
2018-03-06 09:54:12 +00:00
|
|
|
}
|
|
|
|
spec := ls.apiObject.Spec
|
|
|
|
availableVolumes := 0
|
2018-03-29 14:44:23 +00:00
|
|
|
cleanupBeforeTimestamp := time.Now().Add(time.Hour * -24)
|
2023-06-06 22:53:56 +00:00
|
|
|
for _, pv := range volumes {
|
2018-03-06 09:54:12 +00:00
|
|
|
if pv.Spec.StorageClassName != spec.StorageClass.Name {
|
|
|
|
// Not our storage class
|
|
|
|
continue
|
|
|
|
}
|
2023-06-06 22:53:56 +00:00
|
|
|
|
|
|
|
// We are under deletion
|
|
|
|
if pv.DeletionTimestamp != nil {
|
|
|
|
// Do not remove object if we are not the owner
|
|
|
|
if ls.isOwnerOf(pv) {
|
|
|
|
ls.log.Str("name", pv.GetName()).Warn("PV is being deleted")
|
|
|
|
if err := ls.inspectPVFinalizer(pv); err != nil {
|
|
|
|
ls.log.Str("name", pv.GetName()).Warn("Unable to remove finalizers")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-03-06 09:54:12 +00:00
|
|
|
switch pv.Status.Phase {
|
2022-06-30 18:39:07 +00:00
|
|
|
case core.VolumeAvailable:
|
2018-03-29 14:44:23 +00:00
|
|
|
// Is this an old volume?
|
|
|
|
if pv.GetObjectMeta().GetCreationTimestamp().Time.Before(cleanupBeforeTimestamp) {
|
|
|
|
// Let's clean it up
|
2023-06-06 22:53:56 +00:00
|
|
|
if ls.isOwnerOf(pv) {
|
2018-03-29 14:44:23 +00:00
|
|
|
// Cleanup this volume
|
2023-06-06 22:53:56 +00:00
|
|
|
if features.LocalStorageReclaimPolicyPass().Enabled() {
|
|
|
|
ls.removePVObjectWithLog(pv)
|
|
|
|
} else {
|
|
|
|
ls.log.Str("name", pv.GetName()).Debug("Added PersistentVolume to cleaner")
|
|
|
|
ls.pvCleaner.Add(pv)
|
|
|
|
}
|
2018-03-29 14:44:23 +00:00
|
|
|
} else {
|
2022-06-14 07:26:07 +00:00
|
|
|
ls.log.Str("name", pv.GetName()).Debug("PersistentVolume is not owned by us")
|
2018-03-29 14:44:23 +00:00
|
|
|
availableVolumes++
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
availableVolumes++
|
|
|
|
}
|
2022-06-30 18:39:07 +00:00
|
|
|
case core.VolumeReleased:
|
2023-06-06 22:53:56 +00:00
|
|
|
if ls.isOwnerOf(pv) {
|
2018-03-06 09:54:12 +00:00
|
|
|
// Cleanup this volume
|
2023-06-06 22:53:56 +00:00
|
|
|
if !features.LocalStorageReclaimPolicyPass().Enabled() {
|
|
|
|
ls.log.Str("name", pv.GetName()).Debug("Added PersistentVolume to cleaner")
|
|
|
|
ls.pvCleaner.Add(pv)
|
|
|
|
} else {
|
|
|
|
if pv.Spec.PersistentVolumeReclaimPolicy == core.PersistentVolumeReclaimDelete {
|
|
|
|
// We have released PV, now delete it
|
|
|
|
ls.log.Str("name", pv.GetName()).Info("PV With ReclaimPolicy Delete in state Released found, deleting")
|
|
|
|
ls.removePVObjectWithLog(pv)
|
|
|
|
}
|
|
|
|
}
|
2018-03-06 14:37:25 +00:00
|
|
|
} else {
|
2022-06-14 07:26:07 +00:00
|
|
|
ls.log.Str("name", pv.GetName()).Debug("PersistentVolume is not owned by us")
|
2018-03-06 09:54:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return availableVolumes, nil
|
|
|
|
}
|
2023-06-06 22:53:56 +00:00
|
|
|
|
|
|
|
func (ls *LocalStorage) inspectPVFinalizer(pv *core.PersistentVolume) error {
|
|
|
|
currentFinalizers := pv.GetFinalizers()
|
|
|
|
if len(currentFinalizers) == 0 {
|
|
|
|
// No finalizers, nothing to do
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
finalizers := make([]string, 0, len(currentFinalizers))
|
|
|
|
|
|
|
|
for _, finalizer := range pv.GetFinalizers() {
|
|
|
|
switch finalizer {
|
|
|
|
case FinalizerPersistentVolumeCleanup:
|
|
|
|
ls.log.Str("name", pv.GetName()).Str("finalizer", FinalizerPersistentVolumeCleanup).Info("Removing finalizer")
|
|
|
|
if err := ls.removePVFinalizerPersistentVolumeCleanup(pv); err != nil {
|
|
|
|
ls.log.Err(err).Str("name", pv.GetName()).Warn("Unable to remove finalizer")
|
|
|
|
finalizers = append(finalizers, finalizer)
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
finalizers = append(finalizers, finalizer)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// No change in finalizers, all good
|
|
|
|
if len(finalizers) == len(currentFinalizers) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
p := patch.NewPatch()
|
|
|
|
if len(finalizers) == 0 {
|
|
|
|
// Remove them all
|
|
|
|
p.Add(patch.ItemRemove(patch.NewPath("metadata", "finalizers")))
|
|
|
|
} else {
|
|
|
|
p.Add(patch.ItemReplace(patch.NewPath("metadata", "finalizers"), finalizers))
|
|
|
|
}
|
|
|
|
|
|
|
|
data, err := p.Marshal()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := ls.deps.Client.Kubernetes().CoreV1().PersistentVolumes().Patch(context.Background(), pv.GetName(), types.JSONPatchType, data, meta.PatchOptions{}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ls *LocalStorage) removePVFinalizerPersistentVolumeCleanup(pv *core.PersistentVolume) error {
|
|
|
|
if !features.LocalStorageReclaimPolicyPass().Enabled() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find local path
|
|
|
|
localSource := pv.Spec.PersistentVolumeSource.Local
|
|
|
|
if localSource == nil {
|
2024-02-08 14:25:48 +00:00
|
|
|
return errors.WithStack(errors.Errorf("PersistentVolume has no local source"))
|
2023-06-06 22:53:56 +00:00
|
|
|
}
|
|
|
|
localPath := localSource.Path
|
|
|
|
|
|
|
|
// Find client that serves the node
|
|
|
|
nodeName := pv.GetAnnotations()[nodeNameAnnotation]
|
|
|
|
if nodeName == "" {
|
2024-02-08 14:25:48 +00:00
|
|
|
return errors.WithStack(errors.Errorf("PersistentVolume has no node-name annotation"))
|
2023-06-06 22:53:56 +00:00
|
|
|
}
|
|
|
|
client, err := ls.GetClientByNodeName(context.Background(), nodeName)
|
|
|
|
if err != nil {
|
|
|
|
ls.log.Err(err).Str("node", nodeName).Debug("Failed to get client for node")
|
|
|
|
return errors.WithStack(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clean volume through client
|
|
|
|
ctx := context.Background()
|
|
|
|
if err := client.Remove(ctx, localPath); err != nil {
|
|
|
|
ls.log.Err(err).
|
|
|
|
Str("node", nodeName).
|
|
|
|
Str("local-path", localPath).
|
|
|
|
Debug("Failed to remove local path")
|
|
|
|
return errors.WithStack(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ls *LocalStorage) removePVObjectWithLog(pv *core.PersistentVolume) {
|
|
|
|
if pv == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if pv.DeletionTimestamp != nil {
|
|
|
|
// Already deleting. nothing to do
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ls.removePVWithLog(pv.GetName(), string(pv.GetUID()))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ls *LocalStorage) removePVWithLog(name, uid string) {
|
|
|
|
if err := ls.removePV(name, uid); err != nil {
|
|
|
|
ls.log.Str("name", name).Err(err).Warn("PersistentVolume cannot be removed")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ls *LocalStorage) removePV(name, uid string) error {
|
|
|
|
if err := ls.deps.Client.Kubernetes().CoreV1().PersistentVolumes().Delete(context.Background(), name, meta.DeleteOptions{
|
|
|
|
Preconditions: meta.NewUIDPreconditions(uid),
|
|
|
|
}); err != nil {
|
|
|
|
if apiErrors.IsNotFound(err) {
|
|
|
|
// Do not remove if not found
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if apiErrors.IsConflict(err) {
|
|
|
|
// Do not throw error if uid changed
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|