Source code for haddock.gear.config

"""
Module to read and write HADDOCK3 configuration files.

HADDOCK3 user configuration files follow TOML syntax plus additional
features required for HADDOCK3. Therefore, we have implemented our own
TOML-like parser to accommodate the extra features needed for HADDOCK3.
However, HADDOCK3 can also read pure TOML files.

The most relevant features of HADDOCK3 user configuration files are the
hability to prepare TOML-like files but with repeated header names, for
example:

.. code:: toml

        [topoaa]
        # parameters here...

        [rigidbody]
        # parameters here...

        [caprieval]
        # parameters here...

        [flexref]
        # parameters here...

        [caprieval]
        # parameters here...

From the user perspective, everything else just behaves as in TOML.
Actually, HADDOCK3 uses TOML to finally read the configuration files.
Users can also provide pure TOML compatible files, for example:

.. code:: toml

        [topoaa]
        # parameters here...

        [rigidbody]
        # parameters here...

        [caprieval]
        # parameters here...

        [flexref]
        # parameters here...

        ['caprieval.1']
        # parameters here...

Because HADDOCK3 uses TOML in the end to read the configuration files,
all TOML rules apply regarding defining ``parameter = value`` pairs.

Read more about TOML for Python here: https://pypi.org/project/toml/
"""
import collections.abc
import os
import re
from pathlib import Path

import toml

from haddock import EmptyPath
from haddock.core.defaults import RUNDIR
from haddock.core.exceptions import ConfigurationError
from haddock.core.typing import FilePath, ParamDict, ParamMap, ParamMapT, Union
from haddock.libs.libutil import (
    recursive_convert_paths_to_strings,
    transform_to_list,
)


# the re.ASCII parameter makes sure non-ascii chars are rejected in the \w key

# Captures the main headers.
# https://regex101.com/r/9urqti/1
_main_header_re = re.compile(r"^ *\[(\w+)\]", re.ASCII)

# regex by @sverhoeven
# Matches ['<name>.<digit>']
_main_quoted_header_re = re.compile(r"^ *\[\'(\w+)\.\d+\'\]", re.ASCII)

# Captures sub-headers
# https://regex101.com/r/6OpJJ8/1
# thanks https://stackoverflow.com/questions/39158902
_sub_header_re = re.compile(r"^ *\[(\w+)((?:\.\w+)+)\]", re.ASCII)

# regex by @sverhoeven
_sub_quoted_header_re = re.compile(
    r"^ *\[\'(\w+)\.\d+\'((?:\.\w+)+)\]",
    re.ASCII,
)

# Captures parameter uppercase boolean
_uppercase_bool_re = re.compile(r"(_?\w+((_?\w+?)+)?\s*=\s*)(True|False)", re.ASCII)

# Some parameters are paths. The configuration reader reads those as
# pathlib.Path so that they are injected properly in the rest of the
# workflow. In general, any parameter ending with `_fname` is a path,
# but there are also other parameters that are paths. Those need to be
# added to this list bellow:
_keys_that_accept_files = [
    "cns_exec",
    "executable",
    RUNDIR,
]


