statemanager.go 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. /*
  2. Licensed under the Apache License, Version 2.0 (the "License");
  3. you may not use this file except in compliance with the License.
  4. You may obtain a copy of the License at
  5. http://www.apache.org/licenses/LICENSE-2.0
  6. Unless required by applicable law or agreed to in writing, software
  7. distributed under the License is distributed on an "AS IS" BASIS,
  8. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  9. See the License for the specific language governing permissions and
  10. limitations under the License.
  11. */
  12. package statemanager
  13. import (
  14. "context"
  15. "errors"
  16. "fmt"
  17. "strings"
  18. "time"
  19. "github.com/spf13/pflag"
  20. apiextensions "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
  21. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  22. "k8s.io/apimachinery/pkg/runtime"
  23. "sigs.k8s.io/controller-runtime/pkg/client"
  24. "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
  25. genapi "github.com/external-secrets/external-secrets/apis/generators/v1alpha1"
  26. "github.com/external-secrets/external-secrets/pkg/feature"
  27. "github.com/external-secrets/external-secrets/pkg/utils"
  28. )
  29. // Manager takes care of maintaining the state of the generators.
  30. // It provides the ability to commit and rollback the state of the generators,
  31. // which is needed when we have multiple generators that need to be created or
  32. // other operations which can fail.
  33. type Manager struct {
  34. ctx context.Context
  35. scheme *runtime.Scheme
  36. client client.Client
  37. namespace string
  38. resource genapi.StatefulResource
  39. queue []QueueItem
  40. }
  41. type QueueItem struct {
  42. Rollback func() error
  43. Commit func() error
  44. }
  45. var gcGracePeriod time.Duration
  46. func init() {
  47. fs := pflag.NewFlagSet("gc", pflag.ExitOnError)
  48. fs.DurationVar(&gcGracePeriod, "generator-gc-grace-period", time.Minute*2, "Duration after which generated secrets are cleaned up after they have been flagged for gc.")
  49. feature.Register(feature.Feature{
  50. Flags: fs,
  51. })
  52. }
  53. func New(ctx context.Context, client client.Client, scheme *runtime.Scheme, namespace string,
  54. resource genapi.StatefulResource) *Manager {
  55. return &Manager{
  56. ctx: ctx,
  57. scheme: scheme,
  58. client: client,
  59. namespace: namespace,
  60. resource: resource,
  61. }
  62. }
  63. // Rollback will rollback the enqueued operations.
  64. func (m *Manager) Rollback() error {
  65. var errs []error
  66. for _, item := range m.queue {
  67. if err := item.Rollback(); err != nil {
  68. errs = append(errs, err)
  69. }
  70. }
  71. return errors.Join(errs...)
  72. }
  73. // Commit will apply the enqueued changes to the state of the generators.
  74. func (m *Manager) Commit() error {
  75. var errs []error
  76. for _, item := range m.queue {
  77. if err := item.Commit(); err != nil {
  78. errs = append(errs, err)
  79. }
  80. }
  81. return errors.Join(errs...)
  82. }
  83. // EnqueueFlagLatestStateForGC will flag the latest state for garbage collection after Commit.
  84. // It will be cleaned up later by the garbage collector.
  85. func (m *Manager) EnqueueFlagLatestStateForGC(stateKey string) {
  86. m.queue = append(m.queue, QueueItem{
  87. Commit: func() error {
  88. return m.disposeState(stateKey)
  89. },
  90. })
  91. }
  92. // EnqueueMoveStateToGC will move the generator state to GC if Commit() is called.
  93. func (m *Manager) EnqueueMoveStateToGC(stateKey string) {
  94. m.queue = append(m.queue, QueueItem{
  95. Commit: func() error {
  96. return m.disposeState(stateKey)
  97. },
  98. })
  99. }
  100. // EnqueueSetLatest sets the latest state for the given key.
  101. // It will commit the state on success or move the state to GC on failure.
  102. func (m *Manager) EnqueueSetLatest(ctx context.Context, stateKey, namespace string, resource *apiextensions.JSON, gen genapi.Generator, state genapi.GeneratorProviderState) {
  103. m.queue = append(m.queue, QueueItem{
  104. // Stores the state in GeneratorState resource
  105. Commit: func() error {
  106. genState, err := m.createGeneratorState(resource, state, namespace, stateKey)
  107. if err != nil {
  108. return err
  109. }
  110. return m.client.Create(ctx, genState)
  111. },
  112. // Rollback by cleaning up the state.
  113. // In case of failure, create a new GeneratorState, so it will eventually be cleaned up.
  114. // If that also fails we're out of luck :(
  115. Rollback: func() error {
  116. err := gen.Cleanup(ctx, resource, state, m.client, namespace)
  117. if err == nil {
  118. return nil
  119. }
  120. genState, err := m.createGeneratorState(resource, state, namespace, stateKey)
  121. if err != nil {
  122. return err
  123. }
  124. genState.Spec.GarbageCollectionDeadline = &metav1.Time{
  125. Time: time.Now(),
  126. }
  127. return m.client.Create(ctx, genState)
  128. },
  129. })
  130. }
  131. func (m *Manager) createGeneratorState(resource *apiextensions.JSON, state genapi.GeneratorProviderState, namespace, stateKey string) (*genapi.GeneratorState, error) {
  132. genState := &genapi.GeneratorState{
  133. ObjectMeta: metav1.ObjectMeta{
  134. GenerateName: fmt.Sprintf("gen-%s-%s-", strings.ToLower(m.resource.GetObjectKind().GroupVersionKind().Kind), m.resource.GetName()),
  135. Namespace: namespace,
  136. Labels: map[string]string{
  137. genapi.GeneratorStateLabelOwnerKey: ownerKey(
  138. m.resource,
  139. stateKey,
  140. ),
  141. },
  142. },
  143. Spec: genapi.GeneratorStateSpec{
  144. Resource: resource,
  145. State: state,
  146. },
  147. }
  148. if err := controllerutil.SetOwnerReference(m.resource, genState, m.scheme); err != nil {
  149. return nil, err
  150. }
  151. return genState, nil
  152. }
  153. func ownerKey(resource genapi.StatefulResource, key string) string {
  154. return utils.ObjectHash(fmt.Sprintf("%s-%s-%s-%s",
  155. resource.GetObjectKind().GroupVersionKind().Kind,
  156. resource.GetNamespace(),
  157. resource.GetName(),
  158. key),
  159. )
  160. }
  161. func (m *Manager) disposeState(key string) error {
  162. allStates, err := m.GetAllStates(key)
  163. if err != nil {
  164. return err
  165. }
  166. latest := getLatest(allStates)
  167. if latest == nil {
  168. return nil
  169. }
  170. // flag all states for GC except the latest one
  171. // This is to ensure that all "old" states are eventually cleaned up.
  172. // This is needed due to fast reconciles and working with stale cache.
  173. var errs []error
  174. for _, state := range allStates {
  175. if state.Name == latest.Name {
  176. continue
  177. }
  178. if state.Spec.GarbageCollectionDeadline != nil {
  179. continue
  180. }
  181. state.Spec.GarbageCollectionDeadline = &metav1.Time{
  182. Time: time.Now().Add(gcGracePeriod),
  183. }
  184. if err := m.client.Update(m.ctx, &state); err != nil {
  185. errs = append(errs, err)
  186. }
  187. }
  188. return errors.Join(errs...)
  189. }
  190. // GetLatest returns the latest state for the given key.
  191. func (m *Manager) GetAllStates(key string) ([]genapi.GeneratorState, error) {
  192. var stateList genapi.GeneratorStateList
  193. if err := m.client.List(m.ctx, &stateList, &client.MatchingLabels{
  194. genapi.GeneratorStateLabelOwnerKey: ownerKey(
  195. m.resource,
  196. key,
  197. ),
  198. }, client.InNamespace(m.namespace)); err != nil {
  199. return nil, err
  200. }
  201. return stateList.Items, nil
  202. }
  203. // GetLatestState returns the latest state for the given key.
  204. func (m *Manager) GetLatestState(key string) (*genapi.GeneratorState, error) {
  205. var stateList genapi.GeneratorStateList
  206. if err := m.client.List(m.ctx, &stateList, &client.MatchingLabels{
  207. genapi.GeneratorStateLabelOwnerKey: ownerKey(
  208. m.resource,
  209. key,
  210. ),
  211. }, client.InNamespace(m.namespace)); err != nil {
  212. return nil, err
  213. }
  214. if latestState := getLatest(stateList.Items); latestState != nil {
  215. return latestState, nil
  216. }
  217. return nil, nil
  218. }
  219. func getLatest(stateList []genapi.GeneratorState) *genapi.GeneratorState {
  220. var latest *genapi.GeneratorState
  221. for _, state := range stateList {
  222. // if the state is already flagged for GC, skip it
  223. // It can happen that the latest based on creation timestamp is already flagged for GC.
  224. // That is the case when a rollback was performed.
  225. if state.Spec.GarbageCollectionDeadline != nil {
  226. continue
  227. }
  228. if latest == nil {
  229. latest = &state
  230. continue
  231. }
  232. if state.CreationTimestamp.After(latest.CreationTimestamp.Time) {
  233. latest = &state
  234. }
  235. }
  236. return latest
  237. }