The primary purpose of this project is to simplify working with Kubernetes
Custom Resources. To achieve that it provides a base class,
kubecrd.KubeResourceBase
that can create Python
dataclassses into Kubernetes Custom Resources and also generate and install
Custom Resource Definitions for those resource into the K8s cluster directly.
>>> from dataclasses import dataclass, field >>> from uuid import UUID >>> from kubecrd import KubeResourceBase >>> from apischema import schema>>> @dataclass ... class Resource(KubeResourceBase): ... __group__ = 'example.com' ... __version__ = 'v1alpha1' ... ... name: str ... tags: list[str] = field( ... default_factory=list, ... metadata=schema( ... description='regroup multiple resources', ... unique=False, ... ), ... )>>> print(Resource.crd_schema()) apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: name: resources.example.com spec: group: example.com names: kind: Resource plural: resources singular: resource scope: Namespaced versions: - name: v1alpha1 schema: openAPIV3Schema: properties: spec: properties: name: type: string tags: default: [] description: regroup multiple resources items: type: string type: array uniqueItems: false required: - name type: object type: object served: true storage: true <BLANKLINE>
It is also possible to install the CRD in a cluster using a Kubernetes Client object:
from from kubernetes import client, config config.load_kube_config() k8s_client = client.ApiClient() Resource.install(k8s_client)
You can then find the resource in the cluster:
» kubectl get crds/resources.example.com NAME CREATED AT resources.example.com 2022-03-20T03:58:25Z $ kubectl api-resources | grep example.com resources example.com/v1alpha1 true Resource
Installation of resource is idempotent, so re-installing an already installed
resource doesn't raise any exceptions if exist_ok=True
is passed in:
Resource.install(k8s_client, exist_ok=True)
You can serialize a Resource such that it is suitable to POST to K8s:
>>> example = Resource(name='myResource', tags=['tag1', 'tag2']) >>> import json >>> print(json.dumps(example.serialize(), sort_keys=True, indent=4)) { "apiVersion": "example.com/v1alpha1", "kind": "Resource", "metadata": { "name": "..." }, "spec": { "name": "myResource", "tags": [ "tag1", "tag2" ] } }
Objects can also be serialized and saved directly in K8s:
example.save(k8s_client)
Where client
in the above is a Kubernetes client object. You can also use
asyncio with kubernetes_asyncio client and instead do:
await example.async_save(k8s_async_client)
You can deserialize the JSON from Kubernetes API into Python CR objects.
$ cat -p testdata/cr.json { "apiVersion": "example.com/v1alpha1", "kind": "Resource", "metadata": { "generation": 1, "name": "myresource1", "namespace": "default", "resourceVersion": "105572812", "uid": "02102eb3-968b-418a-8023-75df383daa3c" }, "spec": { "name": "bestID", "tags": [ "tag1", "tag2" ] } }
by using from_json
classmethod on the resource:
>>> import json >>> with open('testdata/cr.json') as fd: ... json_schema = json.load(fd) >>> res = Resource.from_json(json_schema) >>> print(res.name) bestID >>> print(res.tags) ['tag1', 'tag2']
This also loads the Kubernetes's V1ObjectMeta
and sets it as the
.metadata
property of CR:
>>> print(res.metadata.namespace) default >>> print(res.metadata.name) myresource1 >>> print(res.metadata.resource_version) 105572812
It is possible to Watch for changes in Custom Resources using the standard Watch API in Kubernetes. For example, to watch for all changes in Resources:
async for happened, resource in Resource.async_watch(k8s_async_client): print(f'Resource {resource.metadata.name} was {happened}')
Or you can use the block sync API for the watch:
for happened, resource in Resource.watch(k8s_client): print(f'Resource {resource.metadata.name} was {happened}')
Kube CRD can be install from PyPI using pip or your favorite tool:
$ pip install kubecrd