2025-01-10 13:56:37 +01:00
|
|
|
from pprint import pprint
|
2025-01-09 13:53:56 +01:00
|
|
|
class ConfigurationResolver:
|
|
|
|
"""
|
|
|
|
A class to resolve `link` entries in a nested configuration structure.
|
|
|
|
Supports navigation through dictionaries, lists, and `subitems`.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, config):
|
|
|
|
self.config = config
|
|
|
|
|
|
|
|
def resolve_links(self):
|
|
|
|
"""
|
|
|
|
Resolves all `link` entries in the configuration.
|
|
|
|
"""
|
|
|
|
self._recursive_resolve(self.config, self.config)
|
|
|
|
|
|
|
|
def _recursive_resolve(self, current_config, root_config):
|
|
|
|
"""
|
|
|
|
Recursively resolves `link` entries in the configuration.
|
|
|
|
"""
|
|
|
|
if isinstance(current_config, dict):
|
|
|
|
for key, value in list(current_config.items()):
|
|
|
|
if key == "link":
|
2025-01-09 13:56:29 +01:00
|
|
|
try:
|
2025-01-10 13:56:37 +01:00
|
|
|
target = self._find_entry(root_config, value.lower(), True)
|
|
|
|
if isinstance(target, list) and len(target) > 2:
|
2025-01-10 14:17:26 +01:00
|
|
|
target = self._find_entry(root_config, value.lower(), False)
|
2025-01-09 13:56:29 +01:00
|
|
|
current_config.clear()
|
|
|
|
current_config.update(target)
|
2025-01-10 13:56:37 +01:00
|
|
|
except Exception as e:
|
2025-01-09 13:56:29 +01:00
|
|
|
raise ValueError(
|
|
|
|
f"Error resolving link '{value}': {str(e)}. "
|
|
|
|
f"Current path: {key}, Current config: {current_config}"
|
|
|
|
)
|
2025-01-09 13:53:56 +01:00
|
|
|
else:
|
|
|
|
self._recursive_resolve(value, root_config)
|
|
|
|
elif isinstance(current_config, list):
|
|
|
|
for item in current_config:
|
|
|
|
self._recursive_resolve(item, root_config)
|
|
|
|
|
2025-01-10 13:56:37 +01:00
|
|
|
def _get_subitems(self,current):
|
|
|
|
if isinstance(current, dict) and ("subitems" in current and current["subitems"]):
|
|
|
|
current = current["subitems"]
|
|
|
|
return current
|
|
|
|
|
|
|
|
def _find_by_name(self,current, part):
|
|
|
|
return next(
|
|
|
|
(item for item in current if isinstance(item, dict) and item.get("name", "").lower() == part),
|
|
|
|
None
|
|
|
|
)
|
|
|
|
|
|
|
|
def _find_entry(self, config, path, subitems):
|
2025-01-09 13:53:56 +01:00
|
|
|
"""
|
|
|
|
Finds an entry in the configuration by a dot-separated path.
|
|
|
|
Supports both dictionaries and lists with `subitems` navigation.
|
|
|
|
"""
|
|
|
|
parts = path.split('.')
|
|
|
|
current = config
|
|
|
|
for part in parts:
|
|
|
|
if isinstance(current, list):
|
|
|
|
# Look for a matching name in the list
|
2025-01-10 13:56:37 +01:00
|
|
|
found = self._find_by_name(current,part)
|
2025-01-09 14:20:59 +01:00
|
|
|
if found:
|
|
|
|
print(
|
|
|
|
f"Matching entry for '{part}' in list. Path so far: {' > '.join(parts[:parts.index(part)+1])}. "
|
|
|
|
f"Current list: {current}"
|
|
|
|
)
|
|
|
|
else:
|
2025-01-09 13:56:29 +01:00
|
|
|
raise ValueError(
|
|
|
|
f"No matching entry for '{part}' in list. Path so far: {' > '.join(parts[:parts.index(part)+1])}. "
|
|
|
|
f"Current list: {current}"
|
|
|
|
)
|
2025-01-09 13:53:56 +01:00
|
|
|
current = found
|
|
|
|
elif isinstance(current, dict):
|
|
|
|
# Case-insensitive dictionary lookup
|
|
|
|
key = next((k for k in current if k.lower() == part), None)
|
|
|
|
if key is None:
|
2025-01-10 13:56:37 +01:00
|
|
|
current = self._find_by_name(current["subitems"],part)
|
|
|
|
if not current:
|
|
|
|
raise KeyError(
|
|
|
|
f"Key '{part}' not found in dictionary. Path so far: {' > '.join(parts[:parts.index(part)+1])}. "
|
|
|
|
f"Current dictionary: {current}"
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
current = current[key]
|
|
|
|
|
2025-01-09 13:53:56 +01:00
|
|
|
else:
|
2025-01-09 13:56:29 +01:00
|
|
|
raise ValueError(
|
|
|
|
f"Invalid path segment '{part}'. Current type: {type(current)}. "
|
|
|
|
f"Path so far: {' > '.join(parts[:parts.index(part)+1])}"
|
|
|
|
)
|
2025-01-10 13:56:37 +01:00
|
|
|
if subitems:
|
|
|
|
current = self._get_subitems(current)
|
2025-01-09 13:53:56 +01:00
|
|
|
|
|
|
|
return current
|
|
|
|
|
|
|
|
def get_config(self):
|
|
|
|
"""
|
|
|
|
Returns the resolved configuration.
|
|
|
|
"""
|
2025-01-09 13:56:29 +01:00
|
|
|
return self.config
|