Optimized dependency loading

This commit is contained in:
Kevin Veen-Birkenbach 2025-05-20 10:34:30 +02:00
parent 5948d7aa93
commit 969a176be1
No known key found for this signature in database
GPG Key ID: 44D8F11FD62F878E
12 changed files with 125 additions and 167 deletions

View File

@ -1,13 +1,10 @@
import os import os
import argparse
import yaml import yaml
import argparse
from collections import defaultdict, deque
def find_roles(roles_dir, prefix=None): def find_roles(roles_dir, prefix=None):
""" """Find all roles in the given directory."""
Yield absolute paths of role directories under roles_dir.
Only include roles whose directory name starts with prefix (if given) and contain meta/main.yml.
"""
for entry in os.listdir(roles_dir): for entry in os.listdir(roles_dir):
if prefix and not entry.startswith(prefix): if prefix and not entry.startswith(prefix):
continue continue
@ -16,87 +13,120 @@ def find_roles(roles_dir, prefix=None):
if os.path.isdir(path) and os.path.isfile(meta_file): if os.path.isdir(path) and os.path.isfile(meta_file):
yield path, meta_file yield path, meta_file
def load_run_after(meta_file):
def load_role_order(meta_file): """Load the 'run_after' from the meta/main.yml of a role."""
"""
Load the meta/main.yml and return the role_run_order field.
Returns a dict with 'before' and 'after' keys. Defaults to empty lists if not found.
"""
with open(meta_file, 'r') as f: with open(meta_file, 'r') as f:
data = yaml.safe_load(f) or {} data = yaml.safe_load(f) or {}
run_order = data.get('role_run_order', {}) return data.get('galaxy_info', {}).get('run_after', [])
before = run_order.get('before', [])
after = run_order.get('after', [])
# If "all" is in before or after, treat it as a special value
if "all" in before:
before.remove("all")
before.insert(0, "all") # Treat "all" as the first item
if "all" in after:
after.remove("all")
after.append("all") # Treat "all" as the last item
return {
'before': before,
'after': after
}
def load_application_id(role_path):
"""Load the application_id from the vars/main.yml of the role."""
vars_file = os.path.join(role_path, 'vars', 'main.yml')
if os.path.exists(vars_file):
with open(vars_file, 'r') as f:
data = yaml.safe_load(f) or {}
return data.get('application_id')
return None
def build_dependency_graph(roles_dir, prefix=None):
"""Build a dependency graph where each role points to the roles it depends on."""
graph = defaultdict(list)
in_degree = defaultdict(int)
roles = {}
def sort_roles_by_order(roles_dir, prefix=None):
roles = []
# Collect roles and their before/after dependencies
for role_path, meta_file in find_roles(roles_dir, prefix): for role_path, meta_file in find_roles(roles_dir, prefix):
run_order = load_role_order(meta_file) run_after = load_run_after(meta_file)
application_id = load_application_id(role_path)
role_name = os.path.basename(role_path) role_name = os.path.basename(role_path)
roles.append({ roles[role_name] = {
'role_name': role_name, 'role_name': role_name,
'before': run_order['before'], 'run_after': run_after,
'after': run_order['after'], 'application_id': application_id,
'path': role_path 'path': role_path
}) }
# Now sort the roles based on before/after relationships # If the role has dependencies, build the graph
for dependency in run_after:
graph[dependency].append(role_name)
in_degree[role_name] += 1
# Ensure roles with no dependencies have an in-degree of 0
if role_name not in in_degree:
in_degree[role_name] = 0
return graph, in_degree, roles
def topological_sort(graph, in_degree):
"""Perform topological sort on the dependency graph."""
# Queue for roles with no incoming dependencies (in_degree == 0)
queue = deque([role for role, degree in in_degree.items() if degree == 0])
sorted_roles = [] sorted_roles = []
unresolved_roles = roles[:]
# First, place roles with "before: all" at the start
roles_with_before_all = [role for role in unresolved_roles if "all" in role['before']]
sorted_roles.extend(roles_with_before_all)
unresolved_roles = [role for role in unresolved_roles if "all" not in role['before']]
while unresolved_roles:
# Find roles with no dependencies in 'before'
ready_roles = [role for role in unresolved_roles if not any(dep in [r['role_name'] for r in unresolved_roles] for dep in role['before'])]
if not ready_roles:
raise ValueError("Circular dependency detected in 'before'/'after' fields")
for role in ready_roles: while queue:
sorted_roles.append(role) role = queue.popleft()
unresolved_roles.remove(role) sorted_roles.append(role)
# Remove from the 'before' lists of remaining roles # Reduce in-degree for roles dependent on the current role
for r in unresolved_roles: for neighbor in graph[role]:
r['before'] = [dep for dep in r['before'] if dep != role['role_name']] in_degree[neighbor] -= 1
if in_degree[neighbor] == 0:
queue.append(neighbor)
# Finally, place roles with "after: all" at the end if len(sorted_roles) != len(in_degree):
roles_with_after_all = [role for role in unresolved_roles if "all" in role['after']] # If the number of sorted roles doesn't match the number of roles,
sorted_roles.extend(roles_with_after_all) # there was a cycle in the graph (not all roles could be sorted)
unresolved_roles = [role for role in unresolved_roles if "all" not in role['after']] raise Exception("Circular dependency detected among the roles!")
return sorted_roles return sorted_roles
def print_dependency_tree(graph):
"""Print the dependency tree visually on the console."""
def print_node(role, indent=0):
print(" " * indent + role)
for dependency in graph[role]:
print_node(dependency, indent + 1)
# Print the tree starting from roles with no dependencies
all_roles = set(graph.keys())
dependent_roles = {role for dependencies in graph.values() for role in dependencies}
root_roles = all_roles - dependent_roles
printed_roles = []
def collect_roles(role, indent=0):
printed_roles.append(role)
for dependency in graph[role]:
collect_roles(dependency, indent + 1)
for root in root_roles:
collect_roles(root)
return printed_roles
def generate_playbook_entries(roles_dir, prefix=None): def generate_playbook_entries(roles_dir, prefix=None):
entries = [] """Generate playbook entries based on the sorted order."""
sorted_roles = sort_roles_by_order(roles_dir, prefix) # Build dependency graph
graph, in_degree, roles = build_dependency_graph(roles_dir, prefix)
for role in sorted_roles: # Print and collect roles in tree order
# entry text tree_sorted_roles = print_dependency_tree(graph)
# Topologically sort the roles
sorted_role_names = topological_sort(graph, in_degree)
# Ensure that roles that appear in the tree come first
final_sorted_roles = [role for role in tree_sorted_roles if role in sorted_role_names]
# Include the remaining unsorted roles
final_sorted_roles += [role for role in sorted_role_names if role not in final_sorted_roles]
# Generate the playbook entries
entries = []
for role_name in final_sorted_roles:
role = roles[role_name]
entry = ( entry = (
f"- name: setup {role['role_name']}\n" f"- name: setup {role['application_id']}\n" # Use application_id here
f" when: (\"{role['role_name']}\" in group_names)\n" f" when: ('{role['application_id']}' in group_names)\n" # Correct condition format
f" include_role:\n" f" include_role:\n"
f" name: {role['role_name']}\n" f" name: {role['role_name']}\n"
) )
@ -104,10 +134,9 @@ def generate_playbook_entries(roles_dir, prefix=None):
return entries return entries
def main(): def main():
parser = argparse.ArgumentParser( parser = argparse.ArgumentParser(
description='Generate an Ansible playbook include file from Docker roles and application_ids, sorted by role_run_order.' description='Generate an Ansible playbook include file from Docker roles, sorted by run_after order.'
) )
parser.add_argument( parser.add_argument(
'roles_dir', 'roles_dir',
@ -123,8 +152,14 @@ def main():
help='Output file path (default: stdout)', help='Output file path (default: stdout)',
default=None default=None
) )
parser.add_argument(
'-t', '--tree',
action='store_true',
help='Display the dependency tree of roles visually'
)
args = parser.parse_args() args = parser.parse_args()
# Generate and output the playbook entries
entries = generate_playbook_entries(args.roles_dir, args.prefix) entries = generate_playbook_entries(args.roles_dir, args.prefix)
output = ''.join(entries) output = ''.join(entries)
@ -135,6 +170,5 @@ def main():
else: else:
print(output) print(output)
if __name__ == '__main__': if __name__ == '__main__':
main() main()

