"""Validation of configuration."""
import logging
import os
from pathlib import Path
from schema import And, Optional, Or, Schema, SchemaError
from topostats.logs.logs import LOGGER_NAME
LOGGER = logging.getLogger(LOGGER_NAME)
# pylint: disable=line-too-long
[docs]
def validate_config(config: dict, schema: Schema, config_type: str) -> None:
"""
Validate configuration.
Parameters
----------
config : dict
Config dictionary imported by read_yaml() and parsed through clean_config().
schema : Schema
A schema against which the configuration is to be compared.
config_type : str
Description of of configuration being validated.
"""
try:
schema.validate(config)
LOGGER.info(f"The {config_type} is valid.")
except SchemaError as schema_error:
raise SchemaError(
f"There is an error in your {config_type} configuration. "
"Please refer to the first error message above for details"
) from schema_error
DEFAULT_CONFIG_SCHEMA = Schema(
{
"base_dir": Path,
"output_dir": Path,
"log_level": Or(
"debug",
"info",
"warning",
"error",
error="Invalid value in config for 'log_level', valid values are 'info' (default), 'debug', 'error' or 'warning",
),
"cores": lambda n: 1 <= n <= os.cpu_count(),
"file_ext": Or(
".spm",
".asd",
".jpk",
".ibw",
".gwy",
".topostats",
error="Invalid value in config for 'file_ext', valid values are '.spm', '.jpk', '.ibw', '.gwy', '.topostats', or '.asd'.",
),
"loading": {"channel": str},
"filter": {
"run": Or(
True,
False,
error="Invalid value in config for 'filter.run', valid values are 'True' or 'False'",
),
"row_alignment_quantile": lambda n: 0.0 <= n <= 1.0,
"threshold_method": Or(
"absolute",
"otsu",
"std_dev",
error=(
"Invalid value in config for 'filter.threshold_method', valid values "
"are 'absolute', 'otsu' or 'std_dev'"
),
),
"otsu_threshold_multiplier": float,
"threshold_std_dev": {
"below": lambda n: n > 0,
"above": lambda n: n > 0,
},
"threshold_absolute": {
"below": Or(
int,
float,
error=(
"Invalid value in config for filter.threshold.absolute.below " "should be type int or float"
),
),
"above": Or(
int,
float,
error=(
"Invalid value in config for filter.threshold.absolute.below " "should be type int or float"
),
),
},
"gaussian_size": float,
"gaussian_mode": Or(
"nearest",
error="Invalid value in config for 'filter.gaussian_mode', valid values are 'nearest'",
),
"remove_scars": {
"run": bool,
"removal_iterations": lambda n: 0 <= n < 10,
"threshold_low": lambda n: n > 0,
"threshold_high": lambda n: n > 0,
"max_scar_width": lambda n: n >= 1,
"min_scar_length": lambda n: n >= 1,
},
},
"grains": {
"run": Or(
True,
False,
error="Invalid value in config for grains.run, valid values are 'True' or 'False'",
),
"smallest_grain_size_nm2": lambda n: n > 0.0,
"threshold_method": Or(
"absolute",
"otsu",
"std_dev",
error=(
"Invalid value in config for 'grains.threshold_method', valid values "
"are 'absolute', 'otsu' or 'std_dev'"
),
),
"otsu_threshold_multiplier": float,
"threshold_std_dev": {
"below": lambda n: n > 0,
"above": lambda n: n > 0,
},
"threshold_absolute": {
"below": Or(
int,
float,
error=(
"Invalid value in config for grains.threshold.absolute.below " "should be type int or float"
),
),
"above": Or(
int,
float,
error=(
"Invalid value in config for grains.threshold.absolute.below " "should be type int or float"
),
),
},
"absolute_area_threshold": {
"above": [
Or(
int,
None,
error=(
"Invalid value in config for 'grains.absolute_area_threshold.above', valid values "
"are int or null"
),
)
],
"below": [
Or(
int,
None,
error=(
"Invalid value in config for 'grains.absolute_area_threshold.below', valid values "
"are int or null"
),
)
],
},
"direction": Or(
"both",
"below",
"above",
error="Invalid direction for grains.direction valid values are 'both', 'below' or 'above",
),
"remove_edge_intersecting_grains": Or(
True,
False,
error="Invalid value in config for 'grains.remove_edge_intersecting_grains', valid values are 'True' or 'False'",
),
"unet_config": {
"model_path": Or(None, str),
"grain_crop_padding": int,
"upper_norm_bound": float,
"lower_norm_bound": float,
},
},
"grainstats": {
"run": Or(
True,
False,
error="Invalid value in config for 'grainstats.run', valid values are 'True' or 'False'",
),
"edge_detection_method": Or(
"binary_erosion",
"canny",
),
"cropped_size": Or(
float,
int,
),
"extract_height_profile": Or(
True,
False,
error="Invalid value in config for 'grainstats.extract_height_profile',"
"valid values are 'True' or 'False'",
),
},
"dnatracing": {
"run": Or(
True,
False,
error="Invalid value in config for 'dnatracing.run', valid values are 'True' or 'False'",
),
"min_skeleton_size": lambda n: n > 0.0,
"skeletonisation_method": Or(
"zhang",
"lee",
"thin",
"topostats",
error="Invalid value in config for 'dnatracing.skeletonisation_method',"
"valid values are 'zhang' or 'lee', 'thin' or 'topostats'",
),
"spline_step_size": lambda n: n > 0.0,
"spline_linear_smoothing": lambda n: n >= 0.0,
"spline_circular_smoothing": lambda n: n >= 0.0,
"pad_width": lambda n: n > 0.0,
# "cores": lambda n: n > 0.0,
},
"plotting": {
"run": Or(
True,
False,
error="Invalid value in config for 'plotting.run', valid values are 'True' or 'False'",
),
"style": And(
str,
Or(
"topostats.mplstyle",
str,
Path,
None,
error="Invalid value in config for 'plotting.style', valid values are 'topostats.mplstyle' or None",
),
),
"savefig_format": Or(
None,
str,
error="Invalid value in config for plotting.savefig_format" "must be a value supported by Matplotlib.",
),
"savefig_dpi": Or(
None,
"figure",
lambda n: n > 0,
error="Invalid value in config for plotting.savefig_dpi, valid" "values are 'figure' or floats",
),
"image_set": Or(
"all",
"core",
error="Invalid value in config for 'plotting.image_set', valid values " "are 'all' or 'core'",
),
"pixel_interpolation": Or(
None,
"none",
"bessel",
"bicubic",
"bilinear",
"catrom",
"gaussian",
"hamming",
"hanning",
"hermite",
"kaiser",
"lanczos",
"mitchell",
"nearest",
"quadric",
"sinc",
"spline16",
"spline36",
error="Invalid interpolation value. See https://matplotlib.org/stable/gallery/images_contours_and_fields/interpolation_methods.html for options.",
),
"zrange": [float, int, None],
"colorbar": Or(
True,
False,
error="Invalid value in config for 'plotting.colorbar', valid values are 'True' or 'False'",
),
"axes": Or(
True,
False,
error="Invalid value in config plotting.for 'axes', valid values are 'True' or 'False'",
),
"num_ticks": Or(
[None, And(int, lambda n: n > 1)],
error="Invalid value in config plotting.for 'num_ticks', valid values are 'null' or integers > 1",
),
"cmap": Or(
None,
str,
error="Invalid value in config for 'plotting.cmap', valid values are 'afmhot', 'nanoscope', "
"'gwyddion' or values supported by Matplotlib",
),
"mask_cmap": str,
"histogram_log_axis": Or(
True,
False,
error=(
"Invalid value in config plotting histogram. For 'log_y_axis', valid values are 'True' or "
"'False'"
),
),
},
"summary_stats": {
"run": Or(
True,
False,
error="Invalid value in config for summary_stats.run, valid values are 'True' or 'False'",
),
"config": Or(
None,
str,
error=(
"Invalid value in config for summary_stats.config, valid values are 'None' or a path to a "
"config file."
),
),
},
}
)
PLOTTING_SCHEMA = Schema(
{
"extracted_channel": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'extracted_channel.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"pixels": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error="Invalid value in config 'pixels.image_type', valid values are 'binary' or 'non-binary'",
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"initial_median_flatten": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'initial_median_flatten.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"initial_tilt_removal": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'initial_tilt_removal.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"initial_quadratic_removal": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'initial_quadratic_removal.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"initial_scar_removal": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'initial_scar_removal.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"initial_zero_average_background": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'initial_zero_average_background.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"initial_nonlinear_polynomial_removal": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'initial_nonlinear_polynomial_removal.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"mask": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error="Invalid value in config 'mask.image_type', valid values are 'binary' or 'non-binary'",
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"masked_median_flatten": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'masked_median_flatten.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"masked_tilt_removal": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'masked_tilt_removal.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"masked_quadratic_removal": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'masked_quadratic_removal.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"masked_nonlinear_polynomial_removal": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'masked_nonlinear_polynomial_removal.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"secondary_scar_removal": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'masked_quadratic_removal.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"scar_mask": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'masked_quadratic_removal.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"final_zero_average_background": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'final_zero_average_background.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"gaussian_filtered": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'gaussian_filtered.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"z_threshed": {
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'z_threshold.image_type', valid values " "are 'binary' or 'non-binary'"
),
),
"core_set": True,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"mask_grains": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'mask_grains.image_type', valid values " "are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"labelled_regions_01": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'labelled_regions_01.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"tidied_border": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'tidied_border.image_type', valid values " "are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"removed_noise": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'removed_noise.image_type', valid values " "are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"removed_small_objects": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'removed_small_objects.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"mask_overlay": {
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'mask_overlay.image_type', valid values " "are 'binary' or 'non-binary'"
),
),
"core_set": True,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"labelled_regions_02": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'labelled_regions_02.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"coloured_regions": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'coloured_regions.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"bounding_boxes": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'bounding_boxes.image_type', valid values " "are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"coloured_boxes": {
"filename": str,
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'coloured_boxes.image_type', valid values " "are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"all_molecule_traces": {
"title": str,
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'all_molecule_traces.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"grain_image": {
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'grain_image.image_type', valid values " "are 'binary' or 'non-binary'"
),
),
"core_set": False,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"grain_mask": {
"image_type": Or(
"binary",
"non-binary",
error=("Invalid value in config 'grain_mask.image_type', valid values " "are 'binary' or 'non-binary'"),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"grain_mask_image": {
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'grain_mask_image.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
"single_molecule_trace": {
"image_type": Or(
"binary",
"non-binary",
error=(
"Invalid value in config 'single_molecule_trace.image_type', valid values "
"are 'binary' or 'non-binary'"
),
),
"core_set": bool,
"savefig_dpi": Or(
lambda n: n > 0,
"figure",
error="Invalid value in config for 'dpi', valid values are 'figure' or > 0.",
),
},
}
)
SUMMARY_SCHEMA = Schema(
{
"base_dir": Path,
"output_dir": Path,
"csv_file": str,
"savefig_format": Or(
"png",
"pdf",
"svg",
"tiff",
"tif",
error=("Invalid value in config 'savefig_format', valid values are 'png', 'pdf', 'svg', 'tiff' or 'tif'"),
),
"var_to_label": Or(
None,
str,
error="Invalid value in config for 'var_to_label', valid values are 'None' or a str",
),
"image_id": str,
"molecule_id": str,
"hist": Or(
True,
False,
error="Invalid value in config for 'hist', valid values are 'True' or 'False'",
),
"bins": lambda n: n > 0,
"stat": Or(
"count",
"frequency",
"probability",
"percent",
"density",
error=(
"Invalid value in config 'stat', valid values are 'count', 'frequency', "
"'probability', 'percent' or 'density'"
),
),
"kde": Or(
True,
False,
error="Invalid value in config for 'kde', valid values are 'True' or 'False'",
),
"violin": Or(
True,
False,
error="Invalid value in config for 'violin', valid values are 'True' or 'False'",
),
"figsize": [lambda n: n > 0],
"alpha": lambda n: n > 0,
"palette": Or(
"colorblind",
"deep",
"muted",
"pastel",
"bright",
"dark",
"Spectral",
"Set2",
error=(
"Invalid value in config 'palette', valid values are 'colorblind', 'deep', "
"'muted', 'pastel', 'bright', 'dark', 'Spectral' or 'Set2'"
),
),
"stats_to_sum": [
Optional("area"),
Optional("area_cartesian_bbox"),
Optional("aspect_ratio"),
Optional("bending_angle"),
Optional("contour_length"),
Optional("end_to_end_distance"),
Optional("height_max"),
Optional("height_mean"),
Optional("height_median"),
Optional("height_min"),
Optional("max_feret"),
Optional("min_feret"),
Optional("radius_max"),
Optional("radius_mean"),
Optional("radius_median"),
Optional("radius_min"),
Optional("smallest_bounding_area"),
Optional("smallest_bounding_length"),
Optional("smallest_bounding_width"),
Optional("volume"),
],
}
)