Repository URL to install this package:
|
Version:
0.24.2 ▾
|
qiskit-terra
/
user_config.py
|
|---|
# 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:<default location>
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