-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathansible-flag-parser.py
118 lines (89 loc) · 3.44 KB
/
ansible-flag-parser.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
#!/usr/bin/env python3
# ansible-flag-parser: run ansible playbooks using any flags
# This is a poc
# Stephen Maher
# todo
#. set ansbile.cfg options ?? forks / ssh etc
#. change path
#. add loggin feature
#. set output method
#. run in container :)
#. run via ansible-navigator !!
#. credentials / store / secret reading ?? hsm ??
#. option to run in nice but simole chui like navigator ?? operator mode
# builtin requirements.yml
# . system wide config file and user config file, ie global log or comtainer mounts/opts
import argparse
import json
import yaml
import os
import subprocess
import sys
import signal
import re
def load_metadata_from_self():
"""Load metadata by reading YAML directly from the script file itself."""
with open(__file__, 'r') as f:
content = f.read()
# Use regex to capture YAML embedded after "--- METADATA ---" until "--- END METADATA ---"
metadata_match = re.search(r"--- METADATA ---\n(.*?)\n--- END METADATA ---", content, re.DOTALL)
if metadata_match:
metadata_yaml = metadata_match.group(1)
return yaml.safe_load(metadata_yaml)
else:
pass # do we care if there is no meta??
# raise ValueError("No metadata found in the script.")
def parse_flags(metadata):
"""Parse flags from the metadata."""
parser = argparse.ArgumentParser(description="Flag parser for ansible-playbook")
# Define flags based on metadata
for flag, properties in metadata.items():
if flag.startswith('_'): # Skip internal variables
continue
# @@SGM we will extend this
parser.add_argument(
f"--{flag}",
help=properties.get("help", ""),
required=properties.get("required", False),
default=properties.get("default", None),
)
# Extra flag for CTRL+C handling and rescuer playbook
# @@SGM these should not be exposed but defined from the meta
parser.add_argument("--no-ctrl-c", action="store_true", help="Disable CTRL+C trapping.")
parser.add_argument("--rescuer", action="store_true", help="Execute rescuer playbook on failure.")
return parser.parse_args()
def set_env_vars(env_vars):
"""Set environment variables based on metadata."""
for key, value in env_vars.items():
os.environ[key] = str(value)
def disable_ctrlc():
"""Disable CTRL+C trapping if --no-ctrlc is set."""
signal.signal(signal.SIGINT, signal.SIG_IGN)
def main():
# Load metadata from the script itself
metadata = load_metadata_from_self()
# Parse arguments based on metadata
args = parse_flags(metadata)
# Disable CTRL+C trapping if --no-ctrlc is set
if args.no_ctrlc:
disable_ctrlc()
# Set environment variables if specified in metadata
env_vars = metadata.get("environment", {})
set_env_vars(env_vars)
# Prepare extra vars and options for ansible-playbook
extra_vars = {flag: getattr(args, flag) for flag in vars(args)}
# also needs extraflags
# Run ansible-playbook command using __file__ as the playbook
command = [
"ansible-playbook",
__file__, # Self-reference as the playbook
"-e", json.dumps(extra_vars),
]
if args.rescuer:
command.append("--rescuer-playbook") # Run rescuer if needed
try:
subprocess.run(command, check=True)
except subprocess.CalledProcessError as e:
sys.exit(e.returncode)
if __name__ == "__main__":
main()