-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathmet-config.py
executable file
·130 lines (105 loc) · 3.75 KB
/
met-config.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
119
120
121
122
123
124
125
126
127
128
129
130
#!/usr/bin/env python3
import sys
import argparse
import json
import pprint
def get_nested(config, keys):
"""Retrieve a nested value using a list of keys."""
for key in keys:
if key in config:
config = config[key]
else:
allowed_keys = "[" + ", ".join(config.keys()) + "]"
raise KeyError(f"Key '{key}' not in {allowed_keys}")
return config
def str2bool(v):
return v.lower() in ("yes", "true", "t", "1")
def set_nested(config, keys, value):
"""Set a nested value using a list of keys, creating dictionaries as needed."""
target = config[keys[0]][keys[1]]
if isinstance(target, bool):
config[keys[0]][keys[1]] = str2bool(value)
elif isinstance(target, str):
config[keys[0]][keys[1]] = value
elif isinstance(target, int):
config[keys[0]][keys[1]] = int(value)
elif isinstance(target, float):
config[keys[0]][keys[1]] = float(value)
elif hasattr(target, "__len__"): # target is list-alike
# handle " and ' shell string escaping if non ambiguous
if "'" in value and '"' not in value:
value = value.replace("'", '"')
data = json.loads(value)
config[keys[0]][keys[1]] = data
else:
raise KeyError("Unsupported target type: ", type(target))
def parsePath(argpath):
keys = argpath.split(".")
if len(keys) >= 1 and keys[0] == "":
keys = keys[1:]
if keys[0] == "":
return None
else:
return keys
def main():
parser = argparse.ArgumentParser(
description="Inspect and modify a configuration dictionary."
)
# Global flag for JSON output.
parser.add_argument("--json", action="store_true", help="Output in JSON format")
subparsers = parser.add_subparsers(
dest="command", required=True, help="Sub-command to run"
)
# 'get' command
parser_get = subparsers.add_parser("get", help="Retrieve a configuration value.")
parser_get.add_argument(
"keypath", help="Dot-separated key path (e.g., machine.version)"
)
# 'set' command
parser_set = subparsers.add_parser("set", help="Set a configuration value.")
parser_set.add_argument(
"keypath", help="Dot-separated key path (e.g., machine.version)"
)
parser_set.add_argument("value", help="Value to set")
args = parser.parse_args()
try:
if args.command == "get": #
keys = parsePath(args.keypath)
from config import MeticulousConfig
if keys is None:
result = MeticulousConfig
else:
result = get_nested(MeticulousConfig, keys)
if args.json:
print(json.dumps(result, indent=2))
else:
if isinstance(result, str):
print(result)
else:
pprint.pprint(result)
elif args.command == "set":
keys = parsePath(args.keypath)
if keys is None:
raise Exception("Cannot set the root of the configuration.")
from config import MeticulousConfig
set_nested(MeticulousConfig, keys, args.value)
result = get_nested(MeticulousConfig, keys)
if args.json:
print(json.dumps(result, indent=2))
else:
print(".".join(keys), "= ", end="")
if isinstance(result, str):
print(result)
else:
pprint.pprint(result)
MeticulousConfig.save()
else:
parser.print_help()
sys.exit(1)
except Exception as e:
print("Error:", e)
import traceback
traceback.print_exception(e)
sys.exit(1)
if __name__ == "__main__":
main()