summaryrefslogtreecommitdiff
path: root/kvmd
diff options
context:
space:
mode:
authorAdam Outler <[email protected]>2023-06-19 22:35:53 -0400
committerGitHub <[email protected]>2023-06-20 05:35:53 +0300
commitdb3f6220238da8138c670924a074477987790fbe (patch)
tree66a086a240d6f44485e9dc460dbc1fe5b326dbf5 /kvmd
parent9879a9f05b07d329b8db140e58d78f6005b2d76a (diff)
Refactoring merge Method into a New Class & Adding Unit Tests (#137)
Diffstat (limited to 'kvmd')
-rw-r--r--kvmd/apps/__init__.py5
-rw-r--r--kvmd/tools.py9
-rw-r--r--kvmd/yamlconf/loader.py8
-rw-r--r--kvmd/yamlconf/merger.py48
4 files changed, 56 insertions, 14 deletions
diff --git a/kvmd/apps/__init__.py b/kvmd/apps/__init__.py
index 04f88ffa..31847bc3 100644
--- a/kvmd/apps/__init__.py
+++ b/kvmd/apps/__init__.py
@@ -53,6 +53,7 @@ from ..yamlconf import Option
from ..yamlconf import build_raw_from_options
from ..yamlconf.dumper import make_config_dump
from ..yamlconf.loader import load_yaml_file
+from ..yamlconf.merger import yaml_merge
from ..validators.basic import valid_stripped_string
from ..validators.basic import valid_stripped_string_not_empty
@@ -177,8 +178,8 @@ def _init_config(config_path: str, override_options: list[str], **load_flags: bo
scheme = _get_config_scheme()
try:
- tools.merge(raw_config, (raw_config.pop("override", {}) or {}))
- tools.merge(raw_config, build_raw_from_options(override_options))
+ yaml_merge(raw_config, (raw_config.pop("override", {}) or {}))
+ yaml_merge(raw_config, build_raw_from_options(override_options), "raw command line options")
_patch_raw(raw_config)
config = make_config(raw_config, scheme)
diff --git a/kvmd/tools.py b/kvmd/tools.py
index 773fcb5a..1c03f8ce 100644
--- a/kvmd/tools.py
+++ b/kvmd/tools.py
@@ -45,15 +45,6 @@ def efmt(err: Exception) -> str:
# =====
-def merge(dest: dict, src: dict) -> None:
- for key in src:
- if key in dest:
- if isinstance(dest[key], dict) and isinstance(src[key], dict):
- merge(dest[key], src[key])
- continue
- dest[key] = src[key]
-
-
def rget(dct: dict, *keys: Hashable) -> dict:
result = functools.reduce((lambda nxt, key: nxt.get(key, {})), keys, dct)
if not isinstance(result, dict):
diff --git a/kvmd/yamlconf/loader.py b/kvmd/yamlconf/loader.py
index 55bc7aa6..ffd69e3e 100644
--- a/kvmd/yamlconf/loader.py
+++ b/kvmd/yamlconf/loader.py
@@ -22,6 +22,8 @@
import os
+from .. import tools
+
from typing import IO
from typing import Any
@@ -30,7 +32,7 @@ import yaml.nodes
import yaml.resolver
import yaml.constructor
-from .. import tools
+from .merger import yaml_merge
# =====
@@ -70,9 +72,9 @@ class _YamlLoader(yaml.SafeLoader):
for child in sorted(os.listdir(inc_path)):
child_path = os.path.join(inc_path, child)
if os.path.isfile(child_path) or os.path.islink(child_path):
- tools.merge(tree, (load_yaml_file(child_path) or {}))
+ yaml_merge(tree, (load_yaml_file(child_path) or {}), child_path)
else: # Try file
- tools.merge(tree, (load_yaml_file(inc_path) or {}))
+ yaml_merge(tree, (load_yaml_file(inc_path) or {}), inc_path)
return tree
diff --git a/kvmd/yamlconf/merger.py b/kvmd/yamlconf/merger.py
new file mode 100644
index 00000000..51a168c8
--- /dev/null
+++ b/kvmd/yamlconf/merger.py
@@ -0,0 +1,48 @@
+# ========================================================================== #
+# #
+# KVMD - The main PiKVM daemon. #
+# #
+# Copyright (C) 2018-2022 Maxim Devaev <[email protected]> #
+# #
+# This program is free software: you can redistribute it and/or modify #
+# it under the terms of the GNU General Public License as published by #
+# the Free Software Foundation, either version 3 of the License, or #
+# (at your option) any later version. #
+# #
+# This program is distributed in the hope that it will be useful, #
+# but WITHOUT ANY WARRANTY; without even the implied warranty of #
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
+# GNU General Public License for more details. #
+# #
+# You should have received a copy of the GNU General Public License #
+# along with this program. If not, see <https://www.gnu.org/licenses/>. #
+# #
+# ========================================================================== #
+
+from typing import Optional
+
+
+# =====
+def _merge(dest: dict, src: dict) -> None:
+ for key in src:
+ if key in dest:
+ if isinstance(dest[key], dict) and isinstance(src[key], dict):
+ _merge(dest[key], src[key])
+ continue
+ dest[key] = src[key]
+
+
+def yaml_merge(dest: dict, src: dict, source_name: Optional[str]=None) -> None:
+ """ Merges the source dictionary into the destination dictionary. """
+
+ # Checking if destination is None
+ if dest is None:
+ # We can't merge into a None
+ raise ValueError(f"Could not merge {source_name} into None. The destination cannot be None")
+
+ # Checking if source is None or empty
+ if src is None:
+ # If src is None or empty, there's nothing to merge
+ return
+
+ _merge(dest, src)