This repository has been archived by the owner on Nov 30, 2023. It is now read-only.
/
kubelock_locker.go
107 lines (87 loc) · 2.17 KB
/
kubelock_locker.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
package locker
import (
"context"
"time"
"github.com/giantswarm/kubelock/v2"
"github.com/giantswarm/microerror"
"github.com/giantswarm/micrologger"
"k8s.io/apimachinery/pkg/runtime/schema"
"k8s.io/client-go/dynamic"
"k8s.io/client-go/rest"
"github.com/giantswarm/azure-operator/v8/pkg/project"
)
const (
lockName = "ipam"
lockNamespaceName = "giantswarm"
)
var (
lockOwner = project.Name() + "@" + project.Version()
lockTTL = 30 * time.Second
)
type KubeLockLockerConfig struct {
Logger micrologger.Logger
RestConfig *rest.Config
}
type KubeLockLocker struct {
logger micrologger.Logger
kubelock kubelock.Interface
}
func NewKubeLockLocker(config KubeLockLockerConfig) (*KubeLockLocker, error) {
if config.Logger == nil {
return nil, microerror.Maskf(invalidConfigError, "%T.Logger must not be empty", config)
}
if config.RestConfig == nil {
return nil, microerror.Maskf(invalidConfigError, "%T.RestConfig must not be empty", config)
}
var err error
var dynClient dynamic.Interface
{
dynClient, err = dynamic.NewForConfig(config.RestConfig)
if err != nil {
return nil, microerror.Mask(err)
}
}
var k kubelock.Interface
{
c := kubelock.Config{
DynClient: dynClient,
GVR: schema.GroupVersionResource{
Group: "",
Version: "v1",
Resource: "namespaces",
},
}
k, err = kubelock.New(c)
if err != nil {
return nil, microerror.Mask(err)
}
}
d := &KubeLockLocker{
logger: config.Logger,
kubelock: k,
}
return d, nil
}
func (d KubeLockLocker) Lock(ctx context.Context) error {
err := d.kubelock.Lock(lockName).Acquire(ctx, lockNamespaceName, kubelock.AcquireOptions{
Owner: lockOwner,
TTL: lockTTL,
})
if kubelock.IsAlreadyExists(err) {
return microerror.Maskf(alreadyExistsError, err.Error())
} else if err != nil {
return microerror.Mask(err)
}
return nil
}
func (d KubeLockLocker) Unlock(ctx context.Context) error {
err := d.kubelock.Lock(lockName).Release(ctx, lockNamespaceName, kubelock.ReleaseOptions{
Owner: lockOwner,
})
if kubelock.IsNotFound(err) {
return microerror.Maskf(notFoundError, err.Error())
} else if err != nil {
return microerror.Mask(err)
}
return nil
}