Source code for nvflare.fuel.hci.reg

# Copyright (c) 2021-2022, 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.

from typing import List


[docs]class CommandSpec(object): valid_confirms = ["none", "yesno", "auth"] def __init__( self, name: str, description: str, usage: str, handler_func, authz_func=None, visible=True, confirm=None ): """Specification of a command within a CommandModuleSpec to register into CommandRegister as a CommandEntry. Args: name: command name description: command description text usage: string to show usage of the command handler_func: function to call for executing the command authz_func: authorization function to run to get a tuple of (valid, authz_ctx) in AuthzFilter visible: whether the command is visible or not confirm: whether the command needs confirmation to execute """ self.name = name self.description = description self.usage = usage self.handler_func = handler_func self.authz_func = authz_func self.visible = visible self.confirm = confirm if not confirm: self.confirm = "none" else: assert confirm in CommandSpec.valid_confirms
[docs]class CommandModuleSpec(object): def __init__(self, name: str, cmd_specs: List[CommandSpec]): """Specification for a command module containing a list of commands in the form of CommandSpec. Args: name: becomes the scope name of the commands in cmd_specs when registered in CommandRegister cmd_specs: list of CommandSpec objects with """ self.name = name self.cmd_specs = cmd_specs
[docs]class CommandModule(object): """Base class containing CommandModuleSpec."""
[docs] def get_spec(self) -> CommandModuleSpec: pass
[docs] def close(self): pass
[docs]class CommandEntry(object): def __init__(self, scope, name, desc, usage, handler, authz_func, visible, confirm): """Contains information about a command. This is registered in Scope within CommandRegister. Args: scope: scope for this command name: command name desc: command description text usage: string to show usage of the command handler: function to call for executing the command authz_func: authorization function to run to get a tuple of (valid, authz_ctx) in AuthzFilter visible: whether the command is visible or not confirm: whether the command needs confirmation to execute """ self.scope = scope self.name = name self.desc = desc self.usage = usage self.handler = handler self.authz_func = authz_func self.visible = visible self.confirm = confirm
class _Scope(object): def __init__(self, name: str): """A container grouping CommandEntry objects inside CommandRegister. Args: name: name of scope grouping commands """ self.name = name self.entries = {} def register_command( self, cmd_name: str, cmd_desc: str, cmd_usage: str, handler_func, authz_func, visible, confirm ): self.entries[cmd_name] = CommandEntry( self, cmd_name, cmd_desc, cmd_usage, handler_func, authz_func, visible, confirm )
[docs]class CommandRegister(object): def __init__(self, app_ctx): """Object containing the commands in scopes once they have been registered. ServerCommandRegister is derived from this class and calls the handler of the command through ``process_command`` and ``_do_command``. This is also used to register commands for the admin client. Args: app_ctx: app context """ self.app_ctx = app_ctx self.scopes = {} self.cmd_map = {} self.modules = [] def _get_scope(self, name: str): scope = self.scopes.get(name, None) if scope is None: scope = _Scope(name) self.scopes[name] = scope return scope
[docs] def get_command_entries(self, cmd_name: str): return self.cmd_map.get(cmd_name, [])
[docs] def register_module(self, module: CommandModule, include_invisible=True): self.modules.append(module) module_spec = module.get_spec() for cmd_spec in module_spec.cmd_specs: assert isinstance(cmd_spec, CommandSpec) if cmd_spec.visible or include_invisible: self.add_command( scope_name=module_spec.name, cmd_name=cmd_spec.name, desc=cmd_spec.description, usage=cmd_spec.usage, handler=cmd_spec.handler_func, authz_func=cmd_spec.authz_func, visible=cmd_spec.visible, confirm=cmd_spec.confirm, )
[docs] def add_command(self, scope_name, cmd_name, desc, usage, handler, authz_func, visible, confirm): scope = self._get_scope(scope_name) scope.register_command( cmd_name=cmd_name, cmd_desc=desc, cmd_usage=usage, handler_func=handler, authz_func=authz_func, visible=visible, confirm=confirm, )
def _add_cmd_entry(self, cmd_name, entry): entry_list = self.cmd_map.get(cmd_name, None) if entry_list is None: entry_list = [] self.cmd_map[cmd_name] = entry_list entry_list.append(entry)
[docs] def finalize(self, add_cmd_func=None): if len(self.cmd_map) > 0: # already finalized return for scope_name, scope in self.scopes.items(): for cmd_name, entry in scope.entries.items(): self._add_cmd_entry(cmd_name, entry) full_cmd_name = "{}.{}".format(scope_name, cmd_name) self._add_cmd_entry(full_cmd_name, entry) if add_cmd_func: add_cmd_func(entry)