rougail/src/rougail/config.py

488 lines
14 KiB
Python
Raw Normal View History

2019-11-23 08:17:35 +01:00
"""
2021-01-30 08:15:26 +01:00
Config file for Rougail
2019-11-23 08:17:35 +01:00
2021-01-30 08:15:26 +01:00
Created by:
EOLE (http://eole.orion.education.fr)
Copyright (C) 2005-2018
Forked by:
Cadoles (http://www.cadoles.com)
Copyright (C) 2019-2021
2022-11-02 23:00:42 +01:00
Silique (https://www.silique.fr)
Copyright (C) 2022-2025
2022-11-02 23:00:42 +01:00
2024-11-01 09:04:27 +01:00
This program is free software: you can redistribute it and/or modify it
under the terms of the GNU Lesser General Public License as published by the
Free Software Foundation, either version 3 of the License, or (at your
option) any later version.
This program is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
details.
You should have received a copy of the GNU Lesser General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
2019-11-23 08:17:35 +01:00
"""
2024-10-29 11:01:30 +01:00
from sys import version_info
from pathlib import Path
from tiramisu import Config
from ruamel.yaml import YAML
from .utils import _, load_modules, normalize_family
from .convert import RougailConvert
from .object_model import get_convert_option_types
if version_info.major == 3 and version_info.minor:
import rougail.structural_commandline.object_model
2024-10-29 11:01:30 +01:00
RENAMED = {
"dictionaries_dir": "main_dictionaries",
"variable_namespace": "main_namespace",
"functions_file": "functions_files",
}
NOT_IN_TIRAMISU = {
"custom_types": {},
}
SUBMODULES = None
def get_sub_modules():
global SUBMODULES
if SUBMODULES is None:
SUBMODULES = {}
for submodule in Path(__file__).parent.iterdir():
2024-10-29 11:01:30 +01:00
if submodule.name.startswith("_") or not submodule.is_dir():
continue
2024-10-29 11:01:30 +01:00
config_file = submodule / "config.py"
if config_file.is_file():
2024-10-29 11:01:30 +01:00
SUBMODULES[submodule.name] = load_modules(
"rougail." + submodule.name + ".config", str(config_file)
)
return SUBMODULES
2024-10-29 11:01:30 +01:00
def get_level(module):
2025-01-02 21:06:13 +01:00
return float(module["level"]) + {
"structural": 0.1,
"user data": 0.2,
"output": 0.3,
}.get(module["process"])
class _RougailConfig:
def __init__(self, backward_compatibility: bool,
add_extra_options: bool):
self.backward_compatibility = backward_compatibility
self.add_extra_options = add_extra_options
self.root = None
def copy(self):
rougailconfig = _RougailConfig(
self.backward_compatibility,
self.add_extra_options,
)
if self.root:
rougailconfig.config.value.importation(self.config.value.exportation())
rougailconfig.config.property.importation(self.config.property.exportation())
rougailconfig.config.property.read_only()
rougailconfig.root = self.root
rougailconfig.config = self.config
rougailconfig.extra_vars = self.extra_vars
for variable in self.not_in_tiramisu:
value = getattr(self, variable)
if not isinstance(value, str):
value = value.copy()
setattr(rougailconfig, variable, value)
return rougailconfig
def generate_config(self):
root, extra_vars = _rougail_config(self.backward_compatibility, self.add_extra_options)
self.root = root
self.config = Config(
2024-10-29 11:01:30 +01:00
self.root,
)
self.extra_vars = extra_vars
self.not_in_tiramisu = NOT_IN_TIRAMISU | extra_vars
for variable, default_value in self.not_in_tiramisu.items():
if not isinstance(default_value, str):
default_value = default_value.copy()
setattr(self, variable, default_value)
self.config.property.read_only()
2024-10-29 11:01:30 +01:00
def __setitem__(
self,
key,
value,
) -> None:
if self.root is None:
self.generate_config()
if key in self.not_in_tiramisu:
setattr(self, key, value)
else:
self.config.property.read_write()
2024-10-29 11:01:30 +01:00
if key == "export_with_import":
key = "not_export_with_import"
key = RENAMED.get(key, key)
option = self.config.option(key)
if option.isoptiondescription() and option.isleadership():
leader = list(value)
option.leader().value.reset()
option.leader().value.set(leader)
follower = option.followers()[0]
for idx, val in enumerate(value.values()):
self.config.option(follower.path(), idx).value.set(val)
2024-10-29 11:01:30 +01:00
elif key == "not_export_with_import":
option.value.set(not value)
else:
option.value.set(value)
self.config.property.read_only()
2024-10-29 11:01:30 +01:00
def __getitem__(
self,
key,
) -> None:
if self.root is None:
self.generate_config()
if key in self.not_in_tiramisu:
return getattr(self, key)
2024-10-29 11:01:30 +01:00
if key == "export_with_import":
key = "not_export_with_import"
option = self.config.option(key)
if option.isoptiondescription() and option.isleadership():
return self.get_leadership(option)
ret = self.config.option(key).value.get()
2024-10-29 11:01:30 +01:00
if key == "not_export_with_import":
return not ret
return ret
2024-10-29 11:01:30 +01:00
def get_leadership(self, option) -> dict:
2024-08-25 12:23:06 +02:00
leader = None
followers = []
for opt, value in option.value.get().items():
if opt.issymlinkoption():
continue
if leader is None:
leader = value
else:
followers.append(value)
return dict(zip(leader, followers))
2024-08-25 12:23:06 +02:00
def parse(self, config) -> str:
for option in config:
if option.isoptiondescription():
yield from self.parse(option)
elif not option.issymlinkoption():
2024-10-29 11:01:30 +01:00
yield f"{option.path()}: {option.value.get()}"
2024-08-25 12:23:06 +02:00
def __repr__(self):
self.config.property.read_write()
try:
values = "\n".join(self.parse(self.config))
except Exception as err:
values = str(err)
self.config.property.read_only()
return values
class FakeRougailConvert(RougailConvert):
2024-10-29 11:01:30 +01:00
def __init__(
self,
add_extra_options: bool,
) -> None:
2024-08-08 09:04:49 +02:00
self.add_extra_options = add_extra_options
super().__init__({})
def load_config(self) -> None:
self.sort_dictionaries_all = False
self.main_namespace = None
2024-10-29 11:01:30 +01:00
self.suffix = ""
self.custom_types = {}
self.functions_files = []
self.modes_level = []
self.extra_annotators = []
self.base_option_name = "baseoption"
self.export_with_import = True
self.internal_functions = []
self.force_optional = False
2025-01-02 21:06:13 +01:00
self.structurals = ["commandline"]
self.user_datas = []
2024-12-11 20:54:03 +01:00
self.output = None
2024-08-08 09:04:49 +02:00
self.add_extra_options = self.add_extra_options
2025-01-02 21:06:13 +01:00
self.tiramisu_cache = False
self.load_unexist_redefine = False
def _rougail_config(
2024-10-29 11:01:30 +01:00
backward_compatibility: bool = True,
add_extra_options: bool = True,
) -> "OptionDescription":
rougail_options = f"""default_dictionary_format_version:
description: Dictionary format version by default, if not specified in dictionary file
alternative_name: v
choices:
- '1.0'
- '1.1'
mandatory: false
functions_files:
2025-01-02 21:06:13 +01:00
description: {_("File with functions")}
alternative_name: c
type: unix_filename
params:
allow_relative: true
test_existence: true
types:
- file
multi: true
mandatory: false
modes_level:
2025-01-02 21:06:13 +01:00
description: {_("All modes level available")}
multi: true
mandatory: false
"""
if backward_compatibility:
2025-01-02 21:06:13 +01:00
rougail_options += """ default:
- basic
- standard
- advanced
"""
2025-01-02 21:06:13 +01:00
rougail_options += f"""
default_family_mode:
2025-01-02 21:06:13 +01:00
description: {_("Default mode for a family")}
default:
jinja: |
2025-01-02 21:06:13 +01:00
{{% if modes_level %}}
{{{{ modes_level[0] }}}}
{{% endif %}}
disabled:
jinja: |
2025-01-02 21:06:13 +01:00
{{% if not modes_level %}}
No mode
2025-01-02 21:06:13 +01:00
{{% endif %}}
validators:
- type: jinja
jinja: |
2025-01-02 21:06:13 +01:00
{{% if default_family_mode not in modes_level %}}
not in modes_level ({{modes_level}})
{{% endif %}}
commandline: false
default_variable_mode:
2025-01-02 21:06:13 +01:00
description: {_("Default mode for a variable")}
default:
jinja: |
2025-01-02 21:06:13 +01:00
{{% if modes_level %}}
{{% if modes_level | length == 1 %}}
{{{{ modes_level[0] }}}}
{{% else %}}
{{{{ modes_level[1] }}}}
{{% endif %}}
{{% endif %}}
disabled:
jinja: |
2025-01-02 21:06:13 +01:00
{{% if not modes_level %}}
No mode
2025-01-02 21:06:13 +01:00
{{% endif %}}
validators:
- type: jinja
jinja: |
2025-01-02 21:06:13 +01:00
{{% if default_variable_mode not in modes_level %}}
not in modes_level ({{modes_level}})
{{% endif %}}
commandline: false
base_option_name:
2025-01-02 21:06:13 +01:00
description: {_("Option name for the base option")}
default: baseoption
commandline: false
not_export_with_import:
2025-01-02 21:06:13 +01:00
description: {_("In cache file, do not importation of Tiramisu and other dependencies")}
negative_description: {_("In cache file, do importation of Tiramisu and other dependencies")}
default: false
commandline: false
tiramisu_cache:
2025-01-02 21:06:13 +01:00
description: {_("Tiramisu cache filename")}
alternative_name: t
type: unix_filename
mandatory: false
params:
allow_relative: true
internal_functions:
2025-01-02 21:06:13 +01:00
description: {_("Name of internal functions that we can use as a function")}
multi: true
mandatory: false
commandline: false
extra_annotators:
2025-01-02 21:06:13 +01:00
description: {_("Name of extra annotators")}
multi: true
mandatory: false
commandline: false
suffix:
2025-01-02 21:06:13 +01:00
description: {_("Suffix add to generated option name")}
default: ''
mandatory: false
commandline: false
force_optional:
2025-01-02 21:06:13 +01:00
description: {_("Every variable in calculation are optional")}
negative_description: {_("Variable in calculation are not optional by default")}
default: False
load_unexist_redefine:
description: {_("Load redefine variable even if there don't already exists")}
negative_description: {_("Load redefine variable even if there don't already exists")}
commandline: false
default: False
2025-01-02 21:06:13 +01:00
"""
2024-10-29 11:01:30 +01:00
processes = {
"structural": [],
"user data": [],
2025-01-02 21:06:13 +01:00
"output": [],
2024-10-29 11:01:30 +01:00
}
for module in get_sub_modules().values():
2025-01-02 21:06:13 +01:00
data = module.get_rougail_config(backward_compatibility=backward_compatibility)
2024-10-29 11:01:30 +01:00
processes[data["process"]].append(data)
# reorder
for process in processes:
processes[process] = list(sorted(processes[process], key=get_level))
2025-01-02 21:06:13 +01:00
rougail_process = "step: # Load and exporter steps"
for process in processes:
if processes[process]:
objects = processes[process]
rougail_process += """
2025-01-02 21:06:13 +01:00
{NAME}:
description: Select for {NAME}
alternative_name: {NAME[0]}
choices:
2024-10-29 11:01:30 +01:00
""".format(
NAME=normalize_family(process),
)
for obj in objects:
rougail_process += f" - {obj['name']}\n"
2024-10-29 11:01:30 +01:00
if process == "structural":
2025-01-02 21:06:13 +01:00
rougail_process += """ commandline: false
multi: true
default:
- directory
"""
2024-10-29 11:01:30 +01:00
elif process == "user data":
2024-08-08 09:04:49 +02:00
rougail_process += """ multi: true
2025-01-02 21:06:13 +01:00
mandatory: false"""
2024-10-29 11:01:30 +01:00
hidden_outputs = [
process["name"]
for process in processes["output"]
if not process.get("allow_user_data", True)
]
if hidden_outputs:
2025-01-02 21:06:13 +01:00
rougail_process += """
hidden:
type: jinja
jinja: |
"""
for hidden_output in hidden_outputs:
rougail_process += """ {% if _.output == 'NAME' %}
Cannot load user data for NAME output
2025-01-02 21:06:13 +01:00
{% endif %}""".replace(
2024-11-01 09:45:54 +01:00
"NAME", hidden_output
)
2024-10-29 12:02:27 +01:00
elif objects:
2024-10-29 11:01:30 +01:00
rougail_process += " default: {DEFAULT}".format(
DEFAULT=objects[0]["name"]
)
else:
2025-01-02 21:06:13 +01:00
if process == "output":
prop = "hidden"
else:
2025-01-02 21:06:13 +01:00
prop = "disabled"
rougail_process += """
{NAME}:
description: Select for {NAME}
2024-08-05 18:09:13 +02:00
mandatory: false
{PROP}: true
2024-08-05 18:09:13 +02:00
multi: true
default: ["You haven't installed \\\"{NAME}\\\" package for rougail"]
validators:
- jinja: Please install a rougail-{NAME}-* package.
2024-10-29 11:01:30 +01:00
""".format(
NAME=normalize_family(process),
PROP=prop,
2024-10-29 11:01:30 +01:00
)
rougail_process += f"""
default_params:
description: {_("Default parameters for option type")}
"""
for typ, params in get_convert_option_types():
rougail_process += f"""
{typ}:
"""
for key, key_type, multi, value in params:
rougail_process += f"""
{key}:
type: {key_type}
multi: {multi}
mandatory: false
default: {value}
"""
rougail_options += rougail_process
2025-01-02 21:06:13 +01:00
# print(rougail_options)
2024-08-08 09:04:49 +02:00
convert = FakeRougailConvert(add_extra_options)
2025-01-02 21:06:13 +01:00
convert.init()
convert.namespace = None
convert.parse_root_file(
2024-10-29 11:01:30 +01:00
"rougail.config",
"",
"1.1",
YAML().load(rougail_options),
)
extra_vars = {}
2025-01-02 21:06:13 +01:00
objects = []
for obj in sorted(
[obj for objects in processes.values() for obj in objects], key=get_level
):
if "extra_vars" in obj:
extra_vars |= obj["extra_vars"]
if not "options" in obj:
continue
if not isinstance(obj["options"], list):
options = [obj["options"]]
else:
options = obj["options"]
for option in options:
convert.parse_root_file(
f'rougail.config.{obj["name"]}',
"",
"1.1",
YAML().load(option),
)
2024-10-29 11:01:30 +01:00
2025-01-02 21:06:13 +01:00
tiram_obj = convert.save()
optiondescription = {}
exec(tiram_obj, {}, optiondescription) # pylint: disable=W0122
return optiondescription["option_0"], extra_vars
def get_rougail_config(
*,
backward_compatibility: bool = True,
add_extra_options: bool = True,
) -> _RougailConfig:
2024-10-29 11:01:30 +01:00
return _RougailConfig(
backward_compatibility,
add_extra_options,
2024-10-29 11:01:30 +01:00
)
RougailConfig = get_rougail_config()