Source code for labscript_utils.labconfig

#####################################################################
#                                                                   #
# labconfig.py                                                      #
#                                                                   #
# Copyright 2013, Monash University                                 #
#                                                                   #
# This file is part of the labscript suite (see                     #
# http://labscriptsuite.org) and is licensed under the Simplified   #
# BSD License. See the license.txt file in the root of the project  #
# for the full license.                                             #
#                                                                   #
#####################################################################
import os
import configparser
from ast import literal_eval
from pprint import pformat
from pathlib import Path
import warnings

from labscript_utils import dedent
from labscript_profile import default_labconfig_path, LABSCRIPT_SUITE_PROFILE

default_config_path = default_labconfig_path()


[docs] class EnvInterpolation(configparser.BasicInterpolation): """Interpolation which expands environment variables in values, by post-filtering BasicInterpolation.before_get()"""
[docs] def before_get(self, *args): value = super(EnvInterpolation, self).before_get(*args) return os.path.expandvars(value)
[docs] class LabConfig(configparser.ConfigParser): NoOptionError = configparser.NoOptionError NoSectionError = configparser.NoSectionError
[docs] def __init__( self, config_path=default_config_path, required_params=None, defaults=None, ): if required_params is None: required_params = {} if defaults is None: defaults = {} # str() below is for py36 compat, where ConfigParser can't deal with Path objs defaults['labscript_suite'] = str(LABSCRIPT_SUITE_PROFILE) if isinstance(config_path, list): self.config_path = config_path[0] else: self.config_path = config_path self.file_format = "" for section, options in required_params.items(): self.file_format += "[%s]\n" % section for option in options: self.file_format += "%s = <value>\n" % option # Load the config file configparser.ConfigParser.__init__( self, defaults=defaults, interpolation=EnvInterpolation() ) # read all files in the config path if it is a list (self.config_path only # contains one string): self.read(config_path) # Rename experiment_name to apparatus_name and raise a DeprectionWarning experiment_name = self.get("DEFAULT", "experiment_name", fallback=None) if experiment_name: msg = """The experiment_name keyword has been renamed apparatus_name in labscript_utils 3.0, and will be removed in a future version. Please update your labconfig to use the apparatus_name keyword.""" warnings.warn(dedent(msg), FutureWarning) if self.get("DEFAULT", "apparatus_name", fallback=None): msg = """You have defined both experiment_name and apparatus_name in your labconfig. Please omit the deprecate experiment_name keyword.""" raise Exception(dedent(msg)) else: self.set("DEFAULT", "apparatus_name", experiment_name) try: for section, options in required_params.items(): for option in options: self.get(section, option) except configparser.NoOptionError: msg = f"""The experiment configuration file located at {config_path} does not have the required keys. Make sure the config file contains the following structure:\n{self.file_format}""" raise Exception(dedent(msg))
[docs] def save_appconfig(filename, data): """Save a dictionary as an ini file. The keys of the dictionary comprise the section names, and the values must themselves be dictionaries for the names and values within each section. All section values will be converted to strings with pprint.pformat().""" # Error checking for section_name, section in data.items(): for name, value in section.items(): try: valid = value == literal_eval(pformat(value)) except (ValueError, SyntaxError): valid = False if not valid: msg = f"{section_name}/{name} value {value} not a Python built-in type" raise TypeError(msg) data = { section_name: {name: pformat(value) for name, value in section.items()} for section_name, section in data.items() } c = configparser.ConfigParser(interpolation=None) c.optionxform = str # preserve case c.read_dict(data) Path(filename).parent.mkdir(parents=True, exist_ok=True) with open(filename, 'w') as f: c.write(f)
[docs] def load_appconfig(filename): """Load an .ini file and return a dictionary of its contents. All values will be converted to Python objects with ast.literal_eval(). All keys will be lowercase regardless of the written contents on the .ini file.""" c = configparser.ConfigParser(interpolation=None) c.optionxform = str # preserve case # No file? No config - don't crash. if Path(filename).exists(): c.read(filename) return { section_name: {name: literal_eval(value) for name, value in section.items()} for section_name, section in c.items() }