View File

@ -28,4 +28,6 @@ galaxy_info:
issue_tracker_url: https://s.veen.world/cymaisissues issue_tracker_url: https://s.veen.world/cymaisissues
documentation: https://s.veen.world/cymais documentation: https://s.veen.world/cymais
logo: logo:
class: "fa-solid fa-chalkboard-teacher" class: "fa-solid fa-chalkboard-teacher"
run_after:
- docker-keycloak

View File

@ -19,4 +19,6 @@ galaxy_info:
documentation: https://s.veen.world/cymais documentation: https://s.veen.world/cymais
logo: logo:
class: "fa-solid fa-comments" class: "fa-solid fa-comments"
run_after:
- docker-wordpress
dependencies: [] dependencies: []

View File

@ -19,9 +19,6 @@ galaxy_info:
documentation: https://s.veen.world/cymais documentation: https://s.veen.world/cymais
logo: logo:
class: "fa-solid fa-lock" class: "fa-solid fa-lock"
run_after:
- docker-matomo
dependencies: [] dependencies: []
role_run_order:
before:
- all
after:
- docker-ldap

View File

@ -19,4 +19,6 @@ galaxy_info:
documentation: https://s.veen.world/cymais documentation: https://s.veen.world/cymais
logo: logo:
class: "fa-solid fa-network-wired" class: "fa-solid fa-network-wired"
run_after:
- docker-keycloak
dependencies: [] dependencies: []

