Skip to content

Commit 6969cef

Browse files
authored
Merge pull request #133 from Anatoscope/fix/add-ns
Fix addition/deletion of namespaces and exception on startup
2 parents 130cd95 + d43cd49 commit 6969cef

6 files changed

+447
-28
lines changed

.gitignore

+4
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,6 @@
11
src/__pycache__/
2+
src/tests/__pycache__/
23
yaml/Object_example/debug-*
4+
.vscode
5+
.coverage
6+
lcov.info

src/handlers.py

+65-25
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import logging
2+
import sys
23
from typing import Any, Dict, List, Optional
34

45
import kopf
@@ -14,15 +15,14 @@
1415

1516
from os_utils import in_cluster
1617

17-
csecs: Dict[str, Any] = {}
18-
19-
# Loading kubeconfig
20-
if in_cluster():
18+
if "unittest" not in sys.modules:
2119
# Loading kubeconfig
22-
config.load_incluster_config()
23-
else:
24-
# Loading using the local kubevonfig.
25-
config.load_kube_config()
20+
if in_cluster():
21+
# Loading kubeconfig
22+
config.load_incluster_config()
23+
else:
24+
# Loading using the local kubevonfig.
25+
config.load_kube_config()
2626

2727
v1 = client.CoreV1Api()
2828
custom_objects_api = client.CustomObjectsApi()
@@ -92,7 +92,7 @@ def on_field_match_namespace(
9292
uid=uid,
9393
name=name,
9494
namespace=namespace,
95-
data=body.get('data'),
95+
body=body,
9696
synced_namespace=updated_matched,
9797
))
9898

