forked from cmu-db/benchbase
-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathmodifyyaml.py
105 lines (81 loc) · 3.89 KB
/
modifyyaml.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
#!/usr/bin/python
import json
import sys
import yaml
from jinja2 import Environment, FileSystemLoader, Undefined
from functools import reduce
from operator import getitem
from collections import OrderedDict
# Custom Undefined handler to return an empty string or leave placeholders as-is
class SilentUndefined(Undefined):
def _fail_with_undefined_error(self, *args, **kwargs):
return '' # Return empty string if undefined
def __str__(self):
return self._fail_with_undefined_error()
def jinja2_modification(context, input_yaml):
template_env = Environment(loader=FileSystemLoader("./"), undefined=SilentUndefined)
template = template_env.get_template(input_yaml)
with open(input_yaml, 'w') as f:
f.write(template.render(context))
# Load the YAML using OrderedDict to preserve the order of elements
def ordered_yaml_loader(stream, Loader=yaml.SafeLoader, object_pairs_hook=OrderedDict):
class OrderedLoader(Loader):
pass
def construct_mapping(loader, node):
loader.flatten_mapping(node)
return object_pairs_hook(loader.construct_pairs(node))
OrderedLoader.add_constructor(
yaml.resolver.BaseResolver.DEFAULT_MAPPING_TAG,
construct_mapping)
return yaml.load(stream, OrderedLoader)
# Custom dumper to preserve empty strings as "" instead of null
def ordered_yaml_dumper(data, stream=None, Dumper=yaml.SafeDumper, **kwds):
class OrderedDumper(Dumper):
pass
# Add custom representer for OrderedDict to preserve order
def _dict_representer(dumper, data):
return dumper.represent_dict(data.items())
# Add custom representer for empty strings to prevent 'null' output
def _str_representer(dumper, value):
if not value or value == '' or value == 'null':
return dumper.represent_scalar('tag:yaml.org,2002:str', '') # Keep as empty string
return dumper.represent_scalar('tag:yaml.org,2002:str', value)
OrderedDumper.add_representer(OrderedDict, _dict_representer)
OrderedDumper.add_representer(str, _str_representer) # Ensure empty strings are represented as ""
return yaml.dump(data, stream, OrderedDumper, **kwds)
def set_nested_item(dataDict, mapList, val):
"""Set item in nested dictionary, handling list indices if present."""
for i, key in enumerate(mapList[:-1]):
if isinstance(reduce(getitem, mapList[:i], dataDict), list):
index = int(key[key.find("[") + 1:key.find("]")]) # Extract index inside brackets
mapList[i] = index # Replace the string with an integer index
reduce(getitem, mapList[:-1], dataDict)[mapList[-1]] = val
return dataDict
def main():
context = sys.argv[1]
yaml_file = sys.argv[2]
data = json.loads(context, strict=False)
jinja2_modification(data, yaml_file)
with open(yaml_file) as f:
doc = ordered_yaml_loader(f, Loader=yaml.FullLoader)
for key, value in data.items():
if key == "warmup":
doc["works"]["work"]["warmup"] = value
elif key == "time":
doc["works"]["work"]["time_secs"] = value
elif key == "url":
doc[key] = doc[key].replace("localhost", value)
if "sslmode" in data:
doc[key] = doc[key].replace("disable", "require") if data["sslmode"] else doc[key].replace(
"require", "disable")
if "load-balance" in data and data['load-balance'] == True:
doc[key] = doc[key] + "&load-balance=true"
elif key == "setAutoCommit":
doc["microbenchmark"]["properties"]["setAutoCommit"] = value
else:
nested_key = key.replace('[', '.[').split('.')
set_nested_item(doc, nested_key, value)
with open(yaml_file, 'w') as fnew:
ordered_yaml_dumper(doc, fnew, Dumper=yaml.SafeDumper, allow_unicode=True)
return
main()