[docs] class ConfigFormatError(Exception): """Exception if there is a format error.""" pass
# main public API
[docs] def load(fpath: FilePath) -> ParamDict: """ Load an HADDOCK3 configuration file to a dictionary. Accepts HADDOCK3 ``cfg`` files or pure ``toml`` files. Parameters ---------- fpath : str or :external:py:class:`pathlib.Path` Path to user configuration file. Returns ------- dictionary Representing the user configuration file where first level of keys are the module names. Step keys will have a numeric suffix, for example: ``module.1``. .. see-also:: * :py:func:`loads` """ try: return loads(Path(fpath).read_text()) except Exception as err: raise ConfigurationError( "Something is wrong with the config file." ) from err # noqa: E501
[docs] def loads(cfg_str: str) -> ParamDict: """ Read a string representing a config file to a dictionary. Config strings are converted to toml-compatible format and finally read by the toml library. All headers (dictionary keys) will be suffixed by an integer starting at ``1``. For example: ``topoaa.1``. If the key is repeated, ``2`` is appended, and so forth. Even if specific integers are provided by the user, the suffix integers will be normalized. Parameters ---------- cfg_str : str The string representing the config file. Accepted formats are the HADDOCK3 config file or pure `toml` syntax. Returns ------- all_configs : dict A dictionary holding all the configuration file steps: - 'raw_input': Original input file as provided by user. - 'cleaned_input': Regex cleaned input file. - 'loaded_cleaned_input': Dict of toml loaded cleaned input. - 'final_cfg': The config in the form of a dictionary. In which the order of the keys matters as it defines the order of the steps in the workflow. """ new_lines: list[str] = [] cfg_lines = cfg_str.split(os.linesep) counter: dict[str, int] = {} # this for-loop normalizes all headers regardless of their input format. for line in cfg_lines: if group := _main_header_re.match(line): name = group[1] counter.setdefault(name, 0) counter[name] += 1 count = counter[name] new_line = f"['{name}.{count}']" elif group := _main_quoted_header_re.match(line): name = group[1] counter.setdefault(name, 0) counter[name] += 1 count = counter[name] new_line = f"['{name}.{count}']" elif group := _sub_header_re.match(line): name = group[1] count = counter[name] # name should be already defined here new_line = f"['{name}.{count}'{group[2]}]" elif group := _sub_quoted_header_re.match(line): name = group[1] count = counter[name] # name should be already defined here new_line = f"['{name}.{count}'{group[2]}]" elif group := _uppercase_bool_re.match(line): param = group[1] # Catches 'param = ' uppercase_bool = group[4] new_line = f"{param}{uppercase_bool.lower()}" # Lowercase bool else: new_line = line new_lines.append(new_line) # Re-build workflow configuration file cfg = os.linesep.join(new_lines) try: cfg_dict = toml.loads(cfg) # Try to load it with the toml library except Exception as err: raise ConfigurationError( "Some thing is wrong with the config file: " f"{str(err)}" ) from err final_cfg = convert_variables_to_paths(cfg_dict) all_configs = { "raw_input": cfg_str, "cleaned_input": cfg, "loaded_cleaned_input": cfg_dict, "final_cfg": final_cfg, } # Return all configuration steps return all_configs
[docs] def convert_variables_to_paths(cfg: ParamMapT) -> ParamMapT: """ Convert config variables to paths. Applies only to those variables HADDOCK3 needs as paths. Parameters ---------- cfg : dictionary A dictionary containing the HADDOCK3 parameters. .. see-also:: :py:func:`match_path_criteria` """ for param, value in cfg.items(): if param == "molecules": # `transform_to_list` is needed to allow strings OR lists in the # `molecule` parameter cfg[param] = [convert_to_path(v) for v in transform_to_list(value)] elif match_path_criteria(param): cfg[param] = convert_to_path(value) elif isinstance(value, collections.abc.Mapping): cfg[param] = convert_variables_to_paths(value) # type: ignore return cfg
[docs] def convert_to_path(value: FilePath) -> Union[Path, EmptyPath]: """Convert a variable to Path if variable evaluates to true.""" if value: return Path(value) else: return EmptyPath()
[docs] def match_path_criteria(param: str) -> bool: """ Confirm a parameter should be converted to path. This function contains the rules defining which parameter need to be converted to Path before be sent to HADDOCK3. """ return param.endswith("_fname") or param in _keys_that_accept_files
[docs] def get_module_name(name: str) -> str: """ Get the module name according to the config parser. Get the name without the integer suffix. """ return name.split(".")[0]
[docs] def save(params: ParamMap, path: FilePath, pure_toml: bool = False) -> None: """ Write a dictionary to a HADDOCK3 config file. Write the HADDOCK3 parameter dictionary to a `.cfg` file. There is also the option to write in pure TOML format. Both are compatible with HADDOCK3. Parameters ---------- params : dict The dictionary containing the parameters. path : str or pathlib.Path File name where to save the configuration file. pure_toml : bool Save config in pure toml format. """ params = recursive_convert_paths_to_strings(params) if pure_toml: with open(path, "w") as fout: toml.dump(params, fout) return cfg_str = toml.dumps(params).split(os.linesep) new_lines: list[str] = [] for line in cfg_str: if group := _main_quoted_header_re.match(line): name = group[1] new_line = f"[{name}]" elif group := _sub_quoted_header_re.match(line): name = group[1] new_line = f"[{name}{group[2]}]" else: new_line = line new_lines.append(new_line) cfg_str = os.linesep.join(new_lines) # type: ignore with open(path, "w") as fout: fout.write(cfg_str) # type: ignore