Handlers¶
Handlers are Python functions with the actual behavior of the custom resources.
They are called when any custom resource (within the scope of the operator) is created, modified, or deleted.
Any operator built with Kopf is based on handlers.
Events & Causes¶
Kubernetes only notifies when something is changed in the object, but does not clarify what was changed.
Moreover, since Kopf stores the state of the handlers on the object itself, these state changes also trigger events, which are seen by the operators and any other watchers.
To hide the complexity of state storing, Kopf provides cause detection: whenever an event happens for the object, the framework detects what actually happened, as follows:
Was the object just created?
Was the object deleted (marked for deletion)?
Was the object edited, and which fields specifically were edited, from what old values into what new values?
These causes, in turn, trigger the appropriate handlers, passing the detected information to the keyword arguments.
Registering¶
To register a handler for an event, use the @kopf.on decorator:
import kopf
from typing import Any
@kopf.on.create('kopfexamples')
def my_handler(spec: kopf.Spec, **_: Any) -> None:
pass
All available decorators are described below.
Kopf only supports simple functions and static methods as handlers. Class and instance methods are not supported. For explanation and rationale, see the discussion in #849 (briefly: the semantics of handlers are ambiguous when multiple instances exist or when multiple sub-classes inherit from the class, thus inheriting the handlers).
If you still want to use classes for namespacing, register the handlers
by using Kopf’s decorators explicitly for specific instances/sub-classes,
thus resolving the mentioned ambiguity and giving meaning to self/cls:
import kopf
from typing import Any
class MyCls:
def my_handler(self, spec: kopf.Spec, **_: Any) -> None:
print(repr(self))
instance = MyCls()
kopf.on.create('kopfexamples')(instance.my_handler)
Event-watching handlers¶
Low-level events can be intercepted and handled silently, without storing the handlers’ status (errors, retries, successes) on the object.
This can be useful if the operator needs to watch over the objects of another operator or controller, without adding its data.
The following event-handler is available:
import kopf
from typing import Any
@kopf.on.event('kopfexamples')
def my_handler(event: kopf.RawEvent, **_: Any) -> None:
pass
The event has the following structure:
class RawBody(TypedDict, total=False):
apiVersion: str
kind: str
metadata: Mapping[str, Any]
spec: Mapping[str, Any]
status: Mapping[str, Any]
class RawEvent(TypedDict, total=True):
type: Literal[None, 'ADDED', 'MODIFIED', 'DELETED']
object: RawBody
The event type None means the initial listing of the resources
before the actual watch-stream begins.
If the event handler fails, the error is logged to the operator’s log, and then ignored.
Note
Kopf invokes the event handlers for every event received from the stream. This includes the first-time listing when the operator starts or restarts.
It is the developer’s responsibility to make the handlers idempotent (re-executable with no duplicate side effects).
State-changing handlers¶
Kopf goes above and beyond: it detects the actual causes of these events, i.e. what happened to the object:
Was the object just created?
Was the object deleted (marked for deletion)?
Was the object edited, and which fields specifically were edited, from which old values to which new values?
Note
Kopf stores the status of the handlers, such as their progress, errors, or retries, in the object itself (in annotations), which triggers low-level events, but these events are not detected as separate causes, as nothing has changed essentially.
The following three core cause-handlers are available:
import kopf
from typing import Any
@kopf.on.create('kopfexamples')
def my_handler(spec: kopf.Spec, **_: Any) -> None:
pass
@kopf.on.update('kopfexamples')
def my_handler(spec: kopf.Spec, old: Any, new: Any, diff: kopf.Diff, **_: Any) -> None:
pass
@kopf.on.delete('kopfexamples')
def my_handler(spec: kopf.Spec, **_: Any) -> None:
pass
Despite the handlers seeing the full body of the resource object, they react
only to _essential_ changes, as implemented by kopf.DiffBaseStorage
or its descendants (Change detection).
In particular, Kopf ignores the whole status stanza as non-essential,
and all fields of metadata except for labels & annotations —
the framework remains blind to changes in these fields unless explicitly told to see them.
For example, to react to changes in the status of kind: Job:
import kopf
from typing import Any
@kopf.on.update('batch/v1', 'jobs', field='status')
def job_status_changes(**_: Any) -> None:
pass
Note
Kopf’s finalizers will be added to the object when delete handlers are
specified. Finalizers block Kubernetes from fully deleting objects; they
will only be deleted when all finalizers are removed, i.e. only if the
Kopf operator is running to remove them (see kubectl freezes on object deletion
for a workaround). If a delete handler is added but finalizers are not
required to block the actual deletion, i.e. the handler is optional,
the optional=True argument can be passed to the delete cause decorator.
Resuming handlers¶
A special kind of handler can be used for cases when the operator restarts and detects an object that existed before:
import kopf
from typing import Any
@kopf.on.resume('kopfexamples')
def my_handler(spec: kopf.Spec, **_: Any) -> None:
pass
This handler can be used to start threads or asyncio tasks or to update a global state to keep it consistent with the actual state of the cluster. With the resuming handler in addition to creation, update, and deletion handlers, no object will be left unattended even if it does not change over time.
The resuming handlers are guaranteed to execute only once per operator lifetime for each resource object (except if errors are retried).
Normally, the resume handlers are mixed into the creation and updating handling cycles, and are executed in the order they are declared.
It is a common pattern to declare both creation and resuming handlers pointing to the same function, so that this function is called either when an object is created while the operator is running, or when the operator starts while the object already exists:
import kopf
from typing import Any
@kopf.on.resume('kopfexamples')
@kopf.on.create('kopfexamples')
def my_handler(spec: kopf.Spec, **_: Any) -> None:
pass
However, the resuming handlers are not called if the object has been deleted during the operator downtime or restart, and the deletion handlers are now being invoked.
This is done intentionally to prevent cases where the resuming handlers start threads/tasks or allocate resources, and the deletion handlers stop/free them: it could happen that the resuming handlers would be executed after the deletion handlers, thus starting threads/tasks and never stopping them. For example:
import asyncio
import kopf
from typing import Any
TASKS: dict[str, asyncio.Task[None]] = {}
@kopf.on.delete('kopfexamples')
async def my_handler(spec: kopf.Spec, name: str, **_: Any) -> None:
if name and name in TASKS:
TASKS[name].cancel()
@kopf.on.resume('kopfexamples')
@kopf.on.create('kopfexamples')
def my_handler(spec: kopf.Spec, **_: Any) -> None:
if name and name not in TASKS:
TASKS[name] = asyncio.create_task(some_coroutine(spec))
In this example, if the operator starts and notices an object that has been marked for deletion, the deletion handler will be called, but the resuming handler is not called at all, despite the object being present. Otherwise, there would be a resource (e.g. memory) leak.
If the resume handlers are still desired during the deletion handling, they
can be explicitly marked as compatible with the deleted state of the object
with deleted=True option:
import kopf
from typing import Any
@kopf.on.resume('kopfexamples', deleted=True)
def my_handler(spec: kopf.Spec, **_: Any) -> None:
pass
In that case, both the deletion and resuming handlers will be invoked. It is the developer’s responsibility to ensure this does not lead to memory leaks.
Field handlers¶
Specific fields can be handled instead of the whole object:
import kopf
from typing import Any
@kopf.on.field('kopfexamples', field='spec.somefield')
def somefield_changed(old: Any, new: Any, **_: Any) -> None:
pass
There is no special detection of the causes for the fields, such as create/update/delete, so the field handler is effective only when the object is updated.
Sub-handlers¶
Warning
Sub-handlers are an advanced topic. Please make sure you understand the regular handlers first, as well as the handling cycle of the framework.
A common use case for this feature involves lists defined in the spec, each element of which should be handled with a handler-like approach rather than explicitly — i.e., with error tracking, retries, logging, progress and status reporting, etc.
This can be used with dynamically created functions, such as lambdas,
partials (functools.partial), or inner functions in closures:
spec:
items:
- item1
- item2
Sub-handlers can be implemented either imperatively
(which requires asynchronous handlers and async/await):
import functools
import kopf
from typing import Any
@kopf.on.create('kopfexamples')
async def create_fn(spec: kopf.Spec, **_: Any) -> None:
fns = {}
for item in spec.get('items', []):
fns[item] = functools.partial(handle_item, item=item)
await kopf.execute(fns=fns)
def handle_item(item: Any, *, spec: kopf.Spec, **_: Any) -> None:
pass
Or declaratively with decorators:
import kopf
from typing import Any
@kopf.on.create('kopfexamples')
def create_fn(spec: kopf.Spec, **_: Any) -> None:
for item in spec.get('items', []):
@kopf.subhandler(id=item)
def handle_item(item: Any = item, **_: Any) -> None:
pass
Both of these ways are equivalent. It is a matter of taste and preference which one to use.
The sub-handlers will be processed by all the standard rules and cycles
of Kopf’s handling cycle, as if they were the regular handlers
with the ids like create_fn/item1, create_fn/item2, etc.
Warning
The sub-handler functions, their code or their arguments, are not stored on the object between handling cycles.
Instead, their parent handler is considered as not finished, and it is called again and again to register the sub-handlers until all the sub-handlers of that parent handler are finished, so that the parent handler also becomes finished.
As such, the parent handler SHOULD NOT produce any side effects
except for read-only parsing of the inputs (e.g. spec)
and generating the dynamic functions of the sub-handlers.