Source code for nvflare.fuel.hci.client.config

# Copyright (c) 2023, 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 os
import re
import sys

from nvflare.apis.fl_constant import WorkspaceConstants
from nvflare.apis.workspace import Workspace
from nvflare.fuel.common.excepts import ConfigError
from nvflare.fuel.sec.security_content_service import LoadResult, SecurityContentManager
from nvflare.fuel.utils.json_scanner import Node
from nvflare.fuel.utils.wfconf import ConfigContext
from nvflare.private.json_configer import JsonConfigurator

from .api_spec import AdminConfigKey
from .event import EventHandler

FL_PACKAGES = ["nvflare"]
FL_MODULES = ["ha"]


[docs] class FLAdminClientStarterConfigurator(JsonConfigurator): """FL Admin Client startup configurator.""" def __init__(self, workspace: Workspace): """Uses the json configuration to start the FL admin client. Args: workspace: the workspace object """ base_pkgs = FL_PACKAGES module_names = FL_MODULES custom_dir = workspace.get_client_custom_dir() if os.path.isdir(custom_dir): sys.path.append(custom_dir) admin_config_file_path = workspace.get_admin_startup_file_path() config_files = [admin_config_file_path] resources_file_path = workspace.get_resources_file_path() if resources_file_path: config_files.append(resources_file_path) JsonConfigurator.__init__( self, config_file_name=config_files, base_pkgs=base_pkgs, module_names=module_names, exclude_libs=True, ) self.workspace = workspace self.admin_config_file_path = config_files self.overseer_agent = None self.handlers = []
[docs] def process_config_element(self, config_ctx: ConfigContext, node: Node): """Process config element. Args: config_ctx: config context node: element node """ element = node.element path = node.path() if re.search(r"^handlers\.#[0-9]+$", path): c = self.build_component(element) if not isinstance(c, EventHandler): raise ConfigError(f"component must be EventHandler but got {type(c)}") self.handlers.append(c) return
def _update_property_path_in_startup(self, admin_config: dict, prop_key: str): """The property value in the admin config is the base name. This method replaces it with absolute path in the startup kit's startup dir. Args: admin_config: the admin config data prop_key: key of the property Returns: """ prop_value = admin_config.get(prop_key) if prop_value: admin_config[prop_key] = self.workspace.get_file_path_in_startup(prop_value) def _update_property_path_in_root(self, admin_config: dict, prop_key: str): """The property value in the admin config is the base name. This method replaces it with absolute path in the startup kit's root dir. Args: admin_config: the admin config data prop_key: key of the property Returns: """ prop_value = admin_config.get(prop_key) if prop_value: admin_config[prop_key] = self.workspace.get_file_path_in_root(prop_value)
[docs] def start_config(self, config_ctx: ConfigContext): """Start the config process. Args: config_ctx: config context """ super().start_config(config_ctx) try: admin = self.get_admin_config() if isinstance(admin, dict): for key in [AdminConfigKey.CLIENT_KEY, AdminConfigKey.CLIENT_CERT, AdminConfigKey.CA_CERT]: self._update_property_path_in_startup(admin, key) for key in [AdminConfigKey.UPLOAD_DIR, AdminConfigKey.DOWNLOAD_DIR]: self._update_property_path_in_root(admin, key) except Exception: raise ValueError(f"Client config error: '{self.admin_config_file_path}'")
[docs] def get_admin_config(self): if not isinstance(self.config_data, dict): return None return self.config_data.get(AdminConfigKey.ADMIN)
[docs] def secure_load_admin_config(workspace: Workspace): mgr = SecurityContentManager(content_folder=workspace.get_startup_kit_dir()) # make sure admin startup config file is not tampered with _, result = mgr.load_json(WorkspaceConstants.ADMIN_STARTUP_CONFIG) if result != LoadResult.OK: raise ConfigError(f"invalid {WorkspaceConstants.ADMIN_STARTUP_CONFIG}: {result}") conf = FLAdminClientStarterConfigurator(workspace=workspace) conf.configure() return conf