View File

@ -20,7 +20,6 @@ galaxy_info:
documentation: https://s.veen.world/cymais documentation: https://s.veen.world/cymais
logo: logo:
class: "fa-solid fa-users" class: "fa-solid fa-users"
#run_after:
# - "0"
dependencies: [] dependencies: []
role_run_order:
before:
- all

View File

@ -21,5 +21,5 @@ galaxy_info:
documentation: "https://s.veen.world/cymais" documentation: "https://s.veen.world/cymais"
logo: logo:
class: "fa-solid fa-bullhorn" class: "fa-solid fa-bullhorn"
role_run_order: run_after:
after: docker-keycloak - docker-keycloak

View File

@ -18,8 +18,5 @@ galaxy_info:
documentation: "https://s.veen.world/cymais" documentation: "https://s.veen.world/cymais"
logo: logo:
class: "fa-solid fa-chart-line" class: "fa-solid fa-chart-line"
role_run_order: run_after:
before: - "docker-ldap"
- all
after:
- docker-keycloak

View File

@ -29,4 +29,5 @@ galaxy_info:
documentation: "https://s.veen.world/cymais" documentation: "https://s.veen.world/cymais"
logo: logo:
class: "fa-solid fa-project-diagram" class: "fa-solid fa-project-diagram"
dependencies: [] run_after:
- docker-keycloak

View File

@ -28,4 +28,5 @@ galaxy_info:
documentation: "https://s.veen.world/cymais" documentation: "https://s.veen.world/cymais"
logo: logo:
class: "fa-solid fa-video" class: "fa-solid fa-video"
dependencies: [] run_after:
- docker-keycloak

View File

@ -27,4 +27,5 @@ galaxy_info:
documentation: "https://s.veen.world/cymais" documentation: "https://s.veen.world/cymais"
logo: logo:
class: "fa-solid fa-blog" class: "fa-solid fa-blog"
dependencies: [] run_after:
- docker-keycloak

View File

@ -1,78 +0,0 @@
import unittest
import os
import yaml
def load_yaml_file(file_path):
"""Load a YAML file and return its content."""
with open(file_path, 'r') as file:
return yaml.safe_load(file) or {}
def get_meta_info(role_path):
"""Extract before and after dependencies from the meta/main.yml of a role."""
meta_file = os.path.join(role_path, 'meta', 'main.yml')
if not os.path.isfile(meta_file):
return [], []
meta_data = load_yaml_file(meta_file)
run_order = meta_data.get('role_run_order', {})
before = run_order.get('before', [])
after = run_order.get('after', [])
return before, after
def resolve_dependencies(roles_dir):
"""Resolve all role dependencies and detect circular dependencies in before/after."""
visited = set() # Tracks roles that have been processed
stack = set() # Tracks roles being processed in the current recursion path
def visit(role_path, stack):
role_name = os.path.basename(role_path)
# Check for circular dependencies in the recursion path (for before/after)
if role_name in stack:
raise ValueError(f"Circular dependency detected in 'before'/'after' between roles: {' -> '.join(stack)} -> {role_name}")
# Check if role is already processed
if role_name in visited:
return []
# Mark role as visited and add to stack
visited.add(role_name)
stack.append(role_name)
# Get before and after dependencies
before, after = get_meta_info(role_path)
# Resolve before and after roles
for before_role in before:
before_role_path = os.path.join(roles_dir, before_role)
visit(before_role_path, stack) # Recurse into before dependencies
for after_role in after:
after_role_path = os.path.join(roles_dir, after_role)
visit(after_role_path, stack) # Recurse into after dependencies
stack.pop() # Remove the current role from the stack
for role_name in os.listdir(roles_dir):
role_path = os.path.join(roles_dir, role_name)
if os.path.isdir(role_path):
try:
visit(role_path, []) # Start recursion from this role
except ValueError as e:
raise ValueError(f"Error processing role '{role_name}' at path '{role_path}': {str(e)}")
return "No Circular Dependency Detected"
class TestRoleBeforeAfterDependencies(unittest.TestCase):
def test_no_circular_before_after_dependencies(self):
roles_dir = "roles" # Path to the roles directory
try:
result = resolve_dependencies(roles_dir)
except ValueError as e:
self.fail(f"Circular dependency detected: {e}")
# If no exception, the test passed
self.assertEqual(result, "No Circular Dependency Detected")
if __name__ == '__main__':
unittest.main()