namespace_controller.go 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  1. /*
  2. Copyright 2015 The Kubernetes Authors.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package namespace
  14. import (
  15. "fmt"
  16. "time"
  17. "golang.org/x/time/rate"
  18. v1 "k8s.io/api/core/v1"
  19. "k8s.io/apimachinery/pkg/api/errors"
  20. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  21. utilruntime "k8s.io/apimachinery/pkg/util/runtime"
  22. "k8s.io/apimachinery/pkg/util/wait"
  23. coreinformers "k8s.io/client-go/informers/core/v1"
  24. clientset "k8s.io/client-go/kubernetes"
  25. corelisters "k8s.io/client-go/listers/core/v1"
  26. "k8s.io/client-go/metadata"
  27. "k8s.io/client-go/tools/cache"
  28. "k8s.io/client-go/util/workqueue"
  29. "k8s.io/component-base/metrics/prometheus/ratelimiter"
  30. "k8s.io/kubernetes/pkg/controller"
  31. "k8s.io/kubernetes/pkg/controller/namespace/deletion"
  32. "k8s.io/klog"
  33. )
  34. const (
  35. // namespaceDeletionGracePeriod is the time period to wait before processing a received namespace event.
  36. // This allows time for the following to occur:
  37. // * lifecycle admission plugins on HA apiservers to also observe a namespace
  38. // deletion and prevent new objects from being created in the terminating namespace
  39. // * non-leader etcd servers to observe last-minute object creations in a namespace
  40. // so this controller's cleanup can actually clean up all objects
  41. namespaceDeletionGracePeriod = 5 * time.Second
  42. )
  43. // NamespaceController is responsible for performing actions dependent upon a namespace phase
  44. type NamespaceController struct {
  45. // lister that can list namespaces from a shared cache
  46. lister corelisters.NamespaceLister
  47. // returns true when the namespace cache is ready
  48. listerSynced cache.InformerSynced
  49. // namespaces that have been queued up for processing by workers
  50. queue workqueue.RateLimitingInterface
  51. // helper to delete all resources in the namespace when the namespace is deleted.
  52. namespacedResourcesDeleter deletion.NamespacedResourcesDeleterInterface
  53. }
  54. // NewNamespaceController creates a new NamespaceController
  55. func NewNamespaceController(
  56. kubeClient clientset.Interface,
  57. metadataClient metadata.Interface,
  58. discoverResourcesFn func() ([]*metav1.APIResourceList, error),
  59. namespaceInformer coreinformers.NamespaceInformer,
  60. resyncPeriod time.Duration,
  61. finalizerToken v1.FinalizerName) *NamespaceController {
  62. // create the controller so we can inject the enqueue function
  63. namespaceController := &NamespaceController{
  64. queue: workqueue.NewNamedRateLimitingQueue(nsControllerRateLimiter(), "namespace"),
  65. namespacedResourcesDeleter: deletion.NewNamespacedResourcesDeleter(kubeClient.CoreV1().Namespaces(), metadataClient, kubeClient.CoreV1(), discoverResourcesFn, finalizerToken),
  66. }
  67. if kubeClient != nil && kubeClient.CoreV1().RESTClient().GetRateLimiter() != nil {
  68. ratelimiter.RegisterMetricAndTrackRateLimiterUsage("namespace_controller", kubeClient.CoreV1().RESTClient().GetRateLimiter())
  69. }
  70. // configure the namespace informer event handlers
  71. namespaceInformer.Informer().AddEventHandlerWithResyncPeriod(
  72. cache.ResourceEventHandlerFuncs{
  73. AddFunc: func(obj interface{}) {
  74. namespace := obj.(*v1.Namespace)
  75. namespaceController.enqueueNamespace(namespace)
  76. },
  77. UpdateFunc: func(oldObj, newObj interface{}) {
  78. namespace := newObj.(*v1.Namespace)
  79. namespaceController.enqueueNamespace(namespace)
  80. },
  81. },
  82. resyncPeriod,
  83. )
  84. namespaceController.lister = namespaceInformer.Lister()
  85. namespaceController.listerSynced = namespaceInformer.Informer().HasSynced
  86. return namespaceController
  87. }
  88. // nsControllerRateLimiter is tuned for a faster than normal recycle time with default backoff speed and default overall
  89. // requeing speed. We do this so that namespace cleanup is reliably faster and we know that the number of namespaces being
  90. // deleted is smaller than total number of other namespace scoped resources in a cluster.
  91. func nsControllerRateLimiter() workqueue.RateLimiter {
  92. return workqueue.NewMaxOfRateLimiter(
  93. // this ensures that we retry namespace deletion at least every minute, never longer.
  94. workqueue.NewItemExponentialFailureRateLimiter(5*time.Millisecond, 60*time.Second),
  95. // 10 qps, 100 bucket size. This is only for retry speed and its only the overall factor (not per item)
  96. &workqueue.BucketRateLimiter{Limiter: rate.NewLimiter(rate.Limit(10), 100)},
  97. )
  98. }
  99. // enqueueNamespace adds an object to the controller work queue
  100. // obj could be an *v1.Namespace, or a DeletionFinalStateUnknown item.
  101. func (nm *NamespaceController) enqueueNamespace(obj interface{}) {
  102. key, err := controller.KeyFunc(obj)
  103. if err != nil {
  104. utilruntime.HandleError(fmt.Errorf("Couldn't get key for object %+v: %v", obj, err))
  105. return
  106. }
  107. namespace := obj.(*v1.Namespace)
  108. // don't queue if we aren't deleted
  109. if namespace.DeletionTimestamp == nil || namespace.DeletionTimestamp.IsZero() {
  110. return
  111. }
  112. // delay processing namespace events to allow HA api servers to observe namespace deletion,
  113. // and HA etcd servers to observe last minute object creations inside the namespace
  114. nm.queue.AddAfter(key, namespaceDeletionGracePeriod)
  115. }
  116. // worker processes the queue of namespace objects.
  117. // Each namespace can be in the queue at most once.
  118. // The system ensures that no two workers can process
  119. // the same namespace at the same time.
  120. func (nm *NamespaceController) worker() {
  121. workFunc := func() bool {
  122. key, quit := nm.queue.Get()
  123. if quit {
  124. return true
  125. }
  126. defer nm.queue.Done(key)
  127. err := nm.syncNamespaceFromKey(key.(string))
  128. if err == nil {
  129. // no error, forget this entry and return
  130. nm.queue.Forget(key)
  131. return false
  132. }
  133. if estimate, ok := err.(*deletion.ResourcesRemainingError); ok {
  134. t := estimate.Estimate/2 + 1
  135. klog.V(4).Infof("Content remaining in namespace %s, waiting %d seconds", key, t)
  136. nm.queue.AddAfter(key, time.Duration(t)*time.Second)
  137. } else {
  138. // rather than wait for a full resync, re-add the namespace to the queue to be processed
  139. nm.queue.AddRateLimited(key)
  140. utilruntime.HandleError(fmt.Errorf("deletion of namespace %v failed: %v", key, err))
  141. }
  142. return false
  143. }
  144. for {
  145. quit := workFunc()
  146. if quit {
  147. return
  148. }
  149. }
  150. }
  151. // syncNamespaceFromKey looks for a namespace with the specified key in its store and synchronizes it
  152. func (nm *NamespaceController) syncNamespaceFromKey(key string) (err error) {
  153. startTime := time.Now()
  154. defer func() {
  155. klog.V(4).Infof("Finished syncing namespace %q (%v)", key, time.Since(startTime))
  156. }()
  157. namespace, err := nm.lister.Get(key)
  158. if errors.IsNotFound(err) {
  159. klog.Infof("Namespace has been deleted %v", key)
  160. return nil
  161. }
  162. if err != nil {
  163. utilruntime.HandleError(fmt.Errorf("Unable to retrieve namespace %v from store: %v", key, err))
  164. return err
  165. }
  166. return nm.namespacedResourcesDeleter.Delete(namespace.Name)
  167. }
  168. // Run starts observing the system with the specified number of workers.
  169. func (nm *NamespaceController) Run(workers int, stopCh <-chan struct{}) {
  170. defer utilruntime.HandleCrash()
  171. defer nm.queue.ShutDown()
  172. klog.Infof("Starting namespace controller")
  173. defer klog.Infof("Shutting down namespace controller")
  174. if !cache.WaitForNamedCacheSync("namespace", stopCh, nm.listerSynced) {
  175. return
  176. }
  177. klog.V(5).Info("Starting workers of namespace controller")
  178. for i := 0; i < workers; i++ {
  179. go wait.Until(nm.worker, time.Second, stopCh)
  180. }
  181. <-stopCh
  182. }