rougail/tests/test_2_makedict.py

256 lines
9.3 KiB
Python
Raw Normal View History

2019-11-26 20:33:24 +01:00
from os.path import isfile, join, isdir
2023-05-19 12:56:13 +02:00
from pytest import fixture
2022-01-19 18:24:00 +01:00
from os import listdir, mkdir, environ
2019-11-26 20:33:24 +01:00
from json import dump, load, dumps, loads
from pathlib import Path
2019-11-26 20:33:24 +01:00
2022-01-19 18:24:00 +01:00
environ['TIRAMISU_LOCALE'] = 'en'
from .custom import CustomOption
2019-12-02 10:31:55 +01:00
from tiramisu import Config
from tiramisu.error import PropertiesOptionError
2019-11-26 20:33:24 +01:00
2020-08-12 08:23:38 +02:00
dico_dirs = 'tests/dictionaries'
2019-11-26 20:33:24 +01:00
test_ok = set()
for test in listdir(dico_dirs):
if isdir(join(dico_dirs, test)):
2020-07-29 08:59:40 +02:00
if isdir(join(dico_dirs, test, 'tiramisu')):
2019-11-26 20:33:24 +01:00
test_ok.add(test)
2020-11-18 22:18:16 +01:00
debug = False
#debug = True
2019-11-26 20:33:24 +01:00
excludes = set([])
2024-10-22 12:40:41 +02:00
#excludes = set([
# '80leadership_subfamily',
# '80valid_enum_variables',
#])
2023-10-12 08:17:30 +02:00
2024-10-22 12:40:41 +02:00
# excludes = set(['60_5family_dynamic_variable_outside_sub_suffix'])
2019-11-26 20:33:24 +01:00
test_ok -= excludes
# test_ok = ['60_0family_dynamic_static']
2019-11-26 20:33:24 +01:00
test_ok = list(test_ok)
test_ok.sort()
no_test_base_multi = False
#no_test_base_multi = True
#no_test_base = True
no_test_base = False
#no_test_multi = True
no_test_multi = False
2019-11-26 20:33:24 +01:00
@fixture(scope="module", params=test_ok)
def test_dir(request):
return request.param
2024-06-19 17:36:18 +02:00
def option_value(parent, key_is_option=False):
for option, value in parent.items():
if option.isoptiondescription():
if not key_is_option and option.isleadership():
ret = []
for idx, datas in enumerate(option_value(value, key_is_option=True)):
sub_option, sub_value = datas
if not idx:
sub_option = sub_option.path()
key = sub_option
for val in sub_value:
ret.append({sub_option: val})
else:
index = sub_option.index()
sub_option = sub_option.path()
ret[index][sub_option] = sub_value
yield key, ret
else:
yield from option_value(value, key_is_option)
elif key_is_option:
yield option, value
else:
yield option.path(), value
2023-05-19 12:56:13 +02:00
def launch_flattener(test_dir,
filename,
):
2019-11-26 20:33:24 +01:00
makedict_dir = join(test_dir, 'makedict')
makedict_file = join(makedict_dir, 'base.json')
makedict_before = join(makedict_dir, 'before.json')
makedict_after = join(makedict_dir, 'after.json')
2023-06-22 12:11:14 +02:00
informations_file = join(test_dir, 'informations.json')
mandatory_file = Path(makedict_dir) / 'mandatory.json'
2020-07-20 18:13:53 +02:00
modulepath = join(test_dir, 'tiramisu', filename + '.py')
2025-01-02 21:06:13 +01:00
if not isfile(modulepath):
return
with open(modulepath) as fh:
optiondescription = {}
exec(fh.read(), {'CustomOption': CustomOption}, optiondescription) # pylint: disable=W0122
config = Config(optiondescription["option_0"])
# change default rights
2023-06-22 12:11:14 +02:00
ro_origin = config.property.default('read_only', 'append')
ro_append = frozenset(ro_origin - {'force_store_value'})
2023-06-22 12:11:14 +02:00
rw_origin = config.property.default('read_write', 'append')
rw_append = frozenset(rw_origin - {'force_store_value'})
2023-05-19 12:56:13 +02:00
config.property.setdefault(ro_append, 'read_only', 'append')
config.property.setdefault(rw_append, 'read_write', 'append')
config.information.set('test_information', 'value')
config.information.set('test_information_list', ['value'])
2023-05-19 12:56:13 +02:00
config.property.read_only()
config.property.remove('mandatory')
config.information.set('info', 'value')
2023-06-22 12:11:14 +02:00
if isfile(informations_file):
with open(informations_file) as informations:
for key, value in load(informations).items():
if filename == 'base':
config.option(key).information.set('test_information', value)
elif filename == 'no_namespace':
config.option(key.split('.', 1)[-1]).information.set('test_information', value)
2023-06-22 12:11:14 +02:00
else:
for root in ['1', '2']:
config.option(f'{root}.{key}').information.set('test_information', value)
#
2025-01-02 21:06:13 +01:00
if not isdir(makedict_dir):
mkdir(makedict_dir)
2024-06-19 17:36:18 +02:00
config_dict = dict(option_value(config.value.get()))
2025-01-02 21:06:13 +01:00
if not isfile(Path(test_dir) / 'tiramisu' / 'base.py'):
tconfig_dict = {f'rougail.{path}': value for path, value in config_dict.items()}
2022-10-01 22:27:22 +02:00
else:
2025-01-02 21:06:13 +01:00
tconfig_dict = config_dict
if not isfile(makedict_file) or debug:
with open(makedict_file, 'w') as fh:
dump(tconfig_dict, fh, indent=4)
fh.write('\n')
if filename == 'no_namespace':
config_dict = {f'rougail.{path}': value for path, value in config_dict.items()}
elif filename != 'base':
2022-10-01 22:27:22 +02:00
config_dict_prefix = {'1': {}, '2': {}}
for key, value in config_dict.items():
prefix, path = key.split('.', 1)
if value and isinstance(value, list) and isinstance(value[0], dict):
new_value = []
for dct in value:
new_dct = {}
for k, v in dct.items():
k = k.split('.', 1)[-1]
new_dct[k] = v
new_value.append(new_dct)
value = new_value
config_dict_prefix[prefix][path] = value
assert loads(dumps(config_dict_prefix['1'])) == loads(dumps(config_dict_prefix['2']))
config_dict = config_dict_prefix['1']
2019-11-26 20:33:24 +01:00
if not isfile(makedict_file):
raise Exception('dict is not empty')
with open(makedict_file, 'r') as fh:
2025-01-02 21:06:13 +01:00
loaded_config_dict = load(fh)
assert loaded_config_dict == loads(dumps(config_dict)), f"error in file {makedict_file}"
#
2025-01-02 21:06:13 +01:00
value_owner(test_dir, makedict_before, config, filename)
# deploy
2023-06-22 12:11:14 +02:00
ro = config.property.default('read_only', 'append')
ro = frozenset(list(ro) + ['force_store_value'])
2023-05-19 12:56:13 +02:00
config.property.setdefault(ro, 'read_only', 'append')
2023-06-22 12:11:14 +02:00
rw = config.property.default('read_write', 'append')
rw = frozenset(list(rw) + ['force_store_value'])
2023-05-19 12:56:13 +02:00
config.property.setdefault(rw, 'read_write', 'append')
config.property.add('force_store_value')
#
2025-01-02 21:06:13 +01:00
value_owner(test_dir, makedict_after, config, filename)
#
2025-01-02 21:06:13 +01:00
mandatory(test_dir, mandatory_file, config.value.mandatory(), filename)
2022-10-01 22:27:22 +02:00
2025-01-02 21:06:13 +01:00
def value_owner(test_dir, makedict_value_owner, config, filename):
ret = {}
2024-06-19 17:36:18 +02:00
for key, value in option_value(config.value.get(), True):
2023-06-22 12:11:14 +02:00
path = key.path()
if not key.issymlinkoption() and key.isfollower():
2024-06-19 17:36:18 +02:00
if path in ret:
continue
ret[path] = {'owner': [],
'value': [],
}
2023-05-19 12:56:13 +02:00
for idx in range(0, key.value.len()):
try:
option = config.option(path, idx)
2024-06-19 17:36:18 +02:00
ret[path]['value'].append(option.value.get())
ret[path]['owner'].append(option.owner.get())
except PropertiesOptionError as err:
2024-06-19 17:36:18 +02:00
ret[path]['value'].append(str(err))
ret[path]['owner'].append('error')
else:
2023-05-19 12:56:13 +02:00
owner = key.owner.get()
2024-06-19 17:36:18 +02:00
ret[path] = {'owner': owner,
'value': value,
}
2025-01-02 21:06:13 +01:00
if not isfile(Path(test_dir) / 'tiramisu' / 'base.py'):
tret = {f'rougail.{path}': value for path, value in ret.items()}
2022-10-01 22:27:22 +02:00
else:
2025-01-02 21:06:13 +01:00
tret = ret
if not isfile(makedict_value_owner) or debug:
with open(makedict_value_owner, 'w') as fh:
dump(tret, fh, indent=4)
fh.write('\n')
if filename == 'no_namespace':
ret = {f'rougail.{path}': value for path, value in ret.items()}
elif filename != 'base':
2022-10-01 22:27:22 +02:00
ret_prefix = {'1': {}, '2': {}}
for key, value in ret.items():
prefix, path = key.split('.', 1)
ret_prefix[prefix][path] = value
assert loads(dumps(ret_prefix['1'])) == loads(dumps(ret_prefix['2']))
ret = ret_prefix['1']
with open(makedict_value_owner, 'r') as fh:
2022-03-05 11:13:05 +01:00
assert load(fh) == loads(dumps(ret)), f"error in file {makedict_value_owner}"
2019-11-26 20:33:24 +01:00
2025-01-02 21:06:13 +01:00
def mandatory(test_dir, mandatory_file, mandatories, filename):
if filename == 'no_namespace':
ret = [f'rougail.{opt.path()}' for opt in mandatories]
else:
ret = [opt.path() for opt in mandatories]
if not mandatory_file.is_file():
with mandatory_file.open('w') as fh:
dump(ret, fh)
2025-01-02 21:06:13 +01:00
if filename == 'multi':
ret_prefix = {'1': [], '2': []}
for key in ret:
prefix, path = key.split('.', 1)
ret_prefix[prefix].append(path)
assert ret_prefix['1'] == ret_prefix['2']
ret = ret_prefix['1']
with mandatory_file.open() as fh:
2024-10-22 12:40:41 +02:00
assert ret == load(fh), f"error in file {mandatory_file}"
2023-05-19 12:56:13 +02:00
def test_dictionary(test_dir):
if no_test_base:
print('FIXME')
return
2019-11-26 20:33:24 +01:00
test_dir = join(dico_dirs, test_dir)
2025-01-02 21:06:13 +01:00
if not (Path(test_dir) / 'tiramisu' / 'base.py').is_file():
return
2023-05-19 12:56:13 +02:00
launch_flattener(test_dir, 'base')
2022-10-01 22:27:22 +02:00
def test_dictionary_no_namespace(test_dir):
test_dir = join(dico_dirs, test_dir)
2025-01-02 21:06:13 +01:00
if not (Path(test_dir) / 'tiramisu' / 'no_namespace.py').is_file():
return
launch_flattener(test_dir, 'no_namespace')
2023-05-19 12:56:13 +02:00
def test_dictionary_multi(test_dir):
if no_test_base_multi:
print('FIXME')
return
2022-10-01 22:27:22 +02:00
test_dir = join(dico_dirs, test_dir)
2025-01-02 21:06:13 +01:00
if not (Path(test_dir) / 'tiramisu' / 'multi.py').is_file():
return
2023-05-19 12:56:13 +02:00
launch_flattener(test_dir, 'multi')