@@ -113,6 +113,8 @@ def on_field_data(
113113
body: Dict[str, Any],
114114
meta: kopf.Meta,
115115
name: str,
116+
namespace: Optional[str],
117+
uid: str,
116118
logger: logging.Logger,
117119
**_,
118120
):
@@ -126,9 +128,14 @@ def on_field_data(
126128

127129
secret_type = body.get('type', 'Opaque')
128130

131+
cached_cluster_secret = csecs_cache.get_cluster_secret(uid)
132+
if cached_cluster_secret is None:
133+
logger.error('Received an event for an unknown ClusterSecret.')
134+
135+
updated_syncedns = syncedns.copy()
129136
for ns in syncedns:
130137
logger.info(f'Re Syncing secret {name} in ns {ns}')
131-
body = client.V1Secret(
138+
ns_sec_body = client.V1Secret(
132139
api_version='v1',
133140
data={str(key): str(value) for key, value in new.items()},
134141
kind='Secret',
@@ -140,14 +147,42 @@ def on_field_data(
140147
),
141148
type=secret_type,
142149
)
143-
logger.debug(f'body: {body}')
150+
logger.debug(f'body: {ns_sec_body}')
144151
# Ensuring the secret still exist.
145152
if secret_exists(logger=logger, name=name, namespace=ns, v1=v1):
146-
response = v1.replace_namespaced_secret(name=name, namespace=ns, body=body)
153+
response = v1.replace_namespaced_secret(name=name, namespace=ns, body=ns_sec_body)
147154
else:
148-
response = v1.create_namespaced_secret(namespace=ns, body=body)
155+
try:
156+
v1.read_namespace(name=ns)
157+
except client.exceptions.ApiException as e:
158+
if e.status != 404:
159+
raise
160+
response = f'Namespace {ns} not found'
161+
updated_syncedns.remove(ns)
162+
logger.info(f'Namespace {ns} not found while Syncing secret {name}')
163+
else:
164+
response = v1.create_namespaced_secret(namespace=ns, body=ns_sec_body)
149165
logger.debug(response)
150166

167+
if updated_syncedns != syncedns:
168+
# Patch synced_ns field
169+
logger.debug(f'Patching clustersecret {name} in namespace {namespace}')
170+
body = patch_clustersecret_status(
171+
logger=logger,
172+
name=name,
173+
new_status={'create_fn': {'syncedns': updated_syncedns}},
174+
custom_objects_api=custom_objects_api,
175+
)
176+
177+
# Updating the cache
178+
csecs_cache.set_cluster_secret(BaseClusterSecret(
179+
uid=uid,
180+
name=name,
181+
namespace=namespace or "",
182+
body=body,
183+
synced_namespace=updated_syncedns,
184+
))
185+
151186

152187
@kopf.on.resume('clustersecret.io', 'v1', 'clustersecrets')
153188
@kopf.on.create('clustersecret.io', 'v1', 'clustersecrets')
@@ -164,8 +199,8 @@ async def create_fn(
164199

165200
# sync in all matched NS
166201
logger.info(f'Syncing on Namespaces: {matchedns}')
167-
for namespace in matchedns:
168-
sync_secret(logger, namespace, body, v1)
202+
for ns in matchedns:
203+
sync_secret(logger, ns, body, v1)
169204

170205
# store status in memory
171206
cached_cluster_secret = csecs_cache.get_cluster_secret(uid)
@@ -176,8 +211,8 @@ async def create_fn(
176211
csecs_cache.set_cluster_secret(BaseClusterSecret(
177212
uid=uid,
178213
name=name,
179-
namespace=namespace,
180-
data=body.get('data'),
214+
namespace=namespace or "",
215+
body=body,
181216
synced_namespace=matchedns,
182217
))
183218

@@ -193,10 +228,10 @@ async def namespace_watcher(logger: logging.Logger, meta: kopf.Meta, **_):
193228
logger.debug(f'New namespace created: {new_ns} re-syncing')
194229
ns_new_list = []
195230
for cluster_secret in csecs_cache.all_cluster_secret():
196-
obj_body = cluster_secret['body']
197-
name = obj_body['metadata']['name']
231+
obj_body = cluster_secret.body
232+
name = cluster_secret.name
198233

199-
matcheddns = cluster_secret['syncedns']
234+
matcheddns = cluster_secret.synced_namespace
200235

201236
logger.debug(f'Old matched namespace: {matcheddns} - name: {name}')
202237
ns_new_list = get_ns_list(logger, obj_body, v1)
@@ -211,11 +246,16 @@ async def namespace_watcher(logger: logging.Logger, meta: kopf.Meta, **_):
211246
)
212247

213248
# if there is a new matching ns, refresh cache
214-
cluster_secret.namespace = ns_new_list
249+
cluster_secret.synced_namespace = ns_new_list
215250
csecs_cache.set_cluster_secret(cluster_secret)
216251

217-
# update ns_new_list on the object so then we also delete from there
218-
return {'syncedns': ns_new_list}
252+
# update ns_new_list on the object so then we also delete from there
253+
patch_clustersecret_status(
254+
logger=logger,
255+
name=cluster_secret.name,
256+
new_status={'create_fn': {'syncedns': ns_new_list}},
257+
custom_objects_api=custom_objects_api,
258+
)
219259

220260

221261
@kopf.on.startup()
@@ -243,8 +283,8 @@ async def startup_fn(logger: logging.Logger, **_):
243283
BaseClusterSecret(
244284
uid=metadata.get('uid'),
245285
name=metadata.get('name'),
246-
namespace=metadata.get('namespace'),
247-
data=item.get('data'),
286+
namespace=metadata.get('namespace', ''),
287+
body=item,
248288
synced_namespace=item.get('status', {}).get('create_fn', {}).get('syncedns', []),
249289
)
250290
)

src/kubernetes_utils.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ def patch_clustersecret_status(
3636
logger.debug(f'Updated clustersecret manifest: {clustersecret}')
3737

3838
# Perform a patch operation to update the custom resource
39-
custom_objects_api.patch_cluster_custom_object(
39+
return custom_objects_api.patch_cluster_custom_object(
4040
group=group,
4141
version=version,
4242
plural=plural,

src/models.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,5 @@ class BaseClusterSecret(BaseModel):
77
uid: str
88
name: str
99
namespace: str
10-
data: Dict[str, Any]
10+
body: Dict[str, Any]
1111
synced_namespace: List[str]

src/requirements.txt

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
kopf===1.35.3
1+
kopf===1.37.2
22
kubernetes===19.15.0
33
setuptools>=65.5.1 # not directly required, pinned by Snyk to avoid a vulnerability
44
pydantic==2.4.0

0 commit comments

Comments
 (0)