# This code is part of Qiskit. # # (C) Copyright IBM 2017, 2019. # # This code is licensed under the Apache License, Version 2.0. You may # obtain a copy of this license in the LICENSE.txt file in the root directory # of this source tree or at http://www.apache.org/licenses/LICENSE-2.0. # # Any modifications or derivative works of this code must retain this # copyright notice, and modified files need to carry a notice indicating # that they have been altered from the originals. """Utils for reading a user preference config files.""" import configparser import os from warnings import warn from qiskit import exceptions DEFAULT_FILENAME = os.path.join(os.path.expanduser("~"), ".qiskit", "settings.conf") class UserConfig: """Class representing a user config file The config file format should look like: [default] circuit_drawer = mpl circuit_mpl_style = default circuit_mpl_style_path = ~/.qiskit: circuit_reverse_bits = True transpile_optimization_level = 1 parallel = False num_processes = 4 """ def __init__(self, filename=None): """Create a UserConfig Args: filename (str): The path to the user config file. If one isn't specified, ~/.qiskit/settings.conf is used. """ if filename is None: self.filename = DEFAULT_FILENAME else: self.filename = filename self.settings = {} self.config_parser = configparser.ConfigParser() def read_config_file(self): """Read config file and parse the contents into the settings attr.""" if not os.path.isfile(self.filename): return self.config_parser.read(self.filename) if "default" in self.config_parser.sections(): # Parse circuit_drawer circuit_drawer = self.config_parser.get("default", "circuit_drawer", fallback=None) if circuit_drawer: if circuit_drawer not in ["text", "mpl", "latex", "latex_source", "auto"]: raise exceptions.QiskitUserConfigError( "%s is not a valid circuit drawer backend. Must be " "either 'text', 'mpl', 'latex', 'latex_source', or " "'auto'." % circuit_drawer ) self.settings["circuit_drawer"] = circuit_drawer # Parse state_drawer state_drawer = self.config_parser.get("default", "state_drawer", fallback=None) if state_drawer: valid_state_drawers = [ "repr", "text", "latex", "latex_source", "qsphere", "hinton", "bloch", ] if state_drawer not in valid_state_drawers: valid_choices_string = "', '".join(c for c in valid_state_drawers) raise exceptions.QiskitUserConfigError( f"'{state_drawer}' is not a valid state drawer backend. " f"Choose from: '{valid_choices_string}'" ) self.settings["state_drawer"] = state_drawer # Parse circuit_mpl_style circuit_mpl_style = self.config_parser.get( "default", "circuit_mpl_style", fallback=None ) if circuit_mpl_style: if not isinstance(circuit_mpl_style, str): warn( "%s is not a valid mpl circuit style. Must be " "a text string. Will not load style." % circuit_mpl_style, UserWarning, 2, ) self.settings["circuit_mpl_style"] = circuit_mpl_style # Parse circuit_mpl_style_path circuit_mpl_style_path = self.config_parser.get( "default", "circuit_mpl_style_path", fallback=None ) if circuit_mpl_style_path: cpath_list = circuit_mpl_style_path.split(":") for path in cpath_list: if not os.path.exists(os.path.expanduser(path)): warn( "%s is not a valid circuit mpl style path." " Correct the path in ~/.qiskit/settings.conf." % path, UserWarning, 2, ) self.settings["circuit_mpl_style_path"] = cpath_list # Parse circuit_reverse_bits try: circuit_reverse_bits = self.config_parser.getboolean( "default", "circuit_reverse_bits", fallback=None ) except ValueError as err: raise exceptions.QiskitUserConfigError( f"Value assigned to circuit_reverse_bits is not valid. {str(err)}" ) if circuit_reverse_bits is not None: self.settings["circuit_reverse_bits"] = circuit_reverse_bits # Parse transpile_optimization_level transpile_optimization_level = self.config_parser.getint( "default", "transpile_optimization_level", fallback=-1 ) if transpile_optimization_level != -1: if transpile_optimization_level < 0 or transpile_optimization_level > 3: raise exceptions.QiskitUserConfigError( "%s is not a valid optimization level. Must be 0, 1, 2, or 3." ) self.settings["transpile_optimization_level"] = transpile_optimization_level # Parse parallel parallel_enabled = self.config_parser.getboolean("default", "parallel", fallback=None) if parallel_enabled is not None: self.settings["parallel_enabled"] = parallel_enabled # Parse num_processes num_processes = self.config_parser.getint("default", "num_processes", fallback=-1) if num_processes != -1: if num_processes <= 0: raise exceptions.QiskitUserConfigError( "%s is not a valid number of processes. Must be greater than 0" ) self.settings["num_processes"] = num_processes def set_config(key, value, section=None, file_path=None): """Adds or modifies a user configuration It will add configuration to the currently configured location or the value of file argument. Only valid user config can be set in 'default' section. Custom user config can be added in any other sections. Changes to the existing config file will not be reflected in the current session since the config file is parsed at import time. Args: key (str): name of the config value (obj): value of the config section (str, optional): if not specified, adds it to the `default` section of the config file. file_path (str, optional): the file to which config is added. If not specified, adds it to the default config file or if set, the value of `QISKIT_SETTINGS` env variable. Raises: QiskitUserConfigError: if the config is invalid """ filename = file_path or os.getenv("QISKIT_SETTINGS", DEFAULT_FILENAME) section = "default" if section is None else section if not isinstance(key, str): raise exceptions.QiskitUserConfigError("Key must be string type") valid_config = { "circuit_drawer", "circuit_mpl_style", "circuit_mpl_style_path", "circuit_reverse_bits", "transpile_optimization_level", "parallel", "num_processes", } if section in [None, "default"]: if key not in valid_config: raise exceptions.QiskitUserConfigError(f"{key} is not a valid user config.") config = configparser.ConfigParser() config.read(filename) if section not in config.sections(): config.add_section(section) config.set(section, key, str(value)) try: with open(filename, "w") as cfgfile: config.write(cfgfile) except OSError as ex: raise exceptions.QiskitUserConfigError( f"Unable to load the config file {filename}. Error: '{str(ex)}'" ) # validates config user_config = UserConfig(filename) user_config.read_config_file() def get_config(): """Read the config file from the default location or env var It will read a config file at either the default location ~/.qiskit/settings.conf or if set the value of the QISKIT_SETTINGS env var. It will return the parsed settings dict from the parsed config file. Returns: dict: The settings dict from the parsed config file. """ filename = os.getenv("QISKIT_SETTINGS", DEFAULT_FILENAME) if not os.path.isfile(filename): return {} user_config = UserConfig(filename) user_config.read_config_file() return user_config.settings