mirror of
https://github.com/optim-enterprises-bv/kubernetes.git
synced 2025-11-12 09:36:36 +00:00
We are going to use PodBackoff for controlling backoff when adding unschedulable pods back to the active scheduling queue. In order to do this more easily, limit the interface for PodBackoff to only this struct (rather than exposing BackoffEntry) and change the backing expiry implementation to be queue based.
198 lines
5.4 KiB
Go
198 lines
5.4 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 util
|
|
|
|
import (
|
|
"fmt"
|
|
"sync"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
ktypes "k8s.io/apimachinery/pkg/types"
|
|
|
|
"k8s.io/klog"
|
|
)
|
|
|
|
type clock interface {
|
|
Now() time.Time
|
|
}
|
|
|
|
type realClock struct{}
|
|
|
|
func (realClock) Now() time.Time {
|
|
return time.Now()
|
|
}
|
|
|
|
// backoffEntry is single threaded. in particular, it only allows a single action to be waiting on backoff at a time.
|
|
// It is also not safe to copy this object.
|
|
type backoffEntry struct {
|
|
initialized bool
|
|
podName ktypes.NamespacedName
|
|
backoff time.Duration
|
|
lastUpdate time.Time
|
|
reqInFlight int32
|
|
}
|
|
|
|
// tryLock attempts to acquire a lock via atomic compare and swap.
|
|
// returns true if the lock was acquired, false otherwise
|
|
func (b *backoffEntry) tryLock() bool {
|
|
return atomic.CompareAndSwapInt32(&b.reqInFlight, 0, 1)
|
|
}
|
|
|
|
// unlock returns the lock. panics if the lock isn't held
|
|
func (b *backoffEntry) unlock() {
|
|
if !atomic.CompareAndSwapInt32(&b.reqInFlight, 1, 0) {
|
|
panic(fmt.Sprintf("unexpected state on unlocking: %+v", b))
|
|
}
|
|
}
|
|
|
|
// backoffTime returns the Time when a backoffEntry completes backoff
|
|
func (b *backoffEntry) backoffTime() time.Time {
|
|
return b.lastUpdate.Add(b.backoff)
|
|
}
|
|
|
|
// getBackoff returns the duration until this entry completes backoff
|
|
func (b *backoffEntry) getBackoff(maxDuration time.Duration) time.Duration {
|
|
if !b.initialized {
|
|
b.initialized = true
|
|
return b.backoff
|
|
}
|
|
newDuration := b.backoff * 2
|
|
if newDuration > maxDuration {
|
|
newDuration = maxDuration
|
|
}
|
|
b.backoff = newDuration
|
|
klog.V(4).Infof("Backing off %s", newDuration.String())
|
|
return newDuration
|
|
}
|
|
|
|
// backoffAndWait Blocks until this entry has completed backoff
|
|
func (b *backoffEntry) backoffAndWait(maxDuration time.Duration) {
|
|
time.Sleep(b.getBackoff(maxDuration))
|
|
}
|
|
|
|
// PodBackoff is used to restart a pod with back-off delay.
|
|
type PodBackoff struct {
|
|
// expiryQ stores backoffEntry orderedy by lastUpdate until they reach maxDuration and are GC'd
|
|
expiryQ *Heap
|
|
lock sync.Mutex
|
|
clock clock
|
|
defaultDuration time.Duration
|
|
maxDuration time.Duration
|
|
}
|
|
|
|
// MaxDuration returns the max time duration of the back-off.
|
|
func (p *PodBackoff) MaxDuration() time.Duration {
|
|
return p.maxDuration
|
|
}
|
|
|
|
// CreateDefaultPodBackoff creates a default pod back-off object.
|
|
func CreateDefaultPodBackoff() *PodBackoff {
|
|
return CreatePodBackoff(1*time.Second, 60*time.Second)
|
|
}
|
|
|
|
// CreatePodBackoff creates a pod back-off object by default duration and max duration.
|
|
func CreatePodBackoff(defaultDuration, maxDuration time.Duration) *PodBackoff {
|
|
return CreatePodBackoffWithClock(defaultDuration, maxDuration, realClock{})
|
|
}
|
|
|
|
// CreatePodBackoffWithClock creates a pod back-off object by default duration, max duration and clock.
|
|
func CreatePodBackoffWithClock(defaultDuration, maxDuration time.Duration, clock clock) *PodBackoff {
|
|
return &PodBackoff{
|
|
expiryQ: NewHeap(backoffEntryKeyFunc, backoffEntryCompareUpdate),
|
|
clock: clock,
|
|
defaultDuration: defaultDuration,
|
|
maxDuration: maxDuration,
|
|
}
|
|
}
|
|
|
|
// getEntry returns the backoffEntry for a given podID
|
|
func (p *PodBackoff) getEntry(podID ktypes.NamespacedName) *backoffEntry {
|
|
entry, exists, _ := p.expiryQ.GetByKey(podID.String())
|
|
var be *backoffEntry
|
|
if !exists {
|
|
be = &backoffEntry{
|
|
initialized: false,
|
|
podName: podID,
|
|
backoff: p.defaultDuration,
|
|
}
|
|
p.expiryQ.Update(be)
|
|
} else {
|
|
be = entry.(*backoffEntry)
|
|
}
|
|
return be
|
|
}
|
|
|
|
// BackoffPod updates the backoff for a podId and returns the duration until backoff completion
|
|
func (p *PodBackoff) BackoffPod(podID ktypes.NamespacedName) time.Duration {
|
|
p.lock.Lock()
|
|
defer p.lock.Unlock()
|
|
entry := p.getEntry(podID)
|
|
entry.lastUpdate = p.clock.Now()
|
|
p.expiryQ.Update(entry)
|
|
return entry.getBackoff(p.maxDuration)
|
|
}
|
|
|
|
// TryBackoffAndWait tries to acquire the backoff lock
|
|
func (p *PodBackoff) TryBackoffAndWait(podID ktypes.NamespacedName) bool {
|
|
p.lock.Lock()
|
|
entry := p.getEntry(podID)
|
|
|
|
if !entry.tryLock() {
|
|
p.lock.Unlock()
|
|
return false
|
|
}
|
|
defer entry.unlock()
|
|
duration := entry.getBackoff(p.maxDuration)
|
|
p.lock.Unlock()
|
|
time.Sleep(duration)
|
|
return true
|
|
}
|
|
|
|
// Gc execute garbage collection on the pod back-off.
|
|
func (p *PodBackoff) Gc() {
|
|
p.lock.Lock()
|
|
defer p.lock.Unlock()
|
|
now := p.clock.Now()
|
|
var be *backoffEntry
|
|
for {
|
|
entry := p.expiryQ.Peek()
|
|
if entry == nil {
|
|
break
|
|
}
|
|
be = entry.(*backoffEntry)
|
|
if now.Sub(be.lastUpdate) > p.maxDuration {
|
|
p.expiryQ.Pop()
|
|
} else {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
// backoffEntryKeyFunc is the keying function used for mapping a backoffEntry to string for heap
|
|
func backoffEntryKeyFunc(b interface{}) (string, error) {
|
|
be := b.(*backoffEntry)
|
|
return be.podName.String(), nil
|
|
}
|
|
|
|
// backoffEntryCompareUpdate returns true when b1's backoff time is before b2's
|
|
func backoffEntryCompareUpdate(b1, b2 interface{}) bool {
|
|
be1 := b1.(*backoffEntry)
|
|
be2 := b2.(*backoffEntry)
|
|
return be1.lastUpdate.Before(be2.lastUpdate)
|
|
}
|