/
release_version.go
123 lines (102 loc) · 2.95 KB
/
release_version.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
package releaseversion
import (
"context"
releasev1alpha1 "github.com/giantswarm/apiextensions/v3/pkg/apis/release/v1alpha1"
"github.com/giantswarm/k8sclient/v5/pkg/k8sclient"
"github.com/giantswarm/microerror"
"k8s.io/apimachinery/pkg/api/meta"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/types"
"github.com/giantswarm/cluster-operator/v3/service/controller/key"
"github.com/giantswarm/cluster-operator/v3/service/internal/releaseversion/internal/cache"
)
type Config struct {
K8sClient k8sclient.Interface
}
type ReleaseVersion struct {
k8sClient k8sclient.Interface
releaseCache *cache.Release
}
func New(c Config) (*ReleaseVersion, error) {
if c.K8sClient == nil {
return nil, microerror.Maskf(invalidConfigError, "%T.K8sClient must not be empty", c)
}
rv := &ReleaseVersion{
k8sClient: c.K8sClient,
releaseCache: cache.NewRelease(),
}
return rv, nil
}
func (rv *ReleaseVersion) Apps(ctx context.Context, obj interface{}) (map[string]ReleaseApp, error) {
cr, err := meta.Accessor(obj)
if err != nil {
return nil, microerror.Mask(err)
}
release, err := rv.cachedRelease(ctx, cr)
if err != nil {
return nil, microerror.Mask(err)
}
apps := make(map[string]ReleaseApp, len(release.Spec.Apps))
for _, v := range release.Spec.Apps {
apps[v.Name] = ReleaseApp{
Catalog: v.Catalog,
Version: v.Version,
}
}
return apps, nil
}
func (rv *ReleaseVersion) ComponentVersion(ctx context.Context, obj interface{}) (map[string]ReleaseComponent, error) {
cr, err := meta.Accessor(obj)
if err != nil {
return nil, microerror.Mask(err)
}
release, err := rv.cachedRelease(ctx, cr)
if err != nil {
return nil, microerror.Mask(err)
}
components := make(map[string]ReleaseComponent, len(release.Spec.Components))
for _, v := range release.Spec.Components {
components[v.Name] = ReleaseComponent{
Catalog: v.Catalog,
Reference: v.Reference,
Version: v.Version,
}
}
return components, nil
}
func (rv *ReleaseVersion) cachedRelease(ctx context.Context, cr metav1.Object) (releasev1alpha1.Release, error) {
var err error
var ok bool
var release releasev1alpha1.Release
{
r := rv.releaseCache.Key(ctx, cr)
if r == "" {
release, err = rv.lookupReleaseVersions(ctx, cr)
if err != nil {
return releasev1alpha1.Release{}, microerror.Mask(err)
}
} else {
release, ok = rv.releaseCache.Get(ctx, r)
if !ok {
release, err = rv.lookupReleaseVersions(ctx, cr)
if err != nil {
return releasev1alpha1.Release{}, microerror.Mask(err)
}
rv.releaseCache.Set(ctx, r, release)
}
}
}
return release, nil
}
func (rv *ReleaseVersion) lookupReleaseVersions(ctx context.Context, cr metav1.Object) (releasev1alpha1.Release, error) {
var re releasev1alpha1.Release
err := rv.k8sClient.CtrlClient().Get(
ctx,
types.NamespacedName{Name: key.ReleaseName(key.ReleaseVersion(cr))},
&re,
)
if err != nil {
return releasev1alpha1.Release{}, microerror.Mask(err)
}
return re, nil
}