Source code for nvflare.private.event

# Copyright (c) 2021, NVIDIA CORPORATION.  All rights reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

import uuid

from nvflare.apis.fl_component import FLComponent
from nvflare.apis.fl_constant import EventScope, FLContextKey
from nvflare.apis.fl_context import FLContext
from nvflare.security.logging import secure_format_exception

# do not use underscore as key name; otherwise it cannot be removed from ctx
_KEY_EVENT_DEPTH = "###event_depth"
_MAX_EVENT_DEPTH = 20


[docs]def fire_event(event: str, handlers: list, ctx: FLContext): """Fires the specified event and invokes the list of handlers. Args: event: the event to be fired handlers: handlers to be invoked ctx: context for cross-component data sharing Returns: N/A """ event_id = str(uuid.uuid4()) event_data = ctx.get_prop(FLContextKey.EVENT_DATA, None) event_origin = ctx.get_prop(FLContextKey.EVENT_ORIGIN, None) event_scope = ctx.get_prop(FLContextKey.EVENT_SCOPE, EventScope.LOCAL) depth = ctx.get_prop(_KEY_EVENT_DEPTH, 0) if depth > _MAX_EVENT_DEPTH: # too many recursive event calls raise RuntimeError("Recursive event calls too deep (>{})".format(_MAX_EVENT_DEPTH)) ctx.set_prop(key=_KEY_EVENT_DEPTH, value=depth + 1, private=True, sticky=False) if handlers: for h in handlers: if not isinstance(h, FLComponent): raise TypeError("handler must be FLComponent but got {}".format(type(h))) try: # since events could be recursive (a handler fires another event) on the same fl_ctx, # we need to reset these key values into the fl_ctx ctx.set_prop(key=FLContextKey.EVENT_ID, value=event_id, private=True, sticky=False) ctx.set_prop(key=FLContextKey.EVENT_DATA, value=event_data, private=True, sticky=False) ctx.set_prop(key=FLContextKey.EVENT_ORIGIN, value=event_origin, private=True, sticky=False) ctx.set_prop(key=FLContextKey.EVENT_SCOPE, value=event_scope, private=True, sticky=False) h.handle_event(event, ctx) except Exception as e: h.log_exception( ctx, f'Exception when handling event "{event}": {secure_format_exception(e)}', fire_event=False ) exceptions = ctx.get_prop(FLContextKey.EXCEPTIONS) if not exceptions: exceptions = {} ctx.set_prop(FLContextKey.EXCEPTIONS, exceptions, sticky=False, private=True) exceptions[h.name] = e ctx.set_prop(key=_KEY_EVENT_DEPTH, value=depth, private=True, sticky=False)