statemanager.go 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  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 item.Rollback == nil {
  68. continue
  69. }
  70. if err := item.Rollback(); err != nil {
  71. errs = append(errs, err)
  72. }
  73. }
  74. return errors.Join(errs...)
  75. }
  76. // Commit will apply the enqueued changes to the state of the generators.
  77. func (m *Manager) Commit() error {
  78. var errs []error
  79. for _, item := range m.queue {
  80. if item.Commit == nil {
  81. continue
  82. }
  83. if err := item.Commit(); err != nil {
  84. errs = append(errs, err)
  85. }
  86. }
  87. return errors.Join(errs...)
  88. }
  89. // EnqueueFlagLatestStateForGC will flag the latest state for garbage collection after Commit.
  90. // It will be cleaned up later by the garbage collector.
  91. func (m *Manager) EnqueueFlagLatestStateForGC(stateKey string) {
  92. m.queue = append(m.queue, QueueItem{
  93. Commit: func() error {
  94. return m.disposeState(stateKey)
  95. },
  96. })
  97. }
  98. // EnqueueMoveStateToGC will move the generator state to GC if Commit() is called.
  99. func (m *Manager) EnqueueMoveStateToGC(stateKey string) {
  100. m.queue = append(m.queue, QueueItem{
  101. Commit: func() error {
  102. return m.disposeState(stateKey)
  103. },
  104. })
  105. }
  106. // EnqueueSetLatest sets the latest state for the given key.
  107. // It will commit the state on success or move the state to GC on failure.
  108. func (m *Manager) EnqueueSetLatest(ctx context.Context, stateKey, namespace string, resource *apiextensions.JSON, gen genapi.Generator, state genapi.GeneratorProviderState) {
  109. m.queue = append(m.queue, QueueItem{
  110. // Stores the state in GeneratorState resource
  111. Commit: func() error {
  112. genState, err := m.createGeneratorState(resource, state, namespace, stateKey)
  113. if err != nil {
  114. return err
  115. }
  116. return m.client.Create(ctx, genState)
  117. },
  118. // Rollback by cleaning up the state.
  119. // In case of failure, create a new GeneratorState, so it will eventually be cleaned up.
  120. // If that also fails we're out of luck :(
  121. Rollback: func() error {
  122. err := gen.Cleanup(ctx, resource, state, m.client, namespace)
  123. if err == nil {
  124. return nil
  125. }
  126. genState, err := m.createGeneratorState(resource, state, namespace, stateKey)
  127. if err != nil {
  128. return err
  129. }
  130. genState.Spec.GarbageCollectionDeadline = &metav1.Time{
  131. Time: time.Now(),
  132. }
  133. return m.client.Create(ctx, genState)
  134. },
  135. })
  136. }
  137. func (m *Manager) createGeneratorState(resource *apiextensions.JSON, state genapi.GeneratorProviderState, namespace, stateKey string) (*genapi.GeneratorState, error) {
  138. genState := &genapi.GeneratorState{
  139. ObjectMeta: metav1.ObjectMeta{
  140. GenerateName: fmt.Sprintf("gen-%s-%s-", strings.ToLower(m.resource.GetObjectKind().GroupVersionKind().Kind), m.resource.GetName()),
  141. Namespace: namespace,
  142. Labels: map[string]string{
  143. genapi.GeneratorStateLabelOwnerKey: ownerKey(
  144. m.resource,
  145. stateKey,
  146. ),
  147. },
  148. },
  149. Spec: genapi.GeneratorStateSpec{
  150. Resource: resource,
  151. State: state,
  152. },
  153. }
  154. if err := controllerutil.SetOwnerReference(m.resource, genState, m.scheme); err != nil {
  155. return nil, err
  156. }
  157. return genState, nil
  158. }
  159. func ownerKey(resource genapi.StatefulResource, key string) string {
  160. return utils.ObjectHash(fmt.Sprintf("%s-%s-%s-%s",
  161. resource.GetObjectKind().GroupVersionKind().Kind,
  162. resource.GetNamespace(),
  163. resource.GetName(),
  164. key),
  165. )
  166. }
  167. func (m *Manager) disposeState(key string) error {
  168. allStates, err := m.GetAllStates(key)
  169. if err != nil {
  170. return err
  171. }
  172. latest := getLatest(allStates)
  173. if latest == nil {
  174. return nil
  175. }
  176. // flag all states for GC except the latest one
  177. // This is to ensure that all "old" states are eventually cleaned up.
  178. // This is needed due to fast reconciles and working with stale cache.
  179. var errs []error
  180. for _, state := range allStates {
  181. if state.Name == latest.Name {
  182. continue
  183. }
  184. if state.Spec.GarbageCollectionDeadline != nil {
  185. continue
  186. }
  187. state.Spec.GarbageCollectionDeadline = &metav1.Time{
  188. Time: time.Now().Add(gcGracePeriod),
  189. }
  190. if err := m.client.Update(m.ctx, &state); err != nil {
  191. errs = append(errs, err)
  192. }
  193. }
  194. return errors.Join(errs...)
  195. }
  196. // GetLatest returns the latest state for the given key.
  197. func (m *Manager) GetAllStates(key string) ([]genapi.GeneratorState, error) {
  198. var stateList genapi.GeneratorStateList
  199. if err := m.client.List(m.ctx, &stateList, &client.MatchingLabels{
  200. genapi.GeneratorStateLabelOwnerKey: ownerKey(
  201. m.resource,
  202. key,
  203. ),
  204. }, client.InNamespace(m.namespace)); err != nil {
  205. return nil, err
  206. }
  207. return stateList.Items, nil
  208. }
  209. // GetLatestState returns the latest state for the given key.
  210. func (m *Manager) GetLatestState(key string) (*genapi.GeneratorState, error) {
  211. var stateList genapi.GeneratorStateList
  212. if err := m.client.List(m.ctx, &stateList, &client.MatchingLabels{
  213. genapi.GeneratorStateLabelOwnerKey: ownerKey(
  214. m.resource,
  215. key,
  216. ),
  217. }, client.InNamespace(m.namespace)); err != nil {
  218. return nil, err
  219. }
  220. if latestState := getLatest(stateList.Items); latestState != nil {
  221. return latestState, nil
  222. }
  223. return nil, nil
  224. }
  225. func getLatest(stateList []genapi.GeneratorState) *genapi.GeneratorState {
  226. var latest *genapi.GeneratorState
  227. for _, state := range stateList {
  228. // if the state is already flagged for GC, skip it
  229. // It can happen that the latest based on creation timestamp is already flagged for GC.
  230. // That is the case when a rollback was performed.
  231. if state.Spec.GarbageCollectionDeadline != nil {
  232. continue
  233. }
  234. if latest == nil {
  235. latest = &state
  236. continue
  237. }
  238. if state.CreationTimestamp.After(latest.CreationTimestamp.Time) {
  239. latest = &state
  240. }
  241. }
  242. return latest
  243. }