Charmed-Kubernetes/containerd/hooks/relations/docker-registry/provides.py

114 lines
3.5 KiB
Python

"""
This is the provides side of the interface layer, for use only by the
docker-registry charm itself.
The flags that are set by the provides side of this interface are:
* **`endpoint.{endpoint_name}.requested`** This flag is set when there is
a new or updated request by a remote unit for docker-registry config.
The docker-registry integration charm should then iterate over each
request, perform whatever actions are necessary to satisfy those requests,
and then mark them as complete.
"""
from operator import attrgetter
from charms.reactive import Endpoint
from charms.reactive import when
from charms.reactive import toggle_flag, clear_flag
class DockerRegistryProvides(Endpoint):
"""
Example usage:
```python
from charms.reactive import when, endpoint_from_flag
from charms import layer
@when('endpoint.docker-registry.joined')
def configure_client():
registry = endpoint_from_flag('endpoint.docker-registry.joined')
registry.set_registry_config(netloc, **data)
@when('endpoint.docker-registry.requests-pending')
def handle_image_request():
registry = endpoint_from_flag('endpoint.docker-registry.requests-pending')
for request in registry.requests:
request.image_data(name, tag)
registry.mark_completed()
```
"""
@when("endpoint.{endpoint_name}.changed")
def check_requests(self):
toggle_flag(self.expand_name("requests-pending"), len(self.requests) > 0)
clear_flag(self.expand_name("changed"))
@property
def requests(self):
"""
A list of the new or updated #RegistryRequests that
have been made.
"""
if not hasattr(self, "_requests"):
all_requests = [RegistryRequest(unit) for unit in self.all_joined_units]
is_changed = attrgetter("is_changed")
self._requests = list(filter(is_changed, all_requests))
return self._requests
def mark_completed(self):
"""
Mark all requests as completed and remove the `requests-pending` flag.
"""
clear_flag(self.expand_name("requests-pending"))
self._requests = []
def set_registry_config(self, registry_netloc, **kwargs):
"""
Set the registry config. Minimally, a network location is required.
Other data (auth, tls, etc) may also be set.
"""
data = {"registry_netloc": registry_netloc}
for k, v in kwargs.items():
data[k] = v
for relation in self.relations:
relation.to_publish.update(data)
class RegistryRequest:
"""
A request from a single remote unit to include an image in our registry.
"""
def __init__(self, unit):
self._unit = unit
@property
def _to_publish(self):
return self._unit.relation.to_publish
@property
def has_image(self):
"""
Whether or not an image has been processed via `image_data`.
"""
return "image" in self._unit.relation.to_publish
@property
def is_changed(self):
"""
Whether this request has changed since the last time it was
marked completed (if ever).
"""
return not self.has_image
@property
def unit_name(self):
return self._unit.unit_name
def image_data(self, image, tag):
"""
Set the image characteristics this request.
"""
data = {
"image": image,
"tag": tag,
}
self._unit.relation.to_publish.update(data)