From b2a5e0ffb441a66cf67ce72fbd926a40ef0f6703 Mon Sep 17 00:00:00 2001 From: "Haoyu (Daniel) YANG" Date: Wed, 11 Dec 2024 09:51:58 +0800 Subject: [PATCH 01/12] `zopen`: explicit binary/text `mode` , and explicit `encoding` as UTF-8 (#4219) * explicit mode for zopen * fix bad mode for cifwrite * remove tag * remove an unnecessary cast to int * TODO: test monty zopen change * fix implicit mode * Revert "TODO: test monty zopen change" This reverts commit c9bf970928e7e1d235a835e3cb4ad45a149ce010. * Revert "Revert "TODO: test monty zopen change"" This reverts commit be411a0f9d13bb3ebf77ecd5997ce5b21af81910. * Revert "Revert "Revert "TODO: test monty zopen change""" This reverts commit e75c196d47c614022e534741178eabd1316f9efe. * explicit text mode for stout * some explicit utf-8 for zopen * TO BE REVERTED: test monty pr * add the rest encoding * Revert "TO BE REVERTED: test monty pr" This reverts commit d899c87e60e58b5719f1123a60f992a9428d0e51. --- dev_scripts/potcar_scrambler.py | 2 +- src/pymatgen/apps/borg/hive.py | 2 +- src/pymatgen/apps/borg/queen.py | 4 +- src/pymatgen/core/structure.py | 18 ++++---- src/pymatgen/core/trajectory.py | 2 +- src/pymatgen/io/adf.py | 2 +- src/pymatgen/io/aims/inputs.py | 4 +- src/pymatgen/io/cif.py | 8 ++-- src/pymatgen/io/common.py | 6 +-- src/pymatgen/io/core.py | 6 +-- src/pymatgen/io/cp2k/inputs.py | 2 +- src/pymatgen/io/cp2k/outputs.py | 16 +++---- src/pymatgen/io/cp2k/utils.py | 2 +- src/pymatgen/io/cssr.py | 4 +- src/pymatgen/io/exciting/inputs.py | 2 +- src/pymatgen/io/feff/inputs.py | 20 ++++---- src/pymatgen/io/feff/outputs.py | 10 ++-- src/pymatgen/io/fiesta.py | 16 +++---- src/pymatgen/io/gaussian.py | 12 ++--- src/pymatgen/io/lammps/data.py | 4 +- src/pymatgen/io/lammps/generators.py | 2 +- src/pymatgen/io/lammps/inputs.py | 4 +- src/pymatgen/io/lammps/outputs.py | 4 +- src/pymatgen/io/lmto.py | 6 +-- src/pymatgen/io/lobster/inputs.py | 4 +- src/pymatgen/io/lobster/outputs.py | 30 ++++++------ src/pymatgen/io/nwchem.py | 6 +-- src/pymatgen/io/pwmat/inputs.py | 10 ++-- src/pymatgen/io/pwmat/outputs.py | 6 +-- src/pymatgen/io/pwscf.py | 2 +- src/pymatgen/io/qchem/inputs.py | 6 +-- src/pymatgen/io/qchem/outputs.py | 2 +- src/pymatgen/io/qchem/sets.py | 2 +- src/pymatgen/io/res.py | 4 +- src/pymatgen/io/template.py | 2 +- src/pymatgen/io/vasp/inputs.py | 28 +++++------ src/pymatgen/io/vasp/outputs.py | 46 +++++++++---------- src/pymatgen/io/xr.py | 4 +- src/pymatgen/io/xyz.py | 4 +- src/pymatgen/io/zeopp.py | 4 +- .../transformations/site_transformations.py | 4 +- src/pymatgen/util/io_utils.py | 2 +- tests/electronic_structure/test_dos.py | 2 +- tests/io/aims/conftest.py | 2 +- tests/io/pwmat/test_inputs.py | 6 +-- tests/io/vasp/test_inputs.py | 2 +- tests/io/vasp/test_outputs.py | 2 +- 47 files changed, 168 insertions(+), 170 deletions(-) diff --git a/dev_scripts/potcar_scrambler.py b/dev_scripts/potcar_scrambler.py index f7115f1996a..7e1aa2579f9 100644 --- a/dev_scripts/potcar_scrambler.py +++ b/dev_scripts/potcar_scrambler.py @@ -124,7 +124,7 @@ def scramble_single_potcar(self, potcar: PotcarSingle) -> str: return scrambled_potcar_str def to_file(self, filename: str) -> None: - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(self.scrambled_potcars_str) @classmethod diff --git a/src/pymatgen/apps/borg/hive.py b/src/pymatgen/apps/borg/hive.py index 932aa1ec7ce..2ea8a1be63b 100644 --- a/src/pymatgen/apps/borg/hive.py +++ b/src/pymatgen/apps/borg/hive.py @@ -443,7 +443,7 @@ def _get_transformation_history(path: PathLike): """Check for a transformations.json* file and return the history.""" if trans_json := glob(f"{path!s}/transformations.json*"): try: - with zopen(trans_json[0]) as file: + with zopen(trans_json[0], mode="rt", encoding="utf-8") as file: return json.load(file)["history"] except Exception: return None diff --git a/src/pymatgen/apps/borg/queen.py b/src/pymatgen/apps/borg/queen.py index 2dd4d74e03a..bfa47b39432 100644 --- a/src/pymatgen/apps/borg/queen.py +++ b/src/pymatgen/apps/borg/queen.py @@ -103,12 +103,12 @@ def save_data(self, filename: PathLike) -> None: that if the filename ends with gz or bz2, the relevant gzip or bz2 compression will be applied. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: json.dump(list(self._data), file, cls=MontyEncoder) def load_data(self, filename: PathLike) -> None: """Load assimilated data from a file.""" - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: self._data = json.load(file, cls=MontyDecoder) diff --git a/src/pymatgen/core/structure.py b/src/pymatgen/core/structure.py index 8e46793c837..8f9fc7f7321 100644 --- a/src/pymatgen/core/structure.py +++ b/src/pymatgen/core/structure.py @@ -2952,7 +2952,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: elif fmt == "json" or fnmatch(filename.lower(), "*.json*"): json_str = json.dumps(self.as_dict()) if filename: - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(json_str) return json_str elif fmt == "xsf" or fnmatch(filename.lower(), "*.xsf*"): @@ -2960,7 +2960,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: res_str = XSF(self).to_str() if filename: - with zopen(filename, mode="wt", encoding="utf8") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(res_str) return res_str elif ( @@ -2986,7 +2986,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: yaml.dump(self.as_dict(), str_io) yaml_str = str_io.getvalue() if filename: - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(yaml_str) return yaml_str elif fmt == "aims" or fnmatch(filename, "geometry.in"): @@ -2994,7 +2994,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: geom_in = AimsGeometryIn.from_structure(self) if filename: - with zopen(filename, mode="w") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(geom_in.get_header(filename)) file.write(geom_in.content) file.write("\n") @@ -3009,7 +3009,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: res_str = ResIO.structure_to_str(self) if filename: - with zopen(filename, mode="wt", encoding="utf8") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(res_str) return res_str elif fmt == "pwmat" or fnmatch(filename.lower(), "*.pwmat") or fnmatch(filename.lower(), "*.config"): @@ -3172,7 +3172,7 @@ def from_file( return struct fname = os.path.basename(filename) - with zopen(filename, mode="rt", errors="replace") as file: + with zopen(filename, mode="rt", errors="replace", encoding="utf-8") as file: contents = file.read() if fnmatch(fname.lower(), "*.cif*") or fnmatch(fname.lower(), "*.mcif*"): return cls.from_str( @@ -3918,7 +3918,7 @@ def to(self, filename: str = "", fmt: str = "") -> str | None: elif fmt == "json" or fnmatch(filename, "*.json*") or fnmatch(filename, "*.mson*"): json_str = json.dumps(self.as_dict()) if filename: - with zopen(filename, mode="wt", encoding="utf8") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(json_str) return json_str elif fmt in {"yaml", "yml"} or fnmatch(filename, "*.yaml*") or fnmatch(filename, "*.yml*"): @@ -3927,7 +3927,7 @@ def to(self, filename: str = "", fmt: str = "") -> str | None: yaml.dump(self.as_dict(), str_io) yaml_str = str_io.getvalue() if filename: - with zopen(filename, mode="wt", encoding="utf8") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(yaml_str) return yaml_str else: @@ -4009,7 +4009,7 @@ def from_file(cls, filename: PathLike) -> Self | None: """ filename = str(filename) - with zopen(filename) as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: contents = file.read() fname = filename.lower() if fnmatch(fname, "*.xyz*"): diff --git a/src/pymatgen/core/trajectory.py b/src/pymatgen/core/trajectory.py index a7bc74ac2d6..53a02ffb063 100644 --- a/src/pymatgen/core/trajectory.py +++ b/src/pymatgen/core/trajectory.py @@ -465,7 +465,7 @@ def write_Xdatcar( xdatcar_str = "\n".join(lines) + "\n" - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(xdatcar_str) def as_dict(self) -> dict: diff --git a/src/pymatgen/io/adf.py b/src/pymatgen/io/adf.py index f5fbf52d9a2..6725c89beb6 100644 --- a/src/pymatgen/io/adf.py +++ b/src/pymatgen/io/adf.py @@ -645,7 +645,7 @@ def _parse_logfile(self, logfile): # The last non-empty line of the logfile must match the end pattern. # Otherwise the job has some internal failure. The TAPE13 part of the # ADF manual has a detailed explanation. - with zopen(logfile, mode="rt") as file: + with zopen(logfile, mode="rt", encoding="utf-8") as file: for line in reverse_readline(file): if line == "": continue diff --git a/src/pymatgen/io/aims/inputs.py b/src/pymatgen/io/aims/inputs.py index 9b7ff838bc3..0997601f750 100644 --- a/src/pymatgen/io/aims/inputs.py +++ b/src/pymatgen/io/aims/inputs.py @@ -133,7 +133,7 @@ def from_file(cls, filepath: str | Path) -> Self: Returns: AimsGeometryIn: The input object represented in the file """ - with zopen(filepath, mode="rt") as in_file: + with zopen(filepath, mode="rt", encoding="utf-8") as in_file: content = in_file.read() return cls.from_str(content) @@ -752,7 +752,7 @@ def from_file(cls, filename: str, label: str | None = None) -> Self: Returns: AimsSpeciesFile """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls(data=file.read(), label=label) @classmethod diff --git a/src/pymatgen/io/cif.py b/src/pymatgen/io/cif.py index 60377486900..27164fdaf7f 100644 --- a/src/pymatgen/io/cif.py +++ b/src/pymatgen/io/cif.py @@ -299,7 +299,7 @@ def from_file(cls, filename: PathLike) -> Self: Returns: CifFile """ - with zopen(filename, mode="rt", errors="replace") as file: + with zopen(filename, mode="rt", errors="replace", encoding="utf-8") as file: return cls.from_str(file.read()) @@ -1757,9 +1757,9 @@ def cif_file(self) -> CifFile: def write_file( self, - filename: str | Path, - mode: Literal["w", "a", "wt", "at"] = "w", + filename: PathLike, + mode: Literal["wt", "at"] = "wt", ) -> None: """Write the CIF file.""" - with zopen(filename, mode=mode) as file: + with zopen(filename, mode=mode, encoding="utf-8") as file: file.write(str(self)) diff --git a/src/pymatgen/io/common.py b/src/pymatgen/io/common.py index b2445b9c356..44b4045a424 100644 --- a/src/pymatgen/io/common.py +++ b/src/pymatgen/io/common.py @@ -351,7 +351,7 @@ def to_cube(self, filename, comment: str = ""): filename (str): Name of the cube file to be written. comment (str): If provided, this will be added to the second comment line """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(f"# Cube file for {self.structure.formula} generated by Pymatgen\n") file.write(f"# {comment}\n") file.write(f"\t {len(self.structure)} 0.000000 0.000000 0.000000\n") @@ -383,7 +383,7 @@ def from_cube(cls, filename: str | Path) -> Self: Args: filename (str): of the cube to read """ - file = zopen(filename, mode="rt") + file = zopen(filename, mode="rt", encoding="utf-8") # skip header lines file.readline() @@ -526,7 +526,7 @@ def __getitem__(self, item): f"No parser defined for {item}. Contents are returned as a string.", UserWarning, ) - with zopen(fpath, "rt") as f: + with zopen(fpath, mode="rt", encoding="utf-8") as f: return f.read() def get_files_by_name(self, name: str) -> dict[str, Any]: diff --git a/src/pymatgen/io/core.py b/src/pymatgen/io/core.py index 5484954afe1..e81cd7a026e 100644 --- a/src/pymatgen/io/core.py +++ b/src/pymatgen/io/core.py @@ -74,7 +74,7 @@ def write_file(self, filename: PathLike) -> None: Args: filename: The filename to output to, including path. """ - with zopen(Path(filename), mode="wt") as file: + with zopen(Path(filename), mode="wt", encoding="utf-8") as file: file.write(self.get_str()) @classmethod @@ -102,7 +102,7 @@ def from_file(cls, path: PathLike) -> None: Returns: InputFile """ - with zopen(Path(path), mode="rt") as file: + with zopen(Path(path), mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) # from_str not implemented @@ -218,7 +218,7 @@ def write_input( if isinstance(contents, InputFile): contents.write_file(file_path) else: - with zopen(file_path, mode="wt") as file: + with zopen(file_path, mode="wt", encoding="utf-8") as file: file.write(str(contents)) if zip_inputs: diff --git a/src/pymatgen/io/cp2k/inputs.py b/src/pymatgen/io/cp2k/inputs.py index aa68870f100..488cd016771 100644 --- a/src/pymatgen/io/cp2k/inputs.py +++ b/src/pymatgen/io/cp2k/inputs.py @@ -692,7 +692,7 @@ def _from_dict(cls, dct: dict): @classmethod def from_file(cls, filename: str | Path) -> Self: """Initialize from a file.""" - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: txt = preprocessor(file.read(), os.path.dirname(file.name)) return cls.from_str(txt) diff --git a/src/pymatgen/io/cp2k/outputs.py b/src/pymatgen/io/cp2k/outputs.py index ee01c1db302..3885b640e67 100644 --- a/src/pymatgen/io/cp2k/outputs.py +++ b/src/pymatgen/io/cp2k/outputs.py @@ -327,7 +327,7 @@ def parse_initial_structure(self): ) coord_table = [] - with zopen(self.filename, mode="rt") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: while True: line = file.readline() if re.search(r"Atom\s+Kind\s+Element\s+X\s+Y\s+Z\s+Z\(eff\)\s+Mass", line): @@ -788,7 +788,7 @@ def parse_atomic_kind_info(self): except (TypeError, IndexError, ValueError): atomic_kind_info[kind]["total_pseudopotential_energy"] = None - with zopen(self.filename, mode="rt") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: j = -1 lines = file.readlines() for k, line in enumerate(lines): @@ -1009,7 +1009,7 @@ def parse_mo_eigenvalues(self): eigenvalues = [] efermi = [] - with zopen(self.filename, mode="rt") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: lines = iter(file.readlines()) for line in lines: try: @@ -1346,7 +1346,7 @@ def parse_hyperfine(self, hyperfine_filename=None): else: return None - with zopen(hyperfine_filename, mode="rt") as file: + with zopen(hyperfine_filename, mode="rt", encoding="utf-8") as file: lines = [line for line in file.read().split("\n") if line] hyperfine = [[] for _ in self.ionic_steps] @@ -1367,7 +1367,7 @@ def parse_gtensor(self, gtensor_filename=None): else: return None - with zopen(gtensor_filename, mode="rt") as file: + with zopen(gtensor_filename, mode="rt", encoding="utf-8") as file: lines = [line for line in file.read().split("\n") if line] data = {} @@ -1404,7 +1404,7 @@ def parse_chi_tensor(self, chi_filename=None): else: return None - with zopen(chi_filename, mode="rt") as file: + with zopen(chi_filename, mode="rt", encoding="utf-8") as file: lines = [line for line in file.read().split("\n") if line] data = {k: [] for k in "chi_soft chi_local chi_total chi_total_ppm_cgs PV1 PV2 PV3 ISO ANISO".split()} @@ -1551,7 +1551,7 @@ def read_table_pattern( row_pattern, or a dict in case that named capturing groups are defined by row_pattern. """ - with zopen(self.filename, mode="rt") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: if strip: lines = file.readlines() text = "".join( @@ -1688,7 +1688,7 @@ def parse_pdos(dos_file=None, spin_channel=None, total=False): """ spin = Spin(spin_channel) if spin_channel else Spin.down if "BETA" in os.path.split(dos_file)[-1] else Spin.up - with zopen(dos_file, mode="rt") as file: + with zopen(dos_file, mode="rt", encoding="utf-8") as file: lines = file.readlines() kind = re.search(r"atomic kind\s(.*)\sat iter", lines[0]) or re.search(r"list\s(\d+)\s(.*)\sat iter", lines[0]) kind = kind.groups()[0] diff --git a/src/pymatgen/io/cp2k/utils.py b/src/pymatgen/io/cp2k/utils.py index 7eca9758a73..9566ce45fbe 100644 --- a/src/pymatgen/io/cp2k/utils.py +++ b/src/pymatgen/io/cp2k/utils.py @@ -80,7 +80,7 @@ def preprocessor(data: str, dir: str = ".") -> str: # noqa: A002 raise ValueError(f"length of inc should be 2, got {len(inc)}") inc = inc[1].strip("'") inc = inc.strip('"') - with zopen(os.path.join(dir, inc)) as file: + with zopen(os.path.join(dir, inc), mode="rt", encoding="utf-8") as file: data = re.sub(rf"{incl}", file.read(), data) variable_sets = re.findall(r"(@SET.+)", data, re.IGNORECASE) for match in variable_sets: diff --git a/src/pymatgen/io/cssr.py b/src/pymatgen/io/cssr.py index c5a4fa4ab82..1068396e14b 100644 --- a/src/pymatgen/io/cssr.py +++ b/src/pymatgen/io/cssr.py @@ -57,7 +57,7 @@ def write_file(self, filename): Args: filename (str): Filename to write to. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self) + "\n") @classmethod @@ -98,5 +98,5 @@ def from_file(cls, filename: str | Path) -> Self: Returns: Cssr object. """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) diff --git a/src/pymatgen/io/exciting/inputs.py b/src/pymatgen/io/exciting/inputs.py index 7ef99482b03..5213c8f622a 100644 --- a/src/pymatgen/io/exciting/inputs.py +++ b/src/pymatgen/io/exciting/inputs.py @@ -172,7 +172,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: ExcitingInput """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: data = file.read().replace("\n", "") return cls.from_str(data) diff --git a/src/pymatgen/io/feff/inputs.py b/src/pymatgen/io/feff/inputs.py index 1b3a11571fc..f9321eb0881 100644 --- a/src/pymatgen/io/feff/inputs.py +++ b/src/pymatgen/io/feff/inputs.py @@ -246,7 +246,7 @@ def header_string_from_file(filename: str = "feff.inp"): Returns: Reads header string. """ - with zopen(filename, mode="r") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.readlines() feff_header_str = [] ln = 0 @@ -434,8 +434,8 @@ def atoms_string_from_file(filename): Returns: Atoms string. """ - with zopen(filename, mode="rt") as fobject: - f = fobject.readlines() + with zopen(filename, mode="rt", encoding="utf-8") as file: + f = file.readlines() coords = 0 atoms_str = [] @@ -527,7 +527,7 @@ def write_file(self, filename="ATOMS"): Args: filename: path for file to be written """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(f"{self}\n") @@ -651,7 +651,7 @@ def write_file(self, filename="PARAMETERS"): Args: filename: filename and path to write to. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(f"{self}\n") @classmethod @@ -665,7 +665,7 @@ def from_file(cls, filename: str = "feff.inp") -> Self: Returns: Tags """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = list(clean_lines(file.readlines())) params = {} eels_params = [] @@ -825,8 +825,8 @@ def pot_string_from_file(filename="feff.inp"): Returns: FEFFPOT string. """ - with zopen(filename, mode="rt") as f_object: - f = f_object.readlines() + with zopen(filename, mode="rt", encoding="utf-8") as file: + f = file.readlines() ln = -1 pot_str = ["POTENTIALS\n"] pot_tag = -1 @@ -931,7 +931,7 @@ def write_file(self, filename="POTENTIALS"): Args: filename: filename and path to write potential file to. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self) + "\n") @@ -973,7 +973,7 @@ def __str__(self): def write_file(self, filename="paths.dat"): """Write paths.dat.""" - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self) + "\n") diff --git a/src/pymatgen/io/feff/outputs.py b/src/pymatgen/io/feff/outputs.py index 16a0de3567d..8357fdfd357 100644 --- a/src/pymatgen/io/feff/outputs.py +++ b/src/pymatgen/io/feff/outputs.py @@ -70,7 +70,7 @@ def from_file(cls, feff_inp_file: str = "feff.inp", ldos_file: str = "ldos") -> dos_index = 1 begin = 0 - with zopen(pot_inp, mode="r") as potfile: + with zopen(pot_inp, mode="rt", encoding="utf-8") as potfile: for line in potfile: if len(pot_read_end.findall(line)) > 0: break @@ -95,7 +95,7 @@ def from_file(cls, feff_inp_file: str = "feff.inp", ldos_file: str = "ldos") -> dicts = Potential.pot_dict_from_str(pot_string) pot_dict = dicts[0] - with zopen(f"{ldos_file}00.dat", mode="r") as file: + with zopen(f"{ldos_file}00.dat", mode="rt", encoding="utf-8") as file: lines = file.readlines() e_fermi = float(lines[0].split()[4]) @@ -172,7 +172,7 @@ def charge_transfer_from_file(feff_inp_file, ldos_file): pot_inp = re.sub(r"feff.inp", r"pot.inp", feff_inp_file) pot_readstart = re.compile(".*iz.*lmaxsc.*xnatph.*xion.*folp.*") pot_readend = re.compile(".*ExternalPot.*switch.*") - with zopen(pot_inp, mode="r") as potfile: + with zopen(pot_inp, mode="rt", encoding="utf-8") as potfile: for line in potfile: if len(pot_readend.findall(line)) > 0: break @@ -203,7 +203,7 @@ def charge_transfer_from_file(feff_inp_file, ldos_file): for idx in range(len(dicts[0]) + 1): if len(str(idx)) == 1: - with zopen(f"{ldos_file}0{idx}.dat", mode="rt") as file: + with zopen(f"{ldos_file}0{idx}.dat", mode="rt", encoding="utf-8") as file: lines = file.readlines() s = float(lines[3].split()[2]) p = float(lines[4].split()[2]) @@ -212,7 +212,7 @@ def charge_transfer_from_file(feff_inp_file, ldos_file): tot = float(lines[1].split()[4]) cht[str(idx)] = {pot_dict[idx]: {"s": s, "p": p, "d": d, "f": f1, "tot": tot}} else: - with zopen(f"{ldos_file}{idx}.dat", mode="rt") as file: + with zopen(f"{ldos_file}{idx}.dat", mode="rt", encoding="utf-8") as file: lines = file.readlines() s = float(lines[3].split()[2]) p = float(lines[4].split()[2]) diff --git a/src/pymatgen/io/fiesta.py b/src/pymatgen/io/fiesta.py index df647a600fa..5ed94ac3e69 100644 --- a/src/pymatgen/io/fiesta.py +++ b/src/pymatgen/io/fiesta.py @@ -63,7 +63,7 @@ def run(self): init_folder = os.getcwd() os.chdir(self.folder) - with zopen(self.log_file, mode="w") as fout: + with zopen(self.log_file, mode="wt", encoding="utf-8") as fout: subprocess.call( [ self._NWCHEM2FIESTA_cmd, @@ -138,7 +138,7 @@ def _gw_run(self): if self.folder != init_folder: os.chdir(self.folder) - with zopen(self.log_file, mode="w") as fout: + with zopen(self.log_file, mode="wt", encoding="utf-8") as fout: subprocess.call( [ "mpirun", @@ -161,7 +161,7 @@ def bse_run(self): if self.folder != init_folder: os.chdir(self.folder) - with zopen(self.log_file, mode="w") as fout: + with zopen(self.log_file, mode="wt", encoding="utf-8") as fout: subprocess.call( [ "mpirun", @@ -214,7 +214,7 @@ def __init__(self, filename): """ self.filename = filename - with zopen(filename) as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: basis_set = file.read() self.data = self._parse_file(basis_set) @@ -533,7 +533,7 @@ def write_file(self, filename: str | Path) -> None: Args: filename: Filename. """ - with zopen(filename, mode="w") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self)) def as_dict(self): @@ -712,7 +712,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: FiestaInput object """ - with zopen(filename) as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) @@ -730,7 +730,7 @@ def __init__(self, filename): """ self.filename = filename - with zopen(filename) as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: data = file.read() chunks = re.split(r"GW Driver iteration", data) @@ -821,7 +821,7 @@ def __init__(self, filename): """ self.filename = filename - with zopen(filename) as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: log_bse = file.read() # self.job_info = self._parse_preamble(preamble) diff --git a/src/pymatgen/io/gaussian.py b/src/pymatgen/io/gaussian.py index cdd98482022..c74c9354406 100644 --- a/src/pymatgen/io/gaussian.py +++ b/src/pymatgen/io/gaussian.py @@ -368,7 +368,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: GaussianInput object """ - with zopen(filename, mode="r") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) def get_zmatrix(self): @@ -447,9 +447,9 @@ def para_dict_to_str(para, joiner=" "): def write_file(self, filename, cart_coords=False): """Write the input string into a file. - Option: see __str__ method + Option: see `__str__` method """ - with zopen(filename, mode="w") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(self.to_str(cart_coords)) def as_dict(self): @@ -661,7 +661,7 @@ def _parse(self, filename): opt_structures = [] route_lower = {} - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: for line in file: if parse_stage == 0: if start_patt.search(line): @@ -1102,7 +1102,7 @@ def read_scan(self): data = {"energies": [], "coords": {}} # read in file - with zopen(self.filename, mode="r") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: line = file.readline() while line != "": @@ -1188,7 +1188,7 @@ def read_excitation_energies(self): transitions = [] # read in file - with zopen(self.filename, mode="r") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: line = file.readline() td = False while line != "": diff --git a/src/pymatgen/io/lammps/data.py b/src/pymatgen/io/lammps/data.py index 13514acd8c2..d241f6c41bf 100644 --- a/src/pymatgen/io/lammps/data.py +++ b/src/pymatgen/io/lammps/data.py @@ -647,7 +647,7 @@ def from_file(cls, filename: str, atom_style: str = "full", sort_id: bool = Fals sort_id (bool): Whether sort each section by id. Default to True. """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.readlines() kw_pattern = r"|".join(itertools.chain(*SECTION_KEYWORDS.values())) section_marks = [idx for idx, line in enumerate(lines) if re.search(kw_pattern, line)] @@ -1436,7 +1436,7 @@ def parse_xyz(cls, filename: str | Path) -> pd.DataFrame: Returns: pandas.DataFrame """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.readlines() str_io = StringIO("".join(lines[2:])) # skip the 2nd line diff --git a/src/pymatgen/io/lammps/generators.py b/src/pymatgen/io/lammps/generators.py index 143860d8764..d541dd1adf4 100644 --- a/src/pymatgen/io/lammps/generators.py +++ b/src/pymatgen/io/lammps/generators.py @@ -67,7 +67,7 @@ def get_input_set(self, structure: Structure | LammpsData | CombinedData) -> Lam data: LammpsData = LammpsData.from_structure(structure) if isinstance(structure, Structure) else structure # Load the template - with zopen(self.template, mode="r") as file: + with zopen(self.template, mode="rt", encoding="utf-8") as file: template_str = file.read() # Replace all variables diff --git a/src/pymatgen/io/lammps/inputs.py b/src/pymatgen/io/lammps/inputs.py index bcb421354c2..5b473a16a28 100644 --- a/src/pymatgen/io/lammps/inputs.py +++ b/src/pymatgen/io/lammps/inputs.py @@ -549,7 +549,7 @@ def write_file( If False, a single block is assumed. """ filename = filename if isinstance(filename, Path) else Path(filename) - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(self.get_str(ignore_comments=ignore_comments, keep_stages=keep_stages)) @classmethod @@ -649,7 +649,7 @@ def from_file(cls, path: str | Path, ignore_comments: bool = False, keep_stages: LammpsInputFile """ filename = path if isinstance(path, Path) else Path(path) - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read(), ignore_comments=ignore_comments, keep_stages=keep_stages) def __repr__(self) -> str: diff --git a/src/pymatgen/io/lammps/outputs.py b/src/pymatgen/io/lammps/outputs.py index 216cc458c66..4efc01fc934 100644 --- a/src/pymatgen/io/lammps/outputs.py +++ b/src/pymatgen/io/lammps/outputs.py @@ -115,7 +115,7 @@ def parse_lammps_dumps(file_pattern): files = sorted(files, key=lambda f: int(re.match(pattern, f)[1])) for filename in files: - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: dump_cache = [] for line in file: if line.startswith("ITEM: TIMESTEP"): @@ -144,7 +144,7 @@ def parse_lammps_log(filename: str = "log.lammps") -> list[pd.DataFrame]: Returns: [pd.DataFrame] containing thermo data for each completed run. """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.readlines() begin_flag = ( "Memory usage per processor =", diff --git a/src/pymatgen/io/lmto.py b/src/pymatgen/io/lmto.py index 1a5d669d4bc..a660ee12510 100644 --- a/src/pymatgen/io/lmto.py +++ b/src/pymatgen/io/lmto.py @@ -139,7 +139,7 @@ def write_file(self, filename="CTRL", **kwargs): """Write a CTRL file with structure, HEADER, and VERS that can be used as input for lmhart.run. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(self.get_str(**kwargs)) @classmethod @@ -153,7 +153,7 @@ def from_file(cls, filename: str | Path = "CTRL", **kwargs) -> Self: Returns: An LMTOCtrl object. """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: contents = file.read() return cls.from_str(contents, **kwargs) @@ -322,7 +322,7 @@ def __init__(self, filename="COPL", to_eV=False): eV, set to True. Defaults to False for energies in Ry. """ # COPL files have an extra trailing blank line - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: contents = file.read().split("\n")[:-1] # The parameters line is the second line in a COPL file. It # contains all parameters that are needed to map the file. diff --git a/src/pymatgen/io/lobster/inputs.py b/src/pymatgen/io/lobster/inputs.py index a4f7902e73b..3bbe0518a11 100644 --- a/src/pymatgen/io/lobster/inputs.py +++ b/src/pymatgen/io/lobster/inputs.py @@ -585,7 +585,7 @@ def from_file(cls, lobsterin: PathLike) -> Self: Returns: Lobsterin object """ - with zopen(lobsterin, mode="rt") as file: + with zopen(lobsterin, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") if not lines: raise RuntimeError("lobsterin file contains no data.") @@ -642,7 +642,7 @@ def _get_potcar_symbols(POTCAR_input: PathLike) -> list[str]: raise ValueError("Lobster only works with PAW! Use different POTCARs") # Warning about a bug in LOBSTER-4.1.0 - with zopen(POTCAR_input, mode="r") as file: + with zopen(POTCAR_input, mode="rt", encoding="utf-8") as file: data = file.read() if isinstance(data, bytes): diff --git a/src/pymatgen/io/lobster/outputs.py b/src/pymatgen/io/lobster/outputs.py index 300e2f68c90..78946c1fc51 100644 --- a/src/pymatgen/io/lobster/outputs.py +++ b/src/pymatgen/io/lobster/outputs.py @@ -121,7 +121,7 @@ def __init__( else: self._filename = "COHPCAR.lobster" - with zopen(self._filename, mode="rt") as file: + with zopen(self._filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") # The parameters line is the second line in a COHPCAR file. @@ -405,7 +405,7 @@ def __init__( # LOBSTER list files have an extra trailing blank line # and we don't need the header. if self._icohpcollection is None: - with zopen(self._filename, mode="rt") as file: + with zopen(self._filename, mode="rt", encoding="utf-8") as file: all_lines = file.read().split("\n") lines = all_lines[1:-1] if "spin" not in all_lines[1] else all_lines[2:-1] if len(lines) == 0: @@ -622,7 +622,7 @@ def __init__(self, filename: PathLike | None = "NcICOBILIST.lobster") -> None: # LOBSTER list files have an extra trailing blank line # and we don't need the header - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n")[1:-1] if len(lines) == 0: raise RuntimeError("NcICOBILIST file contains no data.") @@ -753,7 +753,7 @@ def _parse_doscar(self): tdensities = {} itdensities = {} - with zopen(doscar, mode="rt") as file: + with zopen(doscar, mode="rt", encoding="utf-8") as file: file.readline() # Skip the first line efermi = float([file.readline() for nn in range(4)][3].split()[17]) dos = [] @@ -912,7 +912,7 @@ def __init__( self.loewdin = [] if loewdin is None else loewdin if self.num_atoms is None: - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n")[3:-3] if len(lines) == 0: raise RuntimeError("CHARGES file contains no data.") @@ -1046,7 +1046,7 @@ def __init__(self, filename: PathLike | None, **kwargs) -> None: else: raise ValueError(f"{attr}={val} is not a valid attribute for Lobsterout") elif filename: - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") if len(lines) == 0: raise RuntimeError("lobsterout does not contain any data") @@ -1438,7 +1438,7 @@ def __init__( raise ValueError("No FATBAND files in folder or given") for fname in filenames: - with zopen(fname, mode="rt") as file: + with zopen(fname, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") atom_names.append(os.path.split(fname)[1].split("_")[1].capitalize()) @@ -1472,7 +1472,7 @@ def __init__( eigenvals: dict = {} p_eigenvals: dict = {} for ifilename, filename in enumerate(filenames): - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") if ifilename == 0: @@ -1620,7 +1620,7 @@ def __init__( self.max_deviation = [] if max_deviation is None else max_deviation if not self.band_overlaps_dict: - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") spin_numbers = [0, 1] if lines[0].split()[-1] == "0" else [1, 2] @@ -1760,7 +1760,7 @@ def __init__( self.is_lcfo = is_lcfo self.list_dict_grosspop = [] if list_dict_grosspop is None else list_dict_grosspop if not self.list_dict_grosspop: - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") # Read file to list of dict @@ -1890,7 +1890,7 @@ def _parse_file( imaginary (list[float]): Imaginary parts of wave function. distance (list[float]): Distances to the first point in wave function file. """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") points = [] @@ -2060,7 +2060,7 @@ def __init__( self.madelungenergies_mulliken = None if madelungenergies_mulliken is None else madelungenergies_mulliken if self.ewald_splitting is None: - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n")[5] if len(lines) == 0: raise RuntimeError("MadelungEnergies file contains no data.") @@ -2131,7 +2131,7 @@ def __init__( self.madelungenergies_mulliken: list | float = madelungenergies_mulliken or [] if self.num_atoms is None: - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") if len(lines) == 0: raise RuntimeError("SitePotentials file contains no data.") @@ -2284,7 +2284,7 @@ def __init__( """ self._filename = str(filename) - with zopen(self._filename, mode="rt") as file: + with zopen(self._filename, mode="rt", encoding="utf-8") as file: lines = file.readlines() if len(lines) == 0: raise RuntimeError("Please check provided input file, it seems to be empty") @@ -2472,7 +2472,7 @@ def __init__( self.bin_width = 0.0 if bin_width is None else bin_width if not self.bwdf: - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = file.read().split("\n") if len(lines) == 0: raise RuntimeError("BWDF file contains no data.") diff --git a/src/pymatgen/io/nwchem.py b/src/pymatgen/io/nwchem.py index 5aa772591af..2e7e9f22efd 100644 --- a/src/pymatgen/io/nwchem.py +++ b/src/pymatgen/io/nwchem.py @@ -391,7 +391,7 @@ def write_file(self, filename): Args: filename (str): Filename. """ - with zopen(filename, mode="w") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self)) def as_dict(self): @@ -531,7 +531,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: NwInput object """ - with zopen(filename) as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) @@ -554,7 +554,7 @@ def __init__(self, filename): """ self.filename = filename - with zopen(filename) as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: data = file.read() chunks = re.split(r"NWChem Input Module", data) diff --git a/src/pymatgen/io/pwmat/inputs.py b/src/pymatgen/io/pwmat/inputs.py index aa60c1737cd..92bd77e0374 100644 --- a/src/pymatgen/io/pwmat/inputs.py +++ b/src/pymatgen/io/pwmat/inputs.py @@ -36,7 +36,7 @@ def locate_all_lines(file_path: PathLike, content: str, exclusion: str = "") -> """ row_idxs: list[int] = [] # starts from 1 to be compatible with linecache package row_no: int = 0 - with zopen(file_path, mode="rt") as file: + with zopen(file_path, mode="rt", encoding="utf-8") as file: for row_content in file: row_no += 1 if content.upper() in row_content.upper() and ( @@ -418,7 +418,7 @@ def from_file(cls, filename: PathLike, mag: bool = False) -> Self: Returns: AtomConfig object. """ - with zopen(filename, "rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(data=file.read(), mag=mag) @classmethod @@ -466,7 +466,7 @@ def get_str(self) -> str: def write_file(self, filename: PathLike, **kwargs): """Write AtomConfig to a file.""" - with zopen(filename, "wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(self.get_str(**kwargs)) def as_dict(self): @@ -588,7 +588,7 @@ def write_file(self, filename: PathLike): Args: filename (PathLike): The absolute path of file to be written. """ - with zopen(filename, "wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(self.get_str()) @@ -694,5 +694,5 @@ def get_hsp_row_str(label: str, index: int, coordinate: float) -> str: def write_file(self, filename: PathLike): """Write HighSymmetryPoint to a file.""" - with zopen(filename, "wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(self.get_str()) diff --git a/src/pymatgen/io/pwmat/outputs.py b/src/pymatgen/io/pwmat/outputs.py index 9389e82befb..0183be0bab7 100644 --- a/src/pymatgen/io/pwmat/outputs.py +++ b/src/pymatgen/io/pwmat/outputs.py @@ -135,7 +135,7 @@ def _parse_sefv(self) -> list[dict]: 'atom_forces' and 'virial'. """ ionic_steps: list[dict] = [] - with zopen(self.filename, "rt") as mvt: + with zopen(self.filename, mode="rt", encoding="utf-8") as mvt: tmp_step: dict = {} for ii in range(self.n_ionic_steps): tmp_chunk: str = "" @@ -168,7 +168,7 @@ def __init__(self, filename: PathLike): filename (PathLike): The absolute path of OUT.FERMI file. """ self.filename: PathLike = filename - with zopen(self.filename, "rt") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: self._e_fermi: float = np.round(float(file.readline().split()[-2].strip()), 3) @property @@ -346,7 +346,7 @@ def _parse(self): labels: list[str] = [] labels = linecache.getline(str(self.filename), 1).split()[1:] dos_str: str = "" - with zopen(self.filename, mode="rt") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: file.readline() dos_str = file.read() dos: np.ndarray = np.loadtxt(StringIO(dos_str)) diff --git a/src/pymatgen/io/pwscf.py b/src/pymatgen/io/pwscf.py index 2f32c0c346a..03f7e7d456e 100644 --- a/src/pymatgen/io/pwscf.py +++ b/src/pymatgen/io/pwscf.py @@ -275,7 +275,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: PWInput object """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) @classmethod diff --git a/src/pymatgen/io/qchem/inputs.py b/src/pymatgen/io/qchem/inputs.py index d350a84ba55..7ff5be8487e 100644 --- a/src/pymatgen/io/qchem/inputs.py +++ b/src/pymatgen/io/qchem/inputs.py @@ -373,7 +373,7 @@ def write_multi_job_file(job_list: list[QCInput], filename: str): job_list (list[QCInput]): List of QChem jobs. filename (str): Name of the file to write. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(QCInput.multi_job_string(job_list)) @classmethod @@ -387,7 +387,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: QcInput """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) @classmethod @@ -401,7 +401,7 @@ def from_multi_jobs_file(cls, filename: str) -> list[Self]: Returns: List of QCInput objects """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: # the delimiter between QChem jobs is @@@ multi_job_strings = file.read().split("@@@") # list of individual QChem jobs diff --git a/src/pymatgen/io/qchem/outputs.py b/src/pymatgen/io/qchem/outputs.py index d7525a52ccd..f9cc304c72f 100644 --- a/src/pymatgen/io/qchem/outputs.py +++ b/src/pymatgen/io/qchem/outputs.py @@ -661,7 +661,7 @@ def multiple_outputs_from_file(filename, keep_sub_files=True): 2.) Creates separate QCCalcs for each one from the sub-files. """ to_return = [] - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: text = re.split(r"\s*(?:Running\s+)*Job\s+\d+\s+of\s+\d+\s+", file.read()) if text[0] == "": text = text[1:] diff --git a/src/pymatgen/io/qchem/sets.py b/src/pymatgen/io/qchem/sets.py index 31c5b8fc321..c3321d150d6 100644 --- a/src/pymatgen/io/qchem/sets.py +++ b/src/pymatgen/io/qchem/sets.py @@ -643,7 +643,7 @@ def write(self, input_file: PathLike) -> None: """ self.write_file(input_file) if self.smd_solvent in {"custom", "other"} and self.qchem_version == 5: - with zopen(os.path.join(os.path.dirname(input_file), "solvent_data"), mode="wt") as file: + with zopen(os.path.join(os.path.dirname(input_file), "solvent_data"), mode="wt", encoding="utf-8") as file: file.write(self.custom_smd) diff --git a/src/pymatgen/io/res.py b/src/pymatgen/io/res.py index 9a45a3b9dc2..cfc81f3d649 100644 --- a/src/pymatgen/io/res.py +++ b/src/pymatgen/io/res.py @@ -249,7 +249,7 @@ def _parse_str(cls, source: str) -> Res: def _parse_file(cls, filename: str | Path) -> Res: """Parse the res file as a file.""" self = cls() - with zopen(filename, mode="r") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: self.source = file.read() return self._parse_txt() @@ -335,7 +335,7 @@ def string(self) -> str: def write(self, filename: str) -> None: """Write the res data to a file.""" - with zopen(filename, mode="w") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self)) diff --git a/src/pymatgen/io/template.py b/src/pymatgen/io/template.py index 2ee08031ede..bacfd661e97 100644 --- a/src/pymatgen/io/template.py +++ b/src/pymatgen/io/template.py @@ -52,7 +52,7 @@ def get_input_set( self.filename = str(filename) # Load the template - with zopen(self.template, mode="r") as file: + with zopen(self.template, mode="rt", encoding="utf-8") as file: template_str = file.read() # Replace all variables diff --git a/src/pymatgen/io/vasp/inputs.py b/src/pymatgen/io/vasp/inputs.py index 6fa0f690df2..af0390b0064 100644 --- a/src/pymatgen/io/vasp/inputs.py +++ b/src/pymatgen/io/vasp/inputs.py @@ -284,7 +284,7 @@ def from_file( except Exception: names = None - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read(), names, read_velocities=read_velocities) @classmethod @@ -657,7 +657,7 @@ def write_file(self, filename: PathLike, **kwargs) -> None: """Write POSCAR to a file. The supported kwargs are the same as those for the Poscar.get_str method and are passed through directly. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(self.get_str(**kwargs)) def as_dict(self) -> dict: @@ -892,7 +892,7 @@ def write_file(self, filename: PathLike) -> None: Args: filename (str): filename to write to. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self)) @classmethod @@ -905,7 +905,7 @@ def from_file(cls, filename: PathLike) -> Self: Returns: Incar object """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) @classmethod @@ -1645,7 +1645,7 @@ def from_file(cls, filename: PathLike) -> Self: Returns: Kpoints object """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) @classmethod @@ -1786,7 +1786,7 @@ def write_file(self, filename: PathLike) -> None: Args: filename (PathLike): Filename to write to. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self)) def as_dict(self) -> dict[str, Any]: @@ -2390,9 +2390,9 @@ def write_file(self, filename: str) -> None: """Write PotcarSingle to a file. Args: - filename (str): Filename to write to. + filename (str): File to write to. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self)) def copy(self) -> Self: @@ -2417,7 +2417,7 @@ def from_file(cls, filename: PathLike) -> Self: symbol = match[0] if match else "" try: - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls(file.read(), symbol=symbol or None) except UnicodeDecodeError: @@ -2823,7 +2823,7 @@ def from_file(cls, filename: PathLike) -> Self: Returns: Potcar """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: fdata = file.read() potcar = cls() @@ -2846,7 +2846,7 @@ def write_file(self, filename: PathLike) -> None: Args: filename (PathLike): filename to write to. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self)) def set_symbols( @@ -2982,7 +2982,7 @@ def write_input( for key, value in self.items(): if value is not None: - with zopen(os.path.join(output_dir, key), mode="wt") as file: + with zopen(os.path.join(output_dir, key), mode="wt", encoding="utf-8") as file: file.write(str(value)) if cif_name: @@ -3005,8 +3005,8 @@ def write_input( files_to_transfer = files_to_transfer or {} for key, val in files_to_transfer.items(): with ( - zopen(val, "rb") as fin, - zopen(str(Path(output_dir) / key), "wb") as fout, + zopen(val, mode="rb") as fin, + zopen(str(Path(output_dir) / key), mode="wb") as fout, ): copyfileobj(fin, fout) diff --git a/src/pymatgen/io/vasp/outputs.py b/src/pymatgen/io/vasp/outputs.py index 14b116ffcbf..ba23c9c9f11 100644 --- a/src/pymatgen/io/vasp/outputs.py +++ b/src/pymatgen/io/vasp/outputs.py @@ -309,7 +309,7 @@ def __init__( self.separate_spins = separate_spins self.exception_on_bad_xml = exception_on_bad_xml - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: if ionic_step_skip or ionic_step_offset: # Remove parts of the xml file and parse the string content: str = file.read() @@ -1759,7 +1759,7 @@ def __init__( self.occu_tol = occu_tol self.separate_spins = separate_spins - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: self.efermi = None parsed_header = False in_kpoints_opt = False @@ -2111,7 +2111,7 @@ def __init__(self, filename: PathLike) -> None: # Data from beginning of OUTCAR run_stats["cores"] = None - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: for line in file: if "serial" in line: # Activate serial parallelization @@ -2362,7 +2362,7 @@ def read_table_pattern( if last_one_only and first_one_only: raise ValueError("last_one_only and first_one_only options are incompatible") - with zopen(self.filename, mode="rt") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: text = file.read() table_pattern_text = header_pattern + r"\s*^(?P(?:\s+" + row_pattern + r")+)\s+" + footer_pattern table_pattern = re.compile(table_pattern_text, re.MULTILINE | re.DOTALL) @@ -2448,7 +2448,7 @@ def read_freq_dielectric(self) -> None: data: dict[str, Any] = {"REAL": [], "IMAGINARY": []} count = 0 component = "IMAGINARY" - with zopen(self.filename, mode="rt") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: for line in file: line = line.strip() if re.match(plasma_pattern, line): @@ -2574,7 +2574,7 @@ def read_cs_raw_symmetrized_tensors(self) -> None: row_pattern = r"\s+".join([r"([-]?\d+\.\d+)"] * 3) unsym_footer_pattern = r"^\s+SYMMETRIZED TENSORS\s+$" - with zopen(self.filename, mode="rt") as file: + with zopen(self.filename, mode="rt", encoding="utf-8") as file: text = file.read() unsym_table_pattern_text = header_pattern + first_part_pattern + r"(?P.+)" + unsym_footer_pattern table_pattern = re.compile(unsym_table_pattern_text, re.MULTILINE | re.DOTALL) @@ -3357,7 +3357,7 @@ def read_core_state_eigen(self) -> list[dict]: The core state eigenenergie of the 2s AO of the 6th atom of the structure at the last ionic step is [5]["2s"][-1]. """ - with zopen(self.filename, mode="rt") as foutcar: + with zopen(self.filename, mode="rt", encoding="utf-8") as foutcar: line = foutcar.readline() cl: list[dict] = [] @@ -3399,7 +3399,7 @@ def read_avg_core_poten(self) -> list[list]: The average core potential of the 2nd atom of the structure at the last ionic step is: [-1][1] """ - with zopen(self.filename, mode="rt") as foutcar: + with zopen(self.filename, mode="rt", encoding="utf-8") as foutcar: line = foutcar.readline() aps: list[list[float]] = [] while line != "": @@ -3595,7 +3595,7 @@ def parse_file(filename: PathLike) -> tuple[Poscar, dict, dict]: ngrid_pts = 0 data_count = 0 poscar = None - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: for line in file: original_line = line line = line.strip() @@ -3729,7 +3729,7 @@ def write_spin(data_type: str) -> None: if isinstance(data, Iterable): file.write("".join(data)) - with zopen(file_name, mode="wt") as file: + with zopen(file_name, mode="wt", encoding="utf-8") as file: poscar = Poscar(self.structure) # Use original name if it's been set (e.g. from Chgcar) @@ -4062,7 +4062,7 @@ def _read(self, filename: PathLike, parsed_kpoints: set[tuple[Kpoint]] | None = if parsed_kpoints is None: parsed_kpoints = set() - with zopen(filename, mode="rt") as file_handle: + with zopen(filename, mode="rt", encoding="utf-8") as file: preamble_expr = re.compile(r"# of k-points:\s*(\d+)\s+# of bands:\s*(\d+)\s+# of ions:\s*(\d+)") kpoint_expr = re.compile(r"^k-point\s+(\d+).*weight = ([0-9\.]+)") band_expr = re.compile(r"^band\s+(\d+)") @@ -4093,7 +4093,7 @@ def _read(self, filename: PathLike, parsed_kpoints: set[tuple[Kpoint]] | None = # total and x,y,z) for each band, while non-SOC have only 1 list of projections: tot_count = 0 band_count = 0 - for line in file_handle: + for line in file: if total_expr.match(line): tot_count += 1 elif band_expr.match(line): @@ -4101,7 +4101,7 @@ def _read(self, filename: PathLike, parsed_kpoints: set[tuple[Kpoint]] | None = if band_count == 2: break - file_handle.seek(0) # reset file handle to beginning + file.seek(0) # reset file handle to beginning if tot_count == 1: is_soc = False elif tot_count == 4: @@ -4118,7 +4118,7 @@ def _read(self, filename: PathLike, parsed_kpoints: set[tuple[Kpoint]] | None = skipping_kpoint = False # true when skipping projections for a previously-parsed kpoint ion_line_count = 0 # printed twice when phase factors present proj_data_parsed_for_band = 0 # 0 for non-SOC, 1-4 for SOC/phase factors - for line in file_handle: + for line in file: line = line.strip() if ion_expr.match(line): ion_line_count += 1 @@ -4353,8 +4353,8 @@ def smart_convert(header: str, num: float | str) -> float | str: electronic_pattern = re.compile(r"\s*\w+\s*:(.*)") header: list = [] - with zopen(filename, mode="rt") as fid: - for line in fid: + with zopen(filename, mode="rt", encoding="utf-8") as file: + for line in file: if match := electronic_pattern.match(line.strip()): tokens = match[1].split() data = {header[idx]: smart_convert(header[idx], tokens[idx]) for idx in range(len(tokens))} @@ -4493,10 +4493,10 @@ def __init__( if ionicstep_end is not None and ionicstep_end < 1: raise ValueError("End ionic step cannot be less than 1") - file_len = sum(1 for _ in zopen(filename, mode="rt")) + file_len = sum(1 for _ in zopen(filename, mode="rt", encoding="utf-8")) ionicstep_cnt = 1 ionicstep_start = ionicstep_start or 0 - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: title = None for iline, line in enumerate(file): line = line.strip() @@ -4592,7 +4592,7 @@ def concatenate( raise ValueError("End ionic step cannot be less than 1") ionicstep_cnt = 1 - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: for line in file: line = line.strip() if preamble is None: @@ -4680,7 +4680,7 @@ def write_file(self, filename: PathLike, **kwargs) -> None: **kwargs: The same as those for the Xdatcar.get_str method and are passed through directly. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(self.get_str(**kwargs)) @@ -4702,7 +4702,7 @@ def __init__(self, filename: PathLike) -> None: Args: filename: Name of file containing DYNMAT. """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: lines = list(clean_lines(file.readlines())) self._nspecs, self._natoms, self._ndisps = map(int, lines[0].split()) self._masses = map(float, lines[1].split()) @@ -5394,7 +5394,7 @@ def __init__( self.occu_tol = occu_tol self.separate_spins = separate_spins - with zopen(filename, mode="r") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: self.ispin = int(file.readline().split()[-1]) # Remove useless header information @@ -5545,7 +5545,7 @@ def from_formatted(cls, filename: PathLike) -> Self: Returns: A Waveder object. """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: nspin, nkpts, nbands = file.readline().split() # 1 and 4 are the eigenvalues of the bands (this data is missing in the WAVEDER file) # 6:12 are the complex matrix elements in each cartesian direction. diff --git a/src/pymatgen/io/xr.py b/src/pymatgen/io/xr.py index 834b8d6d797..e86554a83be 100644 --- a/src/pymatgen/io/xr.py +++ b/src/pymatgen/io/xr.py @@ -68,7 +68,7 @@ def write_file(self, filename: str | Path) -> None: Args: filename (str): name of the file to write to. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self) + "\n") @classmethod @@ -155,5 +155,5 @@ def from_file(cls, filename: str | Path, use_cores: bool = True, thresh: float = xr (Xr): Xr object corresponding to the input file. """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read(), use_cores=use_cores, thresh=thresh) diff --git a/src/pymatgen/io/xyz.py b/src/pymatgen/io/xyz.py index 98078baa63c..4f44f66e552 100644 --- a/src/pymatgen/io/xyz.py +++ b/src/pymatgen/io/xyz.py @@ -111,7 +111,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: XYZ object """ - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) def as_dataframe(self): @@ -151,5 +151,5 @@ def write_file(self, filename: str) -> None: Args: filename (str): File name of output file. """ - with zopen(filename, mode="wt") as file: + with zopen(filename, mode="wt", encoding="utf-8") as file: file.write(str(self)) diff --git a/src/pymatgen/io/zeopp.py b/src/pymatgen/io/zeopp.py index 55a1506dc44..64f1557f4d1 100644 --- a/src/pymatgen/io/zeopp.py +++ b/src/pymatgen/io/zeopp.py @@ -146,7 +146,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: ZeoCssr object. """ - with zopen(filename, mode="r") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) @@ -200,7 +200,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: XYZ object """ - with zopen(filename) as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: return cls.from_str(file.read()) def __str__(self) -> str: diff --git a/src/pymatgen/transformations/site_transformations.py b/src/pymatgen/transformations/site_transformations.py index 4672bad1183..6b1bede1f2a 100644 --- a/src/pymatgen/transformations/site_transformations.py +++ b/src/pymatgen/transformations/site_transformations.py @@ -431,9 +431,7 @@ def apply_transformation(self, structure: Structure, return_ranked_list: bool | n_to_remove = round(n_to_remove) num_remove_dict[tuple(idx)] = n_to_remove n = len(idx) - total_combos += int( - round(math.factorial(n) / math.factorial(n_to_remove) / math.factorial(n - n_to_remove)) - ) + total_combos += round(math.factorial(n) / math.factorial(n_to_remove) / math.factorial(n - n_to_remove)) self.logger.debug(f"Total combinations = {total_combos}") diff --git a/src/pymatgen/util/io_utils.py b/src/pymatgen/util/io_utils.py index f8c7d268f43..7bf4efdc2f4 100644 --- a/src/pymatgen/util/io_utils.py +++ b/src/pymatgen/util/io_utils.py @@ -81,7 +81,7 @@ def micro_pyawk(filename, search, results=None, debug=None, postdebug=None): for entry in search: entry[0] = re.compile(entry[0]) - with zopen(filename, mode="rt") as file: + with zopen(filename, mode="rt", encoding="utf-8") as file: for line in file: for entry in search: match = re.search(entry[0], line) diff --git a/tests/electronic_structure/test_dos.py b/tests/electronic_structure/test_dos.py index a7c8dec41b1..edf2c6957ae 100644 --- a/tests/electronic_structure/test_dos.py +++ b/tests/electronic_structure/test_dos.py @@ -108,7 +108,7 @@ class TestCompleteDos(TestCase): def setUp(self): with open(f"{TEST_DIR}/complete_dos.json") as file: self.dos = CompleteDos.from_dict(json.load(file)) - with zopen(f"{TEST_DIR}/pdag3_complete_dos.json.gz") as file: + with zopen(f"{TEST_DIR}/pdag3_complete_dos.json.gz", mode="rt", encoding="utf-8") as file: self.dos_pdag3 = CompleteDos.from_dict(json.load(file)) def test_get_gap(self): diff --git a/tests/io/aims/conftest.py b/tests/io/aims/conftest.py index 3cabe2b3ca8..b2ac971c904 100644 --- a/tests/io/aims/conftest.py +++ b/tests/io/aims/conftest.py @@ -150,7 +150,7 @@ def compare_single_files(ref_file: PathLike, test_file: PathLike) -> None: with open(test_file) as tf: test_lines = tf.readlines()[5:] - with zopen(f"{ref_file}.gz", mode="rt") as rf: + with zopen(f"{ref_file}.gz", mode="rt", encoding="utf-8") as rf: ref_lines = rf.readlines()[5:] for test_line, ref_line in zip(test_lines, ref_lines, strict=True): diff --git a/tests/io/pwmat/test_inputs.py b/tests/io/pwmat/test_inputs.py index 7618e7ecdf6..e1438482623 100644 --- a/tests/io/pwmat/test_inputs.py +++ b/tests/io/pwmat/test_inputs.py @@ -47,7 +47,7 @@ class TestACstrExtractor(PymatgenTest): def test_extract(self): filepath = f"{TEST_DIR}/atom.config" ac_extractor = ACExtractor(file_path=filepath) - with zopen(filepath, mode="rt") as file: + with zopen(filepath, mode="rt", encoding="utf-8") as file: ac_str_extractor = ACstrExtractor(atom_config_str="".join(file.readlines())) assert ac_extractor.n_atoms == ac_str_extractor.get_n_atoms() for idx in range(9): @@ -102,7 +102,7 @@ def test_write_file(self): tmp_file = f"{self.tmp_path}/gen.kpt.testing.lzma" gen_kpt.write_file(tmp_file) tmp_gen_kpt_str = "" - with zopen(tmp_file, mode="rt") as file: + with zopen(tmp_file, mode="rt", encoding="utf-8") as file: tmp_gen_kpt_str = file.read() assert gen_kpt.get_str() == tmp_gen_kpt_str @@ -128,7 +128,7 @@ def test_write_file(self): tmp_filepath = f"{self.tmp_path}/HIGH_SYMMETRY_POINTS.testing.lzma" high_symmetry_points.write_file(tmp_filepath) tmp_high_symmetry_points_str = "" - with zopen(tmp_filepath, "rt") as file: + with zopen(tmp_filepath, "rt", encoding="utf-8") as file: tmp_high_symmetry_points_str = file.read() assert tmp_high_symmetry_points_str == high_symmetry_points.get_str() diff --git a/tests/io/vasp/test_inputs.py b/tests/io/vasp/test_inputs.py index 1616ff4e5e7..5b572d44acd 100644 --- a/tests/io/vasp/test_inputs.py +++ b/tests/io/vasp/test_inputs.py @@ -1560,7 +1560,7 @@ def test_from_file(self): } def test_potcar_map(self): - fe_potcar = zopen(f"{FAKE_POTCAR_DIR}/POT_GGA_PAW_PBE/POTCAR.Fe_pv.gz").read().decode("utf-8") + fe_potcar = zopen(f"{FAKE_POTCAR_DIR}/POT_GGA_PAW_PBE/POTCAR.Fe_pv.gz", mode="rt", encoding="utf-8").read() # specify V instead of Fe - this makes sure the test won't pass if the # code just grabs the POTCAR from the config file (the config file would # grab the V POTCAR) diff --git a/tests/io/vasp/test_outputs.py b/tests/io/vasp/test_outputs.py index e02b9ac027a..e459bec8919 100644 --- a/tests/io/vasp/test_outputs.py +++ b/tests/io/vasp/test_outputs.py @@ -2148,7 +2148,7 @@ def test_consistency(self): wder_ref = np.loadtxt(f"{VASP_OUT_DIR}/WAVEDERF.Si.gz", skiprows=1) def _check(wder): - with zopen(f"{VASP_OUT_DIR}/WAVEDERF.Si.gz") as file: + with zopen(f"{VASP_OUT_DIR}/WAVEDERF.Si.gz", mode="rt", encoding="utf-8") as file: first_line = [int(a) for a in file.readline().split()] assert wder.nkpoints == first_line[1] assert wder.nbands == first_line[2] From 8c7a4c2e2b1e0453fbdbde57eed46d4aa6220919 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Tue, 10 Dec 2024 17:52:16 -0800 Subject: [PATCH 02/12] pre-commit autoupdate (#4213) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * pre-commit autoupdate updates: - [github.com/astral-sh/ruff-pre-commit: v0.7.2 → v0.8.1](https://github.com/astral-sh/ruff-pre-commit/compare/v0.7.2...v0.8.1) - [github.com/MarcoGorelli/cython-lint: v0.16.2 → v0.16.6](https://github.com/MarcoGorelli/cython-lint/compare/v0.16.2...v0.16.6) - [github.com/igorshubovych/markdownlint-cli: v0.42.0 → v0.43.0](https://github.com/igorshubovych/markdownlint-cli/compare/v0.42.0...v0.43.0) - [github.com/kynan/nbstripout: 0.8.0 → 0.8.1](https://github.com/kynan/nbstripout/compare/0.8.0...0.8.1) - [github.com/RobertCraigie/pyright-python: v1.1.387 → v1.1.389](https://github.com/RobertCraigie/pyright-python/compare/v1.1.387...v1.1.389) * pre-commit auto-fixes * Convert to f-syntax to satisfy ruff Signed-off-by: Matthew Horton --------- Signed-off-by: Matthew Horton Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> Co-authored-by: Matthew Horton --- .pre-commit-config.yaml | 10 +++++----- src/pymatgen/io/aims/inputs.py | 2 +- src/pymatgen/io/cp2k/sets.py | 1 - src/pymatgen/io/lobster/outputs.py | 7 ++----- src/pymatgen/io/vasp/outputs.py | 2 +- src/pymatgen/symmetry/groups.py | 2 +- tests/io/lobster/test_outputs.py | 21 ++++++++++++--------- 7 files changed, 22 insertions(+), 23 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 60a633321ac..d57d1fa0207 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -8,7 +8,7 @@ ci: repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.7.2 + rev: v0.8.1 hooks: - id: ruff args: [--fix, --unsafe-fixes] @@ -36,7 +36,7 @@ repos: exclude: src/pymatgen/analysis/aflow_prototypes.json - repo: https://github.com/MarcoGorelli/cython-lint - rev: v0.16.2 + rev: v0.16.6 hooks: - id: cython-lint args: [--no-pycodestyle] @@ -48,7 +48,7 @@ repos: - id: blacken-docs - repo: https://github.com/igorshubovych/markdownlint-cli - rev: v0.42.0 + rev: v0.43.0 hooks: - id: markdownlint # MD013: line too long @@ -59,12 +59,12 @@ repos: args: [--disable, MD013, MD024, MD025, MD033, MD041, "--"] - repo: https://github.com/kynan/nbstripout - rev: 0.8.0 + rev: 0.8.1 hooks: - id: nbstripout args: [--drop-empty-cells, --keep-output] - repo: https://github.com/RobertCraigie/pyright-python - rev: v1.1.387 + rev: v1.1.389 hooks: - id: pyright diff --git a/src/pymatgen/io/aims/inputs.py b/src/pymatgen/io/aims/inputs.py index 0997601f750..3c1aa389540 100644 --- a/src/pymatgen/io/aims/inputs.py +++ b/src/pymatgen/io/aims/inputs.py @@ -590,7 +590,7 @@ def get_content( width = parameters["smearing"][1] if name == "methfessel-paxton": order = parameters["smearing"][2] - order = " %d" % order + order = f" {order:d}" else: order = "" diff --git a/src/pymatgen/io/cp2k/sets.py b/src/pymatgen/io/cp2k/sets.py index ae6db0edd9a..fc4e7770a7b 100644 --- a/src/pymatgen/io/cp2k/sets.py +++ b/src/pymatgen/io/cp2k/sets.py @@ -71,7 +71,6 @@ from pymatgen.io.vasp.inputs import KpointsSupportedModes if TYPE_CHECKING: - from pathlib import Path from typing import Literal __author__ = "Nicholas Winner" diff --git a/src/pymatgen/io/lobster/outputs.py b/src/pymatgen/io/lobster/outputs.py index 78946c1fc51..421816c5bd6 100644 --- a/src/pymatgen/io/lobster/outputs.py +++ b/src/pymatgen/io/lobster/outputs.py @@ -444,11 +444,8 @@ def __init__( for line in lines: if ( ("_" not in line.split()[1] and version != "5.1.0") - or "_" not in line.split()[1] - and version == "5.1.0" - or (line.split()[1].count("_") == 1) - and version == "5.1.0" - and self.is_lcfo + or ("_" not in line.split()[1] and version == "5.1.0") + or ((line.split()[1].count("_") == 1) and version == "5.1.0" and self.is_lcfo) ): data_without_orbitals.append(line) elif line.split()[1].count("_") >= 2 and version == "5.1.0": diff --git a/src/pymatgen/io/vasp/outputs.py b/src/pymatgen/io/vasp/outputs.py index ba23c9c9f11..0e3182cf68b 100644 --- a/src/pymatgen/io/vasp/outputs.py +++ b/src/pymatgen/io/vasp/outputs.py @@ -4517,7 +4517,7 @@ def __init__( else: preamble.append(line) - elif line == "" or "Direct configuration=" in line and len(coords_str) > 0: + elif line == "" or ("Direct configuration=" in line and len(coords_str) > 0): parse_poscar = True restart_preamble = False else: diff --git a/src/pymatgen/symmetry/groups.py b/src/pymatgen/symmetry/groups.py index 4e91b4cbb98..62a0c8df28a 100644 --- a/src/pymatgen/symmetry/groups.py +++ b/src/pymatgen/symmetry/groups.py @@ -29,7 +29,7 @@ from pymatgen.core.lattice import Lattice # Don't import at runtime to avoid circular import - from pymatgen.core.operations import SymmOp # noqa: TCH004 + from pymatgen.core.operations import SymmOp # noqa: TC004 CrystalSystem = Literal[ "cubic", diff --git a/tests/io/lobster/test_outputs.py b/tests/io/lobster/test_outputs.py index 4dac8c4a01b..f46f69b8af1 100644 --- a/tests/io/lobster/test_outputs.py +++ b/tests/io/lobster/test_outputs.py @@ -1670,16 +1670,19 @@ def test_has_good_quality_check_occupied_bands_patched(self): ) # Assert for expected results if ( - actual_deviation == 0.05 - and number_occ_bands_spin_up <= 7 - and number_occ_bands_spin_down <= 7 - and spin is Spin.up - or actual_deviation == 0.05 - and spin is Spin.down + ( + actual_deviation == 0.05 + and number_occ_bands_spin_up <= 7 + and number_occ_bands_spin_down <= 7 + and spin is Spin.up + ) + or (actual_deviation == 0.05 and spin is Spin.down) or actual_deviation == 0.1 - or actual_deviation in [0.2, 0.5, 1.0] - and number_occ_bands_spin_up == 0 - and number_occ_bands_spin_down == 0 + or ( + actual_deviation in [0.2, 0.5, 1.0] + and number_occ_bands_spin_up == 0 + and number_occ_bands_spin_down == 0 + ) ): assert result else: From 6abf9137fdb6e3d7b7ca99c90f5a81cd3b1a2806 Mon Sep 17 00:00:00 2001 From: "Haoyu (Daniel) YANG" Date: Wed, 11 Dec 2024 09:52:44 +0800 Subject: [PATCH 03/12] Increase warnings `stacklevel` to 2, un-ignore RUFF `B028` (#4217) * increase warning stacklevel * unignore b028 ruff rule * fix some * fix more in apps, command_line and ext * remove unnecessary explicit UserWarning * replace explicit stacklevel 1 * fix io * fix entries * fix electronic_structure * avoid manual deprecation warning * fix core * fix analysis * futurewarning for legacy mprester --- dev_scripts/potcar_scrambler.py | 2 +- pyproject.toml | 1 - src/pymatgen/alchemy/materials.py | 2 +- src/pymatgen/analysis/bond_dissociation.py | 10 +++-- .../coordination_geometry_finder.py | 5 +-- src/pymatgen/analysis/chempot_diagram.py | 2 +- src/pymatgen/analysis/elasticity/elastic.py | 11 +++-- src/pymatgen/analysis/ewald.py | 2 +- src/pymatgen/analysis/gb/grain.py | 1 + src/pymatgen/analysis/graphs.py | 11 +++-- src/pymatgen/analysis/interface_reactions.py | 11 +++-- src/pymatgen/analysis/local_env.py | 6 ++- src/pymatgen/analysis/magnetism/analyzer.py | 16 ++++---- src/pymatgen/analysis/magnetism/jahnteller.py | 6 +-- src/pymatgen/analysis/phase_diagram.py | 12 +++--- src/pymatgen/analysis/piezo.py | 2 +- src/pymatgen/analysis/piezo_sensitivity.py | 4 +- src/pymatgen/analysis/pourbaix_diagram.py | 4 +- src/pymatgen/analysis/structure_analyzer.py | 5 ++- .../structure_prediction/dopant_predictor.py | 4 +- .../structure_prediction/volume_predictor.py | 6 +-- src/pymatgen/analysis/surface_analysis.py | 4 +- src/pymatgen/analysis/wulff.py | 2 +- src/pymatgen/analysis/xas/spectrum.py | 2 - src/pymatgen/analysis/xps.py | 2 +- src/pymatgen/apps/borg/hive.py | 6 ++- src/pymatgen/command_line/bader_caller.py | 17 ++++---- src/pymatgen/command_line/chargemol_caller.py | 6 +-- src/pymatgen/command_line/critic2_caller.py | 18 ++++---- src/pymatgen/core/__init__.py | 4 +- src/pymatgen/core/bonds.py | 5 ++- src/pymatgen/core/composition.py | 12 ++++-- src/pymatgen/core/interface.py | 11 ++--- src/pymatgen/core/lattice.py | 2 +- src/pymatgen/core/operations.py | 2 +- src/pymatgen/core/periodic_table.py | 19 +++++---- src/pymatgen/core/structure.py | 11 +++-- src/pymatgen/core/surface.py | 13 +++--- src/pymatgen/core/tensors.py | 6 +-- src/pymatgen/core/trajectory.py | 6 ++- .../electronic_structure/bandstructure.py | 6 ++- .../electronic_structure/boltztrap2.py | 15 ++++--- src/pymatgen/electronic_structure/cohp.py | 6 ++- src/pymatgen/electronic_structure/dos.py | 4 +- src/pymatgen/electronic_structure/plotter.py | 14 ++++--- src/pymatgen/entries/compatibility.py | 33 +++++++++------ src/pymatgen/entries/computed_entries.py | 3 +- src/pymatgen/entries/correction_calculator.py | 16 ++++++-- src/pymatgen/entries/mixing_scheme.py | 28 ++++++++----- src/pymatgen/ext/cod.py | 3 +- src/pymatgen/ext/matproj_legacy.py | 30 ++++++++------ src/pymatgen/io/abinit/netcdf.py | 5 ++- src/pymatgen/io/aims/inputs.py | 1 + src/pymatgen/io/aims/parsers.py | 3 +- src/pymatgen/io/aims/sets/base.py | 4 +- src/pymatgen/io/ase.py | 3 +- src/pymatgen/io/babel.py | 6 ++- src/pymatgen/io/cif.py | 33 ++++++++------- src/pymatgen/io/common.py | 7 +++- src/pymatgen/io/cp2k/outputs.py | 17 ++++---- src/pymatgen/io/cp2k/sets.py | 25 +++++++---- src/pymatgen/io/feff/inputs.py | 5 ++- src/pymatgen/io/feff/sets.py | 2 +- src/pymatgen/io/gaussian.py | 13 ++++-- src/pymatgen/io/icet.py | 5 ++- src/pymatgen/io/lammps/data.py | 5 ++- src/pymatgen/io/lammps/inputs.py | 13 ++++-- src/pymatgen/io/lobster/inputs.py | 11 +++-- src/pymatgen/io/lobster/outputs.py | 18 ++++++-- src/pymatgen/io/multiwfn.py | 25 ++++++++--- src/pymatgen/io/nwchem.py | 5 ++- src/pymatgen/io/openff.py | 3 +- src/pymatgen/io/qchem/outputs.py | 3 +- src/pymatgen/io/qchem/sets.py | 8 ++-- src/pymatgen/io/shengbte.py | 5 ++- src/pymatgen/io/vasp/inputs.py | 39 +++++++++++++++--- src/pymatgen/io/vasp/outputs.py | 40 +++++++++++++----- src/pymatgen/io/vasp/sets.py | 41 ++++++++++++++----- src/pymatgen/symmetry/analyzer.py | 5 ++- src/pymatgen/symmetry/bandstructure.py | 3 +- src/pymatgen/symmetry/groups.py | 12 ++++-- src/pymatgen/symmetry/kpath.py | 35 ++++++++++------ .../advanced_transformations.py | 11 +++-- tests/entries/test_compatibility.py | 2 +- 84 files changed, 547 insertions(+), 295 deletions(-) diff --git a/dev_scripts/potcar_scrambler.py b/dev_scripts/potcar_scrambler.py index 7e1aa2579f9..4a5bb292a82 100644 --- a/dev_scripts/potcar_scrambler.py +++ b/dev_scripts/potcar_scrambler.py @@ -164,7 +164,7 @@ def generate_fake_potcar_libraries() -> None: zpath(f"{func_dir}/{psp_name}/POTCAR"), ] if not any(map(os.path.isfile, paths_to_try)): - warnings.warn(f"Could not find {psp_name} in {paths_to_try}") + warnings.warn(f"Could not find {psp_name} in {paths_to_try}", stacklevel=2) for potcar_path in paths_to_try: if os.path.isfile(potcar_path): os.makedirs(rebase_dir, exist_ok=True) diff --git a/pyproject.toml b/pyproject.toml index 3e2da3bf703..7fba2e4c661 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -193,7 +193,6 @@ ignore = [ # Single rules "B023", # Function definition does not bind loop variable - "B028", # No explicit stacklevel keyword argument found "B904", # Within an except clause, raise exceptions with ... "C408", # unnecessary-collection-call "D105", # Missing docstring in magic method diff --git a/src/pymatgen/alchemy/materials.py b/src/pymatgen/alchemy/materials.py index e1abf62e988..5eb112d8a82 100644 --- a/src/pymatgen/alchemy/materials.py +++ b/src/pymatgen/alchemy/materials.py @@ -362,7 +362,7 @@ def to_snl(self, authors: list[str], **kwargs) -> StructureNL: StructureNL: The generated StructureNL object. """ if self.other_parameters: - warn("Data in TransformedStructure.other_parameters discarded during type conversion to SNL") + warn("Data in TransformedStructure.other_parameters discarded during type conversion to SNL", stacklevel=2) history = [] for hist in self.history: snl_metadata = hist.pop("_snl", {}) diff --git a/src/pymatgen/analysis/bond_dissociation.py b/src/pymatgen/analysis/bond_dissociation.py index 2a89d672e0d..552b434a8e6 100644 --- a/src/pymatgen/analysis/bond_dissociation.py +++ b/src/pymatgen/analysis/bond_dissociation.py @@ -107,7 +107,8 @@ def __init__( if multibreak: warnings.warn( "Breaking pairs of ring bonds. WARNING: Structure changes much more likely, meaning dissociation values" - " are less reliable! This is a bad idea!" + " are less reliable! This is a bad idea!", + stacklevel=2, ) self.bond_pairs = [] for ii, bond in enumerate(self.ring_bonds, start=1): @@ -164,7 +165,8 @@ def fragment_and_process(self, bonds): warnings.warn( f"Missing ring opening fragment resulting from the breakage of {specie[bonds[0][0]]} " f"{specie[bonds[0][1]]} bond {bonds[0][0]} {bonds[0][1]} which would yield a " - f"molecule with this SMILES string: {smiles}" + f"molecule with this SMILES string: {smiles}", + stacklevel=2, ) elif len(good_entries) == 1: # If we have only one good entry, format it and add it to the list that will eventually return @@ -212,14 +214,14 @@ def fragment_and_process(self, bonds): smiles = pb_mol.write("smi").split()[0] for charge in self.expected_charges: if charge not in frag1_charges_found: - warnings.warn(f"Missing {charge=} for fragment {smiles}") + warnings.warn(f"Missing {charge=} for fragment {smiles}", stacklevel=2) if len(frag2_charges_found) < len(self.expected_charges): bb = BabelMolAdaptor(fragments[1].molecule) pb_mol = bb.pybel_mol smiles = pb_mol.write("smi").split()[0] for charge in self.expected_charges: if charge not in frag2_charges_found: - warnings.warn(f"Missing {charge=} for fragment {smiles}") + warnings.warn(f"Missing {charge=} for fragment {smiles}", stacklevel=2) # Now we attempt to pair fragments with the right total charge, starting with only fragments with no # structural change: for frag1 in frag1_entries[0]: # 0 -> no structural change diff --git a/src/pymatgen/analysis/chemenv/coordination_environments/coordination_geometry_finder.py b/src/pymatgen/analysis/chemenv/coordination_environments/coordination_geometry_finder.py index 1a33f3608c0..562b7b8c987 100644 --- a/src/pymatgen/analysis/chemenv/coordination_environments/coordination_geometry_finder.py +++ b/src/pymatgen/analysis/chemenv/coordination_environments/coordination_geometry_finder.py @@ -2051,10 +2051,7 @@ def coordination_geometry_symmetry_measures_fallback_random( The symmetry measures for the given coordination geometry for each permutation investigated. """ if "NRANDOM" in kwargs: - warnings.warn( - "NRANDOM is deprecated, use n_random instead", - category=DeprecationWarning, - ) + warnings.warn("NRANDOM is deprecated, use n_random instead", category=DeprecationWarning, stacklevel=2) n_random = kwargs.pop("NRANDOM") permutations_symmetry_measures = [None] * n_random permutations = [] diff --git a/src/pymatgen/analysis/chempot_diagram.py b/src/pymatgen/analysis/chempot_diagram.py index 6d13ea30e7f..802c519e580 100644 --- a/src/pymatgen/analysis/chempot_diagram.py +++ b/src/pymatgen/analysis/chempot_diagram.py @@ -391,7 +391,7 @@ def _get_3d_plot( if formulas_to_draw: for formula in formulas_to_draw: if formula not in domain_simplexes: - warnings.warn(f"Specified formula to draw, {formula}, not found!") + warnings.warn(f"Specified formula to draw, {formula}, not found!", stacklevel=2) if draw_formula_lines: data.extend(self._get_3d_formula_lines(draw_domains, formula_colors)) diff --git a/src/pymatgen/analysis/elasticity/elastic.py b/src/pymatgen/analysis/elasticity/elastic.py index 4a236e24d3f..c4826a9dfdc 100644 --- a/src/pymatgen/analysis/elasticity/elastic.py +++ b/src/pymatgen/analysis/elasticity/elastic.py @@ -63,7 +63,7 @@ def __new__(cls, input_array, check_rank=None, tol: float = 1e-4) -> Self: if obj.rank % 2 != 0: raise ValueError("ElasticTensor must have even rank") if not obj.is_voigt_symmetric(tol): - warnings.warn("Input elastic tensor does not satisfy standard Voigt symmetries") + warnings.warn("Input elastic tensor does not satisfy standard Voigt symmetries", stacklevel=2) return obj.view(cls) @property @@ -476,7 +476,8 @@ def from_pseudoinverse(cls, strains, stresses) -> Self: # convert the stress/strain to Nx6 arrays of voigt notation warnings.warn( "Pseudo-inverse fitting of Strain/Stress lists may yield " - "questionable results from vasp data, use with caution." + "questionable results from vasp data, use with caution.", + stacklevel=2, ) stresses = np.array([Stress(stress).voigt for stress in stresses]) with warnings.catch_warnings(): @@ -505,7 +506,9 @@ def from_independent_strains(cls, strains, stresses, eq_stress=None, vasp=False, if not set(strain_states) <= set(ss_dict): raise ValueError(f"Missing independent strain states: {set(strain_states) - set(ss_dict)}") if len(set(ss_dict) - set(strain_states)) > 0: - warnings.warn("Extra strain states in strain-stress pairs are neglected in independent strain fitting") + warnings.warn( + "Extra strain states in strain-stress pairs are neglected in independent strain fitting", stacklevel=2 + ) c_ij = np.zeros((6, 6)) for ii in range(6): strains = ss_dict[strain_states[ii]]["strains"] @@ -916,7 +919,7 @@ def find_eq_stress(strains, stresses, tol: float = 1e-10): ) eq_stress = eq_stress[0] else: - warnings.warn("No eq state found, returning zero voigt stress") + warnings.warn("No eq state found, returning zero voigt stress", stacklevel=2) eq_stress = Stress(np.zeros((3, 3))) return eq_stress diff --git a/src/pymatgen/analysis/ewald.py b/src/pymatgen/analysis/ewald.py index 4f858258b3a..02e0de453e6 100644 --- a/src/pymatgen/analysis/ewald.py +++ b/src/pymatgen/analysis/ewald.py @@ -288,7 +288,7 @@ def get_site_energy(self, site_index): self._initialized = True if self._charged: - warn("Per atom energies for charged structures not supported in EwaldSummation") + warn("Per atom energies for charged structures not supported in EwaldSummation", stacklevel=2) return np.sum(self._recip[:, site_index]) + np.sum(self._real[:, site_index]) + self._point[site_index] def _calc_ewald_terms(self): diff --git a/src/pymatgen/analysis/gb/grain.py b/src/pymatgen/analysis/gb/grain.py index 9a5b8c4166a..836978ad1f7 100644 --- a/src/pymatgen/analysis/gb/grain.py +++ b/src/pymatgen/analysis/gb/grain.py @@ -8,4 +8,5 @@ "Grain boundary analysis has been moved to pymatgen.core.interface." "This stub is retained for backwards compatibility and will be removed Dec 31 2024.", DeprecationWarning, + stacklevel=2, ) diff --git a/src/pymatgen/analysis/graphs.py b/src/pymatgen/analysis/graphs.py index 7cfd8217cad..b79fc427f8c 100644 --- a/src/pymatgen/analysis/graphs.py +++ b/src/pymatgen/analysis/graphs.py @@ -383,7 +383,7 @@ def add_edge( # edges if appropriate if to_jimage is None: # assume we want the closest site - warnings.warn("Please specify to_jimage to be unambiguous, trying to automatically detect.") + warnings.warn("Please specify to_jimage to be unambiguous, trying to automatically detect.", stacklevel=2) dist, to_jimage = self.structure[from_index].distance_and_image(self.structure[to_index]) if dist == 0: # this will happen when from_index == to_index, @@ -417,7 +417,7 @@ def add_edge( # this is a convention to avoid duplicate hops if to_index == from_index: if to_jimage == (0, 0, 0): - warnings.warn("Tried to create a bond to itself, this doesn't make sense so was ignored.") + warnings.warn("Tried to create a bond to itself, this doesn't make sense so was ignored.", stacklevel=2) return # ensure that the first non-zero jimage index is positive @@ -439,7 +439,8 @@ def add_edge( if warn_duplicates: warnings.warn( "Trying to add an edge that already exists from " - f"site {from_index} to site {to_index} in {to_jimage}." + f"site {from_index} to site {to_index} in {to_jimage}.", + stacklevel=2, ) return @@ -1826,7 +1827,9 @@ def add_edge( # between two sites existing_edge_data = self.graph.get_edge_data(from_index, to_index) if existing_edge_data and warn_duplicates: - warnings.warn(f"Trying to add an edge that already exists from site {from_index} to site {to_index}.") + warnings.warn( + f"Trying to add an edge that already exists from site {from_index} to site {to_index}.", stacklevel=2 + ) return # generic container for additional edge properties, diff --git a/src/pymatgen/analysis/interface_reactions.py b/src/pymatgen/analysis/interface_reactions.py index 1f42b04fded..c8cf44ebe52 100644 --- a/src/pymatgen/analysis/interface_reactions.py +++ b/src/pymatgen/analysis/interface_reactions.py @@ -484,8 +484,10 @@ def _get_entry_energy(pd: PhaseDiagram, composition: Composition): if not candidate: warnings.warn( - f"The reactant {composition.reduced_formula} has no matching entry with negative formation" - " energy, instead convex hull energy for this composition will be used for reaction energy calculation." + f"The reactant {composition.reduced_formula} has no matching entry " + "with negative formation energy, instead convex hull energy for " + "this composition will be used for reaction energy calculation.", + stacklevel=2, ) return pd.get_hull_energy(composition) min_entry_energy = min(candidate) @@ -545,8 +547,9 @@ def get_chempot_correction(cls, element: str, temp: float, pres: float): # code The correction of chemical potential in eV/atom of the gas phase at given temperature and pressure. """ - if element not in ["O", "N", "Cl", "F", "H"]: - warnings.warn(f"{element=} not one of valid options: ['O', 'N', 'Cl', 'F', 'H']") + valid_elements = {"O", "N", "Cl", "F", "H"} + if element not in valid_elements: + warnings.warn(f"{element=} not one of valid options: {valid_elements}", stacklevel=2) return 0 std_temp = 298.15 diff --git a/src/pymatgen/analysis/local_env.py b/src/pymatgen/analysis/local_env.py index ce4b370b747..a8100cbced4 100644 --- a/src/pymatgen/analysis/local_env.py +++ b/src/pymatgen/analysis/local_env.py @@ -4025,7 +4025,8 @@ def get_nn_data(self, structure: Structure, n: int, length=None): warnings.warn( "CrystalNN: cannot locate an appropriate radius, " "covalent or atomic radii will be used, this can lead " - "to non-optimal results." + "to non-optimal results.", + stacklevel=2, ) diameter = _get_default_radius(structure[n]) + _get_default_radius(entry["site"]) @@ -4231,7 +4232,8 @@ def _get_radius(site): else: warnings.warn( "No oxidation states specified on sites! For better results, set " - "the site oxidation states in the structure." + "the site oxidation states in the structure.", + stacklevel=2, ) return 0 diff --git a/src/pymatgen/analysis/magnetism/analyzer.py b/src/pymatgen/analysis/magnetism/analyzer.py index 0795101a248..a2e8eee5ee7 100644 --- a/src/pymatgen/analysis/magnetism/analyzer.py +++ b/src/pymatgen/analysis/magnetism/analyzer.py @@ -43,7 +43,7 @@ try: DEFAULT_MAGMOMS = loadfn(f"{MODULE_DIR}/default_magmoms.yaml") except (FileNotFoundError, MarkedYAMLError): - warnings.warn("Could not load default_magmoms.yaml, falling back to VASPIncarBase.yaml") + warnings.warn("Could not load default_magmoms.yaml, falling back to VASPIncarBase.yaml", stacklevel=2) DEFAULT_MAGMOMS = loadfn(f"{MODULE_DIR}/../../io/vasp/VASPIncarBase.yaml")["INCAR"]["MAGMOM"] @@ -159,7 +159,7 @@ def __init__( try: structure = trans.apply_transformation(structure) except ValueError: - warnings.warn(f"Could not assign valences for {structure.reduced_formula}") + warnings.warn(f"Could not assign valences for {structure.reduced_formula}", stacklevel=2) # Check if structure has magnetic moments # on site properties or species spin properties, @@ -186,7 +186,8 @@ def __init__( if None in structure.site_properties["magmom"]: warnings.warn( "Be careful with mixing types in your magmom site properties. " - "Any 'None' magmoms have been replaced with zero." + "Any 'None' magmoms have been replaced with zero.", + stacklevel=2, ) magmoms = [m or 0 for m in structure.site_properties["magmom"]] elif has_spin: @@ -207,7 +208,8 @@ def __init__( "This class is not designed to be used with " "non-collinear structures. If your structure is " "only slightly non-collinear (e.g. canted) may still " - "give useful results, but use with caution." + "give useful results, but use with caution.", + stacklevel=2, ) # this is for collinear structures only, make sure magmoms are all floats @@ -313,8 +315,8 @@ def _round_magmoms(magmoms: ArrayLike, round_magmoms_mode: float) -> np.ndarray: except Exception as exc: # TODO: typically a singular matrix warning, investigate this - warnings.warn("Failed to round magmoms intelligently, falling back to simple rounding.") - warnings.warn(str(exc)) + warnings.warn("Failed to round magmoms intelligently, falling back to simple rounding.", stacklevel=2) + warnings.warn(str(exc), stacklevel=2) # and finally round roughly to the number of significant figures in our kde width n_decimals = len(str(round_magmoms_mode).split(".")[1]) + 1 @@ -483,7 +485,7 @@ def ordering(self) -> Ordering: (in which case a warning is issued). """ if not self.is_collinear: - warnings.warn("Detecting ordering in non-collinear structures not yet implemented.") + warnings.warn("Detecting ordering in non-collinear structures not yet implemented.", stacklevel=2) return Ordering.Unknown if "magmom" not in self.structure.site_properties: diff --git a/src/pymatgen/analysis/magnetism/jahnteller.py b/src/pymatgen/analysis/magnetism/jahnteller.py index 650ac15782b..4fa72c7dbbc 100644 --- a/src/pymatgen/analysis/magnetism/jahnteller.py +++ b/src/pymatgen/analysis/magnetism/jahnteller.py @@ -286,7 +286,7 @@ def is_jahn_teller_active( ) active = analysis["active"] except Exception as exc: - warnings.warn(f"Error analyzing {structure.reduced_formula}: {exc}") + warnings.warn(f"Error analyzing {structure.reduced_formula}: {exc}", stacklevel=2) return active @@ -330,7 +330,7 @@ def tag_structure( structure.add_site_property("possible_jt_active", jt_sites) return structure except Exception as exc: - warnings.warn(f"Error analyzing {structure.reduced_formula}: {exc}") + warnings.warn(f"Error analyzing {structure.reduced_formula}: {exc}", stacklevel=2) return structure @staticmethod @@ -380,7 +380,7 @@ def get_magnitude_of_effect_from_species(self, species: str | Species, spin_stat spin_config = self.spin_configs[motif][d_electrons][spin_state] magnitude = JahnTellerAnalyzer.get_magnitude_of_effect_from_spin_config(motif, spin_config) else: - warnings.warn("No data for this species.") + warnings.warn("No data for this species.", stacklevel=2) return magnitude diff --git a/src/pymatgen/analysis/phase_diagram.py b/src/pymatgen/analysis/phase_diagram.py index 3992aff63c6..8e07887b11b 100644 --- a/src/pymatgen/analysis/phase_diagram.py +++ b/src/pymatgen/analysis/phase_diagram.py @@ -759,7 +759,7 @@ def get_decomp_and_e_above_hull( if on_error == "raise": raise ValueError(f"Unable to get decomposition for {entry}") from exc if on_error == "warn": - warnings.warn(f"Unable to get decomposition for {entry}, encountered {exc}") + warnings.warn(f"Unable to get decomposition for {entry}, encountered {exc}", stacklevel=2) return None, None e_above_hull = entry.energy_per_atom - hull_energy @@ -770,7 +770,7 @@ def get_decomp_and_e_above_hull( if on_error == "raise": raise ValueError(msg) if on_error == "warn": - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) return None, None # 'ignore' and 'warn' case def get_e_above_hull(self, entry: PDEntry, **kwargs: Any) -> float | None: @@ -906,7 +906,8 @@ def get_decomp_and_phase_separation_energy( if len(competing_entries) > space_limit and not stable_only: warnings.warn( f"There are {len(competing_entries)} competing entries " - f"for {entry.composition} - Calculating inner hull to discard additional unstable entries" + f"for {entry.composition} - Calculating inner hull to discard additional unstable entries", + stacklevel=2, ) reduced_space = competing_entries - {*self._get_stable_entries_in_space(entry_elems)} | { @@ -922,7 +923,8 @@ def get_decomp_and_phase_separation_energy( if len(competing_entries) > space_limit: warnings.warn( f"There are {len(competing_entries)} competing entries " - f"for {entry.composition} - Using SLSQP to find decomposition likely to be slow" + f"for {entry.composition} - Using SLSQP to find decomposition likely to be slow", + stacklevel=2, ) decomp = _get_slsqp_decomp(entry.composition, competing_entries, tols, maxiter) @@ -1829,7 +1831,7 @@ def get_decomposition(self, comp: Composition) -> dict[PDEntry, float]: return pd.get_decomposition(comp) except ValueError as exc: # NOTE warn when stitching across pds is being used - warnings.warn(f"{exc} Using SLSQP to find decomposition") + warnings.warn(f"{exc} Using SLSQP to find decomposition", stacklevel=2) competing_entries = self._get_stable_entries_in_space(frozenset(comp.elements)) return _get_slsqp_decomp(comp, competing_entries) diff --git a/src/pymatgen/analysis/piezo.py b/src/pymatgen/analysis/piezo.py index 448b4c48a75..ed9afccc41a 100644 --- a/src/pymatgen/analysis/piezo.py +++ b/src/pymatgen/analysis/piezo.py @@ -39,7 +39,7 @@ def __new__(cls, input_array: ArrayLike, tol: float = 1e-3) -> Self: """ obj = super().__new__(cls, input_array, check_rank=3) if not np.allclose(obj, np.transpose(obj, (0, 2, 1)), atol=tol, rtol=0): - warnings.warn("Input piezo tensor does not satisfy standard symmetries") + warnings.warn("Input piezo tensor does not satisfy standard symmetries", stacklevel=2) return obj.view(cls) @classmethod diff --git a/src/pymatgen/analysis/piezo_sensitivity.py b/src/pymatgen/analysis/piezo_sensitivity.py index ce640af48a0..c1410bd08d5 100644 --- a/src/pymatgen/analysis/piezo_sensitivity.py +++ b/src/pymatgen/analysis/piezo_sensitivity.py @@ -49,7 +49,7 @@ def __init__(self, structure: Structure, bec, pointops, tol: float = 1e-3): self.pointops = pointops self.BEC_operations = None if np.sum(self.bec) >= tol: - warnings.warn("Input born effective charge tensor does not satisfy charge neutrality") + warnings.warn("Input born effective charge tensor does not satisfy charge neutrality", stacklevel=2) def get_BEC_operations(self, eigtol=1e-5, opstol=1e-3): """Get the symmetry operations which maps the tensors @@ -182,7 +182,7 @@ def __init__(self, structure: Structure, ist, pointops, tol: float = 1e-3): obj = self.ist if not np.allclose(obj, np.transpose(obj, (0, 1, 3, 2)), atol=tol, rtol=0): - warnings.warn("Input internal strain tensor does not satisfy standard symmetries") + warnings.warn("Input internal strain tensor does not satisfy standard symmetries", stacklevel=2) def get_IST_operations(self, opstol=1e-3) -> list[list[list]]: """Get the symmetry operations which maps the tensors diff --git a/src/pymatgen/analysis/pourbaix_diagram.py b/src/pymatgen/analysis/pourbaix_diagram.py index 50f2e989a7d..e5e9c003e79 100644 --- a/src/pymatgen/analysis/pourbaix_diagram.py +++ b/src/pymatgen/analysis/pourbaix_diagram.py @@ -664,7 +664,9 @@ def _generate_multielement_entries(self, entries, nproc=None): processed_entries = [] total = sum(comb(len(entries), idx + 1) for idx in range(n_elems)) if total > 1e6: - warnings.warn(f"Your Pourbaix diagram includes {total} entries and may take a long time to generate.") + warnings.warn( + f"Your Pourbaix diagram includes {total} entries and may take a long time to generate.", stacklevel=2 + ) # Parallel processing of multi-entry generation if nproc is not None: diff --git a/src/pymatgen/analysis/structure_analyzer.py b/src/pymatgen/analysis/structure_analyzer.py index 14c67ebaae4..d73ea0542f0 100644 --- a/src/pymatgen/analysis/structure_analyzer.py +++ b/src/pymatgen/analysis/structure_analyzer.py @@ -296,7 +296,10 @@ def connectivity_array(self): connectivity[atom_j, atom_i, image_i] = val if -10.101 in vts[v]: - warn("Found connectivity with infinite vertex. Cutoff is too low, and results may be incorrect") + warn( + "Found connectivity with infinite vertex. Cutoff is too low, and results may be incorrect", + stacklevel=2, + ) return connectivity @property diff --git a/src/pymatgen/analysis/structure_prediction/dopant_predictor.py b/src/pymatgen/analysis/structure_prediction/dopant_predictor.py index e21603e13ee..428e93fa8c6 100644 --- a/src/pymatgen/analysis/structure_prediction/dopant_predictor.py +++ b/src/pymatgen/analysis/structure_prediction/dopant_predictor.py @@ -101,7 +101,9 @@ def get_dopants_from_shannon_radii(bonded_structure, num_dopants=5, match_oxi_si try: species_radius = species.get_shannon_radius(cn_roman) except KeyError: - warnings.warn(f"Shannon radius not found for {species} with coordination number {cn}.\nSkipping...") + warnings.warn( + f"Shannon radius not found for {species} with coordination number {cn}.\nSkipping...", stacklevel=2 + ) continue if cn not in cn_to_radii_map: diff --git a/src/pymatgen/analysis/structure_prediction/volume_predictor.py b/src/pymatgen/analysis/structure_prediction/volume_predictor.py index 7674e75705a..c689c68666d 100644 --- a/src/pymatgen/analysis/structure_prediction/volume_predictor.py +++ b/src/pymatgen/analysis/structure_prediction/volume_predictor.py @@ -92,7 +92,7 @@ def predict(self, structure: Structure, ref_structure): return ref_structure.volume * (numerator / denominator) ** 3 except Exception: - warnings.warn("Exception occurred. Will attempt atomic radii.") + warnings.warn("Exception occurred. Will attempt atomic radii.", stacklevel=2) # If error occurs during use of ionic radii scheme, pass # and see if we can resolve it using atomic radii. @@ -180,10 +180,10 @@ def predict(self, structure: Structure, icsd_vol=False): if sp.atomic_radius: sub_sites.extend([site for site in structure if site.specie == sp]) else: - warnings.warn(f"VolumePredictor: no atomic radius data for {sp}") + warnings.warn(f"VolumePredictor: no atomic radius data for {sp}", stacklevel=2) if sp.symbol not in bond_params: - warnings.warn(f"VolumePredictor: bond parameters not found, used atomic radii for {sp}") + warnings.warn(f"VolumePredictor: bond parameters not found, used atomic radii for {sp}", stacklevel=2) else: r, k = bond_params[sp.symbol]["r"], bond_params[sp.symbol]["k"] bp_dict[sp] = float(r) + float(k) * std_x diff --git a/src/pymatgen/analysis/surface_analysis.py b/src/pymatgen/analysis/surface_analysis.py index 67cdf98d03c..a78abecbec6 100644 --- a/src/pymatgen/analysis/surface_analysis.py +++ b/src/pymatgen/analysis/surface_analysis.py @@ -193,7 +193,7 @@ def surface_energy(self, ucell_entry, ref_entries=None): if slab_clean_comp.reduced_composition != ucell_entry.composition.reduced_composition: list_els = [next(iter(entry.composition.as_dict())) for entry in ref_entries] if not any(el in list_els for el in ucell_entry.composition.as_dict()): - warnings.warn("Elemental references missing for the non-dopant species.") + warnings.warn("Elemental references missing for the non-dopant species.", stacklevel=2) gamma = (Symbol("E_surf") - Symbol("Ebulk")) / (2 * Symbol("A")) ucell_comp = ucell_entry.composition @@ -659,7 +659,7 @@ def get_surface_equilibrium(self, slab_entries, delu_dict=None): solution = linsolve(all_eqns, all_parameters) if not solution: - warnings.warn("No solution") + warnings.warn("No solution", stacklevel=2) return solution return {param: next(iter(solution))[idx] for idx, param in enumerate(all_parameters)} diff --git a/src/pymatgen/analysis/wulff.py b/src/pymatgen/analysis/wulff.py index 300cbaa01a6..d8e1219f83d 100644 --- a/src/pymatgen/analysis/wulff.py +++ b/src/pymatgen/analysis/wulff.py @@ -140,7 +140,7 @@ def __init__(self, lattice: Lattice, miller_list, e_surf_list, symprec=1e-5): symprec (float): for reciprocal lattice operation, default is 1e-5. """ if any(se < 0 for se in e_surf_list): - warnings.warn("Unphysical (negative) surface energy detected.") + warnings.warn("Unphysical (negative) surface energy detected.", stacklevel=2) self.color_ind = list(range(len(miller_list))) diff --git a/src/pymatgen/analysis/xas/spectrum.py b/src/pymatgen/analysis/xas/spectrum.py index a1dac4b2224..183af562817 100644 --- a/src/pymatgen/analysis/xas/spectrum.py +++ b/src/pymatgen/analysis/xas/spectrum.py @@ -85,7 +85,6 @@ def __init__( if len(self.y[neg_intens_mask]) / len(self.y) > 0.05: warnings.warn( "Double check the intensities. More than 5% of them are negative.", - UserWarning, stacklevel=2, ) self.zero_negative_intensity = zero_negative_intensity @@ -216,7 +215,6 @@ def stitch(self, other: XAS, num_samples: int = 500, mode: Literal["XAFS", "L23" if abs(mu[idx] - mu[idx - 1]) / (mu[idx - 1]) > 0.1: warnings.warn( "There might exist a jump at the L2 and L3-edge junction.", - UserWarning, stacklevel=2, ) diff --git a/src/pymatgen/analysis/xps.py b/src/pymatgen/analysis/xps.py index 1aca37a1eee..547d01c2ed6 100644 --- a/src/pymatgen/analysis/xps.py +++ b/src/pymatgen/analysis/xps.py @@ -96,5 +96,5 @@ def from_dos(cls, dos: CompleteDos) -> Self: if weight is not None: total += pdos.get_densities() * weight else: - warnings.warn(f"No cross-section for {el}{orb}") + warnings.warn(f"No cross-section for {el}{orb}", stacklevel=2) return XPS(-dos.energies, total / np.max(total)) diff --git a/src/pymatgen/apps/borg/hive.py b/src/pymatgen/apps/borg/hive.py index 2ea8a1be63b..fbd2a15645e 100644 --- a/src/pymatgen/apps/borg/hive.py +++ b/src/pymatgen/apps/borg/hive.py @@ -139,7 +139,7 @@ def assimilate(self, path: PathLike) -> ComputedStructureEntry | ComputedEntry | # Since multiple files are ambiguous, we will always read # the last one alphabetically. filepath = max(vasprun_files) - warnings.warn(f"{len(vasprun_files)} vasprun.xml.* found. {filepath} is being parsed.") + warnings.warn(f"{len(vasprun_files)} vasprun.xml.* found. {filepath} is being parsed.", stacklevel=2) try: vasp_run = Vasprun(filepath) @@ -252,7 +252,9 @@ def assimilate(self, path: PathLike) -> ComputedStructureEntry | ComputedEntry | # alphabetically for CONTCAR and OSZICAR. files_to_parse[filename] = files[0] if filename == "POSCAR" else files[-1] - warnings.warn(f"{len(files)} files found. {files_to_parse[filename]} is being parsed.") + warnings.warn( + f"{len(files)} files found. {files_to_parse[filename]} is being parsed.", stacklevel=2 + ) if not set(files_to_parse).issuperset({"INCAR", "POTCAR", "CONTCAR", "OSZICAR", "POSCAR"}): raise ValueError( diff --git a/src/pymatgen/command_line/bader_caller.py b/src/pymatgen/command_line/bader_caller.py index e2ddc5046a1..1304d3166e8 100644 --- a/src/pymatgen/command_line/bader_caller.py +++ b/src/pymatgen/command_line/bader_caller.py @@ -192,7 +192,8 @@ def temp_decompress(file: str | Path, target_dir: str = ".") -> str: if self.version < 1.0: warnings.warn( - "Your installed version of Bader is outdated, calculation of vacuum charge may be incorrect." + "Your installed version of Bader is outdated, calculation of vacuum charge may be incorrect.", + stacklevel=2, ) # Parse ACF.dat file @@ -460,7 +461,7 @@ def _get_filepath(filename): # kwarg to avoid this! paths.sort(reverse=True) if len(paths) > 1: - warnings.warn(f"Multiple files detected, using {paths[0]}") + warnings.warn(f"Multiple files detected, using {paths[0]}", stacklevel=2) filepath = paths[0] else: msg = f"Could not find {filename!r}" @@ -468,7 +469,7 @@ def _get_filepath(filename): msg += ", interpret Bader results with severe caution." elif filename == "POTCAR": msg += ", cannot calculate charge transfer." - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) return filepath chgcar_filename = _get_filepath("CHGCAR") @@ -515,7 +516,7 @@ def bader_analysis_from_path(path: str, suffix: str = "") -> dict[str, Any]: def _get_filepath(filename: str, msg: str = "") -> str | None: paths = glob(glob_pattern := f"{path}/{filename}{suffix}*") if len(paths) == 0: - warnings.warn(msg or f"no matches for {glob_pattern=}") + warnings.warn(msg or f"no matches for {glob_pattern=}", stacklevel=2) return None if len(paths) > 1: # using reverse=True because, if multiple files are present, @@ -523,7 +524,7 @@ def _get_filepath(filename: str, msg: str = "") -> str | None: # and this would give 'static' over 'relax2' over 'relax' # however, better to use 'suffix' kwarg to avoid this! paths.sort(reverse=True) - warnings.warn(f"Multiple files detected, using {os.path.basename(path)}") + warnings.warn(f"Multiple files detected, using {os.path.basename(path)}", stacklevel=2) return paths[0] chgcar_path = _get_filepath("CHGCAR", "Could not find CHGCAR!") @@ -533,17 +534,17 @@ def _get_filepath(filename: str, msg: str = "") -> str | None: aeccar0_path = _get_filepath("AECCAR0") if not aeccar0_path: - warnings.warn("Could not find AECCAR0, interpret Bader results with severe caution!") + warnings.warn("Could not find AECCAR0, interpret Bader results with severe caution!", stacklevel=2) aeccar0 = Chgcar.from_file(aeccar0_path) if aeccar0_path else None aeccar2_path = _get_filepath("AECCAR2") if not aeccar2_path: - warnings.warn("Could not find AECCAR2, interpret Bader results with severe caution!") + warnings.warn("Could not find AECCAR2, interpret Bader results with severe caution!", stacklevel=2) aeccar2 = Chgcar.from_file(aeccar2_path) if aeccar2_path else None potcar_path = _get_filepath("POTCAR") if not potcar_path: - warnings.warn("Could not find POTCAR, cannot calculate charge transfer.") + warnings.warn("Could not find POTCAR, cannot calculate charge transfer.", stacklevel=2) potcar = Potcar.from_file(potcar_path) if potcar_path else None return bader_analysis_from_objects(chgcar, potcar, aeccar0, aeccar2) diff --git a/src/pymatgen/command_line/chargemol_caller.py b/src/pymatgen/command_line/chargemol_caller.py index 92943751476..58896093774 100644 --- a/src/pymatgen/command_line/chargemol_caller.py +++ b/src/pymatgen/command_line/chargemol_caller.py @@ -126,12 +126,12 @@ def __init__( else: self.chgcar = self.structure = self.natoms = None - warnings.warn("No CHGCAR found. Some properties may be unavailable.", UserWarning) + warnings.warn("No CHGCAR found. Some properties may be unavailable.", stacklevel=2) if self._potcar_path: self.potcar = Potcar.from_file(self._potcar_path) else: - warnings.warn("No POTCAR found. Some properties may be unavailable.", UserWarning) + warnings.warn("No POTCAR found. Some properties may be unavailable.", stacklevel=2) self.aeccar0 = Chgcar.from_file(self._aeccar0_path) if self._aeccar0_path else None self.aeccar2 = Chgcar.from_file(self._aeccar2_path) if self._aeccar2_path else None @@ -164,7 +164,7 @@ def _get_filepath(path, filename, suffix=""): # however, better to use 'suffix' kwarg to avoid this! paths.sort(reverse=True) if len(paths) > 1: - warnings.warn(f"Multiple files detected, using {os.path.basename(paths[0])}") + warnings.warn(f"Multiple files detected, using {os.path.basename(paths[0])}", stacklevel=2) fpath = paths[0] return fpath diff --git a/src/pymatgen/command_line/critic2_caller.py b/src/pymatgen/command_line/critic2_caller.py index a37cdf3578e..4769e715860 100644 --- a/src/pymatgen/command_line/critic2_caller.py +++ b/src/pymatgen/command_line/critic2_caller.py @@ -107,7 +107,7 @@ def __init__(self, input_script: str): stderr = "" if _stderr: stderr = _stderr.decode() - warnings.warn(stderr) + warnings.warn(stderr, stacklevel=2) if rs.returncode != 0: raise RuntimeError(f"critic2 exited with return code {rs.returncode}: {stdout}") @@ -332,7 +332,7 @@ def get_filepath(filename, warning, path, suffix): """ paths = glob(os.path.join(path, f"{filename}{suffix}*")) if not paths: - warnings.warn(warning) + warnings.warn(warning, stacklevel=2) return None if len(paths) > 1: # using reverse=True because, if multiple files are present, @@ -340,7 +340,7 @@ def get_filepath(filename, warning, path, suffix): # and this would give 'static' over 'relax2' over 'relax' # however, better to use 'suffix' kwarg to avoid this! paths.sort(reverse=True) - warnings.warn(f"Multiple files detected, using {os.path.basename(path)}") + warnings.warn(f"Multiple files detected, using {os.path.basename(path)}", stacklevel=2) return paths[0] @@ -550,7 +550,8 @@ def structure_graph(self, include_critical_points=("bond", "ring", "cage")): "Duplicate edge detected, try re-running " "critic2 with custom parameters to fix this. " "Mostly harmless unless user is also " - "interested in rings/cages." + "interested in rings/cages.", + stacklevel=2, ) logger.debug( f"Duplicate edge between points {idx} (unique point {self.nodes[idx]['unique_idx']})" @@ -700,7 +701,8 @@ def _remap_indices(self): if len(node_mapping) != len(self.structure): warnings.warn( f"Check that all sites in input structure ({len(self.structure)}) have " - f"been detected by critic2 ({ len(node_mapping)})." + f"been detected by critic2 ({ len(node_mapping)}).", + stacklevel=2, ) self.nodes = {node_mapping.get(idx, idx): node for idx, node in self.nodes.items()} @@ -755,7 +757,7 @@ def get_volume_and_charge(nonequiv_idx): if zpsp: if len(charge_transfer) != len(charges): - warnings.warn(f"Something went wrong calculating charge transfer: {charge_transfer}") + warnings.warn(f"Something went wrong calculating charge transfer: {charge_transfer}", stacklevel=2) else: structure.add_site_property("bader_charge_transfer", charge_transfer) @@ -764,7 +766,9 @@ def get_volume_and_charge(nonequiv_idx): def _parse_stdout(self, stdout): warnings.warn( "Parsing critic2 standard output is deprecated and will not be maintained, " - "please use the native JSON output in future." + "please use the native JSON output in future.", + DeprecationWarning, + stacklevel=2, ) stdout = stdout.split("\n") diff --git a/src/pymatgen/core/__init__.py b/src/pymatgen/core/__init__.py index 68343cac523..6d074002060 100644 --- a/src/pymatgen/core/__init__.py +++ b/src/pymatgen/core/__init__.py @@ -57,7 +57,9 @@ def _load_pmg_settings() -> dict[str, Any]: except Exception as exc: # If there are any errors, default to using environment variables # if present. - warnings.warn(f"Error loading {file_path}: {exc}.\nYou may need to reconfigure your YAML file.") + warnings.warn( + f"Error loading {file_path}: {exc}.\nYou may need to reconfigure your YAML file.", stacklevel=2 + ) # Override .pmgrc.yaml with env vars (if present) for key, val in os.environ.items(): diff --git a/src/pymatgen/core/bonds.py b/src/pymatgen/core/bonds.py index 6c1ced8fa47..900b5262c60 100644 --- a/src/pymatgen/core/bonds.py +++ b/src/pymatgen/core/bonds.py @@ -189,7 +189,7 @@ def get_bond_order( # Distance shorter than the shortest bond length stored, # check if the distance is too short if dist < lens[-1] * (1 - tol): # too short - warnings.warn(f"{dist:.2f} angstrom distance is too short for {sp1} and {sp2}") + warnings.warn(f"{dist:.2f} angstrom distance is too short for {sp1} and {sp2}", stacklevel=2) # return the highest bond order return trial_bond_order - 1 @@ -226,6 +226,7 @@ def get_bond_length( except (ValueError, KeyError): warnings.warn( f"No order {bond_order} bond lengths between {sp1} and {sp2} found in " - "database. Returning sum of atomic radius." + "database. Returning sum of atomic radius.", + stacklevel=2, ) return sp1.atomic_radius + sp2.atomic_radius # type: ignore[operator] diff --git a/src/pymatgen/core/composition.py b/src/pymatgen/core/composition.py index d269c247ff9..b4317d506b1 100644 --- a/src/pymatgen/core/composition.py +++ b/src/pymatgen/core/composition.py @@ -802,7 +802,8 @@ def charge(self) -> float | None: """ warnings.warn( "Composition.charge is experimental and may produce incorrect results. Use with " - "caution and open a GitHub issue pinging @janosh to report bad behavior." + "caution and open a GitHub issue pinging @janosh to report bad behavior.", + stacklevel=2, ) oxi_states = [getattr(specie, "oxi_state", None) for specie in self] if {*oxi_states} <= {0, None}: @@ -818,7 +819,8 @@ def charge_balanced(self) -> bool | None: """ warnings.warn( "Composition.charge_balanced is experimental and may produce incorrect results. " - "Use with caution and open a GitHub issue pinging @janosh to report bad behavior." + "Use with caution and open a GitHub issue pinging @janosh to report bad behavior.", + stacklevel=2, ) if self.charge is None: if {getattr(el, "oxi_state", None) for el in self} == {0}: @@ -887,7 +889,8 @@ def replace(self, elem_map: dict[str, str | dict[str, float]]) -> Self: if invalid_elems: warnings.warn( "Some elements to be substituted are not present in composition. Please check your input. " - f"Problematic element = {invalid_elems}; {self}" + f"Problematic element = {invalid_elems}; {self}", + stacklevel=2, ) for elem in invalid_elems: elem_map.pop(elem) @@ -917,7 +920,8 @@ def replace(self, elem_map: dict[str, str | dict[str, float]]) -> Self: if el in self: warnings.warn( f"Same element ({el}) in both the keys and values of the substitution!" - "This can be ambiguous, so be sure to check your result." + "This can be ambiguous, so be sure to check your result.", + stacklevel=2, ) return type(self)(new_comp) diff --git a/src/pymatgen/core/interface.py b/src/pymatgen/core/interface.py index 1b146d14335..282201fe958 100644 --- a/src/pymatgen/core/interface.py +++ b/src/pymatgen/core/interface.py @@ -2072,7 +2072,8 @@ def get_rotation_angle_from_sigma( sigmas.sort() warnings.warn( "This is not the possible sigma value according to the rotation axis!" - "The nearest neighbor sigma and its corresponding angle are returned" + "The nearest neighbor sigma and its corresponding angle are returned", + stacklevel=2, ) rotation_angles = sigma_dict[sigmas[-1]] rotation_angles.sort() @@ -2144,7 +2145,7 @@ def slab_from_csl( t_matrix[1] = np.array(np.dot(scale_factor[1], csl)) t_matrix[2] = csl[miller_nonzero[0]] if abs(np.linalg.det(t_matrix)) > 1000: - warnings.warn("Too large matrix. Suggest to use quick_gen=False") + warnings.warn("Too large matrix. Suggest to use quick_gen=False", stacklevel=2) return t_matrix c_index = 0 @@ -2235,7 +2236,7 @@ def slab_from_csl( logger.info("Did not find the perpendicular c vector, increase max_j") while not normal_init: if max_j == max_search: - warnings.warn("Cannot find the perpendicular c vector, please increase max_search") + warnings.warn("Cannot find the perpendicular c vector, please increase max_search", stacklevel=2) break max_j *= 3 max_j = min(max_j, max_search) @@ -2298,7 +2299,7 @@ def slab_from_csl( t_matrix *= -1 if normal and abs(np.linalg.det(t_matrix)) > 1000: - warnings.warn("Too large matrix. Suggest to use Normal=False") + warnings.warn("Too large matrix. Suggest to use Normal=False", stacklevel=2) return t_matrix @staticmethod @@ -2335,7 +2336,7 @@ def reduce_mat(mat: NDArray, mag: int, r_matrix: NDArray) -> NDArray: break if not reduced: - warnings.warn("Matrix reduction not performed, may lead to non-primitive GB cell.") + warnings.warn("Matrix reduction not performed, may lead to non-primitive GB cell.", stacklevel=2) return mat @staticmethod diff --git a/src/pymatgen/core/lattice.py b/src/pymatgen/core/lattice.py index a49399b09c4..4e60e20af0f 100644 --- a/src/pymatgen/core/lattice.py +++ b/src/pymatgen/core/lattice.py @@ -1792,7 +1792,7 @@ def get_integer_index( # Need to recalculate this after rounding as values may have changed int_miller_index = np.round(mi, 1).astype(int) if np.any(np.abs(mi - int_miller_index) > 1e-6) and verbose: - warnings.warn("Non-integer encountered in Miller index") + warnings.warn("Non-integer encountered in Miller index", stacklevel=2) else: mi = int_miller_index diff --git a/src/pymatgen/core/operations.py b/src/pymatgen/core/operations.py index 9b3d8e0b1b4..d4821d986a8 100644 --- a/src/pymatgen/core/operations.py +++ b/src/pymatgen/core/operations.py @@ -450,7 +450,7 @@ def as_xyz_str(self) -> str: """ # Check for invalid rotation matrix if not np.allclose(self.rotation_matrix, np.round(self.rotation_matrix)): - warnings.warn("Rotation matrix should be integer") + warnings.warn("Rotation matrix should be integer", stacklevel=2) return transformation_to_string( self.rotation_matrix, diff --git a/src/pymatgen/core/periodic_table.py b/src/pymatgen/core/periodic_table.py index a981c74578f..bb6907c88ed 100644 --- a/src/pymatgen/core/periodic_table.py +++ b/src/pymatgen/core/periodic_table.py @@ -207,7 +207,7 @@ def __getattr__(self, item: str) -> Any: key = item.capitalize().replace("_", " ") val = self._data.get(key) if val is None or str(val).startswith("no data"): - warnings.warn(f"No data available for {item} for {self.symbol}") + warnings.warn(f"No data available for {item} for {self.symbol}", stacklevel=2) val = None elif isinstance(val, list | dict): pass @@ -246,7 +246,8 @@ def __getattr__(self, item: str) -> Any: and (match := re.findall(r"[\.\d]+", val)) ): warnings.warn( - f"Ambiguous values ({val}) for {item} of {self.symbol}. Returning first float value." + f"Ambiguous values ({val}) for {item} of {self.symbol}. Returning first float value.", + stacklevel=2, ) return float(match[0]) return val @@ -293,7 +294,8 @@ def X(self) -> float: return X warnings.warn( f"No Pauling electronegativity for {self.symbol}. Setting to NaN. This has no physical meaning, " - "and is mainly done to avoid errors caused by the code expecting a float." + "and is mainly done to avoid errors caused by the code expecting a float.", + stacklevel=2, ) return float("NaN") @@ -342,7 +344,7 @@ def data(self) -> dict[str, Any]: def ionization_energy(self) -> float | None: """First ionization energy of element.""" if not self.ionization_energies: - warnings.warn(f"No data available for ionization_energy for {self.symbol}") + warnings.warn(f"No data available for ionization_energy for {self.symbol}", stacklevel=2) return None return self.ionization_energies[0] @@ -1227,12 +1229,12 @@ def ionic_radius(self) -> float | None: oxi_str = str(int(self._oxi_state)) warn_msg = f"No default ionic radius for {self}." if ion_rad := dct.get("Ionic radii hs", {}).get(oxi_str): - warnings.warn(f"{warn_msg} Using hs data.") + warnings.warn(f"{warn_msg} Using hs data.", stacklevel=2) return ion_rad if ion_rad := dct.get("Ionic radii ls", {}).get(oxi_str): - warnings.warn(f"{warn_msg} Using ls data.") + warnings.warn(f"{warn_msg} Using ls data.", stacklevel=2) return ion_rad - warnings.warn(f"No ionic radius for {self}!") + warnings.warn(f"No ionic radius for {self}!", stacklevel=2) return None @classmethod @@ -1339,7 +1341,8 @@ def get_shannon_radius( if key != spin: warnings.warn( f"Specified {spin=} not consistent with database spin of {key}. " - "Only one spin data available, and that value is returned." + "Only one spin data available, and that value is returned.", + stacklevel=2, ) else: data = radii[spin] diff --git a/src/pymatgen/core/structure.py b/src/pymatgen/core/structure.py index 8f9fc7f7321..f645b347534 100644 --- a/src/pymatgen/core/structure.py +++ b/src/pymatgen/core/structure.py @@ -605,7 +605,8 @@ def replace_species( if not sp_in_structure >= sp_to_replace: warnings.warn( "Some species to be substituted are not present in structure. Pls check your input. Species to be " - f"substituted = {sp_to_replace}; Species in structure = {sp_in_structure}" + f"substituted = {sp_to_replace}; Species in structure = {sp_in_structure}", + stacklevel=2, ) for site in site_coll: @@ -1260,7 +1261,7 @@ def from_sites( props[key][idx] = val for key, val in props.items(): if any(vv is None for vv in val): - warnings.warn(f"Not all sites have property {key}. Missing values are set to None.") + warnings.warn(f"Not all sites have property {key}. Missing values are set to None.", stacklevel=2) return cls( lattice, [site.species for site in sites], @@ -1516,7 +1517,8 @@ def charge(self) -> float: if abs(formal_charge - self._charge) > 1e-8: warnings.warn( f"Structure charge ({self._charge}) is set to be not equal to the sum of oxidation states" - f" ({formal_charge}). Use Structure.unset_charge() to reset the charge to None." + f" ({formal_charge}). Use Structure.unset_charge() to reset the charge to None.", + stacklevel=2, ) return self._charge @@ -4753,7 +4755,8 @@ def merge_sites(self, tol: float = 0.01, mode: Literal["sum", "delete", "average else: props[key] = None warnings.warn( - f"Sites with different site property {key} are merged. So property is set to none" + f"Sites with different site property {key} are merged. So property is set to none", + stacklevel=2, ) sites.append(PeriodicSite(species, coords, self.lattice, properties=props)) diff --git a/src/pymatgen/core/surface.py b/src/pymatgen/core/surface.py index f7f692b6a74..304986f8945 100644 --- a/src/pymatgen/core/surface.py +++ b/src/pymatgen/core/surface.py @@ -537,7 +537,8 @@ def get_equi_index(site: PeriodicSite) -> int: "Odd number of sites to divide! Try changing " "the tolerance to ensure even division of " "sites or create supercells in a or b directions " - "to allow for atoms to be moved!" + "to allow for atoms to be moved!", + stacklevel=2, ) continue combinations = [] @@ -627,8 +628,7 @@ def add_adsorbate_atom( # Check if deprecated argument is used if specie is not None: warnings.warn( - "The argument 'specie' is deprecated. Use 'species' instead.", - DeprecationWarning, + "The argument 'specie' is deprecated. Use 'species' instead.", DeprecationWarning, stacklevel=2 ) species = specie @@ -660,8 +660,7 @@ def symmetrically_add_atom( # Check if deprecated argument is used if specie is not None: warnings.warn( - "The argument 'specie' is deprecated. Use 'species' instead.", - DeprecationWarning, + "The argument 'specie' is deprecated. Use 'species' instead.", DeprecationWarning, stacklevel=2 ) species = specie @@ -737,7 +736,7 @@ def get_equi_sites(slab: Slab, sites: list[int]) -> list[int]: self.remove_sites(equi_sites) else: - warnings.warn("Equivalent sites could not be found for some indices. Surface unchanged.") + warnings.warn("Equivalent sites could not be found for some indices. Surface unchanged.", stacklevel=2) def center_slab(slab: Structure) -> Structure: @@ -1556,7 +1555,7 @@ def nonstoichiometric_symmetrized_slab(self, init_slab: Slab) -> list[Slab]: slab.remove_sites([z_coords.index(min(z_coords))]) if len(slab) <= len(self.parent): - warnings.warn("Too many sites removed, please use a larger slab.") + warnings.warn("Too many sites removed, please use a larger slab.", stacklevel=2) break # Check if the new Slab is symmetric diff --git a/src/pymatgen/core/tensors.py b/src/pymatgen/core/tensors.py index f2c72cc9a37..26552a36b04 100644 --- a/src/pymatgen/core/tensors.py +++ b/src/pymatgen/core/tensors.py @@ -346,7 +346,7 @@ def voigt(self) -> NDArray: for ind, v in this_voigt_map.items(): v_matrix[v] = self[ind] if not self.is_voigt_symmetric(): - warnings.warn("Tensor is not symmetric, information may be lost in Voigt conversion.") + warnings.warn("Tensor is not symmetric, information may be lost in Voigt conversion.", stacklevel=2) return v_matrix * self._vscale def is_voigt_symmetric(self, tol: float = 1e-6) -> bool: @@ -531,7 +531,7 @@ def structure_transform( """ sm = StructureMatcher() if not sm.fit(original_structure, new_structure): - warnings.warn("original and new structures do not match!") + warnings.warn("original and new structures do not match!", stacklevel=2) trans_1 = self.get_ieee_rotation(original_structure, refine_rotation) trans_2 = self.get_ieee_rotation(new_structure, refine_rotation) # Get the ieee format tensor @@ -672,7 +672,7 @@ def merge(old, new) -> None: print(f"Iteration {idx}: {np.max(diff)}") if not converged: max_diff = np.max(np.abs(self - test_new)) - warnings.warn(f"Warning, populated tensor is not converged with max diff of {max_diff}") + warnings.warn(f"Warning, populated tensor is not converged with max diff of {max_diff}", stacklevel=2) return type(self)(test_new) def as_dict(self, voigt: bool = False) -> dict: diff --git a/src/pymatgen/core/trajectory.py b/src/pymatgen/core/trajectory.py index 53a02ffb063..bc656711bb2 100644 --- a/src/pymatgen/core/trajectory.py +++ b/src/pymatgen/core/trajectory.py @@ -147,7 +147,8 @@ def __init__( self.lattice = np.tile(lattice, (len(coords), 1, 1)) warnings.warn( "Get constant_lattice=False, but only get a single lattice. " - "Use this single lattice as the lattice for all frames." + "Use this single lattice as the lattice for all frames.", + stacklevel=2, ) else: self.lattice = lattice @@ -161,7 +162,8 @@ def __init__( if base_positions is None: warnings.warn( "Without providing an array of starting positions, the positions " - "for each time step will not be available." + "for each time step will not be available.", + stacklevel=2, ) self.base_positions = base_positions else: diff --git a/src/pymatgen/electronic_structure/bandstructure.py b/src/pymatgen/electronic_structure/bandstructure.py index a9a1853398f..e3352c36328 100644 --- a/src/pymatgen/electronic_structure/bandstructure.py +++ b/src/pymatgen/electronic_structure/bandstructure.py @@ -652,7 +652,8 @@ def from_dict(cls, dct: dict[str, Any]) -> Self: "Trying from_dict failed. Now we are trying the old " "format. Please convert your BS dicts to the new " "format. The old format will be retired in pymatgen " - "5.0." + "5.0.", + stacklevel=2, ) return cls.from_old_dict(dct) @@ -980,7 +981,8 @@ def from_dict(cls, dct: dict[str, Any]) -> Self: "Trying from_dict failed. Now we are trying the old " "format. Please convert your BS dicts to the new " "format. The old format will be retired in pymatgen " - "5.0." + "5.0.", + stacklevel=2, ) return cls.from_old_dict(dct) diff --git a/src/pymatgen/electronic_structure/boltztrap2.py b/src/pymatgen/electronic_structure/boltztrap2.py index de576cec76d..135b39a38fd 100644 --- a/src/pymatgen/electronic_structure/boltztrap2.py +++ b/src/pymatgen/electronic_structure/boltztrap2.py @@ -31,6 +31,7 @@ import matplotlib.pyplot as plt import numpy as np +from monty.dev import deprecated from monty.serialization import dumpfn, loadfn from tqdm import tqdm @@ -182,6 +183,7 @@ def bandana(self, emin=-np.inf, emax=np.inf): return accepted +@deprecated(VasprunBSLoader, category=DeprecationWarning) class BandstructureLoader: """Loader for Bandstructure object.""" @@ -201,8 +203,6 @@ def __init__(self, bs_obj, structure=None, nelect=None, mommat=None, magmom=None ne = vrun.parameters['NELECT'] data = BandstructureLoader(bs,st,ne) """ - warnings.warn("Deprecated Loader. Use VasprunBSLoader instead.") - self.kpoints = np.array([kp.frac_coords for kp in bs_obj.kpoints]) self.structure = bs_obj.structure if structure is None else structure @@ -278,7 +278,8 @@ def set_upper_lower_bands(self, e_lower, e_upper) -> None: range in the spin up/down bands when calculating the DOS. """ warnings.warn( - "This method does not work anymore in case of spin polarized case due to the concatenation of bands !" + "This method does not work anymore in case of spin polarized case due to the concatenation of bands !", + stacklevel=2, ) lower_band = e_lower * np.ones((1, self.ebands.shape[1])) @@ -301,13 +302,12 @@ def get_volume(self): return self.UCvol +@deprecated(VasprunBSLoader, category=DeprecationWarning) class VasprunLoader: """Loader for Vasprun object.""" def __init__(self, vrun_obj=None) -> None: """vrun_obj: Vasprun object.""" - warnings.warn("Deprecated Loader. Use VasprunBSLoader instead.") - if vrun_obj: self.kpoints = np.array(vrun_obj.actual_kpoints) self.structure = vrun_obj.final_structure @@ -1199,7 +1199,10 @@ def merge_up_down_doses(dos_up, dos_dn): Returns: CompleteDos object """ - warnings.warn("This function is not useful anymore. VasprunBSLoader deals with spin case.") + warnings.warn( + "This function is not useful anymore. VasprunBSLoader deals with spin case.", DeprecationWarning, stacklevel=2 + ) + cdos = Dos( dos_up.efermi, dos_up.energies, diff --git a/src/pymatgen/electronic_structure/cohp.py b/src/pymatgen/electronic_structure/cohp.py index cb82939c4fa..3fc2b2cc460 100644 --- a/src/pymatgen/electronic_structure/cohp.py +++ b/src/pymatgen/electronic_structure/cohp.py @@ -1247,7 +1247,9 @@ def get_summed_icohp_by_label_list( for label in label_list: icohp = self._icohplist[label] if icohp.num_bonds != 1: - warnings.warn("One of the ICOHP values is an average over bonds. This is currently not considered.") + warnings.warn( + "One of the ICOHP values is an average over bonds. This is currently not considered.", stacklevel=2 + ) if icohp._is_spin_polarized and summed_spin_channels: sum_icohp += icohp.summed_icohp @@ -1350,7 +1352,7 @@ def extremum_icohpvalue( if not self._is_spin_polarized: if spin == Spin.down: - warnings.warn("This spin channel does not exist. I am switching to Spin.up") + warnings.warn("This spin channel does not exist. I am switching to Spin.up", stacklevel=2) spin = Spin.up for value in self._icohplist.values(): diff --git a/src/pymatgen/electronic_structure/dos.py b/src/pymatgen/electronic_structure/dos.py index 05e524c4758..78b62d2f4a5 100644 --- a/src/pymatgen/electronic_structure/dos.py +++ b/src/pymatgen/electronic_structure/dos.py @@ -604,7 +604,7 @@ def get_fermi_interextrapolated( return self.get_fermi(concentration, temperature, **kwargs) except ValueError as exc: if warn: - warnings.warn(str(exc)) + warnings.warn(str(exc), stacklevel=2) if abs(concentration) < c_ref: if abs(concentration) < 1e-10: @@ -1482,7 +1482,7 @@ def get_site_t2g_eg_resolved_dos( Returns: dict[Literal["e_g", "t2g"], Dos]: Summed e_g and t2g DOS for the site. """ - warnings.warn("Are the orbitals correctly oriented? Are you sure?") + warnings.warn("Are the orbitals correctly oriented? Are you sure?", stacklevel=2) t2g_dos = [] eg_dos = [] diff --git a/src/pymatgen/electronic_structure/plotter.py b/src/pymatgen/electronic_structure/plotter.py index 1443694d386..88263e0798d 100644 --- a/src/pymatgen/electronic_structure/plotter.py +++ b/src/pymatgen/electronic_structure/plotter.py @@ -570,9 +570,9 @@ def _interpolate_bands(distances, energies, smooth_tol=0, smooth_k=3, smooth_np= # reducing smooth_k when the number # of points are fewer then k smooth_k = len(dist) - 1 - warnings.warn(warning_m_fewer_k) + warnings.warn(warning_m_fewer_k, stacklevel=2) elif len(dist) == 1: - warnings.warn("Skipping single point branch") + warnings.warn("Skipping single point branch", stacklevel=2) continue int_distances.append(np.linspace(dist[0], dist[-1], smooth_np)) @@ -587,7 +587,7 @@ def _interpolate_bands(distances, energies, smooth_tol=0, smooth_k=3, smooth_np= int_energies.append(np.vstack(br_en)) if np.any(np.isnan(int_energies[-1])): - warnings.warn(warning_nan) + warnings.warn(warning_nan, stacklevel=2) return int_distances, int_energies @@ -860,7 +860,7 @@ def plot_compare(self, other_plotter, legend=True) -> plt.Axes: Returns: plt.Axes: matplotlib Axes object with both band structures """ - warnings.warn("Deprecated method. Use BSPlotter([sbs1,sbs2,...]).get_plot() instead.") + warnings.warn("Deprecated method. Use BSPlotter([sbs1,sbs2,...]).get_plot() instead.", stacklevel=2) # TODO: add exception if the band structures are not compatible ax = self.get_plot() @@ -928,7 +928,8 @@ def __init__(self, bs: BandStructureSymmLine) -> None: if isinstance(bs, list): warnings.warn( "Multiple band structures are not handled by BSPlotterProjected. " - "Only the first in the list will be considered" + "Only the first in the list will be considered", + stacklevel=2, ) bs = bs[0] @@ -2347,7 +2348,8 @@ def get_plot( warnings.warn( "Cannot get element projected data; either the projection data " "doesn't exist, or you don't have a compound with exactly 2 " - "or 3 or 4 unique elements." + "or 3 or 4 unique elements.", + stacklevel=2, ) bs_projection = None diff --git a/src/pymatgen/entries/compatibility.py b/src/pymatgen/entries/compatibility.py index 48b05774418..09770153e15 100644 --- a/src/pymatgen/entries/compatibility.py +++ b/src/pymatgen/entries/compatibility.py @@ -14,6 +14,7 @@ import numpy as np from joblib import Parallel, delayed from monty.design_patterns import cached_class +from monty.dev import deprecated from monty.json import MSONable from monty.serialization import loadfn from tqdm import tqdm @@ -296,7 +297,7 @@ def get_correction(self, entry: ComputedEntry | ComputedStructureEntry) -> ufloa if entry.data.get("sulfide_type"): sf_type = entry.data["sulfide_type"] elif hasattr(entry, "structure"): - warnings.warn(sf_type) + warnings.warn(sf_type, stacklevel=2) sf_type = sulfide_type(entry.structure) # use the same correction for polysulfides and sulfides @@ -329,7 +330,8 @@ def get_correction(self, entry: ComputedEntry | ComputedStructureEntry) -> ufloa else: warnings.warn( "No structure or oxide_type parameter present. Note that peroxide/superoxide corrections " - "are not as reliable and relies only on detection of special formulas, e.g. Li2O2." + "are not as reliable and relies only on detection of special formulas, e.g. Li2O2.", + stacklevel=2, ) rform = entry.reduced_formula if rform in UCorrection.common_peroxides: @@ -622,7 +624,7 @@ def _process_entry_inplace( if on_error == "raise": raise if on_error == "warn": - warnings.warn(str(exc)) + warnings.warn(str(exc), stacklevel=2) return None for e_adj in adjustments: @@ -640,7 +642,8 @@ def _process_entry_inplace( warnings.warn( f"Entry {entry.entry_id} already has an energy adjustment called {e_adj.name}, but its " f"value differs from the value of {e_adj.value:.3f} calculated here. This " - "Entry will be discarded." + "Entry will be discarded.", + stacklevel=2, ) else: @@ -880,6 +883,11 @@ def explain(self, entry: ComputedEntry) -> None: print(f"The final energy after corrections is {dct['corrected_energy']:f}") +@deprecated( + "MaterialsProject2020Compatibility", + "Materials Project formation energies use the newer MaterialsProject2020Compatibility scheme.", + category=DeprecationWarning, +) class MaterialsProjectCompatibility(CorrectionsList): """This class implements the GGA/GGA+U mixing scheme, which allows mixing of entries. Note that this should only be used for VASP calculations using the @@ -907,11 +915,6 @@ def __init__( check_potcar_hash (bool): Use potcar hash to verify potcars are correct. silence_deprecation (bool): Silence deprecation warning. Defaults to False. """ - warnings.warn( # added by @janosh on 2023-05-25 - "MaterialsProjectCompatibility is deprecated, Materials Project formation energies " - "use the newer MaterialsProject2020Compatibility scheme.", - DeprecationWarning, - ) self.compat_type = compat_type self.correct_peroxide = correct_peroxide self.check_potcar_hash = check_potcar_hash @@ -1130,7 +1133,8 @@ def get_adjustments(self, entry: AnyComputedEntry) -> list[EnergyAdjustment]: else: warnings.warn( "No structure or oxide_type parameter present. Note that peroxide/superoxide corrections " - "are not as reliable and relies only on detection of special formulas, e.g. Li2O2." + "are not as reliable and relies only on detection of special formulas, e.g. Li2O2.", + stacklevel=2, ) common_peroxides = "Li2O2 Na2O2 K2O2 Cs2O2 Rb2O2 BeO2 MgO2 CaO2 SrO2 BaO2".split() @@ -1180,7 +1184,8 @@ def get_adjustments(self, entry: AnyComputedEntry) -> list[EnergyAdjustment]: warnings.warn( f"Failed to guess oxidation states for Entry {entry.entry_id} " f"({entry.reduced_formula}). Assigning anion correction to " - "only the most electronegative atom." + "only the most electronegative atom.", + stacklevel=2, ) for anion in ("Br", "I", "Se", "Si", "Sb", "Te", "H", "N", "F", "Cl"): @@ -1418,7 +1423,8 @@ def __init__( f"You did not provide the required O2 and H2O energies. {type(self).__name__} " "needs these energies in order to compute the appropriate energy adjustments. It will try " "to determine the values from ComputedEntry for O2 and H2O passed to process_entries, but " - "will fail if these entries are not provided." + "will fail if these entries are not provided.", + stacklevel=2, ) # Standard state entropy of molecular-like compounds at 298K (-T delta S) @@ -1604,7 +1610,8 @@ def process_entries( "being assigned the same energy. This should not cause problems " "with Pourbaix diagram construction, but may be confusing. " "Pass all entries to process_entries() at once in if you want to " - "preserve H2 polymorph energy differences." + "preserve H2 polymorph energy differences.", + stacklevel=2, ) # extract the DFT energies of oxygen and water from the list of entries, if present diff --git a/src/pymatgen/entries/computed_entries.py b/src/pymatgen/entries/computed_entries.py index 42afd1749ba..ae76bc15c4b 100644 --- a/src/pymatgen/entries/computed_entries.py +++ b/src/pymatgen/entries/computed_entries.py @@ -661,7 +661,8 @@ def normalize( warnings.warn( f"Normalization of a `{type(self).__name__}` makes " "`self.composition` and `self.structure.composition` inconsistent" - " - please use self.composition for all further calculations." + " - please use self.composition for all further calculations.", + stacklevel=2, ) # TODO: find a better solution for creating copies instead of as/from dict factor = self._normalization_factor(mode) diff --git a/src/pymatgen/entries/correction_calculator.py b/src/pymatgen/entries/correction_calculator.py index 0053b7e5d23..e11bd1fb9c2 100644 --- a/src/pymatgen/entries/correction_calculator.py +++ b/src/pymatgen/entries/correction_calculator.py @@ -130,7 +130,10 @@ def compute_corrections(self, exp_entries: list, calc_entries: dict) -> dict: compound = self.calc_compounds.get(name) if not compound: - warnings.warn(f"Compound {name} is not found in provided computed entries and is excluded from the fit") + warnings.warn( + f"Compound {name} is not found in provided computed entries and is excluded from the fit", + stacklevel=2, + ) continue # filter out compounds with large uncertainties @@ -139,14 +142,17 @@ def compute_corrections(self, exp_entries: list, calc_entries: dict) -> dict: allow = False warnings.warn( f"Compound {name} is excluded from the fit due to high experimental " - f"uncertainty ({relative_uncertainty:.1%})" + f"uncertainty ({relative_uncertainty:.1%})", + stacklevel=2, ) # filter out compounds containing certain polyanions for anion in self.exclude_polyanions: if anion in name or anion in cmpd_info["formula"]: allow = False - warnings.warn(f"Compound {name} contains the poly{anion=} and is excluded from the fit") + warnings.warn( + f"Compound {name} contains the poly{anion=} and is excluded from the fit", stacklevel=2 + ) break # filter out compounds that are unstable @@ -157,7 +163,9 @@ def compute_corrections(self, exp_entries: list, calc_entries: dict) -> dict: raise ValueError("Missing e above hull data") if eah > self.allow_unstable: allow = False - warnings.warn(f"Compound {name} is unstable and excluded from the fit (e_above_hull = {eah})") + warnings.warn( + f"Compound {name} is unstable and excluded from the fit (e_above_hull = {eah})", stacklevel=2 + ) if allow: comp = Composition(name) diff --git a/src/pymatgen/entries/mixing_scheme.py b/src/pymatgen/entries/mixing_scheme.py index af94f2613f0..5da8f99f515 100644 --- a/src/pymatgen/entries/mixing_scheme.py +++ b/src/pymatgen/entries/mixing_scheme.py @@ -156,7 +156,9 @@ def process_entries( # We can't operate on single entries in this scheme if len(entries) == 1: - warnings.warn(f"{type(self).__name__} cannot process single entries. Supply a list of entries.") + warnings.warn( + f"{type(self).__name__} cannot process single entries. Supply a list of entries.", stacklevel=2 + ) return processed_entry_list # if inplace = False, process entries on a copy @@ -210,7 +212,7 @@ def process_entries( adjustments = self.get_adjustments(entry, mixing_state_data) except CompatibilityError as exc: if "WARNING!" in str(exc): - warnings.warn(str(exc)) + warnings.warn(str(exc), stacklevel=2) elif verbose: print(f" {exc}") ignore_entry = True @@ -228,7 +230,8 @@ def process_entries( warnings.warn( f"Entry {entry.entry_id} already has an energy adjustment called {ea.name}, but its " f"value differs from the value of {ea.value:.3f} calculated here. This " - "Entry will be discarded." + "Entry will be discarded.", + stacklevel=2, ) else: # Add the correction to the energy_adjustments list @@ -481,7 +484,8 @@ def get_mixing_state_data(self, entries: list[ComputedStructureEntry]): if not isinstance(entry, ComputedStructureEntry): warnings.warn( f"Entry {entry.entry_id} is not a ComputedStructureEntry and will be ignored. " - "The DFT mixing scheme requires structures for all entries" + "The DFT mixing scheme requires structures for all entries", + stacklevel=2, ) continue @@ -496,12 +500,12 @@ def get_mixing_state_data(self, entries: list[ComputedStructureEntry]): try: pd_type_1 = PhaseDiagram(entries_type_1) except ValueError: - warnings.warn(f"{self.run_type_1} entries do not form a complete PhaseDiagram.") + warnings.warn(f"{self.run_type_1} entries do not form a complete PhaseDiagram.", stacklevel=2) try: pd_type_2 = PhaseDiagram(entries_type_2) except ValueError: - warnings.warn(f"{self.run_type_2} entries do not form a complete PhaseDiagram.") + warnings.warn(f"{self.run_type_2} entries do not form a complete PhaseDiagram.", stacklevel=2) # Objective: loop through all the entries, group them by structure matching (or fuzzy structure matching # where relevant). For each group, put a row in a pandas DataFrame with the composition of the run_type_1 entry, @@ -582,7 +586,8 @@ def _filter_and_sort_entries(self, entries, verbose=False): if not entry.parameters.get("run_type"): warnings.warn( f"Entry {entry_id} is missing parameters.run_type! This field" - "is required. This entry will be ignored." + "is required. This entry will be ignored.", + stacklevel=2, ) continue @@ -590,7 +595,8 @@ def _filter_and_sort_entries(self, entries, verbose=False): if run_type not in [*self.valid_rtypes_1, *self.valid_rtypes_2]: warnings.warn( f"Invalid {run_type=} for entry {entry_id}. Must be one of " - f"{self.valid_rtypes_1 + self.valid_rtypes_2}. This entry will be ignored." + f"{self.valid_rtypes_1 + self.valid_rtypes_2}. This entry will be ignored.", + stacklevel=2, ) continue @@ -598,7 +604,8 @@ def _filter_and_sort_entries(self, entries, verbose=False): if entry_id is None: warnings.warn( f"{entry_id=} for {formula=}. Unique entry_ids are required for every ComputedStructureEntry." - " This entry will be ignored." + " This entry will be ignored.", + stacklevel=2, ) continue @@ -649,7 +656,8 @@ def _filter_and_sort_entries(self, entries, verbose=False): warnings.warn( f" {self.run_type_2} entries chemical system {entries_type_2.chemsys} is larger than " f"{self.run_type_1} entries chemical system {entries_type_1.chemsys}. Entries outside the " - f"{self.run_type_1} chemical system will be discarded" + f"{self.run_type_1} chemical system will be discarded", + stacklevel=2, ) entries_type_2 = entries_type_2.get_subset_in_chemsys(chemsys) else: diff --git a/src/pymatgen/ext/cod.py b/src/pymatgen/ext/cod.py index 39385609540..d6b53f5146e 100644 --- a/src/pymatgen/ext/cod.py +++ b/src/pymatgen/ext/cod.py @@ -89,8 +89,7 @@ def get_structure_by_id(self, cod_id: int, timeout: int | None = None, **kwargs) # TODO: remove timeout arg and use class level timeout after 2025-10-17 if timeout is not None: warnings.warn( - "separate timeout arg is deprecated, please use class level timeout", - DeprecationWarning, + "separate timeout arg is deprecated, please use class level timeout", DeprecationWarning, stacklevel=2 ) timeout = timeout or self.timeout diff --git a/src/pymatgen/ext/matproj_legacy.py b/src/pymatgen/ext/matproj_legacy.py index 4e6cca8e4e9..47521e972ce 100644 --- a/src/pymatgen/ext/matproj_legacy.py +++ b/src/pymatgen/ext/matproj_legacy.py @@ -169,19 +169,22 @@ def __init__( "You are using the legacy MPRester. This version of the MPRester will no longer be updated. " "To access the latest data with the new MPRester, obtain a new API key from " "https://materialsproject.org/api and consult the docs at https://docs.materialsproject.org/ " - "for more information." + "for more information.", + FutureWarning, + stacklevel=2, ) if api_key is not None: self.api_key = api_key else: self.api_key = SETTINGS.get("PMG_MAPI_KEY", "") + if endpoint is not None: self.preamble = endpoint else: self.preamble = SETTINGS.get("PMG_MAPI_ENDPOINT", "https://legacy.materialsproject.org/rest/v2") if self.preamble != "https://legacy.materialsproject.org/rest/v2": - warnings.warn(f"Non-default endpoint used: {self.preamble}") + warnings.warn(f"Non-default endpoint used: {self.preamble}", stacklevel=2) self.session = requests.Session() self.session.headers = {"x-api-key": self.api_key} @@ -219,12 +222,13 @@ def __init__( else: dct["MAPI_DB_VERSION"]["LOG"][db_version] += 1 - # alert user if db version changed + # alert user if DB version changed last_accessed = dct["MAPI_DB_VERSION"]["LAST_ACCESSED"] if last_accessed and last_accessed != db_version: - print( + warnings.warn( f"This database version has changed from the database last accessed ({last_accessed}).\n" - f"Please see release notes on materialsproject.org for information about what has changed." + f"Please see release notes on materialsproject.org for information about what has changed.", + stacklevel=2, ) dct["MAPI_DB_VERSION"]["LAST_ACCESSED"] = db_version @@ -263,7 +267,7 @@ def _make_request( data = json.loads(response.text, cls=MontyDecoder) if mp_decode else json.loads(response.text) if data["valid_response"]: if data.get("warning"): - warnings.warn(data["warning"]) + warnings.warn(data["warning"], stacklevel=2) return data["response"] raise MPRestError(data["error"]) @@ -693,7 +697,8 @@ def get_structure_by_material_id( f"so structure for {new_material_id} returned. This is not an error, see " f"documentation. If original task data for {material_id} is required, use " "get_task_data(). To find the canonical mp-id from a task id use " - "get_materials_id_from_task_id()." + "get_materials_id_from_task_id().", + stacklevel=2, ) return self.get_structure_by_material_id(new_material_id) except MPRestError: @@ -1106,7 +1111,7 @@ def submit_snl(self, snl): response = json.loads(response.text, cls=MontyDecoder) if response["valid_response"]: if response.get("warning"): - warnings.warn(response["warning"]) + warnings.warn(response["warning"], stacklevel=2) return response["inserted_ids"] raise MPRestError(response["error"]) @@ -1132,7 +1137,7 @@ def delete_snl(self, snl_ids): response = json.loads(response.text, cls=MontyDecoder) if response["valid_response"]: if response.get("warning"): - warnings.warn(response["warning"]) + warnings.warn(response["warning"], stacklevel=2) return response raise MPRestError(response["error"]) @@ -1160,7 +1165,7 @@ def query_snl(self, criteria): response = json.loads(response.text) if response["valid_response"]: if response.get("warning"): - warnings.warn(response["warning"]) + warnings.warn(response["warning"], stacklevel=2) return response["response"] raise MPRestError(response["error"]) @@ -1258,7 +1263,7 @@ def get_stability(self, entries): response = json.loads(response.text, cls=MontyDecoder) if response["valid_response"]: if response.get("warning"): - warnings.warn(response["warning"]) + warnings.warn(response["warning"], stacklevel=2) return response["response"] raise MPRestError(response["error"]) raise MPRestError(f"REST error with status code {response.status_code} and error {response.text}") @@ -1556,7 +1561,8 @@ def _print_help_message(nomad_exist_task_ids, task_ids, file_patterns, task_type warnings.warn( f"For {file_patterns=}] and {task_types=}, \n" f"the following ids are not found on NOMAD [{list(non_exist_ids)}]. \n" - f"If you need to upload them, please contact Patrick Huck at phuck@lbl.gov" + f"If you need to upload them, please contact Patrick Huck at phuck@lbl.gov", + stacklevel=2, ) def _check_get_download_info_url_by_task_id(self, prefix, task_ids) -> list[str]: diff --git a/src/pymatgen/io/abinit/netcdf.py b/src/pymatgen/io/abinit/netcdf.py index b986ce5f5e0..5a70d41f7c1 100644 --- a/src/pymatgen/io/abinit/netcdf.py +++ b/src/pymatgen/io/abinit/netcdf.py @@ -25,7 +25,10 @@ import netCDF4 except ImportError: netCDF4 = None - warnings.warn("Can't import netCDF4. Some features will be disabled unless you pip install netCDF4.") + warnings.warn( + "Can't import netCDF4. Some features will be disabled unless you pip install netCDF4.", + stacklevel=2, + ) logger = logging.getLogger(__name__) diff --git a/src/pymatgen/io/aims/inputs.py b/src/pymatgen/io/aims/inputs.py index 3c1aa389540..fd8860deb2b 100644 --- a/src/pymatgen/io/aims/inputs.py +++ b/src/pymatgen/io/aims/inputs.py @@ -566,6 +566,7 @@ def get_content( warn( "Removing spin from parameters since no spin information is in the structure", RuntimeWarning, + stacklevel=2, ) parameters.pop("spin") diff --git a/src/pymatgen/io/aims/parsers.py b/src/pymatgen/io/aims/parsers.py index dde640a1172..929e01fc9f6 100644 --- a/src/pymatgen/io/aims/parsers.py +++ b/src/pymatgen/io/aims/parsers.py @@ -500,8 +500,7 @@ def _parse_structure(self) -> Structure | Molecule: ) < 1e-3: warnings.warn( "Total magnetic moment and sum of Mulliken spins are not consistent", - UserWarning, - stacklevel=1, + stacklevel=2, ) if lattice is not None: diff --git a/src/pymatgen/io/aims/sets/base.py b/src/pymatgen/io/aims/sets/base.py index 37c3b4e502f..f91c686d8de 100644 --- a/src/pymatgen/io/aims/sets/base.py +++ b/src/pymatgen/io/aims/sets/base.py @@ -343,14 +343,14 @@ def _get_input_parameters( warn( "WARNING: the k_grid is set in user_params and in the kpt_settings," " using the one passed in user_params.", - stacklevel=1, + stacklevel=2, ) elif isinstance(structure, Structure) and ("k_grid" not in params): density = kpt_settings.get("density", 5.0) even = kpt_settings.get("even", True) params["k_grid"] = self.d2k(structure, density, even) elif isinstance(structure, Molecule) and "k_grid" in params: - warn("WARNING: removing unnecessary k_grid information", stacklevel=1) + warn("WARNING: removing unnecessary k_grid information", stacklevel=2) del params["k_grid"] return params diff --git a/src/pymatgen/io/ase.py b/src/pymatgen/io/ase.py index d8a8a748682..f3cf8516d6d 100644 --- a/src/pymatgen/io/ase.py +++ b/src/pymatgen/io/ase.py @@ -266,8 +266,7 @@ def get_structure(atoms: Atoms, cls: type[Structure] = Structure, **cls_kwargs) unsupported_constraint_type = True if unsupported_constraint_type: warnings.warn( - "Only FixAtoms is supported by Pymatgen. Other constraints will not be set.", - UserWarning, + "Only FixAtoms is supported by Pymatgen. Other constraints will not be set.", stacklevel=2 ) sel_dyn = [[False] * 3 if atom.index in constraint_indices else [True] * 3 for atom in atoms] else: diff --git a/src/pymatgen/io/babel.py b/src/pymatgen/io/babel.py index 6845dd7e62c..fcb213941da 100644 --- a/src/pymatgen/io/babel.py +++ b/src/pymatgen/io/babel.py @@ -186,7 +186,8 @@ def rotor_conformer(self, *rotor_args, algo: str = "WeightedRotorSearch", forcef warnings.warn( f"This input {forcefield=} is not supported " "in openbabel. The forcefield will be reset as " - "default 'mmff94' for now." + "default 'mmff94' for now.", + stacklevel=2, ) ff = openbabel.OBForceField.FindType("mmff94") @@ -199,7 +200,8 @@ def rotor_conformer(self, *rotor_args, algo: str = "WeightedRotorSearch", forcef "'SystematicRotorSearch', 'RandomRotorSearch' " "and 'WeightedRotorSearch'. " "The algorithm will be reset as default " - "'WeightedRotorSearch' for now." + "'WeightedRotorSearch' for now.", + stacklevel=2, ) rotor_search = ff.WeightedRotorSearch rotor_search(*rotor_args) diff --git a/src/pymatgen/io/cif.py b/src/pymatgen/io/cif.py index 27164fdaf7f..0bb6413402e 100644 --- a/src/pymatgen/io/cif.py +++ b/src/pymatgen/io/cif.py @@ -233,7 +233,7 @@ def from_str(cls, string: str) -> Self: data[k].append(v.strip()) elif issue := "".join(_str).strip(): - warnings.warn(f"Possible issue in CIF file at line: {issue}") + warnings.warn(f"Possible issue in CIF file at line: {issue}", stacklevel=2) return cls(data, loops, header) @@ -684,7 +684,7 @@ def get_lattice( return self.get_lattice(data, lengths, angles, lattice_type=lattice_type) except AttributeError as exc: self.warnings.append(str(exc)) - warnings.warn(str(exc)) + warnings.warn(str(exc), stacklevel=2) else: return None @@ -734,7 +734,7 @@ def get_symops(self, data: CifBlock) -> list[SymmOp]: if isinstance(xyz, str): msg = "A 1-line symmetry op P1 CIF is detected!" - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) self.warnings.append(msg) xyz = [xyz] try: @@ -772,7 +772,7 @@ def get_symops(self, data: CifBlock) -> list[SymmOp]: if spg := space_groups.get(sg): sym_ops = list(SpaceGroup(spg).symmetry_ops) msg = msg_template.format(symmetry_label) - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) self.warnings.append(msg) break except ValueError: @@ -791,7 +791,7 @@ def get_symops(self, data: CifBlock) -> list[SymmOp]: xyz = _data["symops"] sym_ops = [SymmOp.from_xyz_str(s) for s in xyz] msg = msg_template.format(symmetry_label) - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) self.warnings.append(msg) break except Exception: @@ -818,7 +818,7 @@ def get_symops(self, data: CifBlock) -> list[SymmOp]: if not sym_ops: msg = "No _symmetry_equiv_pos_as_xyz type key found. Defaulting to P1." - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) self.warnings.append(msg) sym_ops = [SymmOp.from_xyz_str(s) for s in ("x", "y", "z")] @@ -880,7 +880,7 @@ def get_magsymops(self, data: CifBlock) -> list[MagSymmOp]: if not mag_symm_ops: msg = "No magnetic symmetry detected, using primitive symmetry." - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) self.warnings.append(msg) mag_symm_ops = [MagSymmOp.from_xyzt_str("x, y, z, 1")] @@ -958,7 +958,7 @@ def _parse_symbol(self, sym: str) -> str | None: if parsed_sym is not None and (m_sp or not re.match(rf"{parsed_sym}\d*", sym)): msg = f"{sym} parsed as {parsed_sym}" - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) self.warnings.append(msg) return parsed_sym @@ -1111,7 +1111,7 @@ def get_matching_coord( "the occupancy_tolerance, they will be rescaled. " f"The current occupancy_tolerance is set to: {self._occupancy_tolerance}" ) - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) self.warnings.append(msg) # Collect info for building Structure @@ -1255,7 +1255,7 @@ def get_matching_coord( if self.check_cif: cif_failure_reason = self.check(struct) if cif_failure_reason is not None: - warnings.warn(cif_failure_reason) + warnings.warn(cif_failure_reason, stacklevel=2) return struct return None @@ -1297,7 +1297,7 @@ def parse_structures( "The default value of primitive was changed from True to False in " "https://github.com/materialsproject/pymatgen/pull/3419. CifParser now returns the cell " "in the CIF file as is. If you want the primitive cell, please set primitive=True explicitly.", - UserWarning, + stacklevel=2, ) if primitive and symmetrized: @@ -1317,11 +1317,11 @@ def parse_structures( if on_error == "raise": raise ValueError(msg) from exc if on_error == "warn": - warnings.warn(msg) + warnings.warn(msg, stacklevel=2) self.warnings.append(msg) if self.warnings and on_error == "warn": - warnings.warn("Issues encountered while parsing CIF: " + "\n".join(self.warnings)) + warnings.warn("Issues encountered while parsing CIF: " + "\n".join(self.warnings), stacklevel=2) if not structures: raise ValueError("Invalid CIF file with no structures!") @@ -1560,7 +1560,10 @@ def __init__( to the CIF as _atom_site_{property name}. Defaults to False. """ if write_magmoms and symprec is not None: - warnings.warn("Magnetic symmetry cannot currently be detected by pymatgen, disabling symmetry detection.") + warnings.warn( + "Magnetic symmetry cannot currently be detected by pymatgen, disabling symmetry detection.", + stacklevel=2, + ) symprec = None blocks: dict[str, Any] = {} @@ -1705,7 +1708,7 @@ def __init__( "Site labels are not unique, which is not compliant with the CIF spec " "(https://www.iucr.org/__data/iucr/cifdic_html/1/cif_core.dic/Iatom_site_label.html):" f"`{atom_site_label}`.", - UserWarning, + stacklevel=2, ) blocks["_atom_site_type_symbol"] = atom_site_type_symbol diff --git a/src/pymatgen/io/common.py b/src/pymatgen/io/common.py index 44b4045a424..6a99f044b1a 100644 --- a/src/pymatgen/io/common.py +++ b/src/pymatgen/io/common.py @@ -152,7 +152,10 @@ def linear_add(self, other, scale_factor=1.0): VolumetricData corresponding to self + scale_factor * other. """ if self.structure != other.structure: - warnings.warn("Structures are different. Make sure you know what you are doing...") + warnings.warn( + "Structures are different. Make sure you know what you are doing...", + stacklevel=2, + ) if list(self.data) != list(other.data): raise ValueError("Data have different keys! Maybe one is spin-polarized and the other is not?") @@ -524,7 +527,7 @@ def __getitem__(self, item): warnings.warn( f"No parser defined for {item}. Contents are returned as a string.", - UserWarning, + stacklevel=2, ) with zopen(fpath, mode="rt", encoding="utf-8") as f: return f.read() diff --git a/src/pymatgen/io/cp2k/outputs.py b/src/pymatgen/io/cp2k/outputs.py index 3885b640e67..873301fe1df 100644 --- a/src/pymatgen/io/cp2k/outputs.py +++ b/src/pymatgen/io/cp2k/outputs.py @@ -428,10 +428,10 @@ def convergence(self): if not all(self.data["scf_converged"]): warnings.warn( "There is at least one unconverged SCF cycle in the provided CP2K calculation", - UserWarning, + stacklevel=2, ) if any(self.data["geo_opt_not_converged"]): - warnings.warn("Geometry optimization did not converge", UserWarning) + warnings.warn("Geometry optimization did not converge", stacklevel=2) def parse_energies(self): """Get the total energy from a CP2K calculation. Presently, the energy reported in the @@ -566,7 +566,7 @@ def parse_input(self): if os.path.isfile(os.path.join(self.dir, input_filename + ext)): self.input = Cp2kInput.from_file(os.path.join(self.dir, input_filename + ext)) return - warnings.warn("Original input file not found. Some info may be lost.") + warnings.warn("Original input file not found. Some info may be lost.", stacklevel=2) def parse_global_params(self): """Parse the GLOBAL section parameters from CP2K output file into a dictionary.""" @@ -711,7 +711,8 @@ def parse_cell_params(self): ] warnings.warn( - "Input file lost. Reading cell params from summary at top of output. Precision errors may result." + "Input file lost. Reading cell params from summary at top of output. Precision errors may result.", + stacklevel=2, ) cell_volume = re.compile(r"\s+CELL\|\sVolume.*\s(\d+\.\d+)") vectors = re.compile(r"\s+CELL\| Vector.*\s(-?\d+\.\d+)\s+(-?\d+\.\d+)\s+(-?\d+\.\d+)") @@ -1046,7 +1047,8 @@ def parse_mo_eigenvalues(self): while True: if "WARNING : did not converge" in line: warnings.warn( - "Convergence of eigenvalues for unoccupied subspace spin 1 did NOT converge" + "Convergence of eigenvalues for unoccupied subspace spin 1 did NOT converge", + stacklevel=2, ) next(lines) next(lines) @@ -1073,7 +1075,8 @@ def parse_mo_eigenvalues(self): while True: if "WARNING : did not converge" in line: warnings.warn( - "Convergence of eigenvalues for unoccupied subspace spin 2 did NOT converge" + "Convergence of eigenvalues for unoccupied subspace spin 2 did NOT converge", + stacklevel=2, ) next(lines) next(lines) @@ -1105,7 +1108,7 @@ def parse_mo_eigenvalues(self): "unoccupied": {Spin.up: None, Spin.down: None}, } ] - warnings.warn("Convergence of eigenvalues for one or more subspaces did NOT converge") + warnings.warn("Convergence of eigenvalues for one or more subspaces did NOT converge", stacklevel=2) self.data["eigenvalues"] = eigenvalues diff --git a/src/pymatgen/io/cp2k/sets.py b/src/pymatgen/io/cp2k/sets.py index fc4e7770a7b..b03506c3ba1 100644 --- a/src/pymatgen/io/cp2k/sets.py +++ b/src/pymatgen/io/cp2k/sets.py @@ -237,7 +237,10 @@ def __init__( ): self.kpoints = None if ot and self.kpoints: - warnings.warn("As of 2022.1, kpoints not supported with OT. Defaulting to diagonalization") + warnings.warn( + "As of 2022.1, kpoints not supported with OT. Defaulting to diagonalization", + stacklevel=2, + ) ot = False # Build the global section @@ -546,19 +549,24 @@ def match_elecs(basis_set): if basis is None: if not basis_and_potential.get(el, {}).get("basis"): raise ValueError(f"No explicit basis found for {el} and matching has failed.") - warnings.warn(f"Unable to validate basis for {el}. Exact name provided will be put in input file.") + warnings.warn( + f"Unable to validate basis for {el}. Exact name provided will be put in input file.", + stacklevel=2, + ) basis = basis_and_potential[el].get("basis") if aux_basis is None and basis_and_potential.get(el, {}).get("aux_basis"): warnings.warn( - f"Unable to validate auxiliary basis for {el}. Exact name provided will be put in input file." + f"Unable to validate auxiliary basis for {el}. Exact name provided will be put in input file.", + stacklevel=2, ) aux_basis = basis_and_potential[el].get("aux_basis") if potential is None: if basis_and_potential.get(el, {}).get("potential"): warnings.warn( - f"Unable to validate potential for {el}. Exact name provided will be put in input file." + f"Unable to validate potential for {el}. Exact name provided will be put in input file.", + stacklevel=2, ) potential = basis_and_potential.get(el, {}).get("potential") else: @@ -872,7 +880,8 @@ def activate_hybrid( if max_cutoff_radius < cutoff_radius: warnings.warn( "Provided cutoff radius exceeds half the minimum" - " distance between atoms. I hope you know what you're doing." + " distance between atoms. I hope you know what you're doing.", + stacklevel=2, ) ip_keywords: dict[str, Keyword] = {} @@ -961,7 +970,8 @@ def activate_hybrid( else: warnings.warn( "Unknown hybrid functional. Using PBE base functional and overriding all " - "settings manually. Proceed with caution." + "settings manually. Proceed with caution.", + stacklevel=2, ) pbe = PBE("ORIG", scale_c=gga_c_fraction, scale_x=gga_x_fraction) xc_functional = XCFunctional(functionals=[], subsections={"PBE": pbe}) @@ -1223,7 +1233,8 @@ def activate_vdw_potential( warnings.warn( "Reference functional will not be checked for validity. " "Calculation will fail if the reference functional " - "does not exist in the dftd3 reference data" + "does not exist in the dftd3 reference data", + stacklevel=2, ) keywords["PARAMETER_FILE_NAME"] = Keyword("PARAMETER_FILE_NAME", "dftd3.dat") keywords["REFERENCE_FUNCTIONAL"] = Keyword("REFERENCE_FUNCTIONAL", reference_functional) diff --git a/src/pymatgen/io/feff/inputs.py b/src/pymatgen/io/feff/inputs.py index f9321eb0881..94ad04dd24e 100644 --- a/src/pymatgen/io/feff/inputs.py +++ b/src/pymatgen/io/feff/inputs.py @@ -554,7 +554,10 @@ def __setitem__(self, key, val): value: value associated with key in dictionary """ if key.strip().upper() not in VALID_FEFF_TAGS: - warnings.warn(f"{key.strip()} not in VALID_FEFF_TAGS list") + warnings.warn( + f"{key.strip()} not in VALID_FEFF_TAGS list", + stacklevel=2, + ) super().__setitem__( key.strip(), Tags.proc_val(key.strip(), val.strip()) if isinstance(val, str) else val, diff --git a/src/pymatgen/io/feff/sets.py b/src/pymatgen/io/feff/sets.py index e1605fff5ca..5d200968ac9 100644 --- a/src/pymatgen/io/feff/sets.py +++ b/src/pymatgen/io/feff/sets.py @@ -191,7 +191,7 @@ def __init__( "For Molecule objects with a net charge it is recommended to set one or more" " ION tags in the input file by modifying user_tag_settings." " Consult the FEFFDictSet docstring and the FEFF10 User Guide for more information.", - UserWarning, + stacklevel=2, ) else: raise ValueError("'structure' argument must be a Structure or Molecule!") diff --git a/src/pymatgen/io/gaussian.py b/src/pymatgen/io/gaussian.py index c74c9354406..fd3ab518f4f 100644 --- a/src/pymatgen/io/gaussian.py +++ b/src/pymatgen/io/gaussian.py @@ -790,7 +790,10 @@ def _parse(self, filename): "Density Matrix:" in line or mo_coeff_patt.search(line) ): end_mo = True - warnings.warn("POP=regular case, matrix coefficients not complete") + warnings.warn( + "POP=regular case, matrix coefficients not complete", + stacklevel=2, + ) file.readline() self.eigenvectors = mat_mo @@ -926,7 +929,8 @@ def _parse(self, filename): line = file.readline() if " -- Stationary point found." not in line: warnings.warn( - f"\n{self.filename}: Optimization complete but this is not a stationary point" + f"\n{self.filename}: Optimization complete but this is not a stationary point", + stacklevel=2, ) if standard_orientation: opt_structures.append(std_structures[-1]) @@ -989,7 +993,10 @@ def _parse(self, filename): self.opt_structures = opt_structures if not terminated: - warnings.warn(f"\n{self.filename}: Termination error or bad Gaussian output file !") + warnings.warn( + f"\n{self.filename}: Termination error or bad Gaussian output file !", + stacklevel=2, + ) def _parse_hessian(self, file, structure): """Parse the hessian matrix in the output file. diff --git a/src/pymatgen/io/icet.py b/src/pymatgen/io/icet.py index 2cd40bac345..c7722bde84f 100644 --- a/src/pymatgen/io/icet.py +++ b/src/pymatgen/io/icet.py @@ -120,7 +120,10 @@ def __init__( unrecognized_kwargs = {key for key in self.sqs_kwargs if key not in self.sqs_kwarg_names[sqs_method]} if len(unrecognized_kwargs) > 0: - warnings.warn(f"Ignoring unrecognized icet {sqs_method} kwargs: {', '.join(unrecognized_kwargs)}") + warnings.warn( + f"Ignoring unrecognized icet {sqs_method} kwargs: {', '.join(unrecognized_kwargs)}", + stacklevel=2, + ) self.sqs_kwargs = { key: value for key, value in self.sqs_kwargs.items() if key in self.sqs_kwarg_names[sqs_method] diff --git a/src/pymatgen/io/lammps/data.py b/src/pymatgen/io/lammps/data.py index d241f6c41bf..2e730c218ca 100644 --- a/src/pymatgen/io/lammps/data.py +++ b/src/pymatgen/io/lammps/data.py @@ -833,7 +833,10 @@ def from_ff_and_topologies( df_topology = pd.DataFrame(np.concatenate(topo_collector[key]), columns=SECTION_HEADERS[key][1:]) df_topology["type"] = list(map(ff.maps[key].get, topo_labels[key])) if any(pd.isna(df_topology["type"])): # Throw away undefined topologies - warnings.warn(f"Undefined {key.lower()} detected and removed") + warnings.warn( + f"Undefined {key.lower()} detected and removed", + stacklevel=2, + ) df_topology = df_topology.dropna(subset=["type"]) df_topology = df_topology.reset_index(drop=True) df_topology.index += 1 diff --git a/src/pymatgen/io/lammps/inputs.py b/src/pymatgen/io/lammps/inputs.py index 5b473a16a28..fa7987dfa3a 100644 --- a/src/pymatgen/io/lammps/inputs.py +++ b/src/pymatgen/io/lammps/inputs.py @@ -273,7 +273,8 @@ def add_stage( if commands or stage_name: warnings.warn( "A stage has been passed together with commands and stage_name. This is incompatible. " - "Only the stage will be used." + "Only the stage will be used.", + stacklevel=2, ) # Make sure the given stage has the correct format @@ -467,7 +468,10 @@ def remove_command( self.stages = new_list_of_stages if n_removed == 0: - warnings.warn(f"{command} not found in the LammpsInputFile.") + warnings.warn( + f"{command} not found in the LammpsInputFile.", + stacklevel=2, + ) def append(self, lmp_input_file: LammpsInputFile) -> None: """ @@ -1105,4 +1109,7 @@ def write_lammps_inputs( elif isinstance(data, str) and os.path.isfile(data): shutil.copyfile(data, os.path.join(output_dir, data_filename)) else: - warnings.warn(f"No data file supplied. Skip writing {data_filename}.") + warnings.warn( + f"No data file supplied. Skip writing {data_filename}.", + stacklevel=2, + ) diff --git a/src/pymatgen/io/lobster/inputs.py b/src/pymatgen/io/lobster/inputs.py index 3bbe0518a11..c83b57cfe4d 100644 --- a/src/pymatgen/io/lobster/inputs.py +++ b/src/pymatgen/io/lobster/inputs.py @@ -337,7 +337,10 @@ def write_INCAR( """ # Read INCAR from file, which will be modified incar = Incar.from_file(incar_input) - warnings.warn("Please check your incar_input before using it. This method only changes three settings!") + warnings.warn( + "Please check your incar_input before using it. This method only changes three settings!", + stacklevel=2, + ) if isym in {-1, 0}: incar["ISYM"] = isym else: @@ -654,7 +657,8 @@ def _get_potcar_symbols(POTCAR_input: PathLike) -> list[str]: "Lobster up to version 4.1.0." "\n The keywords SHA256 and COPYR " "cannot be handled by Lobster" - " \n and will lead to wrong results." + " \n and will lead to wrong results.", + stacklevel=2, ) if potcar.functional != "PBE": @@ -697,7 +701,8 @@ def standard_calculations_from_vasp_files( Lobsterin with standard settings """ warnings.warn( - "Always check and test the provided basis functions. The spilling of your Lobster calculation might help" + "Always check and test the provided basis functions. The spilling of your Lobster calculation might help", + stacklevel=2, ) if option not in { diff --git a/src/pymatgen/io/lobster/outputs.py b/src/pymatgen/io/lobster/outputs.py index 421816c5bd6..90291feec7d 100644 --- a/src/pymatgen/io/lobster/outputs.py +++ b/src/pymatgen/io/lobster/outputs.py @@ -418,7 +418,10 @@ def __init__( version = "5.1.0" elif len(lines[0].split()) == 6: version = "2.2.1" - warnings.warn("Please consider using a newer LOBSTER version. See www.cohp.de.") + warnings.warn( + "Please consider using a newer LOBSTER version. See www.cohp.de.", + stacklevel=2, + ) else: raise ValueError("Unsupported LOBSTER version.") @@ -637,7 +640,8 @@ def __init__(self, filename: PathLike | None = "NcICOBILIST.lobster") -> None: self.orbital_wise = True warnings.warn( "This is an orbitalwise NcICOBILIST.lobster file. " - "Currently, the orbitalwise information is not read!" + "Currently, the orbitalwise information is not read!", + stacklevel=2, ) break # condition has only to be met once @@ -1389,8 +1393,14 @@ def __init__( structure (Structure): Structure object. efermi (float): Fermi level in eV. """ - warnings.warn("Make sure all relevant FATBAND files were generated and read in!") - warnings.warn("Use Lobster 3.2.0 or newer for fatband calculations!") + warnings.warn( + "Make sure all relevant FATBAND files were generated and read in!", + stacklevel=2, + ) + warnings.warn( + "Use Lobster 3.2.0 or newer for fatband calculations!", + stacklevel=2, + ) if structure is None: raise ValueError("A structure object has to be provided") diff --git a/src/pymatgen/io/multiwfn.py b/src/pymatgen/io/multiwfn.py index f644ada586c..558da06417f 100644 --- a/src/pymatgen/io/multiwfn.py +++ b/src/pymatgen/io/multiwfn.py @@ -401,7 +401,10 @@ def sort_cps_by_distance( sorted_atoms = sort_cps_by_distance(np.array(cp_desc["pos_ang"]), atom_info) if sorted_atoms[1][0] > dist_threshold_bond: - warnings.warn("Warning: bond CP is far from bonding atoms") + warnings.warn( + "Warning: bond CP is far from bonding atoms", + stacklevel=2, + ) # Assume only two atoms involved in bond modified_organized_cps["bond"][cp_name]["atom_inds"] = sorted([ca[1] for ca in sorted_atoms[:2]]) @@ -428,7 +431,10 @@ def sort_cps_by_distance( sorted_atoms = sort_cps_by_distance(np.array(cp_desc["pos_ang"]), atom_info) if sorted_atoms[1][0] > dist_threshold_bond: - warnings.warn("Warning: bond CP is far from bonding atoms") + warnings.warn( + "Warning: bond CP is far from bonding atoms", + stacklevel=2, + ) bond_atoms_list = sorted([ca[1] for ca in sorted_atoms[:2]]) @@ -439,7 +445,10 @@ def sort_cps_by_distance( max_close_dist = sorted_bonds[2][0] if max_close_dist > dist_threshold_ring_cage: - warnings.warn("Warning: ring CP is far from closest bond CPs.") + warnings.warn( + "Warning: ring CP is far from closest bond CPs.", + stacklevel=2, + ) # Assume that the three closest bonds are all part of the ring bond_names = [bcp[1] for bcp in sorted_bonds[:3]] @@ -466,7 +475,10 @@ def sort_cps_by_distance( # Warn if the three closest bonds are further than the max distance if max_close_dist > dist_threshold_ring_cage: - warnings.warn("Warning: cage CP is far from closest ring CPs.") + warnings.warn( + "Warning: cage CP is far from closest ring CPs.", + stacklevel=2, + ) # Assume that the three closest rings are all part of the cage ring_names = [rcp[1] for rcp in sorted_rings[:3]] @@ -536,7 +548,10 @@ def process_multiwfn_qtaim( remapped_atoms, missing_atoms = map_atoms_cps(molecule, qtaim_descriptors["atom"], max_distance=max_distance_atom) if len(missing_atoms) > 0: - warnings.warn(f"Some atoms not mapped to atom CPs! Indices: {missing_atoms}") + warnings.warn( + f"Some atoms not mapped to atom CPs! Indices: {missing_atoms}", + stacklevel=2, + ) qtaim_descriptors["atom"] = remapped_atoms diff --git a/src/pymatgen/io/nwchem.py b/src/pymatgen/io/nwchem.py index 2e7e9f22efd..523aa9a53a9 100644 --- a/src/pymatgen/io/nwchem.py +++ b/src/pymatgen/io/nwchem.py @@ -139,7 +139,10 @@ def __init__( if NWCHEM_BASIS_LIBRARY is not None: for b in set(self.basis_set.values()): if re.sub(r"\*", "s", b.lower()) not in NWCHEM_BASIS_LIBRARY: - warnings.warn(f"Basis set {b} not in NWCHEM_BASIS_LIBRARY") + warnings.warn( + f"Basis set {b} not in NWCHEM_BASIS_LIBRARY", + stacklevel=2, + ) self.basis_set_option = basis_set_option diff --git a/src/pymatgen/io/openff.py b/src/pymatgen/io/openff.py index 5294f668d19..5bd1682aab1 100644 --- a/src/pymatgen/io/openff.py +++ b/src/pymatgen/io/openff.py @@ -20,7 +20,8 @@ unit = None warnings.warn( "To use the pymatgen.io.openff module install openff-toolkit and openff-units" - "with `conda install -c conda-forge openff-toolkit openff-units`." + "with `conda install -c conda-forge openff-toolkit openff-units`.", + stacklevel=2, ) diff --git a/src/pymatgen/io/qchem/outputs.py b/src/pymatgen/io/qchem/outputs.py index f9cc304c72f..d483e7ecba3 100644 --- a/src/pymatgen/io/qchem/outputs.py +++ b/src/pymatgen/io/qchem/outputs.py @@ -2308,7 +2308,8 @@ def check_for_structure_changes(mol1: Molecule, mol2: Molecule) -> str: if site.specie.symbol != mol2[ii].specie.symbol: warnings.warn( "Comparing molecules with different atom ordering! " - "Turning off special treatment for coordinating metals." + "Turning off special treatment for coordinating metals.", + stacklevel=2, ) special_elements = [] diff --git a/src/pymatgen/io/qchem/sets.py b/src/pymatgen/io/qchem/sets.py index c3321d150d6..d68a1714218 100644 --- a/src/pymatgen/io/qchem/sets.py +++ b/src/pymatgen/io/qchem/sets.py @@ -555,7 +555,7 @@ def __init__( if rem["solvent_method"] != "pcm": warnings.warn( "The solvent section will be ignored unless solvent_method=pcm!", - UserWarning, + stacklevel=2, ) if sec == "smx": smx |= lower_and_check_unique(sec_dict) @@ -589,17 +589,17 @@ def __init__( if self.cmirs_solvent is not None and v == "0": warnings.warn( "Setting IDEFESR=0 will disable the CMIRS calculation you requested!", - UserWarning, + stacklevel=2, ) if self.cmirs_solvent is None and v == "1": warnings.warn( "Setting IDEFESR=1 will have no effect unless you specify a cmirs_solvent!", - UserWarning, + stacklevel=2, ) if k == "dielst" and rem["solvent_method"] != "isosvp": warnings.warn( "Setting DIELST will have no effect unless you specify a solvent_method=isosvp!", - UserWarning, + stacklevel=2, ) svp[k] = v diff --git a/src/pymatgen/io/shengbte.py b/src/pymatgen/io/shengbte.py index eeee8a4fd61..a101edd3e55 100644 --- a/src/pymatgen/io/shengbte.py +++ b/src/pymatgen/io/shengbte.py @@ -180,7 +180,10 @@ def to_file(self, filename: str = "CONTROL") -> None: """ for param in self.required_params: if param not in self.as_dict(): - warnings.warn(f"Required parameter {param!r} not specified!") + warnings.warn( + f"Required parameter {param!r} not specified!", + stacklevel=2, + ) alloc_dict = _get_subdict(self, self.allocations_keys) alloc_nml = f90nml.Namelist({"allocations": alloc_dict}) diff --git a/src/pymatgen/io/vasp/inputs.py b/src/pymatgen/io/vasp/inputs.py index af0390b0064..11167289055 100644 --- a/src/pymatgen/io/vasp/inputs.py +++ b/src/pymatgen/io/vasp/inputs.py @@ -264,6 +264,7 @@ def from_file( warnings.warn( "check_for_POTCAR is deprecated. Use check_for_potcar instead.", DeprecationWarning, + stacklevel=2, ) check_for_potcar = cast(bool, kwargs.pop("check_for_POTCAR")) @@ -468,6 +469,7 @@ def from_str( warnings.warn( f"Elements in POSCAR cannot be determined. Defaulting to false names {atomic_symbols}.", BadPoscarWarning, + stacklevel=2, ) # Read the atomic coordinates @@ -483,6 +485,7 @@ def from_str( warnings.warn( "Selective dynamics values must be either 'T' or 'F'.", BadPoscarWarning, + stacklevel=2, ) # Warn when elements contains Fluorine (F) (#3539) @@ -493,6 +496,7 @@ def from_str( "Make sure the 4th-6th entry each position line is selective dynamics info." ), BadPoscarWarning, + stacklevel=2, ) selective_dynamics.append([value == "T" for value in tokens[3:6]]) @@ -502,6 +506,7 @@ def from_str( warnings.warn( "Ignoring selective dynamics tag, as no ionic degrees of freedom were fixed.", BadPoscarWarning, + stacklevel=2, ) struct = Structure( @@ -625,7 +630,11 @@ def get_str( # VASP is strict about the format when reading this quantity lines.append(" ".join(f" {val: .7E}" for val in velo)) except Exception: - warnings.warn("Lattice velocities are missing or corrupted.", BadPoscarWarning) + warnings.warn( + "Lattice velocities are missing or corrupted.", + BadPoscarWarning, + stacklevel=2, + ) if self.velocities: try: @@ -633,7 +642,11 @@ def get_str( for velo in self.velocities: lines.append(" ".join(format_str.format(val) for val in velo)) except Exception: - warnings.warn("Velocities are missing or corrupted.", BadPoscarWarning) + warnings.warn( + "Velocities are missing or corrupted.", + BadPoscarWarning, + stacklevel=2, + ) if self.predictor_corrector: lines.append("") @@ -647,6 +660,7 @@ def get_str( warnings.warn( "Preamble information missing or corrupt. Writing Poscar with no predictor corrector data.", BadPoscarWarning, + stacklevel=2, ) return "\n".join(lines) + "\n" @@ -2007,7 +2021,10 @@ def __init__(self, data: str, symbol: str | None = None) -> None: try: keywords[key] = self.parse_functions[key](val) # type: ignore[operator] except KeyError: - warnings.warn(f"Ignoring unknown variable type {key}") + warnings.warn( + f"Ignoring unknown variable type {key}", + stacklevel=2, + ) PSCTR: dict[str, Any] = {} @@ -2081,6 +2098,7 @@ def __init__(self, data: str, symbol: str | None = None) -> None: f"POTCAR data with symbol {self.symbol} is not known to pymatgen. Your " "POTCAR may be corrupted or pymatgen's POTCAR database is incomplete.", UnknownPotcarWarning, + stacklevel=2, ) def __eq__(self, other: object) -> bool: @@ -2112,7 +2130,10 @@ def __repr__(self) -> str: def electron_configuration(self) -> list[tuple[int, str, int]] | None: """Electronic configuration of the PotcarSingle.""" if not self.nelectrons.is_integer(): - warnings.warn("POTCAR has non-integer charge, electron configuration not well-defined.") + warnings.warn( + "POTCAR has non-integer charge, electron configuration not well-defined.", + stacklevel=2, + ) return None el = Element.from_Z(self.atomic_no) @@ -2421,7 +2442,10 @@ def from_file(cls, filename: PathLike) -> Self: return cls(file.read(), symbol=symbol or None) except UnicodeDecodeError: - warnings.warn("POTCAR contains invalid unicode errors. We will attempt to read it by ignoring errors.") + warnings.warn( + "POTCAR contains invalid unicode errors. We will attempt to read it by ignoring errors.", + stacklevel=2, + ) with codecs.open(str(filename), "r", encoding="utf-8", errors="ignore") as file: return cls(file.read(), symbol=symbol or None) @@ -2706,7 +2730,10 @@ def _gen_potcar_summary_stats( if os.path.isdir(cpsp_dir): func_dir_exist[func] = func_dir else: - warnings.warn(f"missing {func_dir} POTCAR directory") + warnings.warn( + f"missing {func_dir} POTCAR directory", + stacklevel=2, + ) # Use append = True if a new POTCAR library is released to add new summary stats # without completely regenerating the dict of summary stats diff --git a/src/pymatgen/io/vasp/outputs.py b/src/pymatgen/io/vasp/outputs.py index 0e3182cf68b..acb5f43bc87 100644 --- a/src/pymatgen/io/vasp/outputs.py +++ b/src/pymatgen/io/vasp/outputs.py @@ -157,7 +157,10 @@ def _vasprun_float(flt: float | str) -> float: flt = cast(str, flt) _flt: str = flt.strip() if _flt == "*" * len(_flt): - warnings.warn("Float overflow (*******) encountered in vasprun") + warnings.warn( + "Float overflow (*******) encountered in vasprun", + stacklevel=2, + ) return np.nan raise @@ -349,7 +352,11 @@ def __init__( msg = f"{filename} is an unconverged VASP run.\n" msg += f"Electronic convergence reached: {self.converged_electronic}.\n" msg += f"Ionic convergence reached: {self.converged_ionic}." - warnings.warn(msg, UnconvergedVASPWarning) + warnings.warn( + msg, + UnconvergedVASPWarning, + stacklevel=2, + ) def _parse( self, @@ -484,7 +491,7 @@ def _parse( else: warnings.warn( "Additional unlabelled dielectric data in vasprun.xml are stored as unlabelled.", - UserWarning, + stacklevel=2, ) label = "unlabelled" # VASP 6+ has labels for the density and current @@ -537,7 +544,6 @@ def _parse( raise warnings.warn( "XML is malformed. Parsing has stopped but partial data is available.", - UserWarning, stacklevel=2, ) @@ -689,7 +695,8 @@ def final_energy(self) -> float: warnings.warn( "Calculation does not have a total energy. " "Possibly a GW or similar kind of run. " - "Infinity is returned." + "Infinity is returned.", + stacklevel=2, ) return float("inf") @@ -801,7 +808,10 @@ def run_type(self) -> str: run_type = "LDA" else: run_type = "unknown" - warnings.warn("Unknown run type!") + warnings.warn( + "Unknown run type!", + stacklevel=2, + ) if self.is_hubbard or self.parameters.get("LDAU", True): run_type += "+U" @@ -1216,7 +1226,10 @@ def get_potcars(self, path: PathLike | bool) -> Potcar | None: except Exception: continue - warnings.warn("No POTCAR file with matching TITEL fields was found in\n" + "\n ".join(potcar_paths)) + warnings.warn( + "No POTCAR file with matching TITEL fields was found in\n" + "\n ".join(potcar_paths), + stacklevel=2, + ) return None @@ -4441,7 +4454,10 @@ def get_band_structure_from_vasp_multiple_branches( branches.append(run.get_band_structure(efermi=efermi)) else: # TODO: It might be better to throw an exception - warnings.warn(f"Skipping {dname}. Unable to find {xml_file}") + warnings.warn( + f"Skipping {dname}. Unable to find {xml_file}", + stacklevel=2, + ) return get_reconstructed_band_structure(branches, efermi) @@ -5274,7 +5290,10 @@ def get_parchg( A Chgcar object. """ if phase and not np.all(self.kpoints[kpoint] == 0.0): - warnings.warn("phase is True should only be used for the Gamma kpoint! I hope you know what you're doing!") + warnings.warn( + "phase is True should only be used for the Gamma kpoint! I hope you know what you're doing!", + stacklevel=2, + ) # Scaling of ng for the fft grid, need to restore value at the end temp_ng = self.ng @@ -5608,7 +5627,8 @@ def cder(self) -> np.ndarray: if self.cder_real.shape[0] != self.cder_real.shape[1]: # pragma: no cover warnings.warn( "Not all band pairs are present in the WAVEDER file." - "If you want to get all the matrix elements set LVEL=.True. in the INCAR." + "If you want to get all the matrix elements set LVEL=.True. in the INCAR.", + stacklevel=2, ) return self.cder_real + 1j * self.cder_imag diff --git a/src/pymatgen/io/vasp/sets.py b/src/pymatgen/io/vasp/sets.py index 45898dfef38..11e5e555816 100644 --- a/src/pymatgen/io/vasp/sets.py +++ b/src/pymatgen/io/vasp/sets.py @@ -268,6 +268,7 @@ def __post_init__(self) -> None: "will generate a KPOINTS file and ignore KSPACING." "Remove the `user_kpoints_settings` argument to enable KSPACING.", BadInputSetWarning, + stacklevel=2, ) if self.vdw: @@ -293,6 +294,7 @@ def __post_init__(self) -> None: "the configuration file may not be available in the selected " "functional.", BadInputSetWarning, + stacklevel=2, ) if self.user_potcar_settings: @@ -304,6 +306,7 @@ def __post_init__(self) -> None: "subclass of a desired input set and override the POTCAR in " "the subclass to be explicit on the differences.", BadInputSetWarning, + stacklevel=2, ) for key, val in self.user_potcar_settings.items(): self._config_dict["POTCAR"][key] = val @@ -431,6 +434,7 @@ def structure(self, structure: Structure | None) -> None: "Yb_2 is known to often give bad results since Yb has oxidation state 3+ in most compounds.\n" "See https://github.com/materialsproject/pymatgen/issues/2968 for details.", BadInputSetWarning, + stacklevel=2, ) if self.standardize and self.sym_prec: structure = standardize_structure( @@ -580,7 +584,8 @@ def incar(self) -> Incar: warnings.warn( "Co without an oxidation state is initialized as low spin by default in Pymatgen. " "If this default behavior is not desired, please set the spin on the magmom on the " - "site directly to ensure correct initialization." + "site directly to ensure correct initialization.", + stacklevel=2, ) mag.append(setting.get(str(site.specie))) else: @@ -588,7 +593,8 @@ def incar(self) -> Incar: warnings.warn( "Co without an oxidation state is initialized as low spin by default in Pymatgen. " "If this default behavior is not desired, please set the spin on the magmom on the " - "site directly to ensure correct initialization." + "site directly to ensure correct initialization.", + stacklevel=2, ) mag.append(setting.get(site.specie.symbol, 0.6)) incar[key] = mag @@ -652,6 +658,7 @@ def incar(self) -> Incar: warnings.warn( "LASPH = True should be set for +U, meta-GGAs, hybrids, and vdW-DFT", BadInputSetWarning, + stacklevel=2, ) # Apply previous INCAR settings, be careful not to override user_incar_settings @@ -670,7 +677,6 @@ def incar(self) -> Incar: "multiplet and should typically be an integer. You are likely " "better off changing the values of MAGMOM or simply setting " "NUPDOWN directly in your INCAR settings.", - UserWarning, stacklevel=2, ) auto_updates["NUPDOWN"] = nupdown @@ -687,6 +693,7 @@ def incar(self) -> Incar: warnings.warn( "Hybrid functionals only support Algo = All, Damped, or Normal.", BadInputSetWarning, + stacklevel=2, ) if self.auto_ismear: @@ -740,6 +747,7 @@ def incar(self) -> Incar: "generates an adequate number of KPOINTS, lower KSPACING, or " "set ISMEAR = 0", BadInputSetWarning, + stacklevel=2, ) ismear = incar.get("ISMEAR", 1) @@ -757,7 +765,7 @@ def incar(self) -> Incar: warnings.warn( f"{msg} See VASP recommendations on ISMEAR for metals (https://www.vasp.at/wiki/index.php/ISMEAR).", BadInputSetWarning, - stacklevel=1, + stacklevel=2, ) return incar @@ -964,6 +972,7 @@ def potcar(self) -> Potcar: f"is known to correspond with functionals {p_single.identify_potcar(mode='data')[0]}. " "Please verify that you are using the right POTCARs!", BadInputSetWarning, + stacklevel=2, ) return potcar @@ -1036,7 +1045,8 @@ def override_from_prev_calc(self, prev_calc_dir: PathLike = ".") -> Self: "Use of standardize=True with from_prev_run is not " "recommended as there is no guarantee the copied " "files will be appropriate for the standardized " - "structure." + "structure.", + stacklevel=2, ) files_to_transfer = {} @@ -1355,7 +1365,10 @@ class MPScanRelaxSet(VaspInputSet): def __post_init__(self) -> None: super().__post_init__() if self.vdw and self.vdw != "rvv10": - warnings.warn("Use of van der waals functionals other than rVV10 with SCAN is not supported at this time. ") + warnings.warn( + "Use of van der waals functionals other than rVV10 with SCAN is not supported at this time. ", + stacklevel=2, + ) # Delete any vdw parameters that may have been added to the INCAR vdw_par = loadfn(f"{MODULE_DIR}/vdW_parameters.yaml") for k in vdw_par[self.vdw]: @@ -1554,7 +1567,7 @@ def __post_init__(self) -> None: if self.user_potcar_functional.upper() != default_potcars: warnings.warn( f"{self.user_potcar_functional=} is inconsistent with the recommended {default_potcars}.", - UserWarning, + stacklevel=2, ) if self.xc_functional.upper() == "R2SCAN": @@ -1789,14 +1802,18 @@ def __post_init__(self) -> None: ) if (mode != "uniform" or self.nedos < 2000) and self.optics: - warnings.warn("It is recommended to use Uniform mode with a high NEDOS for optics calculations.") + warnings.warn( + "It is recommended to use Uniform mode with a high NEDOS for optics calculations.", + stacklevel=2, + ) if self.standardize: warnings.warn( "Use of standardize=True with from_prev_run is not " "recommended as there is no guarantee the copied " "files will be appropriate for the standardized" - " structure. copy_chgcar is enforced to be false." + " structure. copy_chgcar is enforced to be false.", + stacklevel=2, ) self.copy_chgcar = False @@ -2804,7 +2821,10 @@ class LobsterSet(VaspInputSet): def __post_init__(self) -> None: super().__post_init__() - warnings.warn("Make sure that all parameters are okay! This is a brand new implementation.") + warnings.warn( + "Make sure that all parameters are okay! This is a brand new implementation.", + stacklevel=2, + ) if self.user_potcar_functional in ["PBE_52", "PBE_64"]: warnings.warn( @@ -2812,6 +2832,7 @@ def __post_init__(self) -> None: "Basis functions for elements with obsoleted, updated or newly added POTCARs in " f"{self.user_potcar_functional} will not be available and may cause errors or inaccuracies.", BadInputSetWarning, + stacklevel=2, ) if self.isym not in {-1, 0}: raise ValueError("Lobster cannot digest WAVEFUNCTIONS with symmetry. isym must be -1 or 0") diff --git a/src/pymatgen/symmetry/analyzer.py b/src/pymatgen/symmetry/analyzer.py index f20fd187820..cd54afaab22 100644 --- a/src/pymatgen/symmetry/analyzer.py +++ b/src/pymatgen/symmetry/analyzer.py @@ -275,7 +275,7 @@ def _get_symmetry(self) -> tuple[NDArray, NDArray]: vectors in scaled positions. """ with warnings.catch_warnings(): - # TODO: DeprecationWarning: Use get_magnetic_symmetry() for cell with magnetic moments. + # TODO: get DeprecationWarning: Use get_magnetic_symmetry() for cell with magnetic moments. warnings.filterwarnings("ignore", message="Use get_magnetic_symmetry", category=DeprecationWarning) dct = spglib.get_symmetry(self._cell, symprec=self._symprec, angle_tolerance=self._angle_tol) @@ -1674,7 +1674,8 @@ def generate_full_symmops( if len(full) > 1000: warnings.warn( f"{len(full)} matrices have been generated. The tol may be too small. Please terminate" - " and rerun with a different tolerance." + " and rerun with a different tolerance.", + stacklevel=2, ) d = np.abs(full - identity) < tol diff --git a/src/pymatgen/symmetry/bandstructure.py b/src/pymatgen/symmetry/bandstructure.py index c30fdec3aaa..aace58d5964 100644 --- a/src/pymatgen/symmetry/bandstructure.py +++ b/src/pymatgen/symmetry/bandstructure.py @@ -204,7 +204,8 @@ def _get_hin_kpath(self, symprec, angle_tolerance, atol, tri): warn( "K-path from the Hinuma et al. convention has been transformed to the basis of the reciprocal lattice" - "of the input structure. Use `KPathSeek` for the path in the original author-intended basis." + "of the input structure. Use `KPathSeek` for the path in the original author-intended basis.", + stacklevel=2, ) return bs diff --git a/src/pymatgen/symmetry/groups.py b/src/pymatgen/symmetry/groups.py index 62a0c8df28a..e6a4c762bd8 100644 --- a/src/pymatgen/symmetry/groups.py +++ b/src/pymatgen/symmetry/groups.py @@ -106,7 +106,8 @@ def is_subgroup(self, supergroup: SymmetryGroup) -> bool: """ warnings.warn( "This is not fully functional. Only trivial subsets are tested right now. " - "This will not work if the crystallographic directions of the two groups are different." + "This will not work if the crystallographic directions of the two groups are different.", + stacklevel=2, ) return set(self.symmetry_ops).issubset(supergroup.symmetry_ops) @@ -121,7 +122,8 @@ def is_supergroup(self, subgroup: SymmetryGroup) -> bool: """ warnings.warn( "This is not fully functional. Only trivial subsets are tested right now. " - "This will not work if the crystallographic directions of the two groups are different." + "This will not work if the crystallographic directions of the two groups are different.", + stacklevel=2, ) return set(subgroup.symmetry_ops).issubset(self.symmetry_ops) @@ -229,7 +231,8 @@ def is_subgroup(self, supergroup: PointGroup) -> bool: raise NotImplementedError warnings.warn( "This is not fully functional. Only trivial subsets are tested right now. " - "This will not work if the crystallographic directions of the two groups are different." + "This will not work if the crystallographic directions of the two groups are different.", + stacklevel=2, ) return set(self.symmetry_ops).issubset(supergroup.symmetry_ops) @@ -376,7 +379,8 @@ def __init__(self, int_symbol: str, hexagonal: bool = True) -> None: self.full_symbol = spg["hermann_mauguin_u"] warnings.warn( f"Full symbol not available, falling back to short Hermann Mauguin symbol " - f"{self.symbol} instead" + f"{self.symbol} instead", + stacklevel=2, ) self.point_group = spg["point_group"] self.int_number = spg["number"] diff --git a/src/pymatgen/symmetry/kpath.py b/src/pymatgen/symmetry/kpath.py index 3a7fd573013..77c247a5538 100644 --- a/src/pymatgen/symmetry/kpath.py +++ b/src/pymatgen/symmetry/kpath.py @@ -152,7 +152,11 @@ def __init__(self, structure: Structure, symprec: float = 0.01, angle_tolerance= """ if "magmom" in structure.site_properties: warn( - "'magmom' entry found in site properties but will be ignored for the Setyawan and Curtarolo convention." + ( + "'magmom' entry found in site properties but will be ignored " + "for the Setyawan and Curtarolo convention." + ), + stacklevel=2, ) super().__init__(structure, symprec=symprec, angle_tolerance=angle_tolerance, atol=atol) @@ -168,7 +172,8 @@ def __init__(self, structure: Structure, symprec: float = 0.01, angle_tolerance= if not np.allclose(self._structure.lattice.matrix, self._prim.lattice.matrix, atol=atol): warn( "The input structure does not match the expected standard primitive! " - "The path may be incorrect. Use at your own risk." + "The path may be incorrect. Use at your own risk.", + stacklevel=2, ) lattice_type = self._sym.get_lattice_type() @@ -182,7 +187,7 @@ def __init__(self, structure: Structure, symprec: float = 0.01, angle_tolerance= elif "I" in spg_symbol: self._kpath = self.bcc() else: - warn(f"Unexpected value for {spg_symbol=}") + warn(f"Unexpected value for {spg_symbol=}", stacklevel=2) elif lattice_type == "tetragonal": if "P" in spg_symbol: @@ -195,7 +200,7 @@ def __init__(self, structure: Structure, symprec: float = 0.01, angle_tolerance= else: self._kpath = self.bctet2(c, a) else: - warn(f"Unexpected value for {spg_symbol=}") + warn(f"Unexpected value for {spg_symbol=}", stacklevel=2) elif lattice_type == "orthorhombic": a = self._conv.lattice.abc[0] @@ -219,7 +224,7 @@ def __init__(self, structure: Structure, symprec: float = 0.01, angle_tolerance= elif "C" in spg_symbol or "A" in spg_symbol: self._kpath = self.orcc(a, b, c) else: - warn(f"Unexpected value for {spg_symbol=}") + warn(f"Unexpected value for {spg_symbol=}", stacklevel=2) elif lattice_type == "hexagonal": self._kpath = self.hex() @@ -253,7 +258,7 @@ def __init__(self, structure: Structure, symprec: float = 0.01, angle_tolerance= if b * cos(alpha * pi / 180) / c + b**2 * sin(alpha * pi / 180) ** 2 / a**2 > 1: self._kpath = self.mclc5(a, b, c, alpha * pi / 180) else: - warn(f"Unexpected value for {spg_symbol=}") + warn(f"Unexpected value for {spg_symbol=}", stacklevel=2) elif lattice_type == "triclinic": kalpha = self._rec_lattice.parameters[3] @@ -269,7 +274,7 @@ def __init__(self, structure: Structure, symprec: float = 0.01, angle_tolerance= self._kpath = self.trib() else: - warn(f"Unknown lattice type {lattice_type}") + warn(f"Unknown lattice type {lattice_type}", stacklevel=2) @property def conventional(self): @@ -909,7 +914,7 @@ def __init__( site_data: list[Composition] = species if not system_is_tri: - warn("Non-zero 'magmom' data will be used to define unique atoms in the cell.") + warn("Non-zero 'magmom' data will be used to define unique atoms in the cell.", stacklevel=2) site_data = zip(species, [tuple(vec) for vec in sp["magmom"]], strict=True) # type: ignore[assignment] unique_species: list[SpeciesLike] = [] @@ -1069,7 +1074,8 @@ def __init__( print("reducible") warn( "The unit cell of the input structure is not fully reduced!" - "The path may be incorrect. Use at your own risk." + "The path may be incorrect. Use at your own risk.", + stacklevel=2, ) if magmom_axis is None: @@ -1153,7 +1159,8 @@ def _get_ksymm_kpath(self, has_magmoms, magmom_axis, axis_specified, symprec, an if "magmom" in self._structure.site_properties: warn( "The parameter has_magmoms is False, but site_properties contains the key magmom." - "This property will be removed and could result in different symmetry operations." + "This property will be removed and could result in different symmetry operations.", + stacklevel=2, ) self._structure.remove_site_property("magmom") sga = SpacegroupAnalyzer(self._structure) @@ -1634,7 +1641,8 @@ def _convert_all_magmoms_to_vectors(self, magmom_axis, axis_specified): if "magmom" not in struct.site_properties: warn( "The 'magmom' property is not set in the structure's site properties." - "All magnetic moments are being set to zero." + "All magnetic moments are being set to zero.", + stacklevel=2, ) struct.add_site_property("magmom", [np.array([0, 0, 0]) for _ in range(len(struct))]) @@ -1654,7 +1662,10 @@ def _convert_all_magmoms_to_vectors(self, magmom_axis, axis_specified): new_magmoms.append(magmom * magmom_axis) if found_scalar and not axis_specified: - warn("At least one magmom had a scalar value and magmom_axis was not specified. Defaulted to z+ spinor.") + warn( + "At least one magmom had a scalar value and magmom_axis was not specified. Defaulted to z+ spinor.", + stacklevel=2, + ) struct.remove_site_property("magmom") struct.add_site_property("magmom", new_magmoms) diff --git a/src/pymatgen/transformations/advanced_transformations.py b/src/pymatgen/transformations/advanced_transformations.py index 2537241b653..7ac06f99026 100644 --- a/src/pymatgen/transformations/advanced_transformations.py +++ b/src/pymatgen/transformations/advanced_transformations.py @@ -360,7 +360,8 @@ def apply_transformation( if structure.is_ordered: warnings.warn( - f"Enumeration skipped for structure with composition {structure.composition} because it is ordered" + f"Enumeration skipped for structure with composition {structure.composition} because it is ordered", + stacklevel=2, ) structures = [structure.copy()] @@ -392,7 +393,7 @@ def apply_transformation( if structures: break except EnumError: - warnings.warn(f"Unable to enumerate for {max_cell_size = }") + warnings.warn(f"Unable to enumerate for {max_cell_size = }", stacklevel=2) if structures is None: raise ValueError("Unable to enumerate") @@ -580,7 +581,8 @@ def __init__( warnings.warn( "Use care when using a non-standard order parameter, " "though it can be useful in some cases it can also " - "lead to unintended behavior. Consult documentation." + "lead to unintended behavior. Consult documentation.", + stacklevel=2, ) self.order_parameter = order_parameter @@ -846,7 +848,8 @@ def apply_transformation( f"Specified max cell size ({enum_kwargs['max_cell_size']}) is " "smaller than the minimum enumerable cell size " f"({enum_kwargs['min_cell_size']}), changing max cell size to " - f"{enum_kwargs['min_cell_size']}" + f"{enum_kwargs['min_cell_size']}", + stacklevel=2, ) enum_kwargs["max_cell_size"] = enum_kwargs["min_cell_size"] else: diff --git a/tests/entries/test_compatibility.py b/tests/entries/test_compatibility.py index 2aaa4cb02ce..ab4f53bf10f 100644 --- a/tests/entries/test_compatibility.py +++ b/tests/entries/test_compatibility.py @@ -594,7 +594,7 @@ def test_process_entries(self): assert len(entries) == 2 def test_parallel_process_entries(self): - # TODO: DeprecationWarning: This process (pid=xxxx) is multi-threaded, + # TODO: get DeprecationWarning: This process (pid=xxxx) is multi-threaded, # use of fork() may lead to deadlocks in the child. # pid = os.fork() with pytest.raises( From 7837761df00a488ef6c53eb09108351e5f0ef174 Mon Sep 17 00:00:00 2001 From: "Haoyu (Daniel) YANG" Date: Wed, 11 Dec 2024 09:53:22 +0800 Subject: [PATCH 04/12] Fix `assert_str_content_equal`, add tests for testing utils (#4205) * clean up comment and docstring * change reference to stable doc instead of latest * fix assert_str_content_equal and add test * clean up assert_str_content_equal usage * make error msg more detailed * don't force positional as err msg contains expected and actual tag * fix quote * make assert_msonable staticmethod * sort methods alphabetically * add list of methods * comment out test_symmetry_ops * skip failing tests * only comment out assert to reduce change * simplify single file module * make module dir private * remove test for module dir * add test for non existent structure * add more tests * more human readable err msg and test --- src/pymatgen/util/testing.py | 207 ++++++++++++++++++++++++++ src/pymatgen/util/testing/__init__.py | 151 ------------------- tests/analysis/test_graphs.py | 6 +- tests/core/test_structure.py | 2 +- tests/symmetry/test_maggroups.py | 11 +- tests/util/test_testing.py | 156 +++++++++++++++++++ 6 files changed, 377 insertions(+), 156 deletions(-) create mode 100644 src/pymatgen/util/testing.py delete mode 100644 src/pymatgen/util/testing/__init__.py create mode 100644 tests/util/test_testing.py diff --git a/src/pymatgen/util/testing.py b/src/pymatgen/util/testing.py new file mode 100644 index 00000000000..4e9bb8bbccf --- /dev/null +++ b/src/pymatgen/util/testing.py @@ -0,0 +1,207 @@ +"""This module implements testing utilities for materials science codes. + +While the primary use is within pymatgen, the functionality is meant to +be useful for external materials science codes as well. For instance, obtaining +example crystal structures to perform tests, specialized assert methods for +materials science, etc. +""" + +from __future__ import annotations + +import json +import pickle # use pickle over cPickle to get traceback in case of errors +import string +from pathlib import Path +from typing import TYPE_CHECKING +from unittest import TestCase + +import pytest +from monty.json import MontyDecoder, MontyEncoder, MSONable +from monty.serialization import loadfn + +from pymatgen.core import ROOT, SETTINGS + +if TYPE_CHECKING: + from collections.abc import Sequence + from typing import Any, ClassVar + + from pymatgen.core import Structure + from pymatgen.util.typing import PathLike + +_MODULE_DIR: Path = Path(__file__).absolute().parent + +STRUCTURES_DIR: Path = _MODULE_DIR / "structures" + +TEST_FILES_DIR: Path = Path(SETTINGS.get("PMG_TEST_FILES_DIR", f"{ROOT}/../tests/files")) +VASP_IN_DIR: str = f"{TEST_FILES_DIR}/io/vasp/inputs" +VASP_OUT_DIR: str = f"{TEST_FILES_DIR}/io/vasp/outputs" + +# Fake POTCARs have original header information, meaning properties like number of electrons, +# nuclear charge, core radii, etc. are unchanged (important for testing) while values of the and +# pseudopotential kinetic energy corrections are scrambled to avoid VASP copyright infringement +FAKE_POTCAR_DIR: str = f"{VASP_IN_DIR}/fake_potcars" + + +class PymatgenTest(TestCase): + """Extends unittest.TestCase with several convenient methods for testing: + - assert_msonable: Test if an object is MSONable and return the serialized object. + - assert_str_content_equal: Test if two string are equal (ignore whitespaces). + - get_structure: Load a Structure with its formula. + - serialize_with_pickle: Test if object(s) can be (de)serialized with `pickle`. + """ + + # dict of lazily-loaded test structures (initialized to None) + TEST_STRUCTURES: ClassVar[dict[PathLike, Structure | None]] = dict.fromkeys(STRUCTURES_DIR.glob("*")) + + @pytest.fixture(autouse=True) + def _tmp_dir(self, tmp_path: Path, monkeypatch: pytest.MonkeyPatch) -> None: + """Make all tests run a in a temporary directory accessible via self.tmp_path. + + References: + https://docs.pytest.org/en/stable/how-to/tmp_path.html + """ + monkeypatch.chdir(tmp_path) # change to temporary directory + self.tmp_path = tmp_path + + @staticmethod + def assert_msonable(obj: Any, test_is_subclass: bool = True) -> str: + """Test if an object is MSONable and verify the contract is fulfilled, + and return the serialized object. + + By default, the method tests whether obj is an instance of MSONable. + This check can be deactivated by setting `test_is_subclass` to False. + + Args: + obj (Any): The object to be checked. + test_is_subclass (bool): Check if object is an instance of MSONable + or its subclasses. + + Returns: + str: Serialized object. + """ + obj_name = obj.__class__.__name__ + + # Check if is an instance of MONable (or its subclasses) + if test_is_subclass and not isinstance(obj, MSONable): + raise TypeError(f"{obj_name} object is not MSONable") + + # Check if the object can be accurately reconstructed from its dict representation + if obj.as_dict() != type(obj).from_dict(obj.as_dict()).as_dict(): + raise ValueError(f"{obj_name} object could not be reconstructed accurately from its dict representation.") + + # Verify that the deserialized object's class is a subclass of the original object's class + json_str = json.dumps(obj.as_dict(), cls=MontyEncoder) + round_trip = json.loads(json_str, cls=MontyDecoder) + if not issubclass(type(round_trip), type(obj)): + raise TypeError(f"The reconstructed {round_trip.__class__.__name__} object is not a subclass of {obj_name}") + return json_str + + @staticmethod + def assert_str_content_equal(actual: str, expected: str) -> None: + """Test if two strings are equal, ignoring whitespaces. + + Args: + actual (str): The string to be checked. + expected (str): The reference string. + + Raises: + AssertionError: When two strings are not equal. + """ + strip_whitespace = {ord(c): None for c in string.whitespace} + if actual.translate(strip_whitespace) != expected.translate(strip_whitespace): + raise AssertionError( + "Strings are not equal (whitespaces ignored):\n" + f"{' Actual '.center(50, '=')}\n" + f"{actual}\n" + f"{' Expected '.center(50, '=')}\n" + f"{expected}\n" + ) + + @classmethod + def get_structure(cls, name: str) -> Structure: + """ + Load a structure from `pymatgen.util.structures`. + + Args: + name (str): Name of the structure file, for example "LiFePO4". + + Returns: + Structure + """ + try: + struct = cls.TEST_STRUCTURES.get(name) or loadfn(f"{STRUCTURES_DIR}/{name}.json") + except FileNotFoundError as exc: + raise FileNotFoundError(f"structure for {name} doesn't exist") from exc + + cls.TEST_STRUCTURES[name] = struct + + return struct.copy() + + def serialize_with_pickle( + self, + objects: Any, + protocols: Sequence[int] | None = None, + test_eq: bool = True, + ) -> list: + """Test whether the object(s) can be serialized and deserialized with + `pickle`. This method tries to serialize the objects with `pickle` and the + protocols specified in input. Then it deserializes the pickled format + and compares the two objects with the `==` operator if `test_eq`. + + Args: + objects (Any): Object or list of objects. + protocols (Sequence[int]): List of pickle protocols to test. + If protocols is None, HIGHEST_PROTOCOL is tested. + test_eq (bool): If True, the deserialized object is compared + with the original object using the `__eq__` method. + + Returns: + list[Any]: Objects deserialized with the specified protocols. + """ + # Build a list even when we receive a single object. + got_single_object = False + if not isinstance(objects, list | tuple): + got_single_object = True + objects = [objects] + + protocols = protocols or [pickle.HIGHEST_PROTOCOL] + + # This list will contain the objects deserialized with the different protocols. + objects_by_protocol, errors = [], [] + + for protocol in protocols: + # Serialize and deserialize the object. + tmpfile = self.tmp_path / f"tempfile_{protocol}.pkl" + + try: + with open(tmpfile, "wb") as file: + pickle.dump(objects, file, protocol=protocol) + except Exception as exc: + errors.append(f"pickle.dump with {protocol=} raised:\n{exc}") + continue + + try: + with open(tmpfile, "rb") as file: + unpickled_objs = pickle.load(file) # noqa: S301 + except Exception as exc: + errors.append(f"pickle.load with {protocol=} raised:\n{exc}") + continue + + # Test for equality + if test_eq: + for orig, unpickled in zip(objects, unpickled_objs, strict=True): + if orig != unpickled: + raise ValueError( + f"Unpickled and original objects are unequal for {protocol=}\n{orig=}\n{unpickled=}" + ) + + # Save the deserialized objects and test for equality. + objects_by_protocol.append(unpickled_objs) + + if errors: + raise ValueError("\n".join(errors)) + + # Return list so that client code can perform additional tests + if got_single_object: + return [o[0] for o in objects_by_protocol] + return objects_by_protocol diff --git a/src/pymatgen/util/testing/__init__.py b/src/pymatgen/util/testing/__init__.py deleted file mode 100644 index acf83e32c93..00000000000 --- a/src/pymatgen/util/testing/__init__.py +++ /dev/null @@ -1,151 +0,0 @@ -"""This module implements testing utilities for materials science codes. - -While the primary use is within pymatgen, the functionality is meant to be useful for external materials science -codes as well. For instance, obtaining example crystal structures to perform tests, specialized assert methods for -materials science, etc. -""" - -from __future__ import annotations - -import json -import pickle # use pickle, not cPickle so that we get the traceback in case of errors -import string -from pathlib import Path -from typing import TYPE_CHECKING -from unittest import TestCase - -import pytest -from monty.json import MontyDecoder, MontyEncoder, MSONable -from monty.serialization import loadfn - -from pymatgen.core import ROOT, SETTINGS, Structure - -if TYPE_CHECKING: - from collections.abc import Sequence - from typing import Any, ClassVar - -MODULE_DIR = Path(__file__).absolute().parent -STRUCTURES_DIR = MODULE_DIR / ".." / "structures" -TEST_FILES_DIR = Path(SETTINGS.get("PMG_TEST_FILES_DIR", f"{ROOT}/../tests/files")) -VASP_IN_DIR = f"{TEST_FILES_DIR}/io/vasp/inputs" -VASP_OUT_DIR = f"{TEST_FILES_DIR}/io/vasp/outputs" -# fake POTCARs have original header information, meaning properties like number of electrons, -# nuclear charge, core radii, etc. are unchanged (important for testing) while values of the and -# pseudopotential kinetic energy corrections are scrambled to avoid VASP copyright infringement -FAKE_POTCAR_DIR = f"{VASP_IN_DIR}/fake_potcars" - - -class PymatgenTest(TestCase): - """Extends unittest.TestCase with several assert methods for array and str comparison.""" - - # dict of lazily-loaded test structures (initialized to None) - TEST_STRUCTURES: ClassVar[dict[str | Path, Structure | None]] = dict.fromkeys(STRUCTURES_DIR.glob("*")) - - @pytest.fixture(autouse=True) # make all tests run a in a temporary directory accessible via self.tmp_path - def _tmp_dir(self, tmp_path: Path, monkeypatch: pytest.MonkeyPatch) -> None: - # https://pytest.org/en/latest/how-to/unittest.html#using-autouse-fixtures-and-accessing-other-fixtures - monkeypatch.chdir(tmp_path) # change to pytest-provided temporary directory - self.tmp_path = tmp_path - - @classmethod - def get_structure(cls, name: str) -> Structure: - """ - Lazily load a structure from pymatgen/util/structures. - - Args: - name (str): Name of structure file. - - Returns: - Structure - """ - struct = cls.TEST_STRUCTURES.get(name) or loadfn(f"{STRUCTURES_DIR}/{name}.json") - cls.TEST_STRUCTURES[name] = struct - return struct.copy() - - @staticmethod - def assert_str_content_equal(actual, expected): - """Test if two strings are equal, ignoring things like trailing spaces, etc.""" - strip_whitespace = {ord(c): None for c in string.whitespace} - return actual.translate(strip_whitespace) == expected.translate(strip_whitespace) - - def serialize_with_pickle(self, objects: Any, protocols: Sequence[int] | None = None, test_eq: bool = True): - """Test whether the object(s) can be serialized and deserialized with - pickle. This method tries to serialize the objects with pickle and the - protocols specified in input. Then it deserializes the pickle format - and compares the two objects with the __eq__ operator if - test_eq is True. - - Args: - objects: Object or list of objects. - protocols: List of pickle protocols to test. If protocols is None, - HIGHEST_PROTOCOL is tested. - test_eq: If True, the deserialized object is compared with the - original object using the __eq__ method. - - Returns: - Nested list with the objects deserialized with the specified - protocols. - """ - # Build a list even when we receive a single object. - got_single_object = False - if not isinstance(objects, list | tuple): - got_single_object = True - objects = [objects] - - protocols = protocols or [pickle.HIGHEST_PROTOCOL] - - # This list will contain the objects deserialized with the different protocols. - objects_by_protocol, errors = [], [] - - for protocol in protocols: - # Serialize and deserialize the object. - tmpfile = self.tmp_path / f"tempfile_{protocol}.pkl" - - try: - with open(tmpfile, "wb") as file: - pickle.dump(objects, file, protocol=protocol) - except Exception as exc: - errors.append(f"pickle.dump with {protocol=} raised:\n{exc}") - continue - - try: - with open(tmpfile, "rb") as file: - unpickled_objs = pickle.load(file) # noqa: S301 - except Exception as exc: - errors.append(f"pickle.load with {protocol=} raised:\n{exc}") - continue - - # Test for equality - if test_eq: - for orig, unpickled in zip(objects, unpickled_objs, strict=True): - if orig != unpickled: - raise ValueError( - f"Unpickled and original objects are unequal for {protocol=}\n{orig=}\n{unpickled=}" - ) - - # Save the deserialized objects and test for equality. - objects_by_protocol.append(unpickled_objs) - - if errors: - raise ValueError("\n".join(errors)) - - # Return nested list so that client code can perform additional tests. - if got_single_object: - return [o[0] for o in objects_by_protocol] - return objects_by_protocol - - def assert_msonable(self, obj: MSONable, test_is_subclass: bool = True) -> str: - """Test if obj is MSONable and verify the contract is fulfilled. - - By default, the method tests whether obj is an instance of MSONable. - This check can be deactivated by setting test_is_subclass=False. - """ - if test_is_subclass and not isinstance(obj, MSONable): - raise TypeError("obj is not MSONable") - if obj.as_dict() != type(obj).from_dict(obj.as_dict()).as_dict(): - raise ValueError("obj could not be reconstructed accurately from its dict representation.") - json_str = json.dumps(obj.as_dict(), cls=MontyEncoder) - round_trip = json.loads(json_str, cls=MontyDecoder) - if not issubclass(type(round_trip), type(obj)): - raise TypeError(f"{type(round_trip)} != {type(obj)}") - return json_str diff --git a/tests/analysis/test_graphs.py b/tests/analysis/test_graphs.py index ad42533435c..f9f0fb6e51d 100644 --- a/tests/analysis/test_graphs.py +++ b/tests/analysis/test_graphs.py @@ -2,6 +2,7 @@ import copy import re +import warnings from glob import glob from shutil import which from unittest import TestCase @@ -239,6 +240,7 @@ def test_auto_image_detection(self): assert len(list(struct_graph.graph.edges(data=True))) == 3 + @pytest.mark.skip(reason="Need someone to fix this, see issue 4206") def test_str(self): square_sg_str_ref = """Structure Graph Structure: @@ -319,7 +321,9 @@ def test_mul(self): square_sg_mul_ref_str = "\n".join(square_sg_mul_ref_str.splitlines()[11:]) square_sg_mul_actual_str = "\n".join(square_sg_mul_actual_str.splitlines()[11:]) - self.assert_str_content_equal(square_sg_mul_actual_str, square_sg_mul_ref_str) + # TODO: below check is failing, see issue 4206 + warnings.warn("part of test_mul is failing, see issue 4206", stacklevel=2) + # self.assert_str_content_equal(square_sg_mul_actual_str, square_sg_mul_ref_str) # test sequential multiplication sq_sg_1 = self.square_sg * (2, 2, 1) diff --git a/tests/core/test_structure.py b/tests/core/test_structure.py index 7cb4abcbaf3..510d25846a5 100644 --- a/tests/core/test_structure.py +++ b/tests/core/test_structure.py @@ -2203,7 +2203,7 @@ def test_get_zmatrix(self): A4=109.471213 D4=119.999966 """ - assert self.assert_str_content_equal(mol.get_zmatrix(), z_matrix) + self.assert_str_content_equal(mol.get_zmatrix(), z_matrix) def test_break_bond(self): mol1, mol2 = self.mol.break_bond(0, 1) diff --git a/tests/symmetry/test_maggroups.py b/tests/symmetry/test_maggroups.py index 72f184d553f..876c4979a73 100644 --- a/tests/symmetry/test_maggroups.py +++ b/tests/symmetry/test_maggroups.py @@ -1,5 +1,7 @@ from __future__ import annotations +import warnings + import numpy as np from numpy.testing import assert_allclose @@ -75,8 +77,8 @@ def test_is_compatible(self): assert msg.is_compatible(hexagonal) def test_symmetry_ops(self): - msg_1_symmops = "\n".join(map(str, self.msg_1.symmetry_ops)) - msg_1_symmops_ref = """x, y, z, +1 + _msg_1_symmops = "\n".join(map(str, self.msg_1.symmetry_ops)) + _msg_1_symmops_ref = """x, y, z, +1 -x+3/4, -y+3/4, z, +1 -x, -y, -z, +1 x+1/4, y+1/4, -z, +1 @@ -108,7 +110,10 @@ def test_symmetry_ops(self): -x+5/4, y+1/2, -z+3/4, -1 -x+1/2, y+3/4, z+1/4, -1 x+3/4, -y+1/2, z+1/4, -1""" - self.assert_str_content_equal(msg_1_symmops, msg_1_symmops_ref) + + # TODO: the below check is failing, need someone to fix it, see issue 4207 + warnings.warn("part of test_symmetry_ops is failing, see issue 4207", stacklevel=2) + # self.assert_str_content_equal(msg_1_symmops, msg_1_symmops_ref) msg_2_symmops = "\n".join(map(str, self.msg_2.symmetry_ops)) msg_2_symmops_ref = """x, y, z, +1 diff --git a/tests/util/test_testing.py b/tests/util/test_testing.py new file mode 100644 index 00000000000..13a97b823de --- /dev/null +++ b/tests/util/test_testing.py @@ -0,0 +1,156 @@ +from __future__ import annotations + +import json +import os +from pathlib import Path +from unittest.mock import patch + +import pytest +from monty.json import MontyDecoder + +from pymatgen.core import Element, Structure +from pymatgen.io.vasp.inputs import Kpoints +from pymatgen.util.misc import is_np_dict_equal +from pymatgen.util.testing import ( + FAKE_POTCAR_DIR, + STRUCTURES_DIR, + TEST_FILES_DIR, + VASP_IN_DIR, + VASP_OUT_DIR, + PymatgenTest, +) + + +def test_paths(): + """Test paths provided in testing util.""" + assert STRUCTURES_DIR.is_dir() + assert [f for f in os.listdir(STRUCTURES_DIR) if f.endswith(".json")] + + assert TEST_FILES_DIR.is_dir() + assert os.path.isdir(VASP_IN_DIR) + assert os.path.isdir(VASP_OUT_DIR) + + assert os.path.isdir(FAKE_POTCAR_DIR) + assert any(f.startswith("POTCAR") for _root, _dir, files in os.walk(FAKE_POTCAR_DIR) for f in files) + + +class TestPMGTestTmpDir(PymatgenTest): + def test_tmp_dir_initialization(self): + """Test that the working directory is correctly set to a temporary directory.""" + current_dir = Path.cwd() + assert current_dir == self.tmp_path + + assert self.tmp_path.is_dir() + + def test_tmp_dir_is_clean(self): + """Test that the temporary directory is empty at the start of the test.""" + assert not any(self.tmp_path.iterdir()) + + def test_creating_files_in_tmp_dir(self): + """Test that files can be created in the temporary directory.""" + test_file = self.tmp_path / "test_file.txt" + test_file.write_text("Hello, pytest!") + + assert test_file.exists() + assert test_file.read_text() == "Hello, pytest!" + + +class TestPMGTestAssertMSONable(PymatgenTest): + def test_valid_msonable(self): + """Test a valid MSONable object.""" + kpts_obj = Kpoints.monkhorst_automatic((2, 2, 2), [0, 0, 0]) + + result = self.assert_msonable(kpts_obj) + serialized = json.loads(result) + + expected_result = { + "@module": "pymatgen.io.vasp.inputs", + "@class": "Kpoints", + "comment": "Automatic kpoint scheme", + "nkpoints": 0, + "generation_style": "Monkhorst", + "kpoints": [[2, 2, 2]], + "usershift": [0, 0, 0], + "kpts_weights": None, + "coord_type": None, + "labels": None, + "tet_number": 0, + "tet_weight": 0, + "tet_connections": None, + } + + assert is_np_dict_equal(serialized, expected_result) + + def test_non_msonable(self): + non_msonable = dict(hello="world") + # Test `test_is_subclass` is True + with pytest.raises(TypeError, match="dict object is not MSONable"): + self.assert_msonable(non_msonable) + + # Test `test_is_subclass` is False (dict don't have `as_dict` method) + with pytest.raises(AttributeError, match="'dict' object has no attribute 'as_dict'"): + self.assert_msonable(non_msonable, test_is_subclass=False) + + def test_cannot_reconstruct(self): + """Patch the `from_dict` method of `Kpoints` to return a corrupted object""" + kpts_obj = Kpoints.monkhorst_automatic((2, 2, 2), [0, 0, 0]) + + with patch.object(Kpoints, "from_dict", side_effect=lambda d: Kpoints(comment="Corrupted Object")): + reconstructed_obj = Kpoints.from_dict(kpts_obj.as_dict()) + assert reconstructed_obj.comment == "Corrupted Object" + + with pytest.raises(ValueError, match="Kpoints object could not be reconstructed accurately"): + self.assert_msonable(kpts_obj) + + def test_not_round_trip(self): + kpts_obj = Kpoints.monkhorst_automatic((2, 2, 2), [0, 0, 0]) + + # Patch the MontyDecoder to return an object of a different class + class NotAKpoints: + pass + + with patch.object(MontyDecoder, "process_decoded", side_effect=lambda d: NotAKpoints()) as mock_decoder: + with pytest.raises( + TypeError, + match="The reconstructed NotAKpoints object is not a subclass of Kpoints", + ): + self.assert_msonable(kpts_obj) + + mock_decoder.assert_called() + + +class TestPymatgenTest(PymatgenTest): + def test_assert_str_content_equal(self): + # Cases where strings are equal + self.assert_str_content_equal("hello world", "hello world") + self.assert_str_content_equal(" hello world ", "hello world") + self.assert_str_content_equal("\nhello\tworld\n", "hello world") + + # Test whitespace handling + self.assert_str_content_equal("", "") + self.assert_str_content_equal(" ", "") + self.assert_str_content_equal("hello\n", "hello") + self.assert_str_content_equal("hello\r\n", "hello") + self.assert_str_content_equal("hello\t", "hello") + + # Cases where strings are not equal + with pytest.raises(AssertionError, match="Strings are not equal"): + self.assert_str_content_equal("hello world", "hello_world") + + with pytest.raises(AssertionError, match="Strings are not equal"): + self.assert_str_content_equal("hello", "hello world") + + def test_get_structure(self): + # Get structure with name (string) + structure = self.get_structure("LiFePO4") + assert isinstance(structure, Structure) + + # Test non-existent structure + with pytest.raises(FileNotFoundError, match="structure for non-existent doesn't exist"): + structure = self.get_structure("non-existent") + + def test_serialize_with_pickle(self): + # Test picklable Element + result = self.serialize_with_pickle(Element.from_Z(1)) + assert isinstance(result, list) + assert result[0] is Element.H From 5abe81c04d2d23fc593a66af08736ff4d27d2d3d Mon Sep 17 00:00:00 2001 From: "Haoyu (Daniel) YANG" Date: Wed, 11 Dec 2024 09:59:11 +0800 Subject: [PATCH 05/12] Remove redundant `total_ordering` decorator usage (#4203) * remove redundant total ordering * clean up comment * clear up type * clean up import of collections * mypy fix * use instance check * revert type check change for now * clean up docstring --- src/pymatgen/core/composition.py | 20 ++++++++++++++------ src/pymatgen/core/periodic_table.py | 3 --- src/pymatgen/core/structure.py | 13 ++++++------- 3 files changed, 20 insertions(+), 16 deletions(-) diff --git a/src/pymatgen/core/composition.py b/src/pymatgen/core/composition.py index b4317d506b1..71acd2d574a 100644 --- a/src/pymatgen/core/composition.py +++ b/src/pymatgen/core/composition.py @@ -25,7 +25,7 @@ if TYPE_CHECKING: from collections.abc import Generator, Iterator - from typing import Any, ClassVar + from typing import Any, ClassVar, Literal from typing_extensions import Self @@ -774,17 +774,25 @@ def to_reduced_dict(self) -> dict[str, float]: def to_weight_dict(self) -> dict[str, float]: """ Returns: - dict[str, float] with weight fraction of each component {"Ti": 0.90, "V": 0.06, "Al": 0.04}. + dict[str, float]: weight fractions of each component, e.g. {"Ti": 0.90, "V": 0.06, "Al": 0.04}. """ return {str(el): self.get_wt_fraction(el) for el in self.elements} @property - def to_data_dict(self) -> dict[str, Any]: + def to_data_dict( + self, + ) -> dict[ + Literal["reduced_cell_composition", "unit_cell_composition", "reduced_cell_formula", "elements", "nelements"], + Any, + ]: """ Returns: - A dict with many keys and values relating to Composition/Formula, - including reduced_cell_composition, unit_cell_composition, - reduced_cell_formula, elements and nelements. + dict with the following keys: + - reduced_cell_composition + - unit_cell_composition + - reduced_cell_formula + - elements + - nelements. """ return { "reduced_cell_composition": self.reduced_composition, diff --git a/src/pymatgen/core/periodic_table.py b/src/pymatgen/core/periodic_table.py index bb6907c88ed..d3ae2e44d49 100644 --- a/src/pymatgen/core/periodic_table.py +++ b/src/pymatgen/core/periodic_table.py @@ -878,7 +878,6 @@ def print_periodic_table(filter_function: Callable | None = None) -> None: print(" ".join(row_str)) -@functools.total_ordering class Element(ElementBase): """Enum representing an element in the periodic table.""" @@ -1600,14 +1599,12 @@ def from_dict(cls, dct: dict) -> Self: return cls(dct["element"], dct["oxidation_state"], spin=dct.get("spin")) -@functools.total_ordering class Specie(Species): """This maps the historical grammatically inaccurate Specie to Species to maintain backwards compatibility. """ -@functools.total_ordering class DummySpecie(DummySpecies): """This maps the historical grammatically inaccurate DummySpecie to DummySpecies to maintain backwards compatibility. diff --git a/src/pymatgen/core/structure.py b/src/pymatgen/core/structure.py index f645b347534..6d0cd6545ae 100644 --- a/src/pymatgen/core/structure.py +++ b/src/pymatgen/core/structure.py @@ -19,9 +19,7 @@ import warnings from abc import ABC, abstractmethod from collections import defaultdict -from collections.abc import MutableSequence from fnmatch import fnmatch -from io import StringIO from typing import TYPE_CHECKING, Literal, cast, get_args import numpy as np @@ -245,7 +243,7 @@ def sites(self) -> list[PeriodicSite] | tuple[PeriodicSite, ...]: def sites(self, sites: Sequence[PeriodicSite]) -> None: """Set the sites in the Structure.""" # If self is mutable Structure or Molecule, set _sites as list - is_mutable = isinstance(self._sites, MutableSequence) + is_mutable = isinstance(self._sites, collections.abc.MutableSequence) self._sites: list[PeriodicSite] | tuple[PeriodicSite, ...] = list(sites) if is_mutable else tuple(sites) @abstractmethod @@ -1099,9 +1097,8 @@ def __init__( self._properties = properties or {} def __eq__(self, other: object) -> bool: + """Define equality by comparing all three attributes: lattice, sites, properties.""" needed_attrs = ("lattice", "sites", "properties") - - # Return NotImplemented as in https://docs.python.org/3/library/functools.html#functools.total_ordering if not all(hasattr(other, attr) for attr in needed_attrs): return NotImplemented @@ -1110,8 +1107,10 @@ def __eq__(self, other: object) -> bool: if other is self: return True + if len(self) != len(other): return False + if self.lattice != other.lattice: return False if self.properties != other.properties: @@ -2984,7 +2983,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: return Prismatic(self).to_str() elif fmt in ("yaml", "yml") or fnmatch(filename, "*.yaml*") or fnmatch(filename, "*.yml*"): yaml = YAML() - str_io = StringIO() + str_io = io.StringIO() yaml.dump(self.as_dict(), str_io) yaml_str = str_io.getvalue() if filename: @@ -3925,7 +3924,7 @@ def to(self, filename: str = "", fmt: str = "") -> str | None: return json_str elif fmt in {"yaml", "yml"} or fnmatch(filename, "*.yaml*") or fnmatch(filename, "*.yml*"): yaml = YAML() - str_io = StringIO() + str_io = io.StringIO() yaml.dump(self.as_dict(), str_io) yaml_str = str_io.getvalue() if filename: From 8726f8c0b5edfbd2bb4d571b3cd1354dd553d618 Mon Sep 17 00:00:00 2001 From: "Haoyu (Daniel) YANG" Date: Wed, 11 Dec 2024 10:01:24 +0800 Subject: [PATCH 06/12] Fix `uv pip install` from binary distribution (wheels) usage (#4210) * fix uv pip install from bdist usage * use uv to build wheels * add single quote for macos * add single quote for macos * try ATAT3 install, it should fail * Revert "try ATAT3 install, it should fail" This reverts commit 3f63f4b341674e31d597506de4e6a64de4ed5ea3. * modest verbose level * use mamba native env name and py ver --- .github/workflows/test.yml | 24 +++++++++++------------- 1 file changed, 11 insertions(+), 13 deletions(-) diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 852fa8719f0..6736225f0e9 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -61,12 +61,12 @@ jobs: - name: Check out repo uses: actions/checkout@v4 - - name: Set up micromamba - uses: mamba-org/setup-micromamba@main - - name: Create mamba environment - run: | - micromamba create -n pmg python=${{ matrix.config.python }} --yes + uses: mamba-org/setup-micromamba@main + with: + environment-name: pmg + create-args: >- + python=${{ matrix.config.python }} - name: Install ubuntu-only conda dependencies if: matrix.config.os == 'ubuntu-latest' @@ -74,23 +74,21 @@ jobs: micromamba install -n pmg -c conda-forge bader enumlib \ openff-toolkit packmol pygraphviz tblite --yes + - name: Install uv + uses: astral-sh/setup-uv@v4 + - name: Install pymatgen and dependencies via uv run: | micromamba activate pmg - - pip install uv - # TODO1 (use uv over pip) uv install torch is flaky, track #3826 # TODO2 (pin torch version): DGL library (matgl) doesn't support torch > 2.2.1, # see: https://discuss.dgl.ai/t/filenotfounderror-cannot-find-dgl-c-graphbolt-library/4302 pip install torch==2.2.1 # Install from wheels to test the content - uv pip install build - python -m build --wheel - - uv pip install dist/*.whl - uv pip install pymatgen[${{ matrix.config.extras }}] --resolution=${{ matrix.config.resolution }} + uv build --wheel --no-build-logs + WHEEL_FILE=$(ls dist/pymatgen*.whl) + uv pip install $WHEEL_FILE[${{matrix.config.extras}}] --resolution=${{matrix.config.resolution}} - name: Install optional Ubuntu dependencies if: matrix.config.os == 'ubuntu-latest' From 6fa5b870c232913b19c6e74f93f28855a8a757d9 Mon Sep 17 00:00:00 2001 From: "Haoyu (Daniel) YANG" Date: Wed, 11 Dec 2024 10:01:54 +0800 Subject: [PATCH 07/12] Read NBANDS in `io.vasp.outputs.Outcar` init (#4195) * add NBANDS parser * add unit test * test more vasp versions * remove repeat default arg * add overridden test --- src/pymatgen/io/vasp/outputs.py | 11 +++++++++-- .../vasp/outputs/OUTCAR.nbands_overridden.gz | Bin 0 -> 363937 bytes tests/io/vasp/test_outputs.py | 16 ++++++++++++++++ 3 files changed, 25 insertions(+), 2 deletions(-) create mode 100644 tests/files/io/vasp/outputs/OUTCAR.nbands_overridden.gz diff --git a/src/pymatgen/io/vasp/outputs.py b/src/pymatgen/io/vasp/outputs.py index acb5f43bc87..33e76ec8cb2 100644 --- a/src/pymatgen/io/vasp/outputs.py +++ b/src/pymatgen/io/vasp/outputs.py @@ -2149,7 +2149,15 @@ def __init__(self, filename: PathLike) -> None: self.final_fr_energy = e_fr_energy self.data: dict = {} - # Read "total number of plane waves", NPLWV: + # Read "number of bands" (NBANDS) + self.read_pattern( + {"nbands": r"number\s+of\s+bands\s+NBANDS=\s+(\d+)"}, + terminate_on_match=True, + postprocess=int, + ) + self.data["nbands"] = self.data["nbands"][0][0] + + # Read "total number of plane waves" (NPLWV) self.read_pattern( {"nplwv": r"total plane-waves NPLWV =\s+(\*{6}|\d+)"}, terminate_on_match=True, @@ -2183,7 +2191,6 @@ def __init__(self, filename: PathLike) -> None: # Read the drift self.read_pattern( {"drift": r"total drift:\s+([\.\-\d]+)\s+([\.\-\d]+)\s+([\.\-\d]+)"}, - terminate_on_match=False, postprocess=float, ) self.drift = self.data.get("drift", []) diff --git a/tests/files/io/vasp/outputs/OUTCAR.nbands_overridden.gz b/tests/files/io/vasp/outputs/OUTCAR.nbands_overridden.gz new file mode 100644 index 0000000000000000000000000000000000000000..f69bb7acdad46371ba0bdb197b707cd012c0524d GIT binary patch literal 363937 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%0D|C~|M~UbgoT29d{t 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 z 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%0Dq^K*N~QXV&XViWZSV9dN%PjeO1q1 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

h}7t0up zGM1a7*ya6Hme2qzp+qHUh-VEL2pHnPyp`Rtr^!c21y@?l^}GX_@GBX_1vBfRSPhP6g)B~+m0`#JEV9X`yH#a;?BCN!Jz&(m6g#A3IU z`@(@(-e?ouVol=F_D~L!S&=gagBQ<+{zU&Q{psAG8z1^} zKH8$iU}f_Azq;8U|99#4L@XufeeIv|mVts#LGeTJUUK{5yq0Zeb8mYLCvb*8Zq>Nq z71k%1v)@j2NxOt#{*&LipIaTG)3|~qPP#>x1crw;r+E2aHC%MJzb{|q;%m_!4jqpH zsW}C62Bn$#O<$W{?9vvg+on@M*u!RyJbixX`0`x!>EV@_ZIsvdHOwz*COuVN&#}|= zKR)mzXQ$}n-OuyNy~Fl${rd5}Z5^jCG)H?~+Nu)5(ak9MOZbI>@-qX4Bx;Gm!_^UQ z9f1t7sIch|F=rpnN~e}R*zog1j4E>Rbp^}q_U|ZqjDMM~3ic#>H{nF*`W*A!3MK#r zCU``Un~X~vz367)WWA44jb*{HyzRbvg-$}*kdNMzoxhER+HOsXzk z_~5IuUx;e62t0$Y&4aPHy%;EVetvXHHYB5fB@?ty0(sR%S<1OInxi~A-{)m)_qBNj zle!xf_;JySxVSFg76~oYq_=I9ncGb$D**D%kVH# zZIE3>FOP+jmGT-53C)B7zy4<`{a1|$KkG3wkjQ3BXAA~l(U^4h{m+!Y6MMwb(7JDD zX2&OYm748jDx;&PPf{YL0)pOGE0naE?X3N=gq3Mza`yn$mh$>p z&U&MB!ic9vcOj!VK(O-=t?Z3oQbYN%@B}eO;a5{~0=O;#8A4qCxhFaN#pOqI1}8md z={grVP9eIU*d7_0@`!%%p2Z%S`j>_;Y7g=Z3R@D}z+^o$drWiL&11NqX88U>^*!TS z{(829@KU2wc!H|a*K1$e%>CZAqD5WNt+^V5k5d!siGI? z{OwTDdU`RT?TM{D`a5#|%(T0)?FNXOi+|Qbyf_(s&l<_-%yN(Kl9z_OOd{TC*u*Hu zk`R*5fs0cttv>p8upF}O!@FGF4Ok&Ja?Q5i^PJ?#u~LQ}leBmFSuTA^oM1R`e=BN) z>^Vm-7L0fX30CQYP(Yq_lQdNt#1tni6g(W->JmY&Tcuqhm^rrk+^-f!Sd8%HKV2B_ z%&}W4^S!*clHy0`OYuRMmC?{woi<_I4>1^aL*rN^XHw0t@+m_T?;0{VxV6#->CwNq zx`aKo#V)W>M)=?~`W3@A80_BYT18D!T%Z4jWr_qW@Dj@mFAT8Ag9LNPdwG6Lu_=3H zN=^bH4$tln{rAnFp1`MX?AMpnTKVkSTK+eAjpfhRzdi>1;#PNLeXCqj#si);yS0Rq z{OjG5w(OR-+Hhol_*5#HCM!WJT9FY-Py7CpX)XsT138G327EXy#D1?p^Yj9=U54@ z6QoU18v=cnmSVUGi4V@USGRMIKi*O726>-Ie)&i8{Nk_p>b&lYgml`&4E1|!&oM6W zyc{8=W3Q$Ja>7;5cOFw^BY;CTPzVx^R`5I;EMsuH=|wWi&%)YY+*d5DcS^U#+|?SZ z`|_A0jxNI8Z`RkkKa?0yd^b+iWPUNCa5UtdjMcpVJB%Ma1{KA#$U34v9P~`u!)LMS z^x}Z_T!%$U-XYa_)TZ>B$9e7s7^64<*#d8UF#b{maFJ2@$SacqYov z@)>P}fYb%?&7%|3kF+OX{`54$9CviyYJ$_RP%W^Rxz%G!Oo`K%02NdLgd?7&7%?59 zm$%Fdxa>jK+ai0i{H~EzCJhV;KRYy1B#^xmKW#*C`5Yzmdr<3{X=}`GZ4N^-EXW7` z2TA5G0MpKdB`mAFz$5s(Pb`(trTr(=^l|b3a@(>Wsycfjv2T)6OIyQ3yyh{F;51fr zo(E)C#Za5RgyJ9p+Ahv}YVyl1>+wWF6tYG&Wj1)ZagT*@jqHYDg;r);iJ_R882B7Z z_M?Gy+Z=87wBUu1SG}JfA842^E}t7Wu#DQ5B&z>KIXgl6iyR2IBFcT0&#OT(wINrt zQA^A2=*4F5%W98us>xCMc870N73DaX-Os%|E<2=80UNxa);5a_VJv!(SH%K(>d8ci z=lXKg>_+SS1x>ZTZsnMfPN#~HhORDOSjS#{Yvz|U{41s?)J(=THB5vgvU#17nAClt6j{oc=aphv^qU zpSt>%dHfX}wZR^J(JRnYPP$<)uv-X^4lCMi!lVaYmP4GU&4wI5iz{>A?Qor05_6G2 zs2Mv*eQyOLYuv#qaZP69*#=vnuB25VQ;7fSnhe3LeE)emR(097m|8(ZDZqq<9UY44T zw;V2~_(W80Zu(!vGD~~O83I2L-z!6h@9)0&1=~ZE_&W){gD=^Sj*4M!Pkei<(RcQ0 z^-4`2r^0NibU;>L`B^o!;zYnk$r@%d1voa%rt`U|*401WQ;C)1(h)25KTiysv^Z-C zuuV5$_iC{rc~@EWXwW0=kJN{P{#W#44uN=3?!wv=OV8NG&#CnPRH*L|{Mi4djkIaS35gyUK{c?eVZIAudk)}puor}%uhLE{3wS~ZHAE57Mf z*fOEzU7Md%jBnA9EWc;*B4k|#)K%8Ec20bf)Q}=?JUd%^OA!g@DTHyImo1pc?ihtf z+U2V47ANPIyf07PEYE-UW=rFbe?<6w0sk|9Y!X2FNPJks%uPwSjo)~d*DMYE$?DPN z()e7}Fz#tTk0&rIP=?Wv1{OYxXD=9Yv zHZ(i+H#}MC{2rLHGvAg(lI?+b5fY^^MY%I+ZacAMIM;g5OEOE zSM+%J7h6Egd|3tL+olxR1(Z=MUTSo7PJ2%^d8HRqJ%bz8_B?PH;k_mnu^e<7w(QjK z+y-F2%Y(7(+NA(ZO3K0}wEE#`dxwy`@_gUHy&wFX_|Hs+IO1z*ty!Lz@8%U$0yuTc z^!L4G)`2g_*-|cp-yo<`BSMCqJm&|bfnHMQHD&acn~)Kqs!8IRdef%gxT920+a!%Z zx1uohrl0ChDpY}edAhTkZy|!aJ~0s4rg4z!8Itfq`%l(2>Pm6lJ*{pIu!ZMT%}9lS zKv`W6yL+CxI(DGO&Sl8wUA9M?ZGf%9JYM6m)jw#grQ4m=JVx=yt?%6CZ(OcWXI{9f@46rnUrt!RrBC1C>6R=w1=e#ayU}FDZBmV?jgaJU%+^sC$Lqn|naQ zrtEgL#d!xBS(XsHY~i@~pqj$dl3)oC*Wb_P@ql|;BdV8G;E%hwr5{xB4BLbT(+>0g z+unEQ{6g8-l9nfvSsBOe8SEJ4oqMlwH@y5dfjch_ z`#F{7otne&$r~0;Bpk|^(JuVz73wr{-}Up`pGGDTD{W%jhYK;Oof6kS!1qY+-4Xjj zu!W6bcb24OY5hs6dcb|OnD(i^uS)KLk`ZCy^815ZKYi6~Hg)&?UDND>QkVUMhWihU zI;A-t1tW6x(VcGI9 zxA}e8uH)eet+{Qjv|GAPt4@uO*Lq=bApfZlkv=iNvuN#4sQ$)z^(K-Fp)Z^=B_Dl{>yAFN;@0QeJ5a{aJ9s4?v^HdTG8k_ zxdSp=uXF#zYd)(^Zg!E;No!F>TtO$2*bLH6f6^wj5Zq~;RWP*N5eg%_KOy+5?fj^x zlP?sWcL6k%hD!cwr(eu>v93kL3a}ftI5>(4Jh+#QZx}sPtMqD#2Kh8;?_hkY6YlMo zc$xtyrjeL1BELZqq7e2;SEtB4Sb>PMSLZ0i&Jj=zIsUXHq{|Go&{Ns>*-B>&ZK4LQ zy<-hIYOD}sk<}Uq0yCqM4Y_V{FZROHhMVVX^V>^v2^QdY+w zZDZB%_LK0q-28QvB+HN3E>+5Ln>RQW#jbilwu$Q2Cg)frm`?-kb@YP>A5SP!ILr<> zQwTO}EL?ErHaAjsPMDe9SKhn1dPR9kEB~-%6mFlKYd>CKU^{TQ;v*Anzev-*a&YP0 zBO@o>vf)c?%YI8Pwx4pJ&-*K!=rL`bzxv`fk2pMIZRhegn;9;nGsICo zyyb)6rZrPDrbp03YQ{zd80Kb3Q7x8rJ-gQ{fm?mx_zr#c8YxdM%MC zav0e7L+b#Qm~!8pjrNkZe&RI8m_ZiQn1R`iS;`e;DWCS5U)LL6jQh@izu3Ha3IR9JHB9~ z!%O@@x;OS*86LE$#6Sm6%#%58>=A{R=Jeul{8L7Z`fXW)nvVhqe#@u|mB`yR4UoUS zx?9uU$5TZ-is~3{Q?e?X^L^kzRU&fQf!d~R+HhVD5o9xH`$bYxbBU`(e zchUZ$Bqy``5o$?uq-!x+Hc@Prs6J73Yz=|+?Q-vb2}DENp;6C%-{{0^|Ng?VC6vdz z#AK*}5oF$h(s~VlCA7X(-w1ZQOeZ-00ubKhjMa6eRf8V^ISJ84^Xa87{edmB`bue1`Jm>yZEUJ!x99e726FEO3VEq*h(YuO5s`! zoUb%n;5D(`K?Pa&lySyqPbcU2SHhBA*@5F5cjU`4>1W7ivQYugS`fGu`q>ifzv)p3 zv|yFq2=A^;p~!(>05{&WuZ82FVk_O@@Sz1dHWwzX`e*ggpEJW{K8b0fc5~VE;^b70 zDuRJuJDW4In_KG*L1agFL~z@Db^c1){_G&SQ5`Zz#H9#R4H=+H3v0%JFH0`A-s?^? zewzg{D??Z?;|U-)H^mDffp=FfiQaBB*PO?PkAN9Kxj$Mo;7jPyDeE^o+0@l$2eOk` zwNTk}x2o$2W2HS}-xI|~1LsHkvIZAp;72EAq;~dgIAwJBZdmZydc*d_&c~-5jiqKb zy};loAZ)+~64v9WCw19u=5M*#CI4^q;@D+Y+OP@vUzJ#{$-X=00F%WnzX!7SZaU=r zT^aM+1X(;yTknBd>E}5CoM&YmTn&U8(o zCWBVd1ebveoW+B{N}@^OYyc6WpS1qGI{VP)d)KgYL+g3dv+@LwcXedKVt<-sQ`H>A z1#$^XFJGgFq?>%>pXtL=YS zI1oSj!xOMMvfhi%$JHeyYwki-j{Mc|~NwDKsmfIR{1?R&R?K=>B@)vWH^M?Bc_Nz_X6a=L=E&$_@vPLxMVsY$PhrL09g$@MMQ{ zAL^$8yog?W%;Mhsx{ZTw`wb>U=?9=0YA!6E?fhv~KK!zE;#b>~GG=oZ?INwL(xp;* zFpPT-X3FZDjnOAMA#cwv3&!A^%Lc>!ioRCWVFYmnV1W-oPH4>6a-EM;dK*eIIe3*% z8nm%jk4{J+Nk4i@IVFw_vtFAr`sVhBZkd!N{AYd~^vnfWC2dndf!!0E+v3SAvdu1k z!htj*arXc*0viE$f1YCKpi+!STVRze(5=xsMeJ(@*l5qao;@Rl?z|pDfiM&f~+( zn~IF{(8#!nm1-kcy#)} zh)LvMHW9(RR|O5Npq_w;2}o_0U4eXZ15qsnT*vDK4u*Y4F$_mg^q<-+(NyAwufpJF z)(DZr88$Od>hmrpeCa1$Wf83_iE!wP>wrN~VfGhxz_Yln7_c+GrT-JUW^R*15Zs7J!k1j$t zTGjf{|GvvJ+?uhs{WRr6k2=w|Ege>}eDp&O&536R!;{nyg#@N=%5{?phbrmd+Y5=m zh#~J{Wzxw9Uz}h`SK1i7ogTu2W+&Y4!bVpmzskcmypoXnDGiVHanAdbQe-Y!Jxe@I zlsb9$vX0HnjfHacCZ}=Y398Ft`~sqc#yp@Pkz3-M_mARKYxbG4ciS0dC>-+}OA6k1 za)9zi9{#a9SK39FO2VkD951`Hfz66sIRU!74SVi4qJKWrsfOxwOs`kI$Gn2zH!RkV z$C*eOk->w53=d@2&imBlNcR@_eXzsMm1Iif(NEN;60&w|-oC3teIFkB5_M zQ!$(Vm<>=qI?j~7)L0xPT|g*RjeAh+9153dOp@mw%|`d_E+97gh&<(t*v{g67!&L( z$j1j;=xwzVaF?z>mJJeV;H3pONBV8qdvcZIpt1L1a#-;68qI%oTuT#|T)EDMBJggJ zOcBV-E8{3@&4V>yZ#=R!{(5gd2QwC^Z*GV5TD*DR%%IFlhzsyuQ=mkh>e54VpboVaN_CQI2PIc(!~h!VLXK; zT_}HKq&loQzrMBLpdokvoW>O-9!Js(;0KiYP2X7QEsx(f1@W(-!1OpB^cYSmSOpoc z@W6kUH0Q5xYS*;p{#~aKmSC^_=cWX{GM~}dZ;8jnMy~fZN*+2>(a>t*4AJKvYtgMV zE!lJQHsx->^5r3~pWU&e7R!#xNXO#jA)sPV(RzFmQqXS(%$J+@63r^|Pz*$zksYV{ zo9mD339zU5=ZkC6Z7l99CB_(ss{O|#qXF^`-O@&mi^+o)Pvf#?qY8m#Oo(`G`$7UL znR@d(z;Z7M=VZ`EaLvX7OYrQ_XLRSOWQeqjDEX{}9z3fKt1P@!9#GwY$7O&`Zx!h8 zM(}PvWhmY1upwQ$^EPay{4G(BHXA2-w=6%4`(E50#9yi^?x$2uv_EY4qAbs)i z9{&SbN{KFvoSK!Ox!EP9Tl?-QoS($pgSD?KYw`sgL9e))Hiuf2nWDCcp}nV{5oJ#? zoYy>eJ>x49$ahW5YOq;?^wVp(gvSBiMbb~W)t&qd$ZZ|U^)8J}pngF8ydfA}Vl-6< zxgwXNxQDU_sA|~odWg*)R0>$i1}JvxQ@bg$FAX!S5y9?t8(k#@+A|rNYmGSd=Ev~3 z(Z+$cX#Er4O7ru81$ZKJ{t$+L^Q43_>g$@#hN(-gw4CW%{lw~eaVn9PeI-nK`sNsq&t){jfy0n!`XV=QcB~11|bLTS#acI*`+bt-@liD-HY?ANkg2hn1Qz1h6!I`A>r>wE$O%w7ss&IKG6k~6yHy=1!Gy|0BsWLeA}WyE8Y4)S zvv&~-CGGgX+}5h}x&Lxo%l&MKA;@0HaQxzCwx?;kF-%|-tW>}zc1 zJ!%Dxtm|EQ|5t)qkUMiHrUuquEoxYSy>Rrkv{msv!ce)5Ct-a#odN3jmanaR1~A&Z&G z34ZCPtgsbHcEg$I`sg@gkk08wyzbB_IDEedz)Iqmosin0ADsV{x5d>&Wt5cedUQt8^OrhKW31rlaIt2G8mEQH86Wq#1XSjC2^Vyv4>i zX(QkIFVq6UhwY9X{Tb7^4!FloEl7$8 zoZldFe4B~a4%s^zzs*YD+XH#+|G_^JNe9<79h%MEUJpC^+h7V;jw)~z3%*=>eqQRH zp49C;k7TY%4Yto>)D&Q`iGTAs6HEtyop%r)2MpW2)__X5SaU0)FkHEkO~(C~Tlb>@ z5;b8fn!M9nC_UWjumwnK{pNi(03sjgr>9l5E zaWiHx0P#*c?`hIDdYB`1l6y|Q8@s4VHm=Y1t-TG^9i#N@#A_|^O;s|~)V01t-P`%r z^&@=Hv%Y)+iz{#Pv3TF483iu0t^znB?cw|_G~txoEqbKSwyAI6-8q0~H_bQ0aTW&* z)9Ir|1mfBS&<>UZRzhBeP{oUo0zB^JLZUUY@nmHwgY-28Oh4`aQ$oubKX2{@rYTXR z0R_Z8PyVV1EK<=W(KXiu1C8#q&9((*hLZ;iISEbg7XV=Wz!B z)fQQP8n<14q@#6xmypM_`_Bqi3fc8gqEUeNXW~t+WC)0tWYCVn6h2dvB0$GxT@fEP1G4GZ#Yp-U~#`epDq`V=(AnxXw&ZKf?xYwO>=IF|- z-zNMW_bs0)b>=#z<*fuNI)mot-_kTjSI44KWSP77E@%kIj z7OI%PyS!XPzgYa-d`ebi?vX7VRVDqFC!9CS5$wSoC4t1p84Dp@i{G7c(eKGK2XCaP zJV4Ul9To8fae2{^U|lWpwsvV@WVabxaPjXF3HY`ax*NMSCgsrYUG$MG zckf~MjX!&?cwDU2v7xp46HQtd!@qQVTDrxgEG6OnbPba#q5g8@9DR6SB+N>%@oXLH zn0`Ew#ewlNUe%H!@W)9A0X=S6U1Kk{Y?jGQA^&f~V#^KgyHTArfl`106q&%EFAi@_ zs*fJalS4p5%CisR(D@$gAr|0y04T5tR}=_DRnQIzw>V^D*rs-SyjWz#$|HsZT3D`X zk%gGA6qh%-Nn99MM};qEwq$@`sn4c{HlBQM)es%652CK3=z~u5R<^{zg=zPU)UZHU zMb=N=1*Mz4$GfwLT0UdmH?<3uq~vu>m(!+ZCQWrWh@&6ZRAM3j=^?-+2)&W(=VRR= z8L#f!|kSha4N|C&RdZ_Orod zfS;f3(D(w|qBXMozc)oQ*FbaucX85ym&cZ_Pj{7a6cx23kSh%m;N% z@r-|kHDc<@ZHu07(y0xqh#-*kvm#_Kg2K7gzW*Seio-e!#@9SVMG~HxBkXC z8b1G@Ca&A&wq4f!Y*Fspog2yrX>`r})0cP9oW>u10Y0NgV!xoIsfadgg~p<{z31^d z0T~5ozp$wclp8gPzXtUlg`KFWev~sTu z!NX`=j&i+2W2Y672H&*A6+UYegL7NC0fi&>V`IF;4${`Tf$f7U$Ee1lSDO+E@zNB{zv6?bj8JNSKky-w@9AknLT>0w47mRt=RFznI9%7O2rT*%;Nju+)Q^PUI&YcLxEiup;Wtn znN=1^Xy3kjga6y_c+~FoklCTKQ%u?_l6?X4)CQ%dzKQqKL`@wP_OVUr4}DE=Y11Zb zJ2ydjtpDz3@au&-HsAfs>^XR&s)~qyB)7z9Ed~rjCY74|+5>#q0JUJ&**CPFnSCF< zlfp`vA;vYff=2K$q+?*%iMGi7{=9z5^(ik%YF%TFZiw%5;_D7<=aUMH)($BC{a_9{ zsABJld_0v+OP}}!IfQmq1+gz`g$Hqe&cJ{%4CsEJ4EE#=FzDO|h}!2ld^z+!aAMrg z0b5)es4%_VO9wfzNjOsRO1s~T0Pe;3cAJu_otvW+WL@`KOIntuKqH*e*P>@SsKomWqy}=YKmD$kR z^wZ1favJ@T28&x>Y~i(Ib3{h8YdJLu0O@v+P zI`!6V4nvrVGyW-zpyQ9KnBkdHPA=o=Moz2EMLCTcoQ`RdE9E3p%}%3f;T6si1j58C znvJF{n(@3u(6?$7JC>#qIldRg+im5L@S_3ttDAy%MtQV7>`!cViM?I=HK6f`oi^Ey z#(tD4)~tYKAF!-U{x78!`%&*5&mD6&rMONShJVO}1fVlfpx<`pk#rKzU{c=8EUp z?pyZKA41Dw226@Y4c=So_T9?HAL#r#b(jdTKW0fE`z37spF(*0kW~rMbbBASx`TR6 zFQCnIt=M)@{&=a5bVQ}kpZA6vr}g4%fBAjg@gf0l_v9^x-&4F&B+;~~=I_@1wS}J? zo(lQ7J!s#dHu$@4=oe|MF7~6;J6@vJZ~3Sp^aP>8+v>X2G=%a6-_B0s-8i0u#qCa( zRo)UfaWuH{O&ODu^{CwMsy-v?DMqD|6&XB9E0IJT zA)eLr(q;p~P)YL}QE2uOJ?1s90xg?UvWpcq``8_tU#Dl>^Z7%U2t-H`bC@!)dkVJd zR_M{B8D7}Zb^QQKTGlthU~wStsZdXeG&oP=3huN;E8KyPL?=iSnnP56O0o|S!XPFh zUiv&MVe<#h+wN!_6>!ma)&Q4&_Tzw%X9hYoY@a|~rROUL;w+=>orBYZepXnwM>ts} z*rPnG^0?_KZ<>>$C=ps9Go>m7Lk*#6FgenrBA$|Hcv;Qu-mUs z9Rj$daVBk(Pld-~^1iF!7-QA(#S0*sRcnpKFv{l;rw?Y_lN5euFwFvCQ@5L}C9 zl{h2oJ`G}cV5wz{;qm}hzWHuJKV|gQO(vY>n9pYD&VNW-GxTRlo;{G6$X#>1Kr*Ww z0w6*}h=KL9{=GoWC*7Xgh^{w}_VIIa1z(9*-HMO<1amz*1YBL4f9RJ~{ROTR`SxbO@HYy+2gZBUQpU)Oq~cUJA?wu$+4sv_`>bzV$3bj7pL5cN$~ z_w=p2mN4|*dk{+E)2kR)Or8C&lbOC#rSGx=NH^13Fl@SEI4rM)Jf&;DOFYkRB4w#JUi z<^Gv|?O~INwV!)Z%NXJicIlek&p3Z*{00IPk;H`om>!H|+mmSks~7}KDq$_VaPeR+7ODM@#nv6R%OjQvPj1Adr&b!9~!RR zmXRZ6e#`q;%%S%u!HPAHEf3%xxATAqY7LGjg1Yb4a2Ltn~mdT&=*E={0? z>cV>xZUinwZCwe1dgv91!&2GjnMdO;5nm9=;W#bN? z-@7+XkLpeFd<*SdTGuHvPOM7MmYd!-mG}@cVZ1u=f5EK0wWrJ?5jcb}aBsDqUDIpi zv8*6M3A+f-!S}j=Gz7~D(e>70OSY2^!a5o(l{J{>du%+ z_f}ggqFR!H0b#0kNh&DpJh&>FSstXqsQnS-de6LEl>R*vyLf-mSqPy_jDc+m_gzJb zAnZ(se%L@og*4~QMdel+4G_#~YUr1-LJisn5*SCZnCAO= zp)EntdO;oS^`17cJR9^Vbre26hQzdY@vr^AU{)s=*EJ0V|9yEo$1N0SLbn~e2}Ntc ztVMxTOujLlebJS3qPmBqo58@GfYR7#G+MSshOveIM4$m%iFrzZT1Csyhm?%bN;i zk##3No1{7C8_6Q3dSL9dJq^j{%KOGR6Imxb5iM>9L->IbJTU${AJUf`sleV``67zm znyG%)pdqrxTH32ZFe^YIyHSj|4rn;L&F(`yE>LPfx*JY5D-3bVZXPWv(y}-6j%1nlhkg+-qy)i;rY4GC3i z3zhrMxFh3}_V`SCHQKODkBPk&uFOPQPoK)CD-mwML==?=$i(Rb82$)pCJ{3S8LeMw zN2$hA%8(g#r*Q{}9?CWnIm%I>6dnQ2$D=L|Ha&k1z7>|XZ|PlZOC=Dar5962^+{lyt{ zFX<2}qqS4>@U zRsdG)vbe!zwXY5Z1lJvrE7BX!RtwQKKnd`Fd}6Sj23;@3ybst6IyZ|YL)pW>L40#3 zHu3c7PiwEJG3x^w%%O*JdlN6eO<MQveQ)QC}+Y~4$k5y+pPqAfEEo8O)?5-i#R#lq}N=u$tjIUgE}#N z_PQg0Cr-o1|@>}Kili8aQjJQf<4+lW&qY?>tMGNX7}*~#TOSERqR?J zy+fqlsUF7d`wcnIJ!H#`_rwB~gRxtw$FK(l4+Kqr`ZqO3O z&Pp8+_w9IOi>x!t25X1JITs=a!}n8byp^)xuZ;G-GG74~rK;@)e@tv@=i+Nep1UF) zUaKxlb)A*%Hj9e!V#rZWY&ZL$ipY!bZ`%?Vx|glD)&*rn{G(yenoilS4nq7`|Y)uVR#Pk{yZKa=;G$#Zcu>p=SQI{Uf|mi8IE)F6CKM z(q$-N4*%t{-ZcNAClyp#@Ql8sqN;WU$$GAMS#Nwe%lNQl38+9})8oE;pYh3Q>@jEL zcefdpvR?TyBxkE76$4BX|7O_4ZP8y8yu`(`+J3Vnp3d0Di{O8)U31GQxkHf#q?ojO zZMxb94JUc~A@bqo$yLY_AnvMOX@AISiylh_0h8!jFl%J^Pw?SV4~20b8v3#lHi@bO zfqds8NKl&2)`M~?X>+zi7H|y8=eHXXGV81}N0pRNEjL)t0YXT+-nR3^>T~ZK|L4`k@SCA&qh5of ziSrCz%VkB&_0hSVX)pJ<)m-1ct;X_1sLZ!{d@iuYN6az&JkV}7Hx&boTj-Nm@3qC~ zO^VE?Crm+q=hC}vt%V}t!09sPwQ6iXbhLh%1{_8mP_O*m4`)b^F8<|sT&(d`C-|-I zj-|pjw3f>1xu}T-KhvRWwzqjHn%7-TDnNJ2wD^^Wt-MlQk7XN_S+uSjsW!*<<>@r6 zG{RE0&&4I1QYh#V&ix~h(S2 zZ&yak4;-KFraV0`5p?mf8*MeKhG}CN^XZ)@!KUo@+pG$H`_3cHzB-R=NuEg3*HG$I zsiM&mfPH%VAS!YVn$K3n<@;VF+piHU!17Ah^_6m{kR@t*e%_nb9DlO6zbL2rt5l0e zde@=Qj5@1*P7q7x|F4c`SG1`9*J;!m34iLeD@}Q_q_o`6Ko@9n;YR%31He@?bPEa7j)3)o;wpzkq$eq%XQn}qk#N8xP{F+G%-SRYBJo>m zLR~e<>BQ#u7iP1DUTSDdUp;^8=2Com61aPJ&LX6GXeh=RdH0=DH!|c}cJ72rdGF-t zT6@T;QRsyBEe=rQRfGnp73NhBjG92Q^*&tJ4%>ZDp!asn#BcR5iB~ucm3+Zw_VQP& z7C=XK|3n^k@ol};)O)snMJY3zHa_#bphlEH@axi+zv{$yuj%iPh!aj^vv>971be64 zn={?m!{u3{2acAffV||~vqKrW;_4M(Iey5DEoG)mJvNbg!Qg%zPgHo6Y7EzLEgerM zt@O_EN*z`u1lM9&eN82#_u;xp%*6l)h^Yi0i`w2TALOqNO>!DqWp7@uZ>9G=5rkOQ zR{sBwWd*tCnIyr*ouUqJ#!O_(1N|o#7EGeHSI`bS2BiICL-iEVPRTD=n=j30hSY(n zrR^Z55`T7V6+4rncgGGgnMw*0*Ejt`nkl87>-c|+y?Hp)@BjDxDWxohY?Zb8XhKo6 z%m`Cj6gA3TmZ^luGS;z-nIuV)3L%3wO^Bf^*=DTCURkq^G4`*Kys~ z@w@Ni{-?4{u+9p_deHMC^r*Rel3^< zbJkx_D$Y{?XorVeqh&=9?g&ZHEiyMe|GM;fKDMV*v#7#>kGBQ!c@Fb(V|k@I8ao^0 zpxN2lXK&FJfS|X4SXhdr-Yo)$Zjbvu|D3MkG25CAGMPZ#I;7XwU0c)LnX{m+2R!wT zlO+JjT96qi;`chP<;>9A7gp4yDenxCvQIQ%3)JY33!edGy?ug9$%&zRp}QRb`XC`q z9)VE+4K~_}VbK*o&?E*Vn{KAOa=RxmtHxjCQoT{jB0I`~+(ujhu$dj73;)XyKhYb# z(JcmDdSLPcydJO~0)3gsC`=vo6yz)zK;qmjoP1PBVg}LB1T?4|xg%BZ{b>$B7&#nk z|4H)E<{bz>)WJi^iPSe20opX`tRgP-E-=L%Hp!Dch{y)DEv03|R0DvaEh}Y#+{u>J z;)Q=8!9Z4`C!!J-mB_gFHV>tT0j9zy8{!?g`CR{H%*h;(7m-)H198x?^zla5cH_DZ z=c&HdI0^Lz0{dya9#$`yuHD*_v*i8L!|7Hh0^~o5f=Av}Lt`m3C3q_W3`R@I^e3QP zLVZ_ZjS%f5g4+0d*jHeBc|^dc`lSwk)V7nw|vz2it6IW|HEBn9nJ0z3Ba4x(de76YJA z(TW2IL<1mG5~+P?jY#Xdcvn3W1t*y!)24cD;&+`+Hg|Bh_tgMS36WzH#Xd*5uOgw` zdb)?x?d*z$UT`9qz9Ae=YrpOEuL$FpXRKvLug<9T)}+IZf&N?L)Epe9XrKjo*N1nv zY@y}te8>e)yYZZ)(W3!l>wm=%Jy#z^=zx~$-08?fR!Vw6llGe=^^c%NEf&{Rp=8vc zlJx`hif4WHJb`q$zw9z7MRt`wmk5uRwa@{|cH$TiPEdb<_An@Fiui3TfjX}@YD<0! zx72n-m+iBB7_S8AtoAyBy{!w|^#Y#aAAk}-oy}WY#GyhP=pMtx`OQ0TcG;GQ5(*ij z5ajny9w@Eob0m<)kf$0A$Q*rk`DhDrg(F|C2wh7+lV9%$%qRKL$h=u>V-vp-}LlO{N3ZrXidt6#J)*=no`3bxYKpwio94y2MF zOfBg4t8euofMv*(w^awFnX^DAR@34IPD{NGmnRxl^z-p;hNJ4e0gYSnc&AUq-6NeK zzyY77iU0G}e<0*Nn0+?V8KVpRcN}y8Vu)+swHp3bfWx@qk?t0c7gs_1o=ys!eDqtc`s$*Y+34%?pksmF!zJ2h zmk(p*pylS5sSHq8?mI3HQFW8Ri);66MR*kfVx0QFXGib;ZL}PfL#_4LAf{sqQ2r@o zZz;mpOUQCYvCiF%ZuWfg!h4W068X{FX|m^5q}3hWgC#Q=MB<~p1=ZJ{gg52Y?yoC$ zy$)-v#a|pPM%8^Ok(YW8u%%W>Aa;8Q0SZGQfy50fBW=0(`8=wk^tsgx5U=0P_s+l< z&u3ZAROD9{epn&AEI-t+ZD3xMu>0hHhf#I4o?tvJCH=v`O;JyGdfOe}t92_JNU>xf zUgD5|zuUF(vxGm6W+)d<%e{@ZwgSTFWobcS@pUao{f#O>c70{i1A_%`2W5zr0|3*~ zxdK>ZS9q@b+yVbn@%Cda1IOYZZpPt2L{vm$!T+SF^1QS2m7o?L4lyh5J3c8!Qfv_R z_j8lf-@mmM*=fd2vN+b?3U2e1=zLOL>~Qr)QzPK1`L}ZDMys>eZ@m3s=oUhacL7+j z%Rvx{&$rToN%ywF3Z?eR{Zl1t8xU1xwNqd=e0C;U)|V3T!qx1?n|2r`J&+KT2hgA( zH#-n$1fQYWp8(E8;f}f zYmwQ4(4$|rbP&UzB&WhIbW*Q_9_A5hV9pK$kmvW*5U`lb%Vj_PC7v;=s`I#}=;2WI zb!LZMJW2{u32uB*{z_z2)?Ef#+DN@%3vWGdgEONOu~q8FX~-4ZyMPbb)cye!YO}ZN zqr4tiBiz8Q^oHa>JHB7zvi#VyGv46IQU}`y{j=a-+8~#t&D+Pls}iMgY2#VOJtn## zGF?uAdA!{>J*g(FEEQJX)-oqbP?-ve{=TcwCKixo!|oi^fIc#9cM@xJmjAY|MGSGG z*TW~xvgz2?`+0!$w4L^qGy9)nwmFjF2a2)3>)mgGUR3|Jj4pr>JWtw@1K=OYZ^K-d zZ#j=1{|66z2arRc3}^X7D)&qV%-M>}@zkHC4)hjT<(3A` z)&4RU1CdHhLn+t&maus>F;Mu}eKJ=AjGFd7J1snP)zMK**&+$xKByba9O?B23*bBT zI9;1_Q|0egd{tUHup*3^!%Uj{2%&PnY2yr>fHsHfc zs`%hUZ$8LmagUvtiMQ6D>OTXDB?E6e&f&La14Xj_s3>6?IJ!Z{-ce5F2@yf}A8Uub z+y%XuV2!9U|3nB98B7LLn^yL#me~0Cl6-CL^pPrs+XJBWb(?_(7W^s~qBx`W3!zri zU=0X9mfc*_#w0h)U9i*Wx;DN4o1hBtyeEtp{#3$$-l&JVGqnNY2;{xjwD7jE7=`5E_g99SPc$sNyb0S-w?uat zGRlCmY;g0rzPGyW9kSIvCkOE z3)*E1ec1K7T*nioXE-Pg{aZ`Bhw@*17XX}AfdBaSV`|hZ&zRtX}e@HQmsMKw*Akdpq!YT*uP1#MssbN}lOJy{7I!MLLkd0ZFyy z*0s-GC@rPlUBGYeR$~bR$3`iW$mp!ZxT?Kh^dUai8_(fs%3w}U3C;$2!>>Pz_~~cB zY|0MGAp*;TK3CX5+S5J8jgRPh64*qrWX*-_acNqC*o%4M76h z$|cZ!2=O7D2OCDDXWCcs9xvyf`bc)oPdjahbK}cK3TR~9gi%9@I~)Ek7dr5~ubRcC zlb;>XJyGjg#IAHbsqf)EEcj?)LHC?*w<|8yk!%p1!Ca68CcZaI3I&}2nO3^q*4&#B zmweTCbb=~MD18x+(whTE0#t5PNw)T;nEPl#(9o+KU?KwHPfxOzGycCZs#n3L>e(G& zACq?lI@sdE73nF1>j?Vci)p~slf}-7fnx80Wn*QyV(*Tgk!PX8e=9R#$q!Di0euh} zB&B%Oci%de?qL6^J4s!&(tJV=cm>>u48!-PS`Z~$5r#tc$v3dZXC8~aTrkShef!)$lsrAMXk6rngr4qfnx;h zDaNWAKzY5@{T7(2;-n)JY1vMoD=@wes`2^$bUgC1m|=`vpaG;H4oFAOt~r5@4tN-z zL`5PV#d>N*C*Jc9KHhjAls=c&=K)AHYwG_Eq)L@f1va$Sq2wBy%X%GgchyxTZd@$)Ulw{J*@h7s-Dhbn$#4HVjH-j4 zH~o94oaR~Ti!A^X?oT(NmW zSt0qsCq!yc_wzR%kyZxbxd}ik&}K<(*qI~_X)Q_zHtQfDcT<8if8MTt;3>1Xqg@+9 z?$583LG(C`QNu-mL_^~>&cg;^(b~1ml--ms#OG(A%RO;&g7%pxz3IDDMH!*^UN4#rIYxox8oSZpiE+pl%+95b&ec zO;Yw*V)&<~9?@-h%hUp5YGEpD_`5@jq{5m(fYx``N7S9X2%u|u{BRmCS$W{e_Ab<2 zH#-<;QeDi|$-##lC8E9pJ7h)JLz+{x)%e+B(6^c_+e~;k=msQA{og_DRentJwa#{-0LkJNK>Weh#@|AeP%`TSq-R3Hc{4$6d>?Lgc<3YV3cJX3iD zWj1}dJEPZcgP!`${MG-ar=I&a7X@gb3;P!6KL&upvin=Jhl5@9@bzIJYg0N2$N;A(`EjqdG$&hm zU)Mdhv1SD#PH8n<9Up~6cYV!|J1i#+4ILV{gUXk_N=)k-eTjWxGE`I#N&rBz33Oxk zn!)vb*#k6&zpwS`g9e}yB@W~#Ji_ZqtIYu#{^8D1&}g&Ew%PamffaPC;oWJ&hIGjX zMh(>tH$MH_i2wpW#&Tg@?8d!z%sK&`&-k%gtkYXtQ=C5tvMjE^+TuRB1!RAMcDNxr z9-{AU^x1G9yDGHgMy>R2AgSLJK;vIwT#{f=_9pyEpWi9>&E^iLP@uddE((8F;{Vkv z_x)?ls$ct1X8nf2zpqKY7A5en7T)At9dd5ri0d8$v}BE$CN5Ul>yTddxAHIg>O-2_h6WZiSWC5ZNRN@GJ!GnbSEMha3jkWdbj!vY(caF^dYnh5E5bR z%b-uWAzr){s0$P>*V=dYOMb7_{I(w?k%|NX?&A$H4etz>LEw@(=GgVKt$F!7@Xu@j zOC+0t0WhVdVX($-irIbGZc7DUL6})KJ zVCOxI7_)IGYd?3wCI6|Sh~Mcm2{o}tf-9lR6K96}Fp&$rCGF*3PaW>_8}IPn5J}bD z0@ht+ggPQs#Jm!aA(j9NJ~7s7_|HJa@$E+N+MJV2C>QWgk9X#`2JQ3GsvB}}4Tq`F z|8sf)c=@-cZU|S`w67`iK&)ldNjq~TACI!S^1rC59aeyovtjZ!gwst0bh<8{R%Z5hdKUZ`JWuNk4!_)#ZK<(6|@y#*3ER zkrSrQaBy}Mle9YwL`*Ly0c`S?=+OC8KWW7OQptIDJrHqksFb9RUyU$rcND7=I7Ko4 zhN>^*i7$_V?bd%vgIxxvu1RU)!5+K^A8FAH%52p4HO*=Ac)ZK0tDuK}6?B&T;&KT= z%Hn`7JdreMc1u)dJJ8vFMS+6g&yCy-1|=J4N_qI3KL$|hs=##>nRgwmZ(tf7V_&Kq zt|)zW-K^suNDLjMZAJP<0(+eZ@C$K45l|%w&^;rDJAgxtdEnJn2||mejnT-JeG9pM zBi*0SO(|txl06Lw{ylrwPv}Ia7NO+xY18L-Scdgm2IwjeDoPVZI%XwVHd^7JNh))4%qg2TjOHAU)c;CVV|? z0M=WuO=JueXN;UDwH$CyPR!4%?KjLEtnkFg$Zj;@4rF{H&XpvSZ$1K0C7>kA7x5E3 z0%D>KJ%Rm^5B~*GrST2_JBaFE-UdYV1Nf%LK+h;?dus~<5A6r_rHn517v|f>zQDTx z9^|d{sGR+soDJ=uMi287VjtEC25a?8*@1ASdAtC)7U0+Zp9s~cY#YSjk5&*xyXd#3 zf%b>Ho<4P_s*o4x*Hu2`p^5U#D40s9XESPoihI#}`K;Qq~oYXBk(F2fhF zHe#jEX#6)E)t6#Gypw;}YE4U;(F3k8Arqh3HGAl2!U3XyV%X+Wu%(08Hi_A=Wk%*a z?ZBfXKvd)47sCJMsCEUbm@atTutxY47J;5!2)yiypT+#Gz1^eZC!HfblDjUsW_H?* zLs*3?b{|8u=@374+QxJNnKq`S0vBFC>!cBA_*H`SQJR3t? zatCr^#V$NvxWKHO(w$0JPOMbIDRPl?^g3gnZ$Tjm7|crkby7cWbiVQwi%7z;2$kSJ zD(5aVrk5=)tXZVDQl``v2jEQc2;QJZEs!9(E9j*@1efdj^O+x-hKH4FnLPr?R zU9Sv2?Es@tAbrB&Ud&2dRzhDHYrN1al)n;oIwqlVW|1j|gRhkZe10^I*Hg!ggC`Acijc|W$u-;D2y^2{f@$l5a=9`rb_&FG*t6hT3H)g{|PyRKT{&$!^atF zZW4>@Xkn`7owbTT!jeUXK?J;}X4`LWRl&v1m_5A3aDiZPrQ*FXgYRTZB(ox%jvxul zMgMhDRGnv_a7C($!ADk8SS)Wt>RPKJ|&o&NAHzqa7ovU7Z^hZw(CnxMI3eEuT#Hb6pB;S z{$1o&aQaWJ=Cgz(T#Qu2hXjE@jcd$>j$S#q83(TaGM`22tdAk-+7g8c5zg?}7AZt> z(^zX$HBPV=-Nsv7p~}WLaky2B;9cMrrK*KZV=Xb&y2hCI%pzh;51+yb$4*lwsDh?# z4d_lGjhT+tb&kNXmYH<5MFCBaJ_WC?CWo)Z>0;MgD;0egM~2|SNedQbB;w~LhM=<( z$6Q|&uK#s@6tP0HKrcG;P&hQ(J#YTAuqiB0mq?0W*CMg&TqJHifK$s?WAjzFByc9*^0;QV$X(Z0Phq9A2Oq*3koms3=S zSZlw2>dTPs;xgpJU|15a=#0P^t#hCKG%YC?x(&JAb%yAo2bj zbpgxjYE9`BhC$(2uC0)Zy)RtJnr-Sj%I~Ajv!r!V-P)d5zJIV;K3H#!uo& zMm5ZZi4`W{pPqaPi*w@m0f>{{LS7=4!0a0&ze_z z!nHftSZ^o4Cs^2`D&RVs%p0ajoAzCZb`Vy!GV?eq+^JE44U~sj z8m!MZD9c!At_b52wn-x>J7av+Na9o53SD?8cjoO3A5YZD%2lGq+%nm<}iFq|Pv*$CVUCIJuVCWYUb7#3;f9bBU zyIrdDf!_pm&r!_`wFgHCltC3N$+#v-RkemPaY12dMwf095_0%=DWy=^`QLY>p*taL zr>jdb#GSZ$e~waRJ6`akh#&71JVo}xe20*eX})>E=Vos5< z{)cl|mg~56UJncOan??Setm98H#Vxo&3KAmaAzU8YKG0rm1-w?LWjG^nEv=)wX0Ht z))qflN2}5%;@8Ub`OIwzTwhj3JK2aHF51f14@V3h;8)Jp-Vfq(-8&aD`D=f{ zgc;`^P7Gx+iJ8Omi*b-nEPrkaDa7KUFic-$1b^;Sj3|R<*2gE1POq9ZZJu3N@(C*? zuOOm_6W_Fz{GGnqf@a~^dfL12i?)bHX2RWkpE;2MO4}?GrQK@^ACrvNe!-%)r2M3b;@9hu*o@p*yyzJQ;taR z3iEZQ(cihmdG;QA?n=VZ(+cf)gp3+-{$sbL|%F)M^AB8e<)>N?$_ zx(nSt+y&<6YFrm7-2OBSQ;pj)vBK5={`9Q{8Mn-D?aajD*r&^-=}0b{izDM!3WRGc z-7GBbUM03y&rF)7N)nOpCLGg+DBbt*Hj8uHo~7tE?(Ay={AeWf=JqC1g2eJJfp zczJjU%LCnsK>LNosB)Su$oWF5`xN9WW{rhw7tp70?BPYfi{*Qwh<{^Zsk&Kw`qT%H zy~5F8$ME$jcQQXk7aIqk4!tv-AX~4C6EcZdfi|bMdh0v2A#wyyn#W8*3KrCKaD}sX zKU9VxRY^|ET%pDSsh_(vMYv2EIT&%%cj6oCAaAc~)Y;tA*ajSPj*i4I;Kp}0hqLAs z$!wmK9sJ(x+~i)n7+*ry*G^s1$J-2SR~t+zMZ^zZvp{j~zz}pxrNT^yGLVN?Kgyx` zUcJ8I--MeMk#kokv%u0eBWRY=b*^Sp8wcyP$GkIqQt|X3%SAP?cH^D*s=3|9)u(^0 zeyhZN$#Tr;fUU?p&g%8O`whO;f&ar+C~SFs}=XeG7*1rd` zF!qFT_}kE;1*!NaP8F5E)rRMvCED< zO-((`+u2kvFbzmYPpyY&cV_bMgFo0u`hE`1xftwDG=jC0r1~iQ_ah7IcU&Cj*CJXR zKA$F7EC`pE-oQvfLUtZPb+j-^~yn7EDjz37n85)kSG#<1LC{C&xoaL^gvxM~F zG9r$Xkiy!?beQS6eKCO4pOXIrVp@swQw|qZTBF<%er5@lCNYN&Es_MjNEOxoZEQ?_XCh0@d0!_f!cVv?#Bezi?Dh4}c6I1w&q}54k&(eSX?;B-irHwS z&6M0ulGe9@L7#);<1Li(?D83rU&gcBZVCFOYG#^#3)WeUQ>!_Ny zmI2I{;k!R`_|BfPv(MGa8~?sqrYeLacX`CMspsF;Kj|#1hQhrjs{Q=^XFfds znD*mMu(^i%@6Qrn@Bi^7QhuIWa290@1ROQhx$hU zsQWni)hXmY1z~a%r$_$+IqXXUZiZcyNkm`fs2KLvd!tUtSqTcHr%QO1xGGjrIzLQe z3ropxWX0y_Ye8~>QNo1Phl;AW!h$9)`*rvsV@&yd@`&j$j>^cyHl!QQt=03+8IL$A zaW9hvA!_NI9|~FZG(P7Mhi4jAMOegCnJV-M9w(sjXk-E!jj8G~uNL$RgPpiYen^vW zi9@4w(u{sdeoUV3NMQXTuZF0(+&%T^d_q(By{LE>|4ieEDvH^&d0O?z`)#L?)caA( z^!0uY&3Ki`Fjmc+$DR;yLdc}ZCAJ-|sfo+Hq^y>3)Y8M}4VfMr)+d+{sEr+?nD~_Q zMeEn;`50!xsi?;FW5}4zD^s=)xGp5Ac~`Q>5@{_}2{JGxMgjw=|VA5H=g86Kk6HF5qP?dv7_3Wh1lBb&{Jr7QCG zsj;%Nui<)V#)P7Z<<;-xH~H{x8i{AD%5pe1KwO;dL^GgbYB&sL$i>y4&+tVn3c0u; zcg~4o%je5wyDl~@MK@LZ_8s5`+hyQ(&Yx>;CM<>wA>SGqE)Xe}gTsx;?4Ne3tekZR z8utEAXEjzI`7kX_m7>PNcE7_3n3uRQf z*+`>X9ppwAoENTT#yo*X{S>8T`MS}JJ5qMIqdSGGmW#jx)$ z%;s5WU?+-je4g-90-wX(Jt-_n;G;bRop9s8b<%I#Zp`VR_2dB4+|%fFAy$o>-(|7F zp<%=3`V2lPBSYp=1|sg!(rzp!@a5u&Wihwc*+uZ&2EGeCAvLkxZi#Qct7;%dHbsp~ zLxy4JQtTI4yKz%G@Ddbm4(YnUMxqD(B8=9$Tt4F#*8+}|9g4ITo|9ZZw};Cc@ac2( zsVbajG1$WUF}zs851Fg}(;OvZt0pyJFnS#q-Ze;>Wn@V>-EzbYArgcb>dJj5-1d2e z@nf&Q4au=R9(WrH=!67ZUq=GY4*jVt?kjTdGE(ZTYabY66HT}$0@e~+%5~_mV*X=(V~x#+(dOGwLHXT?R+NWOI;W3E|RSj9JKmpYB6J7vE{)u=X(;Vy3k?I zXzR7!CN@)V)TdQwvga3OKBotc+9Qn}cEV}#)n_VGHmyr(%+SdRdZWb5%`aXbrtElM zsi#geXRZ_6QC6aNt40r%3xi#2PD-w^u8D3b<~iZ;2-gQv=N=1^_0)$8&gM%lWjiKZ z6&krdfG2O#oE%=F^l*~(MrZ$AJTrLY6f<9Px6t2^E-NVv!0g;EcRlH~3@`Y}Lq`mGaVx<*ih?A6;{wV%ccq&hF6+$i8Bp<1#B9{N2@Sdz|Tc;_+)28JbkM>JG}sI%$z2#s**-uktcqAOu9%KcR3G# zr1v*I63wkP#0y0X104^OFYIoce@CPDUHr|=6*JMc=wmZ~X?9fp!d{iyOB<h6nx_ z>o!rfWD^dqkKbA@!WlkXcckH&0&E=3{7 zr`FQFIVp7VdMl2P6Y{1UKC<~u6sxM9#OjeBUxi0Igezp;M3$1;cDxr{XLFO&uI6?tKSGYRm(932Xcs`+e zkb@JZ;Mj2CDOQ%SkG>5T(a2fR9#LN+^U83%GKqNO_{v`n_ZsIVhHwE2w+u-3BtdJi z)Np3sH@@D*I3eFc$kD|Krf}T#YC}OQ_jHnAhC>n#Ub-(N3nrR6aTWXw90~UX>+`s> znkDR_ZCho}_3Ww^NU9|iF0s$@e`;(qkja9~GN}aFnb7x5-81L0yj7yD?s)?Sk1tu| z{Z5t{GBy>*)ewFzfEoO@fbb6NDtmV8dRX=d+@j$%<9tZww2rj_gPqo^rgIRZG80-y zNztv!_}M9&Z;Mfx2pMDOd&3={!-btk(~N#>C$obMe@~pB2Vefk&2QDM{Tu7KsiAsE zcW;xJ(C}px|H!ggJuM%`!zK7e7|YC`$6*9hvNPa4D6Y{Mc%_U(ow9vRTP?b$VindK zI^mMoqlI0uS-9PGKTpP^_E*q0UIp{9Po{pdL6YzmHCr%vnM;rwwf1k4Dz44>{*Y}6 zz0i1WlBSbPktd~O%r4{6XKj#S5ikG!Ml)~2zthoX^8E3 zOT(I+j~AtOM0U+Ss%bVAF(*SV^7ZOH{6O@=xnkLQ_bh|joIs(WpOKm>gd|Bo{usy{ zn|@TABN~G|pklEbZ&s5~Rg=RXYWw!ks`_T$Lx?0oakI=Te(0T55!9F%fhiVc_2s4X zAyi8b8B?Z@-*1PI{bc;ms$Qzt8jHDLsa^|-{ zvZMhOxACFcy+|v+?a(MI?+olGtBGzI5Dm~mvCN9Zzd=demVwqLW)>y$A_J%*@J)D z%C6py_gq4mjCKbcw3oo+O}A))#=^RT8&o4Q(U*0q@susX<*{(3QvsE$O34%_j23Bx zV_x`7p9Mc|@@%_8l$GPHVpWSv6RY>?`%q)DH({`Z9VxXQ-*GeGS{=z%&A}_zal{Dp z^EA=!(c@Kr0^cixi&s(qm6SD4V~y~*VjdP!p(Fe5r!jguwPy` zBw!3Ok3yY%?>>B3lM@r+&}V9S8mlV>Emzb0x4{mc5mMJ2-#K5JuSD_wB)%XrVR2=?*$*-9f)^x?g*E)WSL^X zO`I-j@whwOB^H&4j*uW6)%)n|nYRZ4UUHW2z$<8(`2-u=_q8;r-*`KIz{ced7)Q_S z0Y!dS->%Js)qX+cBQRuZb0!`^I5Q2hh`!jzlzdx+Du_(Ca#sJPMOaqePt+px#PSTU zCpPsK8d?O#9I33ydHm?@322Yw(*4!0SO{LH#uan?Dk}FOaj$Aga;>IrZu~ZUU4q~5 zE08}7@aoKm+cB6Y1u+-L%6n*bzkgWo$Rh?DGT6~9Iyw&W(Lc<$c>(;jyd9ugJtO#` zZKzy(SbHjG3nFdmc$fEZslQoStEs^|?_n8)s<}`MfuXshT8u}vPe8NVkoBY1p7jZD zb~adeXW*Wnw*!RHJl!a(0E`J3+aJO`=on*b#9!)Rkn|!_29&9PLI7r?PSl#(Ei+{S z?eWWA#yplPu?USy46t!W&Vy$*sW3imw1Ea9%q`?WhDaiMFwIJ->#Z#Qjp>$~6CacR zt#7XEyS>JLRI3>tQ<7YxS(-vTR-Yq=AM4!91-^inF3TH(Ts(8JMiQZO(eO6P>==!5 z2URiq&liuDjVq#%d8oN92Nmaunw4>o&cWM)2VTcd`z{FUN%z>8MQjk7cha?!yKJ8M zWN#^J5!>XB5*LdkqLz0eJ>VJA&>rg<-`_2o?5S;G$FT|TIMy)lZDV~dr9DURy%1H@ zUMmBy=q-raeJ6pxz#vp+f!ruGiXvqw#fPy~%d*iXnQtbnO#l|2YtD|J07wKKWZ zGw!)4fA-ElEzRM)y|swrr+RGpDu`%bo*aVu4`Eas3-qduVdtsi2E7-=Aq_yqDDbw9 z;KDZ(Mwgtgk9KRf=*uA(C9m+Kfe~6?{3|M6c|qp01fEc*Y$xr(QXNM%S&|1ZkGTFL zN=VOBLF~qR!b^hkWD}$;Nm>e6mHis`%J^c3f$MzOcpEk(?U= z?_7LoSrEAFLWQWrJh6Qqs0!3`iucBaPkzd*v_)WZ@v3>nQ!Qe(%OUGCv%AKECyR}5 zgYoAq09v&?*^SuMP$1}e)c(N+VLbK|^AHF()ld~9F0q>w$IlPjAVyp!j=Ri?K%L;x z-VSIcRmI#V(kl2f9Spn=o>5jba{vIBI)nugfOcMLTM&u|q(m4mLJ=+~@wo8gQC4~O z5I{%mV0`q?%_yFF7Pn;i zwoCMiP1)+>m+o)Jf5nA@Y-t5?#^SEq&RG|}oU}H3KwiE4DE5?rmc|k$3$p(THas`! z!PEgQ4V2991IboR$5#H;7~M!Y$ZQ(csvGw_cRsa5)eUj(=_SbdM|-|A^r5b#$JaE^ z-u@6j)N6w|p0nL5>933Qu=8l3k`2U&vD*gVuH4r5mPBs<#*iodwZn`_6~`VWp4yjr zRI}Dz)f~v}UaTX4{UwS2&cEd)GG`ipXSgJ&7AaM81pjz@7tP|zg;dxdy0m8Pg|QTl z*W<41`>^~jca*_UZ*on1)+Mo5=T@JXRDJb6ABcF)4}CA~%LWpItuZ|*HJS$vfk)u; zPfMu$m(2h|KXqJ`U}>Lr#DMT9zT=~`d)lXdJMli-_tI##YPM@47rSp2Y;r(09_pf)>rY#n8_thD8qm&-ST- zLh`P_iwRGU7DfDO$)R2XQ(NJuw5lKs@w2#}sy?`S=r0;u5X?XaJ4F}t%U#OP$QO!_ z0`B&LJ0j?jBDfcdl+v~2eImDp!7qF}{#5MIl@*$SiR4IikY*vQaZcOLv@`=&PEQ_t zc0r4fmu`ZI?a3i3zt0Q>lPx;>DNG03$!^1Rrq1`y1) zt0I1o5*yz|)H3OI$!*kZrtb6(L^k7^)9ARgxHw^I>#q8lx_i@Xpwc^rT8pf?j3Pa- zY83Z*n0UHIv&2q?BT4YTq_I1dvmF764)I_QrO#=#J>^=-p46{ad-!&R2-^(t$G|^p1 zeKWcmW*!XRhw?Pk)q_Tli=o6JQ%mPx$TGK1>EpEq?MT`A+zDG4*r2?!&?hXg4-gF? zufv|2RDC~}X%)PC@t9_VpYeNdrv}M?ONd(v!T^DP4&U_?$U(#C_M*2?X3wsB(DfwD zAY5?@yBV+o(@mqS?reQEV5iV^EC;_!;4Fu5(tj6*yOJb{*q0Ob6aHTRg4v7x+ArE1 z>;|!u$Crmfq>qUEbuIuKj>#tveS&j2M-LjRa7`S4CF1)-t6clqSd*&b2+S@T5tl#e z1}#a9n20L%!MuB>SLLDJA*%QrX5;hv?qoMK_(FBi;gix_ae9~>LgL+nemiCEmLT8Pskm@8OzmFkOuF`d!Sf(JU(d4HQF<38<$b$FOJ(SkC@m3!K!%j<`17t4MaeoN0w#()pNa0i!CeoRtzMrnTG%#${4uw z>m@cvTb>T=x9Q+#{`uY_iBNfpuSF3{3ybsLgpDqc?9viip z*Otgn(rP$1es!>-F3cS`Lhd;*dM}VV^#K)Y2O#EA-izb!xEDS#F=eS zcH1LAj#MN@WEW=GQoFx<$odYCRm%H1jj8XtPxp8EkQYWr=o1QG9C%Wua~z{GmcL?X zozmj*G8wQy8bLNo(jFu5R>z@FA~b0IPJ^S@dn;=f59jD~JTQoHSqK)U^5Q)#H{ z^I}=(iHiw9lm}SOW_U4Il=#*r;by|vb^qKSlnXD2?f3!xh}YD>7HPGOX}}AlC;<=p zsCB6Gzsa%*HiVW~aOxS*JXaOg>m`QY+cdTZZ%Jbm?8eD6KaGe^v@G#AsLMHTcx(?Y0f3}QAf<72+abI-B@!Glev_6J_WQ%% ziThQFq-zuA&tSa1P>H~=39Z$HlrqKfna4WJp(S=T?Wm_-Jbq^B!t817zT=bm+Aoy~ zJk+uG=miF&{}A#T&KK^+zX7+BQC9TEOn%JkPA>oxU)XgrWogZ_Iw%4jus2iVVnS{6 zk-%v%p6tDDc@OunGoFzJTOewG|2+UKP_iIaMDEpl_BDBlK7Rx`7b7B4lA$bc+l9e?GFPV7Iia( z0{f={Sn$hCJb4r0$H9PA);K1(*z0uv?$dMN%y@X03&@tm@TJ1Tc7%?g0*Z4d=`OEJ&>PP~sT zXa0;~g2S4(L&|@ic2m@to5#a0&1Ch=fr?~3htck(4=1yC7229virMFVLEZd3tdabn z4|XMI%bdL|R6{{$FJ7lmIS`S&11?VZaSOnZ;=$lP43mc*xho?|XsEG+YVdsI5j*pt zz*8r0N*!PNecTS=5FN62PcqnwL2L8hfmF$iw?n;0*1m1WE8GT%I)c)jt%OE_VYJn% zJ@7Ab?w{)1F521EDT${dt>35kXxbUwAjD*e*bo3Mu2jPwFKXdlq`9+Ugxe{=ZNjfo_M zTSb6p-~!UlOrt~Ww|qNMqw*)YDvqtCz7Nt zr+&z%g`M}I8%(`0cd$E(w3HwF@+4DktV#gDAqIV*67e%n%Vu9D*bw|!(FeI*5-2l` zTi?EbU^lSb<;I#$T1|jS4vpt(dOJ-~Nh7_CTh2aot>-&GwTLa=ApCoU=M&_sQn?gF zBM$2JJj>&1g9O7@ns{;IupHDz;2?(p6Bx~R(z{xmP~9=(IUv61E(1-Tw``x@sf2?U z6UQ5+ZwypC-T`o7bM=#dS_FUiZ`g zYxkW4e39qQ2l}e$ZUKG2n-RKuF0wjzv5V?x&%<;CBhg2q#wH7=+2 zp~HU$RH%x@{7hPKmMwIHMn^?Vd?@WmP}$*-VWs?AVq1pQ7y&@g#anhkpo~{rn7w`l zNg(r>A_m~n0HIZ@`MB!`WfmrKnc8qoIy!__^-t^-&7s4WCz3)gB@+SRE`@_Qz_^#?HO)6le&2~0bU@d7yu71{ zTFx+Dwwh>o?RZ%Oh5rP&P=+diTXy>uC|p`R62br3FWoq54eWj(il!`_e#eB{QdSaV zv~a+x?x$dI7GHq$a0OU81B>h}h>UG3U~};?Bi&@EtZSqquiCI3ujlJ83zhe2mP9aL zJW9hA|5yN1iT&K~WDtT7+CM6%P-z2Po;(A96nbNKA$6;)83an*8fgoX1qO2>8V{Oz zA;qpAFu8+4nyMN#W%RdcPVyf{<$d)zJbqlp&Ev2=Skq2^Pu<>SC~BUOXu5Bzc=Vrh zv5gLBqYzM@sCQt6kzw_*`VH<^A~q8`o}U0lzSnJ|)`pZ zm0h+9X|kjgnh|LsSvq2D8A~WGS;kffB}OD9Cgh+JV`wZl#*)23BrPLki?PL&LPF&J zUhe1f`#paDe82zQd-Ld;Ip@63>-Ajys{c{i*6~j(jg1^YW`H9eUWuKR`a$R8e zXj+Iqgmz1ArmRl0uhr1pF0g_KRYx}-Ox8bvx#&x_39sEpC7)r4EE&*oBIgqq>O>)- z?ZGH46xOoPVJ$MCwGXQSAdPt>2C?>KS5?cJ^&m2)Z^)g}9JE z=K^$I#*4F#0Awxt8H|32kK=_=Q*eCjEiWyra{Q*6bL7~^h8CG$a6QfcCgHJy3|mF$ z;=}%BFlr@v<*1>&6C!g%<{)!3zx7*D1gZJSFY%JMPXju2^P&53bJzoHypiIK_l6Mt-dePC3F* zD(H*o>BTiJ_vTwD?1qeM|8b0uBkq&!;qN&gGetkI7Isr1NhS4=IT_!*Rv(IG!xl0k4PYy!*4nEXsfet6XQDc*vzbroDf*U+Vu?Hmdc~J$GdsZgr5h<4K zOz1rk#$DQ2ntVeeP3Fs(yKxZW?P`OazilzrbK%1yg|5_7y2PZHT;TgKB}j&B&4uQH zjF?Y9$cRfIR_EO}RhKOJ$n3vUBuUV)cDi{~Os~)8{&dCpQ@p4|0%TUN1`#|6{x)Rh zz4uIBXiHO*%*J1@|BR0HSWN%)A>-pAv~yc3#S*GmD4o=hx?ef)?S&#qjR5E+Io}Rr zS8Jtp&SaRJSIoUfF9|rOSQUKvJ5BZV7oY;vpcW?wf9$}=p!KsE13D7&~7) z`y>lbt!oC_XS}|=r(vafR%fC0l-XkMQuhn%z2rUL7SXol4?kimgGp5|FU#J}+=kkZ z(}F(qhE}j3w$AaU2vcVLD%Cw;Yc}&Xndvp3+BB0fEq3gvk91~>VER6ng>zYPhj0Tn zi>L_UqUPaAG3{x|H}@d|!BbTEB#Ur33P0{sKTG?shBv8Pp30S-3UQhnR=Cw}DFOI9 zf}cQf@PR}4Sggn8*%PwGflw_r8(}vndV#Wap`@4DR(T6=qy7Q+x>aBCq(-$Pk7um(_37;U%`Uz0{6JT z=|1q#GcZfNa)Ua_szRElbr+q|j7`e2afxfX+uenQ_YwSq{Dw8&DqnL_QuoRBSkAVA z&)kJd=Pcu26Ex68n*-2$t}9~uu=7CaP*?DTD!U_8_!P2A(W%Z~7sc01`gbsHM{wsS!Z-I{ zZTPnDL4<3aLf>Vt=K08ZbQO1ZnC6YUh1|xQuYFV9Fk)XH|L&6$QdY`%bQ~N-(17+x?j-aUhLRc*<0UpP+AKi$UbxU z8V`ju>DDczFTjx5+J+p#NbqSM6Ta|DnT(&J`}_hPK>$Kp1l zJ)tsai)V_JI+ZITSAL{qd8gAD+wx7-xfjcT7`GMgsZvi6sKGb~JS zYba(jZ!$f{OQ*HnC0ur?VAwZoVI~n&2iw@08snG_ErhmWmiXVx49738YL*BA_ZY8s0zi?Wb!9{#xEA% zHmLK7Fb!X>P1X=xfMoV7y@Sd3GnuE#9svZ|b%k*4zKEKQ!A6lTC1YjISd5jJHv!}S z2{9@v;dZUE6on{VNf&CjXZf9zcN=qj>QlEZa#31;o6E%oZd7%l(9j3zvalGeyRzer zc~6aoF-C**)^{j$J-|T{jvzfHpq9iu=NEhv@1^@mL^+C|fOW+1$v6~P-SRQ>lRf^y zXVPkWRWKAtzRX~H`>{Xt6R!)o#2W6o6UGjae#$uy2+PO%D&Ls5xi+V3UWub>*ipHI z;OZ`XF-h+(zhi8W0WTGZblU`2Y(uRMz`)GYGf+E){pR4y4zZZ2h?smTbkHFi8gtij zg7=JsK8KmEW|s!mI(f{C50lffAF5Jh_$WbLLgwLs@r3b*zQ{d*AWZg1ya=JuROdnnl2Ur9X|UMS@7bA5Rvg-qFn@I6l>?sF?wk^p zE5~ug5+@Fwj+0{jqu}{>RT0m{{8pG_j7}(cz!p;xl3x_a61TR*k>I>vn3EfBXj%PX zkE@Be)cLVOoPSKPp-E@KTKZ<(p(Eq$dD|B$iRD@w59UWd(SsFhN$tbEy&qc__NRy# zMCRW&Ilp|Jy-tdQgQ}Y9Q63e4==lw@VHear0Hm{Hv`6o4m;h2Ude_j8Epcj#Cwr(o zpeSP(E?;D{&|a1DY6bGd{P;3`+117t!uw92gdMcg6CI^BKMAydcJLluoj6i(mY1XW z1#}niYmp&FP}y6(!cD}TRb-@y|21eko80YszMgjd!nf5XTnzLQ2W(Z~4vq?e8DD^LVMi$>_$1dgQ+t>`j)m2+R%49L zLvnGpyNV!q0heJ2mi1pSu!NLC!1ym%9s)Ku4&$rwmyEAKvha|vvlJOWe%FMG>iDIJ zV!4wXEwJ;6xlQr6Kh;KD1_X;(lyNSfs1w3&yA>S=s)N?=DwMBK$CcPP2M}p?peJ!Q z;UlN(`XuVP)T`v=zPhr<@uL6Nx1!j}2$3mb_k7=ZKRF0;nVLZO=0kq;I(`|OvVcoK z+bp^wDjl;}=HGtCUerAQ(w^=I4m4E4>l(DFD0IF1;cGb>^vBEsvd}uCqFD?=Rll7p zaUse4>2!SU&LaJWW|tFF{>$S&!SzXW+qmWVUB}*_IV|)tB;BH}M<{l7`$uyz0nm72 zbGrWa$Q$AU99Vgs%1OIlJ`V5-Pkq8U-z7!np@=2%%_k{>r!|B-g&y4gO*QFzljiAg zmj`5e$q&Kwir_6-b}-2Q!y^~Bc>le!chclG7L8CiOrsGhV__ArzZ!)vqR@TI$*_gB zebEgJJ3jf6yJ$HRe;0!#Gd3sh8_%x{0WK%veoYlsnZ73j^ZQUY;aarl+Xew-uEryj z8$+0_yoL8%3!d2co!Ourc?9JsRm|O(DJ3y;c?Pl7Hasj6Q*nIJz%=|M@bC$LLGs1u z*$Ywl2|)-#$UulHGh$4_r*X=4b_gEuD+U9v{DP+X?6M@*@zJmiQdQ(ZzR9gTKjZ##^pnakX`;z59J-Qev$nTJxH0ie}_ z>faCs{tR_Di{QfKKr@TP7Y*zZY0h4S61mXZN{V$DwUW|`G05%DTl7EQvU@ z^4e`#`E9&8R0&3h7NbK7uZKp(RH9o)D?|dM(SS{$5I=Dd2F3ABu>MoQy%eF1vK+v_ zI0K&4_9Zq(g!f!k=-NV|I~UE|-XDCOaCG8{?%|_Ya+|c0CtLT}v;fku$4(3poX;on zg}02RjgK^lv|W247w17Q4L6p%sOk z=hWk*)T8B}`<;Glr`VkrK5&FmEDoek^Nh+J^O0JSO~np3oySzfeX{P{6}EkAB_hJS zzC;+b;?URLnpwc{-7-}z|Ut6KtVEd6H*V#!D)@`$NVD4UOPU}%WY+? zit^DlL{W%I!yb5GWULZ3{sL4M9PM|{inw2VQ_<|1KWBs0&=x?n%G}Sg5iKp4%W-e&*OTz0ApO)yJRi()H+A6{Nc&4tz$+`qhcDIp10th;6)GrdjRIC zLxTf4?QJ5Um$<&xNy=fsWhOH=F)eA(z(j=KMXyJDBmpp>gjbY`A;1T9o@(8zLuu{Cy~j<$ z5?-sBRW5u|MdQZh#FR~vpD)S)k#i0r%lS9kH_9CFLw>$8CRO%wj9O7hVr{=T;Jsu^EcNbDt7C&<`2m282>vEM|;coU?MI5;pPt;H$II9Je%n6zh%{6p;SfbQrRB8eH<+`KL8`k(kW z#M%8V>-0{lM6a2C-B5ZEOw+*HlQUV+pTNu^*9maXApujcaW2fi)Cq?t=N`en#I(CYWALRG`NJ#gu zE+u-o_w6JzUz{3JH>(bizMpP;r0tJESk9M}#l|-)Y)REOuNMulUFg7)-B{hZXnX=8 zn|Y`@$jm6iAj2CeFRSTyTKc>>^DY$3J{+}@wLt(HxQQRXt{8>y7~+#c4_9fVMv%5( z+CDdBo`+{}-D0g%;nN8`Z`peb=n5|~g8u9^szO&?A1+aL0Z*{i^cR@hW|WHaIPbsQu_~IBJ_aH0-#L#B+yvFpYoj z0{r2+s{o=G&{GN-70>2#^uI(t&LDB{Pyz|d$Mv(uCSa%4v%ED=3Q0@>=){$?hE}Y` zc`piUM4_fVg3Bfq*FFx#OwknqcfK~HMyJYFA*psJ77eL=w7BDK6@YZmI~QX9pQSW7o`!dQH8+@@BmT=rBcj?wr8?hq^F5 z_xNTFLxCLIZtFHQd?NmSQ@s8HAH@(j3j1fS_XcZUslG8Z5wbgS0Gc-r7ZDe@%YlX0 zb(hE%z46F;+UgSutp|n@sYAW<1&;N!EAY+@5pn6gvsBbV{lUlO!|pOY<1_rDbslXK zV;y&c$OS&of&l+v2g;$JpIJ0h2T7o912!QQ!*@`|Dvrj!3GI$Kj9p#7^Ii%O4n_ca zC%3j+k6Nw9K*Gp&K>2y^sDHflF{w#8@MXTtjU!j`K_do~eBfPB2LWeOGEn6D^jV9i zMe^B8eC>-fUWFw7^Fhe7+qPKXN(C6h7Sj{SGB*^p0VwIJBwHblw^#wH$tT3Ja7#@e z&VV{nt-K;C{sNey5NLLh*e2~S({NMn?kAhz<>h!To6H0#ENtqN9)L|dk++ft)KSaCvG zxXLo;BAhnm$L8v+@LE^^={vT*x8*BTuXTF*cN~O0_)n;`HW0`eyM1s%2jKvwxrt+^ zadP($mG1jBN-It{b|{LuSpH8fEf5qc5}5&db=x`pb8)qXV~k@!%~M{b*@QgLZsB#l z-S)E~vC9KH8C^A(_f0h~fTeu5K;)2}RZK^}5DDu?$sT2#+CoFC=>WqA<_jSHDbkR5 ze~FWXpHdBnBZcaIp{)euFlC!>W^C1o1ytm71%P%`u9W7(mB9c&&6fY_?d*`}z+|WD zO&ZxQm+`2gIyP$f$%<^;@Ks?SMSPo1&z~+&@2Ii*7D&l^LJ&Y`7x#2>QA|ofoMQ2< zgvjFO7;gCBq(wM_^4&SqNTn@G4pzTUDXV!^dxgYQ=3V?9F=k*q+JlVLcDCIQm0NvG zGY!`$;hb+3F-{RtyY!cW(ifv_5c15adYm%8WHMIh4F5TH&DD7VqAYK;AoDaBehBW6zl}E10>(%r)wj76;A&ZkPel z%_QMMpW%*`EQN7XgN!jT=3xIlJs;;Wv;9%{9(#qAzK`cOin^DnTy9Zns?WKRgu1_? zm?Sh3Z)Z7PN zHaA3EwA#!~gT|-#g&Zk_qFDp%zs^5+4C3n49xlWF~O)s(8%#qZ+GHX5k^Fi~W zW-8EYyDNj;TpM>}y||;_`{+yC=b#Tz0Xx)S8S)E6N^W-%KL2%sfqb{=7AyJgJ8F)C z*Sv@2Lt-P~ru#qQ*~TD@()G=Q7}sS_F~I z4==R~s{T0@-qr&Qc4$gVJmn;8Pb?!qEdej2x~OV~vax|D+!1lgZb-UIv&J5n0-mtx z9`Jcid5?hCrW%-(ReR#G0VPVF~=eE+|`iWKTa+C_~FN#$nO8#L(~ zHJ(S5qX^r$HOV(W4lA@c6m}WLNWoQAFuT@^lQYanIkNL9tynh$GOU;i1(~A1IW>8i zGTLF^p*t|bjQ<+nyuZQ^xeQPq@Vw`tur9P-vb+^kM^p7|jK~)>l<21IBJwE)d1RRo zwR}7qNB217O8N>$qaoq}-NZXvg{O>kD8gp7M>m{^_Avk~>smjtK9-yMSg9-UG z!q>A8I^(T<`7j&9@HBo?#QVOd11o=JJw+0$vI2T z*}vLY!?$Y};AIjRQXD8P#VQ?BKZ1$?Bufh@O7O1&@KQVhn&G5%3amJGU3-0^ZvI?W z7pjwRvXbT&pKpWcTJ`B-ob>j)){T!#4xS{iS6(}e%34VtE=sXx38U& zIx*lbJQoV`4p6lv2ZPk9*Xlj7>&ne5)pJKTMi#VwfLSAZ zpmx`0NhFIX8O?Ns(big`=SLYnSHc+d7oml@t7^bb+WcdYGaGf%a)8 zW#_+{5p_b@(l6dvO*7g=$Rc}_q*6dw6`V5ED?So|vE;Z>OEXQ_BNB~Q7m&!r9r$VE zy6!k30sE}B2b3QEvQ)uz|LKXs-c|%Qttu8I2MrTYmi>0V+8*aG3n=~9QMmAQo-313 zXaDD4gnf!Q|bm%4ZN%u_FKrOMp@<+#hV(mkEblRTK5pRlu z8R#hQgeOBQ+#U13fjBbC#*a=}0KaW@K*20YFY7569Gru-cWt*ljH%gN5Agk^vC_Z9 zLHIzJ1_C}4KUw;?D!9Efe)l;qXWo$OIHLmH+yyjd=EcN~=@n45&+?bqUFuw-$<_s{ ze-B1phA{jwR|)By$5zay#RfVEwz9n;0EvgbA;-R1g%jhk&<(8}Z;32{|E4G9MSDzi zs>h2qG$!;WejfYqjov&444in21S^Tt;@cTK4k)v0s0jwoU5MVOB;>|zS7!}@8 zW;os*r29IQT=S-Y7WS#oG7#Lgk%7>l3X1phCq~osKDSDtB1QvHt2iQa;~;jjKJC%& zf!ZCxi1PEtZnv1AE!M5G#!kX3#ilwmt^-!3tNtWB{6@?a+K|ZmMO<>U%t*v+4byx?ap~QTE?fYK3rf=krm1+K-6B zld#tZDgs=+&rUL&PS?{2m#&$rKl<7niEUajzFU1X;G z;e2Zz^eiuS!TBjylok_i93lOIF6wRAcuPjNA?NP9*&d;MrL$kPpBP&FZ%yx;WK-Y5A$|CiM)D3eg*Kz;F{lr<_+RXvx)!zi$O;+u|11)&Ard+7# ztTONOg_fS0fK%Ze&_)BehLxQCaV|vbt15*b$r0XfoDuxF*HO!9K>CIsr@t(e0GxVR z0}F!Qk4OpF=_h?(PWiHrUp+#RAnw@!9&<+>XWu|AYvsOk5R%ypa+NwQI~7V&cd!rp z@b&ded*+1HiUK>mEgp1IfpOBR6%{b$9Oh?p)Ikx68eNcsA^icBO1HHXV zRlz39qEMG?wqIjIox#)FyBaPXcboCsuOqOiniT;JHa7nQZbQ-1w5iFZ&QmgbEjg0t z*$Zfhl@GDD_H}ebzLA95s}pt4jnAlHoazb{x?zkCA<#_BE@jrE*QE2;{f2l940B)RWZ8rz(+^wMGM zM+o=^Vet`c<${VsUG911Ek}J=v;^IZqgcn!oxKgL_Iehbmhk!#XLPD*=_CH=j7h3QzcHCjW3n2 zZ=5KOxU4UZU470lLvbFau@1y)dp*;N!*`tvk&So;F?dDlQTVCLg1}He(+Z~UoBoF% zwVV!P@UZCJF5H;D5LL&poI-OAzL%cQ$fL2Z-9bD#WE_$6cS3g46pj6C4ixlNyW^6% zw}P?}7D=9<&cFfnkKU8-dLyiGt@27u$(zdA6y+SKQo*}XDHXvErhgOqYJPu+TJda- zB6QIJ{o6fQ+bx(bz>vE>0QYl^`hCDJj3FO3W!gqWyYCa`Zy!ei*|{e^a(y14H3?IN z`YxUBs@{1T z*vg$h{Di?|S=dBX13a=HP2e(4H1jWYeJ6!1mCY}|{CHj&2B5Z$6#yHb^qqvJP?e+D z{1)CZL*&O-`0#z{O`=bPmctkSCVD-y%C+R}UvQyv$rF!(iZ?!xfA3r69CxJBUk7CE zV4qXoGEF!~Q>{xz1&{~&wGVORw#V20gtAl`uDS%e`I6srDEc_=sLM4R zsFk1-y>90}+}%RG{tL>HB)(W+xu1nunKaq1h-8W&3TA_Y4Y!y}{(6(R0}biUcwA*y zJO>BGaC;35$=WcCC!BlXx_&u0vuWnHfOqO4cAOOM3%V|Om;uVY5A%QEW>PB$bU9}s z?_9WJ(WmybxxXkQSX#$Zt`efpWitM!w_157H2g|m=A5_MJbqb}G7{5gPxoNhx#pP>ytRQKr!8nY{2QawIxkH4NmD3bY|3vyL=XSPiJf~rQ$ zuW1Ou{^KJaJjjOZEXz?lZWs3E5&+2{W(Tc(_@MGX^g|J&X z8Z1S$Bl#$dwF^%J_~^Q}XtLu4{7}h`;=hhcXeO0m$nCX?>C?s(ZI@l5DRXF>hJ~-jn zN%*y=#7Lq>+h2EeWIyFn4p5>RXP~WnLy^1hL0xn#3b24XpkI@m^(BcO_2I$fUMRar zT;L}_is<7NQ1~os)X#-Uy7s`o`>*6!AtQ^RDqY=Bk<8u%u>6Y9mUk)O{Ye`C4aK0< zMyZ`0*(#Jyjc0(Kcw|D;sRsEztYN<_#aJW{H4X}C#X=QhqXEeC#NUJ(8b8H~!`WPl z)e!ig29^c7B#WM8ucq#_;EkY*G8SKjV;P|I z#8}mtZI>UIV-x>+>@z9mOIW(fIse$g%OMwF!g9t3U$$!-j}_$Gi{^f%mpt%MIhkyS zix~7OEPKZTFE_p=SOF8W;aAN)?>7wGL~H6crd}?WDz@g9dN}Z#?>zxd``L@L*@!!} z-rsxq+TjiLeK>oZ_^wPD)PBsih{*D_ykBN6ETCfwd18OAl7f& zhO!1lI@WkM@zh<)M0+1dkk{Ib;3rRexUNGP^@PT*#76$*>}KVH#dz-S zdmIy%F7oX7J8*1B!GC8m50=NIPJiVrFV*aJ1&h$~@03Q6LgY{1xvPZ!(xD9UTH$AY zi6__Oa!|yanVKG0^2=Z39jcTZxSxDn*y7llaj0g^Vcz0&y1xhXPboEB;Ar*;fV~A- z*9t`{a~^p8GPF;H!5ky=s>`$7?t~(KXPyW%Yx5)vy=JC#(K=m)aY4NZo~S@&AGu;` zL1)MP-PddQXp-1U_HxUya|SW172c(zJ=jdbfQ}@H9%41Zztf7Jqa)=&87^Z%Ex&o0lZ?O4>nTg(~A#M(Q5IB?g+dn7SWZ0bM8p| zpHL8c-0R5{ZU0cz9e-r27_w5oA5}T(14qTPEnsQUIC%e+Sk8x;6GOX?Zm3Jd6m|1v zlLKQ{c%k~O5B?Z>LkuMffS--(S`6t;-YEw)EAdsLs+-cQ=qdm!0!6iRn`@~B~Y^KqQp|5y~;`U+CBJ;#>P>Aop}%0$)xO$jihORFTK`Kn=>;7A4EVD#gDuhc3u?>2?%oO)M3!a> z?V72zn=0b3aWH>A2+>a%>dhZ+bN;^mn6`GvyTqUukR+^?;aH5*KgztRixAkaYw=qv zMLDuu-+YvDZZL?@UgY&22P*O@oH#uPN;JU(^J~2AbAT1p@*NP$#=9$}HBobTSmk2| zo<-;nlgkn=@DQNKw0)e*uNGaM-VF{wd_P6RfoNY5%ZG`%)k6Iz)NZsQ?zcFWtcdcR z`J)1)xRzDK+!+W4ZyXDc#bdplSIcOoq~cu~8&`cPd&y0ULnf*i=w*q;mS*+nr{89MZi2vG$ow{zpOdS?2M{DJYQW*$4z?Zx#D|^4zg<=xli?O-PVCg3 z1Lp?jILtCXc+V+~%BVAOq*eptr(Axfd%uJ4o>sO&xZ8MJU|!?EP)V$GiZa^!)JJZ| zaxPIiz+!~=Yc8f_HlSE&w++n^G*7d`pL-)RQ2;0I1-pf_N9p8^`B%L@-UmEan z{8G0zH~-q}Woeu^8h_~AO5H8|L{lvWZKEr|?Kh@*|Esf-;D4Loxk205z@#jtDEu)W z=lRT!!#otQcK?_^*{LYa=AyJ*y9I^io_RKiTyP%;#S(!shgV=BHl!!M>WUxl1L)u@ znwpvFW_v;eTlQOw2am_Iw4iTM~Q)sj@2sJh5unKqE>D)jbkz; zdoPkiPoErRoW_%w@&!jLRmQFEShrSzH2c2O7iLSJ;kfC&$a-?XG99zGeV=l-Yj1<9^YlPny>0eq~;wP561m+D~{0SHhyGk6< zB|$uZ^J!4;L->5rrXrsbm*)jDV%0XZ<AxXmifHCBt{F#huPM5n1N*H`Pru@M*&G z{4eoMTaDKwmq|2J!QtE(!-k)R>hx{_ts)qe^jgdnx>{+EFU3*vs(i~kpsIa6>H&$JHXH*(kqzEhWZ%*9+KBug3;RFfT-xfuA?|z|+V1iu3I=OWt3(a^)v|+3c8I4| z1eeQ!#tsV>kRFk(y;A6{X&CD#e(}w3q5SbGfN7U(SqcvYl5QD`Mv)0B6+OJJh) zv4aePF+QDuV%X}0oi}PSb5tx4OTjRR+t~tEbe#yp(zN+Q;6>&PwZuoc+*zI^WCP@D>^s z_jT83X5+s(NM2Am=~unYbhKxH9prs0by{HF?-p}g*7zDM@BNRphI=7BAQ*9o;6+uk zZd*??1Pftaa(c(^0&BlGGNcBJtjj(Hr9v>jgVQ7@N2oAYIr$Hhn=NtW zewH4{vx5(hFm~urUV&nvV~m%S+OFe2(BS%TX5@N^VAd5kv&cg9@Q$0}Nc{4XS0{Aq z+)s~~pEdj5OgQ>aEnTCY6CtTWcGBH7f%*LniQ=fjZ4rJnERv?>v1w+mvIY&i#{Lf)na_v+hU+`r^T4VZLR;|L2~1eih1txq4bnzx#c$uo%f-Fud!xLG!Fs`h(sL7t zELfzn=9j$&wsX#>A%nKJK!L~?#nglwapk!AXwQXV8kI|p36J=ZGq9un6Lz+$^iTt^ z9Hv$*&Ak1qOi`V3r`INRzq_-?`~clq2*_%`^s<)_3K*38N*;2 zi9I!RY9x$*9(cl#_3f6ocD{f2*vIh_h>1-`j0yxWFB@ntPP$wt^tG+I5paT>zys+) zSckbH4?}k>C{h88l#O zoy)kT>+ifw=z}q-44Bnc*mWUrRS+pzSb7n{*4^<$99hd4+2;FvjTh7{oz1YX_x=DN zMRxFSE@ZohGY4hJ`SM@OqBgi-?T&4zNrWB*76EUw+Zvg_eb6qzoX?-m!;dDJYWawd zfq}8iHVW^(zXvbuJ_TtNRtOnCPszl;aUjHRdmiWrBD|0ykP-bhTR?|A_vEZDKLK08 z`+D$?4w(R=uqX<-G#M%RL=_mw=*CFn@uBZJ6tUG!29TfKXl>HuA9OgdQ_T+a?|(*g z^f>}U3O81S??&8uM*atEf$v~|^oL7ZRS-GY`!Qv)Ji%;!=}_SBcZ6wGer_=&+;kt& zt8VW}gNy{tzX0zC6g{Aa)Q z`dy6p+q1wZxilo2DZ3p?E<=)t%j7o1_;r3+qCFikohk;l@n`eNR)dM%4`JnB3ixbpHcE^B6 zC{k|IHI@I6WbF%kZi8r#)vf2J*3+hg#Q8AT4h55LA@k4|N$WN^fBV!X9>*vF=Xev;V`nE9^S< z5(8wd6^O2%C%7;O(<)n3n~9HIp$60*=b}V~v}i)MMv2GDPnU2%c8QIjOlH6_{~us> zy;(AY4naZ)(JK-*9%^Z+)4yx+?WL}okq*szqngJB{|=07Qr^zl^NX&Y_~vT=&Et`G zKP9#gmeBa;0MniY!F8qtc0M8zc(8jQ*0XMlZR*HXz;k$j+7&n&5H{Wr1RZl73a!9@ zc51hWei`UhYD*umh4mXIh2}Z|hG^9#9!iDXzsf$K5=&ErpJ%p&XvQB}CgWv1i_L#J z1mh##2Hu&em_Hg+0{D2%_ryle%+tQaBGPH7HYJfkmDt`3_0KOSE#7u|4t)$@G4ldy zTe;nqmKJy9?Y>3IKBc=7oi@tUr4a3jWoO6VTXA%+LuA+UhO_Pnzj&$@G6x2i$=Vj> zfNEe6Wk2h6bj90Rc@olbv$jADZwF;YuMVV3 zWib%r_3DZ0#!Hd zU_WW)#%TM#g`#ukj?ZKe^Rb^>n9;`mt-WyMZqaXSNCLs2D#h;wEg@kY8fPQ5dhA}r z3opY}rcX$t3Pv>`Fbw1c-F#0&H!4lJ%l47`Twy%fvkZ@G`NT7`X#9lZ5tt|)1SeP9 z%smi_F6PKpbq^%r&C7vaz)H6>k~Qkpk{}=E>ama4uV@MZ$kjvzBT`dG;6=du4zQ3r zXw!8~5;JS=H-s};Nhc3O*>&!;rq$FotPfk_WJ5CD_vevPx8?tyHXou1dD;-Z_ocXI zsv7GH@Ga%dz!|yUg;#&YKDdh>n2j-he}^ox_(PIgYC$<42doL8Z1+2<<=kxm@?OXU zZN;WufA+uNgcQGBhSL+k4zP`C>uE1$8V?BNbN_3XNu_cXyzo6x&<;>9x!RfduC47P zS3>j5Y~Xeu$M6@8wJ zeN3Xt#}TsgaQJ- za`~~>Xw<4-+MC!J?+3+#DUGr8V2GNye0dE}0}yZ;qvw=}_WfbjF*N`V@%hVq->!5r zG#YQF+X2pivNuAFqZ4v{njjp!S{GMULi9?rMv_Fixsd$1U_|kENNM1#{;A7?H8*nN z$fvCZ5Q*C_{{!A2j?edj!L-$KigMokc)xfwo>vZ{&P=v^%d^KB`kt^hxoQ4|N*iy3 z=o#~>U~=CRRTQg#ax|@9b1%`$`$0sD=U%=aQ)BD8{(qDPGnofmm-i^0O~jZ$Nu&dz zqf7bm1!$r6=>L*;<$5Vvbm4edZ5coDlw?Dba_BRq9HQ6Dlf**wRM1vzYimnDF}$@4 zXnN)qn`n&s;nO>95r3<^7rGQ|fb98i8!`CZQdOnaqmZM1W0j)+d38a5_oZ6D=(qKB zc8A=-n}3pT{8`k8nECbB`hVDs>x=qMvizbgbivnYFwaQLMCSGUj?tj$%hk3i#TeRdVvvfr)u;N!a`B9Y{ ziD8hs#<#yRXX=Yun>H4wJ^;Q^m9f&7o&qUlTWzuaU%=S1xO1BkKpuUsJMP;a(^1?PD(11w zhf1jMKrds|A!L+$`04N^YI;-0cO&bU9j^oSMBz`Sz{*CCIEq!ljCjMMzHS1uzC3^y zDZ@v51?ipABwplJ4y7pI7XHCNkGW*{$Kh!ET54?9 z@U~Ema5OALPfmPT0|c}pRXiP>oz|0n2_DyFJxB2gfXDO>+vu&P9swolTC z%uy((cJA-B-Ta=0=f9N5na7o(d1_+Fg&!a7!*1iZ^Rg9cCfK@hVRc%6l6&k*vNwNH z>#N(N&a;W&7mVn?c&#LTG`8nTmPM8Bd`zlL?w}9hXv3hH_>d^#FCeKM_;Y?Wztgj= z2SYRgQ@BBzvAI-mc<{c+F=A&SKgqF!1GD|F^P7>XhmF36$ZJ&wUw2s*B7HwD&PW*L1Rv-+(5bkkFj$yNZ?@#!{H#pBUgf@Dp-CYJP1E-vwt?U{ElKIOHmVA9XO zQX(!SEr;@&c0n4g z{h8gbygQCnI}Nb-%;C+w8UPB@6OXJBrCf)D#iCD6*f9zdbUVwvheoY9owYlab|yq0t_-hPsLZZkeL2me(J zx$Y8L8!0Ed@YGm!UCZFb1>AR=tt<+?Ti@u3DjZ$M%`Ns|kIRGD*9a^?-IpO*<=6u| zDKr4f``Yx_g#|Al=EfNQk5hF!rxewa14?GU7VANng2<-PC}ap;QSq0@IWTXfxuZne z55mN2WPD+A0xNWKbYeKtqHK3>Otv$2$hGHa%fftz!nXpK+0$69x7+Qb@jo^*duqM( zHxpA+$neH9l-onJ&pw2;zWVo<|IJI~zwzuC#Q4@QX!q9A^i8wR-X1ej*a{k3wDdg?WLVe}?;9}#Tgor!< zz>LxT5fv{Suox%6~eH#e7MjcQ+;$`Tc7Is>@>(a2h_*Jv~;D81?(I zy_k1Ez)R2p*P6VD9g8)`l2YPckVWp&eRsPPwHy*0&3PyT@a~?C;eGAM>XQdXB1;tf zp+`*vCxdP=Sq5dIiQ})wA+NT>w!t$?h4Rg3m=>V3LUPpeN$W%z5Nad}7;JfDuInS1 z6(zW*BQlx~=p`v~KeEg4ykV>3DN|`|u9z%*OUuX58;^ ztsULOh5y2S|3doAT?Ts{BSNOr#2|2ql4$v9zHnf)XT=i2wC3WHTg9Zd3wQ55#e%{v0cU?d75dzM;v@TXS#nfueQT8qkD}gOh2&V+ zV;DPj8Vt4@d<{}*36fZ)<#~Z1Qq4I}Tgaus5I16;5zz`7!>nygTM7M#X-&B8p#Unn znif4nOfFEu zY=axaCR+a+e(G5nX)DHx=T5W4S@)EiCESwIgIw9KPE=HSx@H?H-K&{emGoU4qQu_VN^sntOl1WiKmWskE@`w^NtvTKd3^@B9vQa^sI{o8pf;sGx^^WrRr8 z{IXHL#5)P35HO+^uZbbzs@1#*`mZXL%ST`6;BBj0QPkwoAj`P)t9j>=Y}_}uN3k5dS^XUi5PDf&DrV!(?uyS(1X=zU*eCn4z8?}w zCOcCBJZc~Q(0d?2wx{9N=LGkj$`U+vV;ARSePU*@>gp1G?)R%}4Uf`0sSAg|M=;W> zOzeaJBFo#ve~N)Y%}4@ejn$PSJ{e^aJ@$zr6Y?a5h11 z5+pKaOn1cgbi+s7k|#gc%Z{Ggs@VL_wI^osPSTj;@m1Z8HqJ-TfsTfSlQ0vVKO;4w zDdZ8I@CB7Ho%nxqPuKo>qZtsl;|Qfrh%YXpwcPytR@M+7$cFr1W%b*N+;{{=#W#6f zXh{0aZuIc~J^3^MTXS|Z`833WXvftGc>Z%xWlYev&eGh%jR}1_@59lTdLTNL5DwJ^ z?^m+;H+0321R|Fv*=KcRe+?$6(-G=|)ceaW|)3W+jHT`t$@Y<5 zmK0GFF-BuoC|M(G}mZAK5<5+Q8}ZGk>s;3PcDB0BthWr1uzyahr` zp$B*|7|UsK zd4r2$Dc{3IAs82?Sk&fGSs99zkgu7I0(2;?2sA#RGp+Q4#oxr(o4T3=DPvKF4?X*) z-IX`R^;j@?`LM(^lD8olhG(E&DSuhjTyxY4mY}aXv7argWyYG;)8E%+kPqMQRF{+|J;B?BD$co(%}?haV0#Eg_u`R8vezJew*X-%Lr%+H{ZZnWQ_&9T#_GjGyY#VsMha78i>H)ig3qrK?Tw_xz{44)2jBl5;HgQME+!D#y$%Ls5f z5$5B}0n8yH>Cl1H=(8TsLKqEK&?Ledk(@7iF$<#viN!twt-xcOO1xsvAO zE_vLf^mRc+VwHNM=~cbPbW;wDI6kJQhJ0eVZDSL~=6CIKzhP8JYhCSM2<*e?k^KRI?$cIRTQV$S!f5;-#>icYf|tf*?8PzuN#;EuYn5U#kPLkRD^9 z3HPfbrRTC|KaPk-G+CJ=N{d(ZAJlEA9#aXWHUMzrTf=^$b@YfR663y6LZXGeq+#d8 zrLwNv(rP5g%kEUv&}rX=&iGCx`doE`iV3h05+G-%CEezuXu1u0n}iUIl}pKc<``1{ zFDO)@*e>ucmR4s8eNzv^CI#(wfteIE?YngDuY^KCL< zRDeMS>az@~58~DyU>lW>fsWjBLOxo?yLe1RuG}QVBgEzm)Vvy}AR;*nKuJYf5I16U zyoncy`t8e0X@eH{d;8)XXPoHw{=nXlyNj9E#CQj~a`4C#vrwGaq?~LU3P|g!$c!@s zwtB$)UDpGsl}IZ~3cg+SeR$G&o=_chxW)+;8eJGS>*ZX8mhX`C!!wg27c-s`|HEFd ziJ5Ds@VEYj{L_2jyMO$uQ}+TOIX0u;n95auEu z?4JWxA4tCPyTImw`;Lf9DX0zYLUV_v@S*otlfX7YV*X>7gUZBKgLt8Lr9*Jqqa03C zsU1+nhP5S}mcobwT9N;YNGd58mHW7#7vuE?`kH4NU|fh|syRZ~B$kKrp0ohUlv#L~ zwsO>enI(){1>}&|FE-UaGnPHkYKbe|M+K-XvoYi@E|)>=A!Jz)*^SiqZvvg(Yj3E6 zw&1zqM3f8T-n@Sz`8fQ9q*f+&rGM`DpZm893j(b0lV^mcuCpqnp`+e|XZH1r-pU7O zvgo^n)9A-L5)i$uY8w;rHGd)3v`m z(qFMq;p&B^wKQLw0ve&f6Jl7j^^3qHk{3LP)Ak-%TYnlTd-#4E^m|;DMG&VwYL6_u z_yH#$5pRLGmc(`Bfc5mI-S}lOfE*_q)R5C*ug1hGwn3N3quh*LfB|ym6u?iZ-!@>F z0T@Lm4m$$6(xNiwbj=lVKA7u~9XOAFyGqOwEPeDR6MyVtJSUwlsDmw4wk}**Yjhf_ zHwbwus#O5(vS?HKnAw-&Kioi$hg*M9>J)1*2-?shQS z%#i+1y`|?J*m~5WXL)ch(bnBIw8Xk=lzlo_h~MekAmF%jgcVZu3}J4#QR| z{7pZVfU@a3m(9eBCP(q2R^OEyzdoEcWw7w@g)21C^5Oz3TCWvgStOoT%dif=q`{?!Hj8buQYZlsp31JggBzdQE2`0=W!j|}g>(xwrnHxmSZtgDVFv0nL-U`G-eWLch z&vM&;h$36(B!tnOT2K>hhi^U=2R3-}?cES7UfspGK>X^3zK4>_eQU*X#e;_2n!PnI~JGyf+6mC2qQ6H5O#Kvf33>74&^A><-2QxV2w z$zu7UiXzP4q33_H+^GPt?q3FSkUmfk7LJ8#Pk%rYbZIs2}WcNbt}|33myZN4(g+@6kz!2p*Z z>EX(pfJ5GrK+l65;azPA@R~hJr61|dk>1QGO@G0)-jVb7e8v#AxWIO2b3}YY&S3>* zNC)n|rll@uZ>9ykr~QX7FRFf8zUN&th>V#DK2HebhDkl1ywVE?ou))S<)vS{Psg6w z04Cx}l=Fi^8kQmH6EE~6(2R`y4^q4t8ld642a<#mp&4!NCoM-wDN;pMntqg%Ikkor zB6=bWyC7)|1K!m2CE#>23E$k~kpg3PU{n;x!dOB_j&y>uWNjbTb-T{ysz6!SqfG_E zic8Ok!KT0d4*GwDk+Ll7WsN)9fr&8`f5z=2W1Ft$IC;h#;C2bLx^qQ|O@0jfIq$>KM}?do4Tb@aIQD8(M^$yS3~{rG4&e??Ff9K8RMn+n^<@Jvq%5i!;}mCL zm|fSv!#3;@HD*Z<@rVt;)MOJ%dR|W?toGlet9LfGY(^cJm4|@9SU4r}=z%N%el4WZ zZOT_XBa?I@^sTQ|t|R*~(Wx#4FDal(6443Lz7zJuIss4DX+Kvm;RKV8gz#RE`P z%@}~HPLAM~!~|UxOmBR`Hsv`z$ALSw$>PVM#7`p; z=)k*cefXk+#Ekx;9}VHC?b!OmlAd`HMqKM515je~H&L#76Gj}mva-qd0~2}BPc0RA zs>`lHWZLY)C!BA%=XcpS;k0iEh5Y4iZa$Z**K@*df1s-BZvNMLK^hEJ6JO<5opM?q z)KUBx*@4aeYLSve(rV$}&#mHx&x6iyC7lZ6eur=FzF;_N#4j+;tgidet)bS~?tTj~ z(qmP%rSkz*RXYc$s&Ul|JgAuBp5m~z?7f!;l)ZBC9?QgC@|t{jS<+I#k;pBcPyZE4 z7nYwAMKaR7R7no@*a%dnJYyTqsnIc=2etTc1RL(3u_(js(Rn+KWG(*fM0UuU8nsvh zw`-T9SzeL22BgW4^VnR0;tSlms2DQI0PLt%lm2?l)7bd!A4)XRO~;M5aZs+t*lo2v zd{PShHB^`51y1)(2eQaUeRwi%?%*G5#L;=j)IpNO&jnt@=}3dN49sPL@kjU;`eXWb zB>CE_oE``tVB$>G=c%l@=757Q=j1gkL#*(>`XDLqr=&`$k<^*j`dE8_^x9SMxlGr{ z_arMXbxN{^!-yVunJ+wvQr}0?(}VyD#!c&8Tg1rY10A>ct^4t)m?tw%F(Qb<1?^M#UyhmhSEO zGiu_=;DR2Gn2L;SVN?$0jNX`3>wZq;=JIH`vi?=mq)tACk;(6Z2nTom)qkb>sT0t{M4voBwTrfCaCSts*uRyUQFLo z%$)dK<#wV@lHW;Fho~|ZbgkxP;^n9F$#nI`M>T+_=y1)!`?|F!fl~A0hAvAE^JG6& z{{BeMWJ%2CB!q711=y5#mDyy?g^PTQB+L-u6gm;NiQ@W>Z4Nfy2`T|5~ zSGM7r@2QGuQ2~8&fO>SOErNd*W67al?!sj5vNsW&*4;-Z?8achTYBjFsSO-yM&9Q} zA;^joda^|Yo!&$|Pp4u#uNL~GT(_DWe9@ry^zcO0A{lmyBye>;Q?x9qpmKZUM* zlfdt)Wo}0M9LD0`qhe=jxsg}f=pjB+&#KNMNe?cpdvG7a8L~0?e-9f zEKlD{`-pn0he(k;boJCVmyCz~)Ig=p=#0bnpkset_m-^ob4xq>Rz8%l+V*1_?1DwY z&I{@#HH#(5pJPe>Erjy$fAgXikj{-KTV z)azp#0YkgR-FuK5nbXovoeDC&=2zFaJgG|g7SqbxOia2|6m_3gIP3I$`w?)y>45<3=gq@mCB zp(gkH2@@Vny3wd8a#($6pu&Qe5izp=H(7gYz`rNQu8Q$gaUwB_;Lyy#4D+G^0}a^F zbVK(k>67We#W~%SRKGc2?*YB4iH&{boV&uj73sHqKV`Vpz)iuluU|#>VzTx&aF=P_ z=0sE{PBjB1slvYNN@XHd69ozQ@CBA;;m}>HiH}lHp%CJ$qB~|Gtj}JB-Hr(h=8uHr=d-+k}VAHFvuvRNv>@2t=3kY*zCX4)r-< zz;exu@BB2kiDGrJ4G(0}{J<9$88G2eKB;o5td!&iRB) z;exkq5q!)0Nh*AYmp5S^?NWd!^}gvK(^s=9pgg*Esf?Me{l0)pJ%Y{BxMy!|1f>PX z#kg*#(0XjKe^!0YlWS$Pb`>igSjr)tL8fxAjN?msNYleXrv4H-$yiF9en4x!h&DBH()2gISAl07UobqyyP4Uk|hZt(gk1t(k!fYA{;0x+^+6I|dmce~f;OuD9(xQ(Bs0zCS| zJTM@nie6^YJx)nmHtRzgdE%P;Pmh=T~?K7tv>gvOVCNj5p2fgNXv0@7A{;s1-PpCXWV1_C&lWkiaXN znBClP(!HyGMdjR!#Yfy_pKMyix_BunpH%&rX8TCAORwe8D&}O_u8A5MTT8RvKcmkV zKf3WUimHWX{a9kR(>RglJu|4^U43oudl2yL#sWb_JcbHOlhgbo7v-uldz3X7PV~fa zA>50$oVP)y%wx{%L6*7`1&&6Cf`$G6Twm{?4gPDlRT)UD6%#Lkv^vWst@^z3oH@a) zE>W@(<2~O^*vAPdNu6e1O4ZS{WvAm~dkX_!tNnR>cY`^7Z=uRM`)W+oIi{Hv*E3=+ zTqyXF`#{Z+8$NIp)AcSMJE;sYf+%=DQYNdc+-Sns`ZD zTmAEyma|%10maX`69h@_#Us(T%>{oHF(Qox3qnItD3Euz`oOPeM$Q6)i3>yQi0frE zxhg&LPqNBR+03Sw{0EgkfScoGIKEz&BdLf^{1cq8VrvXmeAxn`brZDPn-B5eVQOKY zZ5CIIDqTi)Y~OQUOq!r1^h}(b-{BXzApq3XjWzMK_Bql{*lqOb1HJ0?UDG94fjv75 zlpzw|Q_7lMM{*Pnp8fR*4n-BM1P$-8PPq;nupM1GsXfd}1mBHAmFCKplJZ4O_ zX}CCStq`_Cd;phu;HC zpmT_n4S@;M_&AZ<<3L(1Ch%!m|*@ zO5Q|?e=Fz|Vww#TwsT*?7{tX*!HB)DvYg9()6+Yq;5n8RJrLe~ByIUswFPbtRUVoS z6@|F%;7Y%fn?h=9oO@qY{pJa4LHliSBG`u0JbNKx`vDJSrdly3(JK2u*)LIS!?TLR zbMl(U99|n-IS)~F{T~we@uR?%U4zS0lB}QSisDWp=~QS`zrBI^{nIspzrvh7Ya4%E z&?ENs?Y7R&IY#J5sRP#y>vES3{)|Og8H4iTuZqJDwG8dAz{Z#G(TX-0 zRooC{TGE=&6&Um%YL%}JP^&aHYPAtit6MduMUaoCtFoxjqplZwdvssXu%BZdJsvkx zBa8^Q##L8oP@}~3n|Lv;OXuNj+Yghk0&c-{GUAm>pDYJVEL@OeM@{2!NgiM_cDqUW zuHSb@Slcj9U=x+hO`SvTUEKC)M3w6J2G&n%*n#uiScg0H2WU;}DIyHe1!)r}$!>Pi z#MeDP!_KJM#J}EgMcq?@HXBE$dd&HTopM8SbK5zG$cqBU@0HB~(s_OnC1*P-)zz;m@&`JKxD$p>#KfyGr%b$El-a zD&9#cSD%^1q4%i6=|tys`^qfI&CvV(hX+h}&4O22XtgG1^xg{P@3#!OFYQu!^@PM7 zo9h+2EK&=troR1k_9ZeRpt&Lcj;KH>LJ*hW^Ww9-0u9@eVpa2@5a~i^zZ^)o`9>pO4ZU%GZAz);jq}p7* zRp0)|t{c0we#FpzJj+mq(qxS_4i}#l20~A9)Ubav^}hTmr|x8{Y%}lc^t6lY>6W@gh9A&Y^G?C5%BHQ3I{qn@6;MAH{=8!NlQ2-{=q;;b}J5_ZV5AX56RwS+^R;mKC^S{v3>n z%9=70R4`@JRxtvLpbzNpuA;Snv%Lb>T2lErBpPCtc-_0UN9kGn>FS2Pe^Q8UxhfFKslq~Hh@lihIh3fDw*3uStrBnzh@V?_j+FwT! z&TplymMDdS?(jnXb)g$MIDwU|0rj>bfv=TwyKVI)oa(LD42w2A$ckrE*fO1CAFwB2 zMiX0gCKc4FUvcmR{N?T#5hQ|TVIp`~fd@_!+Vh>vl9)FF?|a+lZz0aC>GyB%1_|RT z?M`WC&0T-lWgNTO4-ql3@03}4vPBz=%Mb_!*noCElq0ib zD=IY;1t~+lYzjH~S`DOLf+XWi`Ct8&CN3pU7jJ`^D{B{~-)l9wciTAwn-#BR!%P&P zbOYX_jF@q_zw4J2xHJwcaxc;_H%&Zq6?i_B#KEf&6}h-dS}Rc9D!3A-u2VD4cj7#uR8Zm!j?N{k$-p zC`QHf!5E?d*svA@^wh?fVw^{G)+2vt;66BFPc2Y~(EgPK9pn=aYSJKL{(@_hQ=W<7 zkB?~hxX5vJ3g#R#=&5jIEKyu>5BuE)P(W@Vd^T`-9w{xjrwu|W{q$c&A|rb~XKzo@ z2PiN1fyc}}<^|Vpl2*o=|I){LQ4UP8J}Y;+BbgS_Z3i-Ob6}|Hzxo=X^p!)7{1lsu zP1`6XyNKV;rkh@}eak@!nLi7T4>v<_(N+p6m}v;3UT?^u|4~;T$jZrXaXl(SAxBKe z;Hfr=x!QV{y(#1?SI)v!8`@9sv%N_=_D3mz?-+@}rmjwYhfCS&Z|HfX0d$L2WZ7zqcLWMs?~nVL`lBa=2wQNG|xs;MUWjj;c{rjILz(cPDb$0 zd%mXjsei1st~knzTKAR332}8M%t31g3aX6WcYrDJ2(!Q#)P2C?5ovBDPXB`p?dnjm zu3o+2K9I8)x135>7H8iQNUhJG`mOUms+Of5l?v4D#x4hbA&jVz7*yRMOR0=YJ;~UQ zb{Xl)H%0$0d~xyphf?)VXQ2V%uoKi z%p;ai^~Fy2lVgX5rp9+6v+P|8I!v1Vg!*<;b<6C%%3#yiR*%gw2EB%$1Ao&HSnS}< zl1G2llhn-P;1wv{uZpSDXGKn=9)u|KUwml+Try;AHV90So%Z{wotiZcP@^PcPs~f+ z@eK%$R5m85Ya^PMZq(=4Q5zSrlAe4NFO3BOde!ZUOGna$*t_}gr}UnjK#~r@E-XRk zDG*JpPee(Aog4gUk&6|J38_kO;I8_!-VOx{ zT_Htd2zF-)p?=|(8ND*Z*Vk2^$0h8#=E3i@a>d%RGF(#|8TNy#iVsgVaS0AH?6+l* z0FlF+Q5xsb1iHNl5^U%rOVT@ZD1GdfO;zTY*|v;+E&6xSk9-+Em4IF)?X&Kh;$1&v zNFhshCJOy;^y-a6s+91ed)efu+sDkqJ&n>}6X%aWD?VN@Vu^hoVd&L&6av7+-C#OwWf$?>%^Hq*QR)gN732;(d>Wp9lj6nRjF7ue03q3 z7gH_CpCq4sgdPMNwFsOq5hzEtz~g@IevMAT$Y}teb-Zj(mAK=b&!t9Of)Vw$RgYFh z%O1WfQSyY0ll3+iP?6HYWJw(wn655P)9JRYi?Se6?sfrH;&NwLJ+^wHEvKPfg`;`v z(LEHhVhdjwbeckNV!*V&X*pOSWY)TTkq5O-_P?(7u{AIyZ9Qe2S-AHPTvf=LoNU$p z+rBy&o^V%tj0xiQSwvLhqbfCOcml)iard%ux+zo4DFRjb(X4waV4Qe>V0ZPnVFaom z8MnrYth%nXi+IEG(?O~}6v*nB`CfV(L7w)#`g4_6&K$xL(pLyOJ^U*U36<(-7Szt- z3e|~-+LFNkyMU--chmc&;DXzwZ546-A2v@U_Q(U*G2=qtTD71gNOmoDr-r%v??h#q zw(MA#(R;WZ^GD1gyqB#Jajfe{I{goUwX7_C1FbOA8hih`({NpHa^I-6V8QT#{c*wq zm&|rJk4r?A6mTh-^Py_ap1(_CLZkMRNWPmfy^5#Q7t5t>va=p}c;w?QX1gM+4TlYi z2+VJeq2)X|WNyUT=ug_)%TraJ&ox&bqocx)o26em!vX`MEK^=On|%8#^iZ$TsC6IMu#J;8r)rd>AYvJ61V|&d;}UWrUi2E<^KVkR zZ*L;J+M;P`1#OT3n5F&d9;kc7H*IaaCvNW{>V=X&A{K?a3PBrP({({`xh8xJc=1mN zr~Izt2I!#(uCgY5jrHxr4G^HM2+RWsr>&JFl&*OQ;V;%cDGi*g*!y`urc7f84v#<= zIMsinNyTYgMF|d)HZ1Y=fTjZRHBad*F9k$Q8H~TJXJiVjH)uhgzIsAO5;h9B?s41NeHidQWIs8udU|`boV#7Vpgwe1I8z{{|0zHe^rhKHU-+_WT#>kc*w0mc|23dBj z37}W~t?G;3PiFwCCf&amy4qHzVK6zjx*ls5qjT<5K(wb-LpwCQ#q;Ht%%f@G0F=dz zOQ=$22j>siTP8LejJ>n5yf~sr%$`<>FK9@7iMKvyi{`?W<7169l`SJ;+Yn!nf_^3wfuj>*)@ zEaAt=c6wnrkq^rc?`@*Y4)7pjS1SP=#5;WM=$Kheqs|VTFK}3~iuZLQ2GK<>-BN=i z8e7|6IX?#LJUrC&omCH11(8Hk1|^d*1L4mOs1C?I+Wt zKBv$mkR&^n91FKODT*m4qEyJf%l?EKTiinz-shw*fd@BZzqTD`N226%KWs%=`*epzPx);cPBZk1u79Rqi zcw9RFxo)he+MB$OqKa$kDrqcoUbv!ie7`wtAP1=Ot5aI(a3?#Xnz@B?qmma)uutI7 zR-Q4&ZrU>*4sfKpK2&1&dh-oIH!3$;H2_|SX1OLDdd~i;0;&5Ik z7jk6TWd}|+`W&UiZDC(+9{xq-s6ePGARo>lj*k`y41TmK5<73Z^~RO*Y$%o^9P@VI zeLEk>KA#fduRix6#OHk2MBDlp#(N6|>O9KI_VfggR}K=yj-&?xI@UjkXF)%RjH4Ix ze|V)?K+eT4*2_XI5u?-8i(N{6QXj$EpD;LFl=!4EVtpCz1QlOtEJUL$YGBYj3{2(R|kWTSFT&PlDNK7*-A&#lt1M8;n|Q#j-Sr+QEAk~Ctl+E zXJbnWtk~BDUIn@ok@x;)&+WKSZ3YGA*@F<6*=%USQE~@{G@cX)HjVhQO^%rR)#5u0 zz<7OnBUJ`7^gBw;BqR~qqvK7U#s^5e0rw6kW8>bWGy#|MeD#6@mkfN532ENs|Q?{q`4Y!RaHsGxyq)8{sSO%|J@P9BOOM{sZodBNgr`B7c@I43eVq?Z2V7?p zS=-EWHc{5s!0G8(jdXn zP0W0P_G|aAoS$%a@OB4e+8K!Tjp=Woyt!4k+(sA%7i1`*7hs}5a-)f-iU#)7H!0m* z;e-B1A~J;?VFt)}TqKRKqi7p$GPKHOyh(Q`DB*#E)lTB;sIp7JrvENJr{zcweSf^F zcs9=-jD5>m5$LPQb@=)Ujs)DuquNM5>-=$5OrzItXxpRDb(vSa;!K?-``cAQ-*jo~ z6~5HdJvB`9!$MOWhFH;0Z@i2X&PxH)N^vO_?XOIi5AN0w@UR)|wC$P_qepEU)lCGRFpAYXzoDZX360;iz=A2PWis|qQ#rxz zSar*2Q_jP4NQC|rCn5`-Vw^{%%~;9xd&zAP_H)d85yns8Z;HmyEpx}=Q7xf>WwjQp zIGPsqh0SC&1SYGf0r=@sS=aj3T)ZaQ-i)gKQK|y_$uQ?52yI?D8aNq27@@tjm%M{- zxR9RETCgU9_h5i#+yS1c5on1M(i?rw@-Z}RoFsDF+v8`@iXj7D#;m5Q?$!Y_=*+k| z^aGdd!z&7T-fH?ir9e5dIKTe=1%DRw0kKP6mF=$c$|SireBb;kz1}S-qE$g846Q4U zk`Y~^1ao|+g7a2=n;#18Y&14X7KIhjmcgkK*z7LCP~`m{kZ~NET{`w>^d5F80tKzP z8>A$yra%W#Fi;&K>hdydIz{S;PZA#sYd9Ejd%U^{TRrkGbaZbn9w0F+HFUSew8nXh z>*vEv9FbU0qUwNt3%C4s^#sF({3)hlnve^#$vC#Y-LbKkc@MAjG5=oG zP5YcqK5IXgYtu`ev0HPGtZ418e+0vCjkM3NuLIID^=5YjUj|f<((!PFPR3WhM>+JhCZSY7*G}9<@C)~E z^oQCcyhnL8B%yzB#Ohx~Q>$#O+M8fgmEZ(bpFJ~r^|?ys;mgYxFb1oYa5#Q2gAD-P z<3jp5{ynKCg&-r-wYfHCTGT1NjM#yeu&2-P*h1;^=DdPwg&qa`6X|Q!I|XyqF|)5wttM|`?ob=S z7=Wz4t}TK%?oR0>UX0q3C{F9!ea!+$LA4w@gyMXq;^;4d$!b?+T3lZ53m5X)vk%e=-+$>_mw&4i!7XY(iKEjTyz6sbJW6XLL=RtV$g$Je z0$O=)&OwZUTk>uSIVteP`=lqYw-#iPXJ>K<>8)Gagi)dkbToBe>22K^|H-CYi>eQ< zs`1RXhevGtIB^$G#=JSxvuL6Bc7AYvEKw!1*)HSu!Z9teE35d^uT=K%E|%<+R`p{= zjy|}rm%1{_iR3K#0-H4fHgwGp(>+zU6l8tixt6fWtQB@=v{UXo%16Kyr^NTJF}>hV z;e8q2--abs+R&>;E_AQ%^~uTUi5T*O*5l@r5s6O<*Nk}?ySjgrKmsrck)ikv+kDiY zf(C>=&ex26B9?xgYyg{S#ctvSWj71lAnb+-cLec++i?ose5hCqe61;lgA*Yf88#g& zQC`|j?D!ZH3F%fm7)EH@>y|y`f~Sk2t%ZBM=JH?!3~9&G^fRivK_kq4Mua|J`__ct z958js*l$m#NOcN~(qm-ZRHr(cmx4rOsrMTu#z3mI&+TP)!B9jUAx}_X)@Cmpe0nf7 z?6YW7U_)e|2pmWq3bf&`=zzED<64!z9NP!nxJvLsLazSCOX6vGGgXcYz>jND$3ZCx zrQgjjXJ2s1Q2bBZ^ad|S!WsGxSkZ=^e{~4cs~+NJ;#x*Kz9v=(S>>}k8cyDeaB?Ty zP@BH!fCa@NH}MO~u(+3syBOcr?opn47V)A8mV;XtdR{MwGyAG;c4G_oQ?kiJn>O== z33sWz(ZFfHe`+LS;1v-67;N~k1z>E?Of$tqp$5|2DdT(T2sFxBbuSjZC-!~L3k`ez zw91LMT9shk+WZ?jp|*+FIEGlf?gM6!z08mu&Y1oWz+7x-zf}`^^>HnzZyXe=VZkVc zY)g69mNQmVo=r||y#w&Lavm4!)@e(ST%DZpR6ZDPH;BT=;I*g;cvd=xC^AH^l46Su z?M46LS=F$F^fPQc>wq?r#Io2=lH1;DkUMBQ6%N-?EBh?s>EqDti*;pcjEQjeib@kB z$(t#w+3aTQ1&<*=68uX<2HZVE=yQp+|8pwo#+e>xT>bqc{Quyywmbss6L+Ez-YI60 z2i4rOXqD z-$fQ5+H<|6X(!j}{5Qf7_Us9EntJ!m<9{VCaaqnSGxZ`IGodYp()dh|S z_`x10T+uy~DC#<|B18GIs~OyOx1~8SotRioWX`7g=0@|}%^Mq&>w;;Q8b6P4BBjbA zJgCINahWbb`1jSf`-_pb<9^n&%e=DGJsJHgtDpTE6M(DaVWsZN4nsL*?W9exBJh z0hCr-Hl_6k@M~<>g>X(8TNUCGT1`r65mXzvSwFoe0j65GC#Jo0(>dKm>`416Z)Cta z{prDdx$%lc%h&^Of~84txYZL>4?z2JG81momvJo#B}z-r&s&#Y|FRRZ+i|HJ7^P|# z>uG2awiKnd2hmTH)D1VBe6wf~zu$gvjnUEoAIhaOx~$p!a;0LOWW>J6HWV}vk9XO7 z-FzG(w&AXeH)faT+qtEzJ$RhJKiscz15sKQN$LEU>%MiM-#;{AVNMf@!2U=}RZO`q z>sQ_z?%{2e*Sf3>&A?E3Jx&B)%sClEr?H)wR51y>c{C>&AgkMQZjNj0m}0!YpxM+| zR_=C$jcE-yr4NN`$v$Y*KOsh-ym>iITN4tj(qZb1PvQ*v*6|V;LVM5jskd5?=!B7$ zq|2!L<1%&4!mo35%MQ^0?(-Y-pn8`Zt2SY-;m3t=sg9vOVf4l8FL^Njnt{!ogC+}MKbtKC_$mqNa8b9b^N zf1XDQSNqTz+tA{gUwA{uZYzh5JshM~pK_ddb+ky~!j1BTx?t0r?a~60;+7#DU*Df+*;efSZwotiwkoeg$vK9) zMAnu=*FcMzSgwidpR7Oe=*)DYe5-b2(i1VMd$L5i$<-4FaTiww?;OODi}5Gw;!9#Q zO5h9^y8w36=g{(*n!54<@4+FbN(tHg-P&^r_Wp&N>G)|5+e?wLQ(wSGrE#1L$LegAHE=QC$F(FhH&it%a!lr`sKu#GqpbzU<#JC z*o6*4x`st~*rm`@h1+&G+urJd9^2r)fb&~VPi<8KQHoFa^A3C{&GGmPn6|qgt~#4@ z#{big66AwftqYQ&KH~?sJV52v#m=A7TpYu^!(TKTK7`7&`jD(2UHl8yS&*v=pMFd1 zU*uLK*iC+c>(w?ktkvcurq!!K2=6fpRhX6iG5>I|m1j(thbA-t@US`$zw7uhda1=V zAb~<9L5t|@0?eJKan{~>x7{&p*Dnp;=0~XlH~lnVh8K|&`Mk$J-2dwkNH@jUqWYZs zLvBrM5uB|jM!3>on_e?JKVIT)KVHOUwaRrJW3yUSDcUv#YJW(XPtI@U5YRW-ad1L^ zOr(NfDTDDcuuK>3eGFIMvQ}v4bdKbDeoF68h3--Hh2YVQ8(H>p zYIYVm`G_2aoblm|dSK|{3w4b2r!2zm%xT^3_OVIz!S4`dSi>t?8=Ye&;-)^d<*4{& z?ajdb62kqQ8&)s#ut=x3oT>SK)~D&1QdX2DcvdeC)bqoH-`RbiwH^*EV%vG84@1Yi zk?mnnlK#>EGvO%bs|z@8wI%Z%Q&3CxYP-FGG;XM&`-t#(qTSg;=YiBJfujAsuF9gg zi?usdsd=r!cl_;#-F`*JN=912LMVx)hC`(-V}>jNgj#3g`;%n4rXJ9*ptJ3Jfhh1%roTI5;pdYFx-_19u?QUGLSF&vYif z=UZOWYF-fpNJ<8pTO4*-> zfG_()UL>V|vh<0MhGvWl(OU3uKuk98B9uSX*AMqU7XATf4XKj?U`m6_M3Y=Op?*{5 z)?E39OZ#e@ZCJ)=Ekzhb2jN;X2h;o5uvTAywHC%L z1evzO@a}|>i`lr~xmxBqU*m8VnQwUBMSLxZZ7}_f4!hMi z&>lbuFtDt%pVIbA283WU10j4W1ZTon;B=c(0^u)U6&|Xz@eAlM<=I0lOT%ouWGCwY zQ7&2<`b}p{zy#}D4_6sNetiU{16(M(nWD%J$=K}ARmR4-MGq7A&UQ`FXp8U`RP-P1 ztE&3eAw^VSr|kG#a3PWeyFs*{oP^K5&w*Kh{KeM>&C%sZV^4|w&TNn29W`93W#Nu1 z+0FG}X;YErl1HVC64PWb0g0O^h37v(Y+`n+EEw0bUP$JGyUcO|%yx$WB_TzruP7Gd1$s%VD z_xC#Ee528fxssm48+9=D;`!5G1BN9PA3@2`pASm7CU5xw<*1I1N(`c|mQI-sIN2Jp z*4U;Twy#Y{e-$3ilBSDuBwF_TsY2!&p_?#0ow(aSHBa~;3tkn+Wi7>U_nUr4Tk=fZ zM{w!6r6O|So>Tyk4*s6ED4ssK=*K1!p*D-D6GNJ58BQZie7X zu=fZRX4Sz7fuY&%`~ugJP<~*a7VF2KMM7=hAurM7Va52c696AN;3kUc#l$}Id|mz6 z)L&n~h=u%GxDsw+?Y1I_kiX>XXX1bUgS77vz#R%?>0e ziZ`^Jyy_y#952OfZaz{=Lq%L_jTmZr`0lXRCa~70wSCRu_K^0$ zHAD=Z<3jj)Hw6nzw#cCz?EGCMbUoV|OQE509IghAzY_aB0NP48G7p}l3RRzaEYAt{ zkfd4xLspE=nHT0mKgd(zx z9BORCg8D5%0{IgwRFFR=k{?6eM(NYoq8691>~o3_-Eg@4GaM41SyWrg#O|QG$wb*Q zja{&$*5li-SEZ%NK%Oc6xytqZuE3>@)ITL*RqMaz7HTWkX9Y5A*MGCbGwc4$u#S&z ze4VEs{(O67y}I1~P3=0P+p^$qd3C|xwW+-7-p{vxwyx$0A3c0${w5aby}hyYE1~)i zv*2VX%V_bJY1N;^f|E%sBgK%vQ**~NZ+mPgoUHI?xOm*LE0~P6uluvQTz)5D-OxYT z{S$xfpQ#a6!R^1@d**NdSRENz_*}F8E3LY0Xt*|`?v3P~Phrcn(rvF#KBs%L=I5Jd z;Qy6I*JXaZqwF^QE9>ddov`(}AHuD3do*MGebw~(M==$32{{{^S*_K+b{oH>KmS=> z{Pi1+S#fu}^UHtj@?x##hP!*GMiQr3?8&;r0;7M2)+>iIMjs7L|NQpn*Y5-C4Zl3= z{?{LEm`Jz2wCD4m@7{O5{oH+G{@42QrSfuC>!?=d(2u{>BfX4uZ({Y(Lhrms%-_J@ ze|DdoZ&+E+=>7Xu(DN^!1N%Sx-qmzB_wnEUA~k=O1giu8X7U!be(v==Gb>dX2#>zb z)m43ZZTi;Vyxu?GFTJzs?AGIc_)csCQe&YO) zp{SemU$Iv|{;rfixplmEdUZjk;P3cyeQ)UJ*5|#GfzMNiKKlEe++eKEPyIe5{;%!7 zD;2GGx_;#T{b6R~WAuMf_2%JF{%_d-M^Pc!DTPvuE!t#8ma?RzQH>&NB}A4%_AMb> zsO%Y(Fp04n`@Rz@`_3R~Y-8+Wc`o1QIDXIZ`^Vv!Ke*?<-}igDuJe4IzEGkV!)>fX>%?sC#`c+XxopocrVMvg-CA_7 zX9@#>TAH@UIZzdl*!`7^vzG;xc;d+z31ci)~D@8G0#iur$A`HH*F4D%*c z-kqx`!EAlo=bhA@!HTT)yhfW<9HB7K@bRcXqriwQp}_vdX$>Yc}3FTj?4DdA`%u+ueDGD!q5fYgcg3WP4kGbEm3#aj(gC zs6rteJ5jT~vanDmy}J>hwYXJ>il;pfiJ&{MJN&3>Yh9~qqkLz zCvUp>kok$F5M5&t2LTR`t&(1M!@IWgl}R$5zbi`IJ-;cTvYb&b8U6s^i#J-$H}&p^bthl~Ufw zsvPN|Yo4V|NfkSD-bR%@@wSa;s}#oi9#Y~lt)8xFS&?q8p^YO$FK0W$H|kHG-<=st zTFfbX6z85k8lw1nVpR2pw^zMF`_5|LSf#t?x^s+dUxn4!mtAbt z)>??P;!fA>_69k-scQY2#olO9y7$(`%wC_g(TJnzpE2)^7WJg^jiQ@`qQ%NeTEWPK zj_uw~R_yTZ+@P4>?%{$&d7=47@?bgjiT3| zeeKl3R?tXYwu7=Ot0U8>v0aOh(QV4etVY@6t(w&b%Pgph*@b1Pq_Npq=^5!8HoR%% z!`HhscG?baF#@|n;7o}wxutS@A^{!#H+Xizrov)p32mGyIAOChm8Sl8XpqY=Tc5qwgDawhRC(U7wJBD?AZ>(0Wwj`Wt|=GUst?RLR2 zFN(x2J_rPVfc6F%i=kCbr zP?5|03Z}}_eQ|NIhsWKU_P~B`c9${MevfQ0GEMi)`@u}t;g`cZoL2IsRT4?dv(7tf z^X^!;18E^Mi{5!zMoHHh`|yUf)9L+QNejD-NO!j!UPCvx$81|#Yuk%ri2|7lnXKzd z4QvZZuV)XE7RP-0jaz5=kqSLD#JE9eRp0yHJM%+Ql8diH@rWl2Nl(4SQO2V_ycIIj zyJtlwmg%iT`w;g`%01`@#*}=zuDR^dXO?)-ryD8GVZEexHH|Ov{@anz6d6;B(QRGj z9(=Z79OJOL?D%SuThDt}eyFd-aP-;B!KObQMb4A_1ru(2v)fFxJ+D;?LU8BL*KqIK zF^KKW_NJ=Mr5WjtGmgwrd+V%KTRVMY+Z*n#`5p4{PMx#T7|k#56QW1A=MNUAd&Y&V zuEb1_j3=zc8~r$L1kZaTZj^s`)|j8*L7Srw1p+1qKC^>(4n%g!%aSM5wD z?b-}i_gbm^W2&MK3VQDRSvU6DBJEvJs*HcsW2tgv=i^?v+OlWSP8oZ2`PNC3-9@S~ zEqrfigR*DwYj7lkn*Ue!NMqrP(i<&)i6XCFI~9&N7k4^H4cb1*2;%qiHkk|Al^ycl zx-3gypj5TBA*-~vQDx$DMnv*LLHB$oO=icTN_cO3LTTq;pYDOny(E46vH0zL9r)|=SK93vlP`X{ z=+{E2VR@-FRyU~>R2HuHX?-^JefRIrf`xneN?up4uZ^{^ziNsKcrnY&vf;fuRO-&|E4cibQ+QLqB(_+qPZcQ=G6?72E0v$wTnsC7iz zVqizMBk_~Xosi>I+bio8qAPnX>y%ou6e{rwllS(@NUBzI!cDzG)n5W5Di#;YX*6fs zy_HRJHu%q}g6AL4`FA@Td+mJf3ZYSgR;7xR=1(N)z1ynVA#d_1Rqi!Crzwq&Erg$+ zEW0+AWy+>dt+3NoRkTiJ9m5rHj%`o0_c`09-T9Ho%zoZ>K7M}Z(BNp4?e0>kd6sn6 zVn?6%)>eF$7LdkWtEy+N31v4)*^#Ui+w@mKeG2WrV55KiTPBRV9 zC)3f(qz8Hmgesav)fkx+oz{b4uNv$+KFyoGH@SPwq4JeCEnKPWq@91S;15YicX&>$7+35wpgs zyyUjG&yCAN@f*`z*~2@3syyx8o@ILtJWHq48{wRPS)wHcVp`I9R{kw2ZBJ_{ov=0g z`)EY zYamyR+LW-W#8R`9oa#4u$L#M42B-RP?OIqB@#_VYcJ;(CEh((Q#w{OqoEz@+_ zijF^4WB>w`GcY(jx}%3=xrzquX-ZN=1054-tIE#k40_we)^J7doieSe9a21P#%XR>Gqag?Zw-SLeNJt?)|yqIunV z4k4hM^b1p+GuDe4+IuNCv^H!@ZqSL>`p>1CC}5%yKGds{)3Jd$x}-AP>&Y}yyRJZV z#3(zzLyqKbX=8=}cXhP-xkiV1?j z4x{9kPL-?n`r#`V3ycGHh@XBJ&`@(AMcsOl zSp0W*;8ULMVrsaSMPImd_~__jPe6F{;K?>j%kJT_3VnSt&1#WYF|2O2nEO_Vda= z?rn*sW_Hi#Yl@v7Dr)r#Os3sia1f+ffDxC8xu%n~a@^@NE7G~B-QAWEG&nZ$(dN`V5o=eh40zPGmeaM65%&`0EQCsTy zOPM`!O*YgsgAx4gXNjBL=C-;tm61QRABF2;9W$htyll3_&Ddnu@Pm_xJ=f){g5Lc@ z<@g7#ZKQUac&!jp`H&6OC3D1UXUCelkwx799#kJAeXWJ{A&)(K`DgL0U*>x|!X+yy z=iZvPl2k^lsPmZMg(ZO>CdADLR#`K*rYEkB?NVO#jz5iOZ5X5-8JCW0pm}}#O?1^M zd)7l_6;7I(qFi}VXuFP|jNO|Xs>rTPdtVik?Ai8D#dFwc@j()s>(wAd+x7HtLhdg` zx3bROdUfk{1^V-URGS3%A^}DuWAM`H2mas6vWU*$9=zaKvW4 zfSYJ+8?qLQTQ~OheyqUEZ7^WH-d{IFgpJ_>tG?aX{|1*_2LAXrH9A`I&Q~Ihx8j9h zDHCDdb+&w@%t&W0R9HmOwQSwBXrw86K+tA=;^L5JL6-CUl9Ql&f?e)NJchZ$*saW2 zAqr)^o|{Od5i3$CW0mtmG)bi$o%iFLgUrekw1G8Wg_nN?a_7~%Y#f}Xm27Fbu`jl7 zBPqq!Rq|Kc)_ds}-MQxiO6QBh*DHuST%@yk9-KR@;%%LsCG+ZvVPURi^MaN?!-<}5 z^=stvp*6~X<$G>r^ClKM7tL4qg7UYOlZBg z-`4FM+AG0S_*?FA2X9xXC`RKdW9F4oAM8vL_eQ&tI+WkjGRNY6F7K)*O$U(Bhc1Q> z?E1cp>0w2ZwRSyI+^lKqC5mP=S#Lu7>*Af5sxYBF=aJpZyqj%l&5L@sl$~^AZcS(3 z`M$2>9i4Rl#4oQ3opIVyU|LaEuAP1Qn!<%K-c7?f--$$1nsZtHHcxjE%lw9bfJ3zH zd^*7={ALl(nzLg*vv4wVFL`Ls6=5fxu}GcX`Y&cQ@;R+T&xYvjp4iZ5{k~&w zN1FZ6In2m9*P()h5e2EEOQSI!N_s1bM{g0WT_zO1|F=+(uOxDGX+GnzcTTWppZ=NA z%SVq4-7M#!FL=Eaz;&p*D(q(YBHy9$vunn--~VHGrtXp#^%!!rm=BJURg}iu3dM?2 zqOVSOrFT=(th7G*H%(1>vJ$26Oug)xv;M=`lf@c)leea+ZGv?gos=07NW7EM_uJ1iXeO;%!Z^sg(eizr4 z;Aj#9rJ}WGw1)L|a<(+aa&gpiYcpOJ5zGWJ!930)Q|*}QMO*E8g7Y$Rn+F&}?2 zshK)8BWS$za?vP-=UZBv%fvNj8ngF$a&e59QKQzFKUumX`i0`p{V7$6F$!DPR+mEYZ()W3Mfc?O+#jPpIJI7=^AEoxiz z2q|Ub;+K&};p_1CVyyle#xTxjaL23{lwbK}cz3{3hB8{QD)+^+BDpoVt$vG`oZ?=# zY@zHP)^~ecX3Jgrbiz*o!^XJjm>cVVOqmt2{#98=cn}|rMG6Eni_34%ny#Z$xo1n0 zD-=H^cqRNyG>XR5Y)=H!*1HE59=jsHRG5w}*tg%6kg9xrSYpmm&ABFxd*MmLQ3M|Q zeMVyA#6-8yKNj3a#;`}_yg19N1rTzxai1Q2LR!ba>lqN49(o8k7Aa3eiW`AH?jIYL z6US!kl_~mm6yuK>2h56?9}Dgg7}3`3O`TQ@+;T;b=N}!xZBP5Ja!*$pDActWNH=~; z4*d^sB-086>-f;HM*v<+vQz8NI*!#qSd;4Dj|d~m!fJUD6qOA1{ zt{0l^9_1eI0B!HSzQ>J|S4VTEB_6=*UW|UO3=mA!q#xWcD!5vC-n?2S^|R4KFb_wk-%B4FMaN__m8k; zx4uOUhU~ku_!Af{AO)+c|CUQ)2j6y24-4*H6PM~lf-1;c8r1r@PEK};~1JDD|Wd>;*mu3|3VL@OibuhR017>+c6dwEJOfuF!NI-33 z>diT?o~^^!OIb%fdOk|+yE%n_*Y&iyf?XsY@d z%JBwQgb?}u7(vW+34nl@C}@&DR=12a^>=ck99tQ-mo4;kIG&kQryDRr5I8dzi1LrP z#2vvYPk(eBh5IM|;iWN*Zz23g=9>x}sDZp79LaG^Ukxp`{MBzh*v0UJLd-&!qngZd zZf0CwO^VTYFiP|W;&_O?b#mTiq&!}Qm0^ZDnGM6;vRLV|LtTMqlCPXyvx)BZ$W z+=r%cAsgP^4^QkBrf-{8W~$KN%lY$4gM9S^rtC9_di%li7{A#Cib=K;1qacydcg*O zeQDx2h`vf1NU3g72W`+r(2q{CB*P=CqYMP%l5<_4lR5tp+H>iVcc&stveL2NT|k$4 z$s`h^Lo;|LfslC9Te*N>*8&Gr?isAOMMQ5DANxsY3);aXz~77Gb9*#Z z*%@nD18qa_+l{7`T*d`gw;+-lTPa?d#>(?Xw{%h&z=I&xLZ;+#iA^ z44ehZmT1O(aMVB4u?zd3Rg@&aoPaW4db2jfeb$~S5tAn>u@n(6gf9T0M!+JN(VumL zwI&mK(QJQv&=w=NfEwwH>V!_Dq?z$TvEY1hU9&|6X_@EBc+8vhiPsTjNh==OY+4(l zre&mMpO|QylbVB4g61KFqawM`6Y`bR^ved5UbinF=*--Fx|r2|ze@*-)1oE11Iz9T zH*vU5hH$6&vN%jRSPTd=`#Pe{fd3>mIC@{Nz(+@U#)UlL3&3KAEDaW9Lp5XH zYs|#D%2_bkq^P2^u>RK7Ekl0qp{ZCd3|4}<3g3vDt~Lpr+W>#(t!!+op+g}2-9*;E zK&C4!uJXfZ2445UG#qsE!8B)r%)7yG>iPH7f%a^$2JLyvv#Bd4{}>@@OIpV#V&W3@HAWv)E$WjOhrJ^E$6~QK4w0-k*&cO;6U5O(7+3u zciVd(m9R#8L zEyi~$4+}LIc?Rb{^lr93JD84Yc%A;`FHSOJ1PCR$b6G+^0%wAO0mBs%Sska65HTWN zKcuDghI1bQA!Rk*o%I#6VNLQ<DE6ZUjUSvkguiHAjXrY1m?2 zMQaP5YTtV!y>@e^``6^~Q@y_M;uFRyXN-!#hJb{QCy)Ooy#6Ix*y>p;t#vGP#3zT& z;Lv|->TRoAMNeUgN|y&c}6m5$88hcdNcVi%*6O&7G{HZ!&>A zCOo;qu^$VX*?$2VjK$mV3IR0(%tKkI)R+c- zH@n%meNpg*fYFk}#xXE|iXqYXo3!8{NrO}SE9fV7)bz{X(nyYazx`P&@`R+2u&BUf z@)_ZrxTIOOr)v`;Im$wac*@St(b9JWlL-on|X{p13+`FK|qzBILiPY z8}PqzUX%RR5p+@9!jM`0m=^it@z;{zH@Thz9tZ`gKxaB(qCOau#xD9}_^r?KqX8X- z>*CCcem`8DF6F~T21s;~U-S5i0>c6k_z^}QHcPRR*22Eq!?E^78XA-=Ap*Da_!W?B%nIJ?W_iNswv$xz!ZyQqN(uM zKyxUjgq)BAZxE_mR17BszdTmWtrmNyfdG%pR06m08wNDXVVquVp`9>N>yCG^GGJHn4)%vkGx$)sx157ND=hfjPS#T3q zV^o0F4R&hV1Qo73dO9Q4Zm7B-h5w5eOO$@$A5q8sTfgW9(+3dW*0$sWZAR>~w}WIai`R=zxB;^+u*MKIS%lg#C?_a)W2BK zdMdF4gQwLc7gfP(JoK>$$E zMO+QVk*rNxW~C3{@7@3X8O!vy<}+5K$UNdQOk}%?T6*SXk6gR-V3EN(&a)Q>ZC$2`qN=sl{dm%m)aN_!WjB;Nltem_KJxEjo)HP=HzVl6SO`+%^sDItN0I z;0zg$0qU|xS{>BS8F3>2G_aDj@}Z||K~}Fh02JvgD@GU-P5^a|hLiFEn!$Gox=Ovf z@(;8IuD>?CffUGtw zFP}nhHxYIrgqgml5)O7 zYXwR)BxQh@^pY;e!-NRHH5~DpIb(F{WM6RP>q%Ar2-9ge8}hUE{P5%(26p}@L)(PV zVZ=8up8eoS!2FpxBZnx9;E?R#UM=XbhHNk6kZLg_i6jGuOA0|Fe{sW)G~V;dy@IKO zk_5x$*-#*085-{Z%%j=~Ku^w&UyN(SkeUJr(WmqWA7XOE-jb`;B+L$AITaHqWvglD%dr0p4@;e1H+Fi1!0}p0G4pQR}4oT zxpqj$#sSLW{M%|R*b#*vz?VAKJ})`~o~;Zzx8jp)PKRPTI{jF zLow{2ZklQdUo=+lIEb?k^btT)ukfY*5-2md&>w`O58%sR8lUK*V?&QxiR>!odI}#} zP>%o>Z!N?>z~#G5E&`!X)I*RDWeq}fN*rusCG^3e#wTpOJyOD<+a@kzZCtaQbrT&z z1R}k*OF76LHArF?KK14v1!o@cKVWy2l@p{~e!3EWq>aR;>Q zBX@xg6ou)TL8;k+~`8JDd6a?|*STZygUH*8CnI&x>pP)A$_YQMMpZ-GZ`~w3wJ@ z!CA@vR9ku1B{ySx)}I<|r^m~LFaUA6|K`)U-<8ke3$oO{#y5-6%e!(Rel&QPT&3lSiOJl#gleR-guFByHuET(_a}#@TfVd4x+?;C! zYKTV1YYedoz))Qf4iasWU(#L1E-QoQ$|1af7j-81W3RANE*CPlsO%q3UVrd5Ex68H zj=TQ8tqd7=3;<*DTH2cD2dnzCK;`^W8Gy}q4Wjt4!C{79Aw_lmY5VMz)2}dW#!!94 z*LDG~+e1zfA|2$*?53t`ZUvq|?>^OAfAVGoGyZ%-5|EEhJWL41P_X;(*fZf;oxX`^ z`S+|vH9W=@WOu_uBinp4(>n*M-X=bKI9zJpFZ2}7Gt5bI+}4DJ?HYSp#vGqa&;?2l zI$CUKEehCG#Jw|asx!}Zf$07G+@Ppu)&>+ve(~ZzaJKz-&RzqolYmmJ7Nbr*g(wWA z*4GhLVf>Hx-AF9IcKbVEI6Yms@X80lh-I;Tw=uVKNF4sG17*ugTpemzP==Wv>G5dpggi^kX*6W`R0{mZtNdl(K zfNH1LH(qe>GS1O(8TQb5H;59PP~e<=3n`%|-hUULf&n*QRLs(2i0>(8m}fW5_8 zZAM%7;dwI$*aS|2x71Tp=*QhDu6?_xUEJfBXfxTOj)y#wx7ve*i@_y*=`O$$eMQ9R zB)+^=mem)6Ri;>yC3;;JE+ zg?W&NJY4!Z>U6c#cPVM0{J zS6(XZhye^#n-l}3*g^Q)@T6=_P$v~$Ay+A=ZBp4Qt1W)j;Dt03Dal@FN%eckc`i~0 z6`)_mBB(lxz775UyCA@~TWh|QN)Ywg$5DkeexySz4mey#lO<{1tO=O&k{0)ncJv52 z;Ny6tdNzy>Cg7k-T=IH<-;>iMgAbO&n};)ewis(-p09*KBqZ;K;XYg z(6zDef00{YZ$nvN#OSFh^oNK6vHE}$znK6yj+9<6-upOpYS6fWBQ4?&ojMhM);Rn+bh(^jdI_Hb0O)9N zA2=#j6F^NnN>2Cm=SMN14Vb%|gG|dIuBFihOhN)_pL*5iUf^!31hvG6?mAuRZaFxc zbK1sT+e6pk->Rx#Ed&`t^&Ip9vL7P!BYy-$em=f{J1K|PrQC;C?)jl%B;n;NusLqu zoLNaJ=qSwQ5wMiNahwDU%;~&%@N(XzY_7=x|530PDcg zxd-)lVG6QlHC<%vYrU?38%Nz3dI+a@>oM7Kb4+jmXIl@!sGtawB`C}d53--dMMaFi z%c-wl^rqOQ)5KNI*zhRBBl=hJTnd{9aB~Aj4ok4X`=6hG8~Is;NAd`6A7FK@AI>_( z3w|Z+Pa6-jK-NsH2*LehM|#Mxt9SxL1Is`;OVUdlIX|ewv=4SCfju{J^M1h}03kkb zU>epHl0zis;6GqU9Jx?zRk<;w)8Fu5y#C)8xU*lS(@}PEt40izhaPD{npwOoNVP18 zg*rvlcxjvPZcJ+QaWF1Si##a2R~@|up#R8{{ahda5U9*xj~26|3P!V*^3Hrhg$a~=Vfp!ITykRAgcj#7hQaXFli4b zWjlJj3<9#m(H<=!${zY`T60q0_wnn4m9?GP%%r#O+wjh#EFv0>+c*tG*hP}n)ndVn zQ;dE2@l!;DctS12aO?*%y)e1Oh`SpI?~_%DI7GIE1lh+(+s?M;z~Cf)iv`}7zE0R# zL>I|95qAk_$N_iT(^Y|0j@4MW3V`>>t9VB(;2sd_jArWb&jIaeK>qNl>(NZ4%fBGV zQsRO<==@W_lzYTF``R}TxR1>GEiI5Ar>m?Q8GuC;QB-H>wGv3D2Tf{>%|-m{}I_G*W?Qa%|z2r zkXn?4j)FW6iDIs|-c zJ)L#b)a!It&GtaUA!M5?ZlLu^eqwhY1g?Y>k^FI*E ze*4Vg=}(B~bP2o`bOF$v9}e>~;w&e=cTX$Jh5nGwL^-mH#4Hqg`ulS}V9U0efXF(5 z-V3)Gb;}gSBYF6Q!2oI2aT{W^(6Pe#>prSzQ>a*--w)|)mA?seiuvW(f)kJJJoI$j zRGLmCr1b>buk%0D3qlU@c37>)p3CEbAsNbbn9nqR-< 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

    )ZGrY?HV{w`+t-+#I@FV2s$21e87(uYuT1; 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

    gNxdT(buqiI`-e~G1*`Vbl^smrh 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�vInP9}a%30=J=m+REnkPxbAq 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

    FOWnm{yaVwHWMNN@<2 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 literal 0 HcmV?d00001 diff --git a/tests/io/vasp/test_outputs.py b/tests/io/vasp/test_outputs.py index e459bec8919..8044a130e55 100644 --- a/tests/io/vasp/test_outputs.py +++ b/tests/io/vasp/test_outputs.py @@ -1327,6 +1327,22 @@ def test_onsite_density_matrix(self): outcar = Outcar(f"{VASP_OUT_DIR}/OUTCAR_merged_numbers2") assert "onsite_density_matrices" in outcar.as_dict() + def test_nbands(self): + # Test VASP 5.2.11 + nbands = Outcar(f"{VASP_OUT_DIR}/OUTCAR.gz").data["nbands"] + assert nbands == 33 + assert isinstance(nbands, int) + + # Test VASP 5.4.4 + assert Outcar(f"{VASP_OUT_DIR}/OUTCAR.LOPTICS.vasp544").data["nbands"] == 128 + + # Test VASP 6.3.0 + assert Outcar(f"{VASP_OUT_DIR}/OUTCAR_vasp_6.3.gz").data["nbands"] == 64 + + # Test NBANDS set by user but overridden by VASP + # VASP 6.3.2 + assert Outcar(f"{VASP_OUT_DIR}/OUTCAR.nbands_overridden.gz").data["nbands"] == 32 + def test_nplwvs(self): outcar = Outcar(f"{VASP_OUT_DIR}/OUTCAR.gz") assert outcar.data["nplwv"] == [[34560]] From 461b56c1afa6b5031c74a04cc07618f393ed692f Mon Sep 17 00:00:00 2001 From: "Haoyu (Daniel) YANG" Date: Wed, 11 Dec 2024 10:02:11 +0800 Subject: [PATCH 08/12] Fix VASP/ABINIT link in main `README.md` (#4201) * update link * add markdown check hook * try to migrate the lint workflow * Revert "add markdown check hook" This reverts commit 2786a73e5605ef04cb7837965bd0bfc4a2e9fc2a. * comment out ruff for now * try linkspector * fall back to markdown-link-check * revert md-link-check --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 9b48fb2154a..27056b715c8 100644 --- a/README.md +++ b/README.md @@ -21,7 +21,7 @@ Pymatgen (Python Materials Genomics) is a robust, open-source Python library for materials analysis. These are some of the main features: 1. Highly flexible classes for the representation of `Element`, `Site`, `Molecule` and `Structure` objects. -2. Extensive input/output support, including support for [VASP](https://cms.mpi.univie.ac.at/vasp), [ABINIT](https://abinit.org), [CIF](https://wikipedia.org/wiki/Crystallographic_Information_File), [Gaussian](https://gaussian.com), [XYZ](https://wikipedia.org/wiki/XYZ_file_format), and many other file formats. +2. Extensive input/output support, including support for [VASP](https://www.vasp.at/), [ABINIT](https://abinit.github.io/abinit_web/), [CIF](https://wikipedia.org/wiki/Crystallographic_Information_File), [Gaussian](https://gaussian.com), [XYZ](https://wikipedia.org/wiki/XYZ_file_format), and many other file formats. 3. Powerful analysis tools, including generation of phase diagrams, Pourbaix diagrams, diffusion analyses, reactions, etc. 4. Electronic structure analyses, such as density of states and band structure. 5. Integration with the [Materials Project] REST API. From ac663b425ad613f3dd31fce6d45a046cf7fc41e4 Mon Sep 17 00:00:00 2001 From: "Haoyu (Daniel) YANG" Date: Wed, 11 Dec 2024 10:03:13 +0800 Subject: [PATCH 09/12] Fix NumPy array indexing in 32-bit system in `analysis.chemenv.coordination_environments` (#4194) * fix index in 32bit system * pre-commit auto-fixes --------- Co-authored-by: Haoyu Yang Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> --- .../coordination_geometry_finder.py | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/src/pymatgen/analysis/chemenv/coordination_environments/coordination_geometry_finder.py b/src/pymatgen/analysis/chemenv/coordination_environments/coordination_geometry_finder.py index 562b7b8c987..86c5d0f118a 100644 --- a/src/pymatgen/analysis/chemenv/coordination_environments/coordination_geometry_finder.py +++ b/src/pymatgen/analysis/chemenv/coordination_environments/coordination_geometry_finder.py @@ -218,7 +218,9 @@ def points_wcs_csc(self, permutation=None): """ if permutation is None: return self._points_wcs_csc - return np.concatenate((self._points_wcs_csc[:1], self._points_wocs_csc.take(permutation, axis=0))) + return np.concatenate( + (self._points_wcs_csc[:1], self._points_wocs_csc.take(np.array(permutation, dtype=np.intp), axis=0)) + ) def points_wocs_csc(self, permutation=None): """ @@ -227,7 +229,7 @@ def points_wocs_csc(self, permutation=None): """ if permutation is None: return self._points_wocs_csc - return self._points_wocs_csc.take(permutation, axis=0) + return self._points_wocs_csc.take(np.array(permutation, dtype=np.intp), axis=0) def points_wcs_ctwcc(self, permutation=None): """ @@ -239,7 +241,7 @@ def points_wcs_ctwcc(self, permutation=None): return np.concatenate( ( self._points_wcs_ctwcc[:1], - self._points_wocs_ctwcc.take(permutation, axis=0), + self._points_wocs_ctwcc.take(np.array(permutation, dtype=np.intp), axis=0), ) ) @@ -250,7 +252,7 @@ def points_wocs_ctwcc(self, permutation=None): """ if permutation is None: return self._points_wocs_ctwcc - return self._points_wocs_ctwcc.take(permutation, axis=0) + return self._points_wocs_ctwcc.take(np.array(permutation, dtype=np.intp), axis=0) def points_wcs_ctwocc(self, permutation=None): """ @@ -262,7 +264,7 @@ def points_wcs_ctwocc(self, permutation=None): return np.concatenate( ( self._points_wcs_ctwocc[:1], - self._points_wocs_ctwocc.take(permutation, axis=0), + self._points_wocs_ctwocc.take(np.array(permutation, dtype=np.intp), axis=0), ) ) @@ -273,7 +275,7 @@ def points_wocs_ctwocc(self, permutation=None): """ if permutation is None: return self._points_wocs_ctwocc - return self._points_wocs_ctwocc.take(permutation, axis=0) + return self._points_wocs_ctwocc.take(np.array(permutation, dtype=np.intp), axis=0) @property def cn(self): @@ -1976,6 +1978,7 @@ def _cg_csm_separation_plane_optim2( stop_search = False # TODO: do not do that several times ... also keep in memory if sepplane.ordered_plane: + separation_indices = [arr.astype(np.intp) for arr in separation_indices] inp = self.local_geometry.coords.take(separation_indices[1], axis=0) if sepplane.ordered_point_groups[0]: pp_s0 = self.local_geometry.coords.take(separation_indices[0], axis=0) From 9ff725dd287ff00cde87b64702ddc79caf5ddcf5 Mon Sep 17 00:00:00 2001 From: Shyue Ping Ong Date: Tue, 10 Dec 2024 18:21:22 -0800 Subject: [PATCH 10/12] =?UTF-8?q?Revert=20"`zopen`:=20explicit=20binary/te?= =?UTF-8?q?xt=20`mode`=20,=20and=20explicit=20`encoding`=20as=20UTF?= =?UTF-8?q?=E2=80=A6"=20(#4221)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This reverts commit b2a5e0ffb441a66cf67ce72fbd926a40ef0f6703. --- dev_scripts/potcar_scrambler.py | 2 +- src/pymatgen/apps/borg/hive.py | 2 +- src/pymatgen/apps/borg/queen.py | 4 +- src/pymatgen/core/structure.py | 18 ++++---- src/pymatgen/core/trajectory.py | 2 +- src/pymatgen/io/adf.py | 2 +- src/pymatgen/io/aims/inputs.py | 4 +- src/pymatgen/io/cif.py | 8 ++-- src/pymatgen/io/common.py | 6 +-- src/pymatgen/io/core.py | 6 +-- src/pymatgen/io/cp2k/inputs.py | 2 +- src/pymatgen/io/cp2k/outputs.py | 16 +++---- src/pymatgen/io/cp2k/utils.py | 2 +- src/pymatgen/io/cssr.py | 4 +- src/pymatgen/io/exciting/inputs.py | 2 +- src/pymatgen/io/feff/inputs.py | 20 ++++---- src/pymatgen/io/feff/outputs.py | 10 ++-- src/pymatgen/io/fiesta.py | 16 +++---- src/pymatgen/io/gaussian.py | 12 ++--- src/pymatgen/io/lammps/data.py | 4 +- src/pymatgen/io/lammps/generators.py | 2 +- src/pymatgen/io/lammps/inputs.py | 4 +- src/pymatgen/io/lammps/outputs.py | 4 +- src/pymatgen/io/lmto.py | 6 +-- src/pymatgen/io/lobster/inputs.py | 4 +- src/pymatgen/io/lobster/outputs.py | 30 ++++++------ src/pymatgen/io/nwchem.py | 6 +-- src/pymatgen/io/pwmat/inputs.py | 10 ++-- src/pymatgen/io/pwmat/outputs.py | 6 +-- src/pymatgen/io/pwscf.py | 2 +- src/pymatgen/io/qchem/inputs.py | 6 +-- src/pymatgen/io/qchem/outputs.py | 2 +- src/pymatgen/io/qchem/sets.py | 2 +- src/pymatgen/io/res.py | 4 +- src/pymatgen/io/template.py | 2 +- src/pymatgen/io/vasp/inputs.py | 28 +++++------ src/pymatgen/io/vasp/outputs.py | 46 +++++++++---------- src/pymatgen/io/xr.py | 4 +- src/pymatgen/io/xyz.py | 4 +- src/pymatgen/io/zeopp.py | 4 +- .../transformations/site_transformations.py | 4 +- src/pymatgen/util/io_utils.py | 2 +- tests/electronic_structure/test_dos.py | 2 +- tests/io/aims/conftest.py | 2 +- tests/io/pwmat/test_inputs.py | 6 +-- tests/io/vasp/test_inputs.py | 2 +- tests/io/vasp/test_outputs.py | 2 +- 47 files changed, 170 insertions(+), 168 deletions(-) diff --git a/dev_scripts/potcar_scrambler.py b/dev_scripts/potcar_scrambler.py index 4a5bb292a82..23cd1403eb9 100644 --- a/dev_scripts/potcar_scrambler.py +++ b/dev_scripts/potcar_scrambler.py @@ -124,7 +124,7 @@ def scramble_single_potcar(self, potcar: PotcarSingle) -> str: return scrambled_potcar_str def to_file(self, filename: str) -> None: - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(self.scrambled_potcars_str) @classmethod diff --git a/src/pymatgen/apps/borg/hive.py b/src/pymatgen/apps/borg/hive.py index fbd2a15645e..7045438fb81 100644 --- a/src/pymatgen/apps/borg/hive.py +++ b/src/pymatgen/apps/borg/hive.py @@ -445,7 +445,7 @@ def _get_transformation_history(path: PathLike): """Check for a transformations.json* file and return the history.""" if trans_json := glob(f"{path!s}/transformations.json*"): try: - with zopen(trans_json[0], mode="rt", encoding="utf-8") as file: + with zopen(trans_json[0]) as file: return json.load(file)["history"] except Exception: return None diff --git a/src/pymatgen/apps/borg/queen.py b/src/pymatgen/apps/borg/queen.py index bfa47b39432..2dd4d74e03a 100644 --- a/src/pymatgen/apps/borg/queen.py +++ b/src/pymatgen/apps/borg/queen.py @@ -103,12 +103,12 @@ def save_data(self, filename: PathLike) -> None: that if the filename ends with gz or bz2, the relevant gzip or bz2 compression will be applied. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: json.dump(list(self._data), file, cls=MontyEncoder) def load_data(self, filename: PathLike) -> None: """Load assimilated data from a file.""" - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: self._data = json.load(file, cls=MontyDecoder) diff --git a/src/pymatgen/core/structure.py b/src/pymatgen/core/structure.py index 6d0cd6545ae..63c1445fc05 100644 --- a/src/pymatgen/core/structure.py +++ b/src/pymatgen/core/structure.py @@ -2953,7 +2953,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: elif fmt == "json" or fnmatch(filename.lower(), "*.json*"): json_str = json.dumps(self.as_dict()) if filename: - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(json_str) return json_str elif fmt == "xsf" or fnmatch(filename.lower(), "*.xsf*"): @@ -2961,7 +2961,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: res_str = XSF(self).to_str() if filename: - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt", encoding="utf8") as file: file.write(res_str) return res_str elif ( @@ -2987,7 +2987,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: yaml.dump(self.as_dict(), str_io) yaml_str = str_io.getvalue() if filename: - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(yaml_str) return yaml_str elif fmt == "aims" or fnmatch(filename, "geometry.in"): @@ -2995,7 +2995,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: geom_in = AimsGeometryIn.from_structure(self) if filename: - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="w") as file: file.write(geom_in.get_header(filename)) file.write(geom_in.content) file.write("\n") @@ -3010,7 +3010,7 @@ def to(self, filename: PathLike = "", fmt: FileFormats = "", **kwargs) -> str: res_str = ResIO.structure_to_str(self) if filename: - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt", encoding="utf8") as file: file.write(res_str) return res_str elif fmt == "pwmat" or fnmatch(filename.lower(), "*.pwmat") or fnmatch(filename.lower(), "*.config"): @@ -3173,7 +3173,7 @@ def from_file( return struct fname = os.path.basename(filename) - with zopen(filename, mode="rt", errors="replace", encoding="utf-8") as file: + with zopen(filename, mode="rt", errors="replace") as file: contents = file.read() if fnmatch(fname.lower(), "*.cif*") or fnmatch(fname.lower(), "*.mcif*"): return cls.from_str( @@ -3919,7 +3919,7 @@ def to(self, filename: str = "", fmt: str = "") -> str | None: elif fmt == "json" or fnmatch(filename, "*.json*") or fnmatch(filename, "*.mson*"): json_str = json.dumps(self.as_dict()) if filename: - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt", encoding="utf8") as file: file.write(json_str) return json_str elif fmt in {"yaml", "yml"} or fnmatch(filename, "*.yaml*") or fnmatch(filename, "*.yml*"): @@ -3928,7 +3928,7 @@ def to(self, filename: str = "", fmt: str = "") -> str | None: yaml.dump(self.as_dict(), str_io) yaml_str = str_io.getvalue() if filename: - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt", encoding="utf8") as file: file.write(yaml_str) return yaml_str else: @@ -4010,7 +4010,7 @@ def from_file(cls, filename: PathLike) -> Self | None: """ filename = str(filename) - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename) as file: contents = file.read() fname = filename.lower() if fnmatch(fname, "*.xyz*"): diff --git a/src/pymatgen/core/trajectory.py b/src/pymatgen/core/trajectory.py index bc656711bb2..3e4d702998d 100644 --- a/src/pymatgen/core/trajectory.py +++ b/src/pymatgen/core/trajectory.py @@ -467,7 +467,7 @@ def write_Xdatcar( xdatcar_str = "\n".join(lines) + "\n" - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(xdatcar_str) def as_dict(self) -> dict: diff --git a/src/pymatgen/io/adf.py b/src/pymatgen/io/adf.py index 6725c89beb6..f5fbf52d9a2 100644 --- a/src/pymatgen/io/adf.py +++ b/src/pymatgen/io/adf.py @@ -645,7 +645,7 @@ def _parse_logfile(self, logfile): # The last non-empty line of the logfile must match the end pattern. # Otherwise the job has some internal failure. The TAPE13 part of the # ADF manual has a detailed explanation. - with zopen(logfile, mode="rt", encoding="utf-8") as file: + with zopen(logfile, mode="rt") as file: for line in reverse_readline(file): if line == "": continue diff --git a/src/pymatgen/io/aims/inputs.py b/src/pymatgen/io/aims/inputs.py index fd8860deb2b..8304a6eb0d5 100644 --- a/src/pymatgen/io/aims/inputs.py +++ b/src/pymatgen/io/aims/inputs.py @@ -133,7 +133,7 @@ def from_file(cls, filepath: str | Path) -> Self: Returns: AimsGeometryIn: The input object represented in the file """ - with zopen(filepath, mode="rt", encoding="utf-8") as in_file: + with zopen(filepath, mode="rt") as in_file: content = in_file.read() return cls.from_str(content) @@ -753,7 +753,7 @@ def from_file(cls, filename: str, label: str | None = None) -> Self: Returns: AimsSpeciesFile """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls(data=file.read(), label=label) @classmethod diff --git a/src/pymatgen/io/cif.py b/src/pymatgen/io/cif.py index 0bb6413402e..253f7d86372 100644 --- a/src/pymatgen/io/cif.py +++ b/src/pymatgen/io/cif.py @@ -299,7 +299,7 @@ def from_file(cls, filename: PathLike) -> Self: Returns: CifFile """ - with zopen(filename, mode="rt", errors="replace", encoding="utf-8") as file: + with zopen(filename, mode="rt", errors="replace") as file: return cls.from_str(file.read()) @@ -1760,9 +1760,9 @@ def cif_file(self) -> CifFile: def write_file( self, - filename: PathLike, - mode: Literal["wt", "at"] = "wt", + filename: str | Path, + mode: Literal["w", "a", "wt", "at"] = "w", ) -> None: """Write the CIF file.""" - with zopen(filename, mode=mode, encoding="utf-8") as file: + with zopen(filename, mode=mode) as file: file.write(str(self)) diff --git a/src/pymatgen/io/common.py b/src/pymatgen/io/common.py index 6a99f044b1a..abaf3c9e4c5 100644 --- a/src/pymatgen/io/common.py +++ b/src/pymatgen/io/common.py @@ -354,7 +354,7 @@ def to_cube(self, filename, comment: str = ""): filename (str): Name of the cube file to be written. comment (str): If provided, this will be added to the second comment line """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(f"# Cube file for {self.structure.formula} generated by Pymatgen\n") file.write(f"# {comment}\n") file.write(f"\t {len(self.structure)} 0.000000 0.000000 0.000000\n") @@ -386,7 +386,7 @@ def from_cube(cls, filename: str | Path) -> Self: Args: filename (str): of the cube to read """ - file = zopen(filename, mode="rt", encoding="utf-8") + file = zopen(filename, mode="rt") # skip header lines file.readline() @@ -529,7 +529,7 @@ def __getitem__(self, item): f"No parser defined for {item}. Contents are returned as a string.", stacklevel=2, ) - with zopen(fpath, mode="rt", encoding="utf-8") as f: + with zopen(fpath, "rt") as f: return f.read() def get_files_by_name(self, name: str) -> dict[str, Any]: diff --git a/src/pymatgen/io/core.py b/src/pymatgen/io/core.py index e81cd7a026e..5484954afe1 100644 --- a/src/pymatgen/io/core.py +++ b/src/pymatgen/io/core.py @@ -74,7 +74,7 @@ def write_file(self, filename: PathLike) -> None: Args: filename: The filename to output to, including path. """ - with zopen(Path(filename), mode="wt", encoding="utf-8") as file: + with zopen(Path(filename), mode="wt") as file: file.write(self.get_str()) @classmethod @@ -102,7 +102,7 @@ def from_file(cls, path: PathLike) -> None: Returns: InputFile """ - with zopen(Path(path), mode="rt", encoding="utf-8") as file: + with zopen(Path(path), mode="rt") as file: return cls.from_str(file.read()) # from_str not implemented @@ -218,7 +218,7 @@ def write_input( if isinstance(contents, InputFile): contents.write_file(file_path) else: - with zopen(file_path, mode="wt", encoding="utf-8") as file: + with zopen(file_path, mode="wt") as file: file.write(str(contents)) if zip_inputs: diff --git a/src/pymatgen/io/cp2k/inputs.py b/src/pymatgen/io/cp2k/inputs.py index 488cd016771..aa68870f100 100644 --- a/src/pymatgen/io/cp2k/inputs.py +++ b/src/pymatgen/io/cp2k/inputs.py @@ -692,7 +692,7 @@ def _from_dict(cls, dct: dict): @classmethod def from_file(cls, filename: str | Path) -> Self: """Initialize from a file.""" - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: txt = preprocessor(file.read(), os.path.dirname(file.name)) return cls.from_str(txt) diff --git a/src/pymatgen/io/cp2k/outputs.py b/src/pymatgen/io/cp2k/outputs.py index 873301fe1df..a230d139baf 100644 --- a/src/pymatgen/io/cp2k/outputs.py +++ b/src/pymatgen/io/cp2k/outputs.py @@ -327,7 +327,7 @@ def parse_initial_structure(self): ) coord_table = [] - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="rt") as file: while True: line = file.readline() if re.search(r"Atom\s+Kind\s+Element\s+X\s+Y\s+Z\s+Z\(eff\)\s+Mass", line): @@ -789,7 +789,7 @@ def parse_atomic_kind_info(self): except (TypeError, IndexError, ValueError): atomic_kind_info[kind]["total_pseudopotential_energy"] = None - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="rt") as file: j = -1 lines = file.readlines() for k, line in enumerate(lines): @@ -1010,7 +1010,7 @@ def parse_mo_eigenvalues(self): eigenvalues = [] efermi = [] - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="rt") as file: lines = iter(file.readlines()) for line in lines: try: @@ -1349,7 +1349,7 @@ def parse_hyperfine(self, hyperfine_filename=None): else: return None - with zopen(hyperfine_filename, mode="rt", encoding="utf-8") as file: + with zopen(hyperfine_filename, mode="rt") as file: lines = [line for line in file.read().split("\n") if line] hyperfine = [[] for _ in self.ionic_steps] @@ -1370,7 +1370,7 @@ def parse_gtensor(self, gtensor_filename=None): else: return None - with zopen(gtensor_filename, mode="rt", encoding="utf-8") as file: + with zopen(gtensor_filename, mode="rt") as file: lines = [line for line in file.read().split("\n") if line] data = {} @@ -1407,7 +1407,7 @@ def parse_chi_tensor(self, chi_filename=None): else: return None - with zopen(chi_filename, mode="rt", encoding="utf-8") as file: + with zopen(chi_filename, mode="rt") as file: lines = [line for line in file.read().split("\n") if line] data = {k: [] for k in "chi_soft chi_local chi_total chi_total_ppm_cgs PV1 PV2 PV3 ISO ANISO".split()} @@ -1554,7 +1554,7 @@ def read_table_pattern( row_pattern, or a dict in case that named capturing groups are defined by row_pattern. """ - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="rt") as file: if strip: lines = file.readlines() text = "".join( @@ -1691,7 +1691,7 @@ def parse_pdos(dos_file=None, spin_channel=None, total=False): """ spin = Spin(spin_channel) if spin_channel else Spin.down if "BETA" in os.path.split(dos_file)[-1] else Spin.up - with zopen(dos_file, mode="rt", encoding="utf-8") as file: + with zopen(dos_file, mode="rt") as file: lines = file.readlines() kind = re.search(r"atomic kind\s(.*)\sat iter", lines[0]) or re.search(r"list\s(\d+)\s(.*)\sat iter", lines[0]) kind = kind.groups()[0] diff --git a/src/pymatgen/io/cp2k/utils.py b/src/pymatgen/io/cp2k/utils.py index 9566ce45fbe..7eca9758a73 100644 --- a/src/pymatgen/io/cp2k/utils.py +++ b/src/pymatgen/io/cp2k/utils.py @@ -80,7 +80,7 @@ def preprocessor(data: str, dir: str = ".") -> str: # noqa: A002 raise ValueError(f"length of inc should be 2, got {len(inc)}") inc = inc[1].strip("'") inc = inc.strip('"') - with zopen(os.path.join(dir, inc), mode="rt", encoding="utf-8") as file: + with zopen(os.path.join(dir, inc)) as file: data = re.sub(rf"{incl}", file.read(), data) variable_sets = re.findall(r"(@SET.+)", data, re.IGNORECASE) for match in variable_sets: diff --git a/src/pymatgen/io/cssr.py b/src/pymatgen/io/cssr.py index 1068396e14b..c5a4fa4ab82 100644 --- a/src/pymatgen/io/cssr.py +++ b/src/pymatgen/io/cssr.py @@ -57,7 +57,7 @@ def write_file(self, filename): Args: filename (str): Filename to write to. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(str(self) + "\n") @classmethod @@ -98,5 +98,5 @@ def from_file(cls, filename: str | Path) -> Self: Returns: Cssr object. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls.from_str(file.read()) diff --git a/src/pymatgen/io/exciting/inputs.py b/src/pymatgen/io/exciting/inputs.py index 5213c8f622a..7ef99482b03 100644 --- a/src/pymatgen/io/exciting/inputs.py +++ b/src/pymatgen/io/exciting/inputs.py @@ -172,7 +172,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: ExcitingInput """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: data = file.read().replace("\n", "") return cls.from_str(data) diff --git a/src/pymatgen/io/feff/inputs.py b/src/pymatgen/io/feff/inputs.py index 94ad04dd24e..d9365c25499 100644 --- a/src/pymatgen/io/feff/inputs.py +++ b/src/pymatgen/io/feff/inputs.py @@ -246,7 +246,7 @@ def header_string_from_file(filename: str = "feff.inp"): Returns: Reads header string. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="r") as file: lines = file.readlines() feff_header_str = [] ln = 0 @@ -434,8 +434,8 @@ def atoms_string_from_file(filename): Returns: Atoms string. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: - f = file.readlines() + with zopen(filename, mode="rt") as fobject: + f = fobject.readlines() coords = 0 atoms_str = [] @@ -527,7 +527,7 @@ def write_file(self, filename="ATOMS"): Args: filename: path for file to be written """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(f"{self}\n") @@ -654,7 +654,7 @@ def write_file(self, filename="PARAMETERS"): Args: filename: filename and path to write to. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(f"{self}\n") @classmethod @@ -668,7 +668,7 @@ def from_file(cls, filename: str = "feff.inp") -> Self: Returns: Tags """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = list(clean_lines(file.readlines())) params = {} eels_params = [] @@ -828,8 +828,8 @@ def pot_string_from_file(filename="feff.inp"): Returns: FEFFPOT string. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: - f = file.readlines() + with zopen(filename, mode="rt") as f_object: + f = f_object.readlines() ln = -1 pot_str = ["POTENTIALS\n"] pot_tag = -1 @@ -934,7 +934,7 @@ def write_file(self, filename="POTENTIALS"): Args: filename: filename and path to write potential file to. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(str(self) + "\n") @@ -976,7 +976,7 @@ def __str__(self): def write_file(self, filename="paths.dat"): """Write paths.dat.""" - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(str(self) + "\n") diff --git a/src/pymatgen/io/feff/outputs.py b/src/pymatgen/io/feff/outputs.py index 8357fdfd357..16a0de3567d 100644 --- a/src/pymatgen/io/feff/outputs.py +++ b/src/pymatgen/io/feff/outputs.py @@ -70,7 +70,7 @@ def from_file(cls, feff_inp_file: str = "feff.inp", ldos_file: str = "ldos") -> dos_index = 1 begin = 0 - with zopen(pot_inp, mode="rt", encoding="utf-8") as potfile: + with zopen(pot_inp, mode="r") as potfile: for line in potfile: if len(pot_read_end.findall(line)) > 0: break @@ -95,7 +95,7 @@ def from_file(cls, feff_inp_file: str = "feff.inp", ldos_file: str = "ldos") -> dicts = Potential.pot_dict_from_str(pot_string) pot_dict = dicts[0] - with zopen(f"{ldos_file}00.dat", mode="rt", encoding="utf-8") as file: + with zopen(f"{ldos_file}00.dat", mode="r") as file: lines = file.readlines() e_fermi = float(lines[0].split()[4]) @@ -172,7 +172,7 @@ def charge_transfer_from_file(feff_inp_file, ldos_file): pot_inp = re.sub(r"feff.inp", r"pot.inp", feff_inp_file) pot_readstart = re.compile(".*iz.*lmaxsc.*xnatph.*xion.*folp.*") pot_readend = re.compile(".*ExternalPot.*switch.*") - with zopen(pot_inp, mode="rt", encoding="utf-8") as potfile: + with zopen(pot_inp, mode="r") as potfile: for line in potfile: if len(pot_readend.findall(line)) > 0: break @@ -203,7 +203,7 @@ def charge_transfer_from_file(feff_inp_file, ldos_file): for idx in range(len(dicts[0]) + 1): if len(str(idx)) == 1: - with zopen(f"{ldos_file}0{idx}.dat", mode="rt", encoding="utf-8") as file: + with zopen(f"{ldos_file}0{idx}.dat", mode="rt") as file: lines = file.readlines() s = float(lines[3].split()[2]) p = float(lines[4].split()[2]) @@ -212,7 +212,7 @@ def charge_transfer_from_file(feff_inp_file, ldos_file): tot = float(lines[1].split()[4]) cht[str(idx)] = {pot_dict[idx]: {"s": s, "p": p, "d": d, "f": f1, "tot": tot}} else: - with zopen(f"{ldos_file}{idx}.dat", mode="rt", encoding="utf-8") as file: + with zopen(f"{ldos_file}{idx}.dat", mode="rt") as file: lines = file.readlines() s = float(lines[3].split()[2]) p = float(lines[4].split()[2]) diff --git a/src/pymatgen/io/fiesta.py b/src/pymatgen/io/fiesta.py index 5ed94ac3e69..df647a600fa 100644 --- a/src/pymatgen/io/fiesta.py +++ b/src/pymatgen/io/fiesta.py @@ -63,7 +63,7 @@ def run(self): init_folder = os.getcwd() os.chdir(self.folder) - with zopen(self.log_file, mode="wt", encoding="utf-8") as fout: + with zopen(self.log_file, mode="w") as fout: subprocess.call( [ self._NWCHEM2FIESTA_cmd, @@ -138,7 +138,7 @@ def _gw_run(self): if self.folder != init_folder: os.chdir(self.folder) - with zopen(self.log_file, mode="wt", encoding="utf-8") as fout: + with zopen(self.log_file, mode="w") as fout: subprocess.call( [ "mpirun", @@ -161,7 +161,7 @@ def bse_run(self): if self.folder != init_folder: os.chdir(self.folder) - with zopen(self.log_file, mode="wt", encoding="utf-8") as fout: + with zopen(self.log_file, mode="w") as fout: subprocess.call( [ "mpirun", @@ -214,7 +214,7 @@ def __init__(self, filename): """ self.filename = filename - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename) as file: basis_set = file.read() self.data = self._parse_file(basis_set) @@ -533,7 +533,7 @@ def write_file(self, filename: str | Path) -> None: Args: filename: Filename. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="w") as file: file.write(str(self)) def as_dict(self): @@ -712,7 +712,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: FiestaInput object """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename) as file: return cls.from_str(file.read()) @@ -730,7 +730,7 @@ def __init__(self, filename): """ self.filename = filename - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename) as file: data = file.read() chunks = re.split(r"GW Driver iteration", data) @@ -821,7 +821,7 @@ def __init__(self, filename): """ self.filename = filename - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename) as file: log_bse = file.read() # self.job_info = self._parse_preamble(preamble) diff --git a/src/pymatgen/io/gaussian.py b/src/pymatgen/io/gaussian.py index fd3ab518f4f..bfbd1938c7b 100644 --- a/src/pymatgen/io/gaussian.py +++ b/src/pymatgen/io/gaussian.py @@ -368,7 +368,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: GaussianInput object """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="r") as file: return cls.from_str(file.read()) def get_zmatrix(self): @@ -447,9 +447,9 @@ def para_dict_to_str(para, joiner=" "): def write_file(self, filename, cart_coords=False): """Write the input string into a file. - Option: see `__str__` method + Option: see __str__ method """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="w") as file: file.write(self.to_str(cart_coords)) def as_dict(self): @@ -661,7 +661,7 @@ def _parse(self, filename): opt_structures = [] route_lower = {} - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: for line in file: if parse_stage == 0: if start_patt.search(line): @@ -1109,7 +1109,7 @@ def read_scan(self): data = {"energies": [], "coords": {}} # read in file - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="r") as file: line = file.readline() while line != "": @@ -1195,7 +1195,7 @@ def read_excitation_energies(self): transitions = [] # read in file - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="r") as file: line = file.readline() td = False while line != "": diff --git a/src/pymatgen/io/lammps/data.py b/src/pymatgen/io/lammps/data.py index 2e730c218ca..d0d565fb9f9 100644 --- a/src/pymatgen/io/lammps/data.py +++ b/src/pymatgen/io/lammps/data.py @@ -647,7 +647,7 @@ def from_file(cls, filename: str, atom_style: str = "full", sort_id: bool = Fals sort_id (bool): Whether sort each section by id. Default to True. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.readlines() kw_pattern = r"|".join(itertools.chain(*SECTION_KEYWORDS.values())) section_marks = [idx for idx, line in enumerate(lines) if re.search(kw_pattern, line)] @@ -1439,7 +1439,7 @@ def parse_xyz(cls, filename: str | Path) -> pd.DataFrame: Returns: pandas.DataFrame """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.readlines() str_io = StringIO("".join(lines[2:])) # skip the 2nd line diff --git a/src/pymatgen/io/lammps/generators.py b/src/pymatgen/io/lammps/generators.py index d541dd1adf4..143860d8764 100644 --- a/src/pymatgen/io/lammps/generators.py +++ b/src/pymatgen/io/lammps/generators.py @@ -67,7 +67,7 @@ def get_input_set(self, structure: Structure | LammpsData | CombinedData) -> Lam data: LammpsData = LammpsData.from_structure(structure) if isinstance(structure, Structure) else structure # Load the template - with zopen(self.template, mode="rt", encoding="utf-8") as file: + with zopen(self.template, mode="r") as file: template_str = file.read() # Replace all variables diff --git a/src/pymatgen/io/lammps/inputs.py b/src/pymatgen/io/lammps/inputs.py index fa7987dfa3a..125261a903b 100644 --- a/src/pymatgen/io/lammps/inputs.py +++ b/src/pymatgen/io/lammps/inputs.py @@ -553,7 +553,7 @@ def write_file( If False, a single block is assumed. """ filename = filename if isinstance(filename, Path) else Path(filename) - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(self.get_str(ignore_comments=ignore_comments, keep_stages=keep_stages)) @classmethod @@ -653,7 +653,7 @@ def from_file(cls, path: str | Path, ignore_comments: bool = False, keep_stages: LammpsInputFile """ filename = path if isinstance(path, Path) else Path(path) - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls.from_str(file.read(), ignore_comments=ignore_comments, keep_stages=keep_stages) def __repr__(self) -> str: diff --git a/src/pymatgen/io/lammps/outputs.py b/src/pymatgen/io/lammps/outputs.py index 4efc01fc934..216cc458c66 100644 --- a/src/pymatgen/io/lammps/outputs.py +++ b/src/pymatgen/io/lammps/outputs.py @@ -115,7 +115,7 @@ def parse_lammps_dumps(file_pattern): files = sorted(files, key=lambda f: int(re.match(pattern, f)[1])) for filename in files: - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: dump_cache = [] for line in file: if line.startswith("ITEM: TIMESTEP"): @@ -144,7 +144,7 @@ def parse_lammps_log(filename: str = "log.lammps") -> list[pd.DataFrame]: Returns: [pd.DataFrame] containing thermo data for each completed run. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.readlines() begin_flag = ( "Memory usage per processor =", diff --git a/src/pymatgen/io/lmto.py b/src/pymatgen/io/lmto.py index a660ee12510..1a5d669d4bc 100644 --- a/src/pymatgen/io/lmto.py +++ b/src/pymatgen/io/lmto.py @@ -139,7 +139,7 @@ def write_file(self, filename="CTRL", **kwargs): """Write a CTRL file with structure, HEADER, and VERS that can be used as input for lmhart.run. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(self.get_str(**kwargs)) @classmethod @@ -153,7 +153,7 @@ def from_file(cls, filename: str | Path = "CTRL", **kwargs) -> Self: Returns: An LMTOCtrl object. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: contents = file.read() return cls.from_str(contents, **kwargs) @@ -322,7 +322,7 @@ def __init__(self, filename="COPL", to_eV=False): eV, set to True. Defaults to False for energies in Ry. """ # COPL files have an extra trailing blank line - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: contents = file.read().split("\n")[:-1] # The parameters line is the second line in a COPL file. It # contains all parameters that are needed to map the file. diff --git a/src/pymatgen/io/lobster/inputs.py b/src/pymatgen/io/lobster/inputs.py index c83b57cfe4d..18a6a53c654 100644 --- a/src/pymatgen/io/lobster/inputs.py +++ b/src/pymatgen/io/lobster/inputs.py @@ -588,7 +588,7 @@ def from_file(cls, lobsterin: PathLike) -> Self: Returns: Lobsterin object """ - with zopen(lobsterin, mode="rt", encoding="utf-8") as file: + with zopen(lobsterin, mode="rt") as file: lines = file.read().split("\n") if not lines: raise RuntimeError("lobsterin file contains no data.") @@ -645,7 +645,7 @@ def _get_potcar_symbols(POTCAR_input: PathLike) -> list[str]: raise ValueError("Lobster only works with PAW! Use different POTCARs") # Warning about a bug in LOBSTER-4.1.0 - with zopen(POTCAR_input, mode="rt", encoding="utf-8") as file: + with zopen(POTCAR_input, mode="r") as file: data = file.read() if isinstance(data, bytes): diff --git a/src/pymatgen/io/lobster/outputs.py b/src/pymatgen/io/lobster/outputs.py index 90291feec7d..a0ca239bfe4 100644 --- a/src/pymatgen/io/lobster/outputs.py +++ b/src/pymatgen/io/lobster/outputs.py @@ -121,7 +121,7 @@ def __init__( else: self._filename = "COHPCAR.lobster" - with zopen(self._filename, mode="rt", encoding="utf-8") as file: + with zopen(self._filename, mode="rt") as file: lines = file.read().split("\n") # The parameters line is the second line in a COHPCAR file. @@ -405,7 +405,7 @@ def __init__( # LOBSTER list files have an extra trailing blank line # and we don't need the header. if self._icohpcollection is None: - with zopen(self._filename, mode="rt", encoding="utf-8") as file: + with zopen(self._filename, mode="rt") as file: all_lines = file.read().split("\n") lines = all_lines[1:-1] if "spin" not in all_lines[1] else all_lines[2:-1] if len(lines) == 0: @@ -622,7 +622,7 @@ def __init__(self, filename: PathLike | None = "NcICOBILIST.lobster") -> None: # LOBSTER list files have an extra trailing blank line # and we don't need the header - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n")[1:-1] if len(lines) == 0: raise RuntimeError("NcICOBILIST file contains no data.") @@ -754,7 +754,7 @@ def _parse_doscar(self): tdensities = {} itdensities = {} - with zopen(doscar, mode="rt", encoding="utf-8") as file: + with zopen(doscar, mode="rt") as file: file.readline() # Skip the first line efermi = float([file.readline() for nn in range(4)][3].split()[17]) dos = [] @@ -913,7 +913,7 @@ def __init__( self.loewdin = [] if loewdin is None else loewdin if self.num_atoms is None: - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n")[3:-3] if len(lines) == 0: raise RuntimeError("CHARGES file contains no data.") @@ -1047,7 +1047,7 @@ def __init__(self, filename: PathLike | None, **kwargs) -> None: else: raise ValueError(f"{attr}={val} is not a valid attribute for Lobsterout") elif filename: - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n") if len(lines) == 0: raise RuntimeError("lobsterout does not contain any data") @@ -1445,7 +1445,7 @@ def __init__( raise ValueError("No FATBAND files in folder or given") for fname in filenames: - with zopen(fname, mode="rt", encoding="utf-8") as file: + with zopen(fname, mode="rt") as file: lines = file.read().split("\n") atom_names.append(os.path.split(fname)[1].split("_")[1].capitalize()) @@ -1479,7 +1479,7 @@ def __init__( eigenvals: dict = {} p_eigenvals: dict = {} for ifilename, filename in enumerate(filenames): - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n") if ifilename == 0: @@ -1627,7 +1627,7 @@ def __init__( self.max_deviation = [] if max_deviation is None else max_deviation if not self.band_overlaps_dict: - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n") spin_numbers = [0, 1] if lines[0].split()[-1] == "0" else [1, 2] @@ -1767,7 +1767,7 @@ def __init__( self.is_lcfo = is_lcfo self.list_dict_grosspop = [] if list_dict_grosspop is None else list_dict_grosspop if not self.list_dict_grosspop: - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n") # Read file to list of dict @@ -1897,7 +1897,7 @@ def _parse_file( imaginary (list[float]): Imaginary parts of wave function. distance (list[float]): Distances to the first point in wave function file. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n") points = [] @@ -2067,7 +2067,7 @@ def __init__( self.madelungenergies_mulliken = None if madelungenergies_mulliken is None else madelungenergies_mulliken if self.ewald_splitting is None: - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n")[5] if len(lines) == 0: raise RuntimeError("MadelungEnergies file contains no data.") @@ -2138,7 +2138,7 @@ def __init__( self.madelungenergies_mulliken: list | float = madelungenergies_mulliken or [] if self.num_atoms is None: - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n") if len(lines) == 0: raise RuntimeError("SitePotentials file contains no data.") @@ -2291,7 +2291,7 @@ def __init__( """ self._filename = str(filename) - with zopen(self._filename, mode="rt", encoding="utf-8") as file: + with zopen(self._filename, mode="rt") as file: lines = file.readlines() if len(lines) == 0: raise RuntimeError("Please check provided input file, it seems to be empty") @@ -2479,7 +2479,7 @@ def __init__( self.bin_width = 0.0 if bin_width is None else bin_width if not self.bwdf: - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = file.read().split("\n") if len(lines) == 0: raise RuntimeError("BWDF file contains no data.") diff --git a/src/pymatgen/io/nwchem.py b/src/pymatgen/io/nwchem.py index 523aa9a53a9..d9e4f47a463 100644 --- a/src/pymatgen/io/nwchem.py +++ b/src/pymatgen/io/nwchem.py @@ -394,7 +394,7 @@ def write_file(self, filename): Args: filename (str): Filename. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="w") as file: file.write(str(self)) def as_dict(self): @@ -534,7 +534,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: NwInput object """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename) as file: return cls.from_str(file.read()) @@ -557,7 +557,7 @@ def __init__(self, filename): """ self.filename = filename - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename) as file: data = file.read() chunks = re.split(r"NWChem Input Module", data) diff --git a/src/pymatgen/io/pwmat/inputs.py b/src/pymatgen/io/pwmat/inputs.py index 92bd77e0374..aa60c1737cd 100644 --- a/src/pymatgen/io/pwmat/inputs.py +++ b/src/pymatgen/io/pwmat/inputs.py @@ -36,7 +36,7 @@ def locate_all_lines(file_path: PathLike, content: str, exclusion: str = "") -> """ row_idxs: list[int] = [] # starts from 1 to be compatible with linecache package row_no: int = 0 - with zopen(file_path, mode="rt", encoding="utf-8") as file: + with zopen(file_path, mode="rt") as file: for row_content in file: row_no += 1 if content.upper() in row_content.upper() and ( @@ -418,7 +418,7 @@ def from_file(cls, filename: PathLike, mag: bool = False) -> Self: Returns: AtomConfig object. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, "rt") as file: return cls.from_str(data=file.read(), mag=mag) @classmethod @@ -466,7 +466,7 @@ def get_str(self) -> str: def write_file(self, filename: PathLike, **kwargs): """Write AtomConfig to a file.""" - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, "wt") as file: file.write(self.get_str(**kwargs)) def as_dict(self): @@ -588,7 +588,7 @@ def write_file(self, filename: PathLike): Args: filename (PathLike): The absolute path of file to be written. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, "wt") as file: file.write(self.get_str()) @@ -694,5 +694,5 @@ def get_hsp_row_str(label: str, index: int, coordinate: float) -> str: def write_file(self, filename: PathLike): """Write HighSymmetryPoint to a file.""" - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, "wt") as file: file.write(self.get_str()) diff --git a/src/pymatgen/io/pwmat/outputs.py b/src/pymatgen/io/pwmat/outputs.py index 0183be0bab7..9389e82befb 100644 --- a/src/pymatgen/io/pwmat/outputs.py +++ b/src/pymatgen/io/pwmat/outputs.py @@ -135,7 +135,7 @@ def _parse_sefv(self) -> list[dict]: 'atom_forces' and 'virial'. """ ionic_steps: list[dict] = [] - with zopen(self.filename, mode="rt", encoding="utf-8") as mvt: + with zopen(self.filename, "rt") as mvt: tmp_step: dict = {} for ii in range(self.n_ionic_steps): tmp_chunk: str = "" @@ -168,7 +168,7 @@ def __init__(self, filename: PathLike): filename (PathLike): The absolute path of OUT.FERMI file. """ self.filename: PathLike = filename - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, "rt") as file: self._e_fermi: float = np.round(float(file.readline().split()[-2].strip()), 3) @property @@ -346,7 +346,7 @@ def _parse(self): labels: list[str] = [] labels = linecache.getline(str(self.filename), 1).split()[1:] dos_str: str = "" - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="rt") as file: file.readline() dos_str = file.read() dos: np.ndarray = np.loadtxt(StringIO(dos_str)) diff --git a/src/pymatgen/io/pwscf.py b/src/pymatgen/io/pwscf.py index 03f7e7d456e..2f32c0c346a 100644 --- a/src/pymatgen/io/pwscf.py +++ b/src/pymatgen/io/pwscf.py @@ -275,7 +275,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: PWInput object """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls.from_str(file.read()) @classmethod diff --git a/src/pymatgen/io/qchem/inputs.py b/src/pymatgen/io/qchem/inputs.py index 7ff5be8487e..d350a84ba55 100644 --- a/src/pymatgen/io/qchem/inputs.py +++ b/src/pymatgen/io/qchem/inputs.py @@ -373,7 +373,7 @@ def write_multi_job_file(job_list: list[QCInput], filename: str): job_list (list[QCInput]): List of QChem jobs. filename (str): Name of the file to write. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(QCInput.multi_job_string(job_list)) @classmethod @@ -387,7 +387,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: QcInput """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls.from_str(file.read()) @classmethod @@ -401,7 +401,7 @@ def from_multi_jobs_file(cls, filename: str) -> list[Self]: Returns: List of QCInput objects """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: # the delimiter between QChem jobs is @@@ multi_job_strings = file.read().split("@@@") # list of individual QChem jobs diff --git a/src/pymatgen/io/qchem/outputs.py b/src/pymatgen/io/qchem/outputs.py index d483e7ecba3..a3e9ad038a8 100644 --- a/src/pymatgen/io/qchem/outputs.py +++ b/src/pymatgen/io/qchem/outputs.py @@ -661,7 +661,7 @@ def multiple_outputs_from_file(filename, keep_sub_files=True): 2.) Creates separate QCCalcs for each one from the sub-files. """ to_return = [] - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: text = re.split(r"\s*(?:Running\s+)*Job\s+\d+\s+of\s+\d+\s+", file.read()) if text[0] == "": text = text[1:] diff --git a/src/pymatgen/io/qchem/sets.py b/src/pymatgen/io/qchem/sets.py index d68a1714218..dddd478a966 100644 --- a/src/pymatgen/io/qchem/sets.py +++ b/src/pymatgen/io/qchem/sets.py @@ -643,7 +643,7 @@ def write(self, input_file: PathLike) -> None: """ self.write_file(input_file) if self.smd_solvent in {"custom", "other"} and self.qchem_version == 5: - with zopen(os.path.join(os.path.dirname(input_file), "solvent_data"), mode="wt", encoding="utf-8") as file: + with zopen(os.path.join(os.path.dirname(input_file), "solvent_data"), mode="wt") as file: file.write(self.custom_smd) diff --git a/src/pymatgen/io/res.py b/src/pymatgen/io/res.py index cfc81f3d649..9a45a3b9dc2 100644 --- a/src/pymatgen/io/res.py +++ b/src/pymatgen/io/res.py @@ -249,7 +249,7 @@ def _parse_str(cls, source: str) -> Res: def _parse_file(cls, filename: str | Path) -> Res: """Parse the res file as a file.""" self = cls() - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="r") as file: self.source = file.read() return self._parse_txt() @@ -335,7 +335,7 @@ def string(self) -> str: def write(self, filename: str) -> None: """Write the res data to a file.""" - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="w") as file: file.write(str(self)) diff --git a/src/pymatgen/io/template.py b/src/pymatgen/io/template.py index bacfd661e97..2ee08031ede 100644 --- a/src/pymatgen/io/template.py +++ b/src/pymatgen/io/template.py @@ -52,7 +52,7 @@ def get_input_set( self.filename = str(filename) # Load the template - with zopen(self.template, mode="rt", encoding="utf-8") as file: + with zopen(self.template, mode="r") as file: template_str = file.read() # Replace all variables diff --git a/src/pymatgen/io/vasp/inputs.py b/src/pymatgen/io/vasp/inputs.py index 11167289055..f5357b96dd7 100644 --- a/src/pymatgen/io/vasp/inputs.py +++ b/src/pymatgen/io/vasp/inputs.py @@ -285,7 +285,7 @@ def from_file( except Exception: names = None - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls.from_str(file.read(), names, read_velocities=read_velocities) @classmethod @@ -671,7 +671,7 @@ def write_file(self, filename: PathLike, **kwargs) -> None: """Write POSCAR to a file. The supported kwargs are the same as those for the Poscar.get_str method and are passed through directly. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(self.get_str(**kwargs)) def as_dict(self) -> dict: @@ -906,7 +906,7 @@ def write_file(self, filename: PathLike) -> None: Args: filename (str): filename to write to. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(str(self)) @classmethod @@ -919,7 +919,7 @@ def from_file(cls, filename: PathLike) -> Self: Returns: Incar object """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls.from_str(file.read()) @classmethod @@ -1659,7 +1659,7 @@ def from_file(cls, filename: PathLike) -> Self: Returns: Kpoints object """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls.from_str(file.read()) @classmethod @@ -1800,7 +1800,7 @@ def write_file(self, filename: PathLike) -> None: Args: filename (PathLike): Filename to write to. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(str(self)) def as_dict(self) -> dict[str, Any]: @@ -2411,9 +2411,9 @@ def write_file(self, filename: str) -> None: """Write PotcarSingle to a file. Args: - filename (str): File to write to. + filename (str): Filename to write to. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(str(self)) def copy(self) -> Self: @@ -2438,7 +2438,7 @@ def from_file(cls, filename: PathLike) -> Self: symbol = match[0] if match else "" try: - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls(file.read(), symbol=symbol or None) except UnicodeDecodeError: @@ -2850,7 +2850,7 @@ def from_file(cls, filename: PathLike) -> Self: Returns: Potcar """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: fdata = file.read() potcar = cls() @@ -2873,7 +2873,7 @@ def write_file(self, filename: PathLike) -> None: Args: filename (PathLike): filename to write to. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(str(self)) def set_symbols( @@ -3009,7 +3009,7 @@ def write_input( for key, value in self.items(): if value is not None: - with zopen(os.path.join(output_dir, key), mode="wt", encoding="utf-8") as file: + with zopen(os.path.join(output_dir, key), mode="wt") as file: file.write(str(value)) if cif_name: @@ -3032,8 +3032,8 @@ def write_input( files_to_transfer = files_to_transfer or {} for key, val in files_to_transfer.items(): with ( - zopen(val, mode="rb") as fin, - zopen(str(Path(output_dir) / key), mode="wb") as fout, + zopen(val, "rb") as fin, + zopen(str(Path(output_dir) / key), "wb") as fout, ): copyfileobj(fin, fout) diff --git a/src/pymatgen/io/vasp/outputs.py b/src/pymatgen/io/vasp/outputs.py index 33e76ec8cb2..4694183b8c5 100644 --- a/src/pymatgen/io/vasp/outputs.py +++ b/src/pymatgen/io/vasp/outputs.py @@ -312,7 +312,7 @@ def __init__( self.separate_spins = separate_spins self.exception_on_bad_xml = exception_on_bad_xml - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: if ionic_step_skip or ionic_step_offset: # Remove parts of the xml file and parse the string content: str = file.read() @@ -1772,7 +1772,7 @@ def __init__( self.occu_tol = occu_tol self.separate_spins = separate_spins - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: self.efermi = None parsed_header = False in_kpoints_opt = False @@ -2124,7 +2124,7 @@ def __init__(self, filename: PathLike) -> None: # Data from beginning of OUTCAR run_stats["cores"] = None - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: for line in file: if "serial" in line: # Activate serial parallelization @@ -2382,7 +2382,7 @@ def read_table_pattern( if last_one_only and first_one_only: raise ValueError("last_one_only and first_one_only options are incompatible") - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="rt") as file: text = file.read() table_pattern_text = header_pattern + r"\s*^(?P(?:\s+" + row_pattern + r")+)\s+" + footer_pattern table_pattern = re.compile(table_pattern_text, re.MULTILINE | re.DOTALL) @@ -2468,7 +2468,7 @@ def read_freq_dielectric(self) -> None: data: dict[str, Any] = {"REAL": [], "IMAGINARY": []} count = 0 component = "IMAGINARY" - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="rt") as file: for line in file: line = line.strip() if re.match(plasma_pattern, line): @@ -2594,7 +2594,7 @@ def read_cs_raw_symmetrized_tensors(self) -> None: row_pattern = r"\s+".join([r"([-]?\d+\.\d+)"] * 3) unsym_footer_pattern = r"^\s+SYMMETRIZED TENSORS\s+$" - with zopen(self.filename, mode="rt", encoding="utf-8") as file: + with zopen(self.filename, mode="rt") as file: text = file.read() unsym_table_pattern_text = header_pattern + first_part_pattern + r"(?P.+)" + unsym_footer_pattern table_pattern = re.compile(unsym_table_pattern_text, re.MULTILINE | re.DOTALL) @@ -3377,7 +3377,7 @@ def read_core_state_eigen(self) -> list[dict]: The core state eigenenergie of the 2s AO of the 6th atom of the structure at the last ionic step is [5]["2s"][-1]. """ - with zopen(self.filename, mode="rt", encoding="utf-8") as foutcar: + with zopen(self.filename, mode="rt") as foutcar: line = foutcar.readline() cl: list[dict] = [] @@ -3419,7 +3419,7 @@ def read_avg_core_poten(self) -> list[list]: The average core potential of the 2nd atom of the structure at the last ionic step is: [-1][1] """ - with zopen(self.filename, mode="rt", encoding="utf-8") as foutcar: + with zopen(self.filename, mode="rt") as foutcar: line = foutcar.readline() aps: list[list[float]] = [] while line != "": @@ -3615,7 +3615,7 @@ def parse_file(filename: PathLike) -> tuple[Poscar, dict, dict]: ngrid_pts = 0 data_count = 0 poscar = None - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: for line in file: original_line = line line = line.strip() @@ -3749,7 +3749,7 @@ def write_spin(data_type: str) -> None: if isinstance(data, Iterable): file.write("".join(data)) - with zopen(file_name, mode="wt", encoding="utf-8") as file: + with zopen(file_name, mode="wt") as file: poscar = Poscar(self.structure) # Use original name if it's been set (e.g. from Chgcar) @@ -4082,7 +4082,7 @@ def _read(self, filename: PathLike, parsed_kpoints: set[tuple[Kpoint]] | None = if parsed_kpoints is None: parsed_kpoints = set() - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file_handle: preamble_expr = re.compile(r"# of k-points:\s*(\d+)\s+# of bands:\s*(\d+)\s+# of ions:\s*(\d+)") kpoint_expr = re.compile(r"^k-point\s+(\d+).*weight = ([0-9\.]+)") band_expr = re.compile(r"^band\s+(\d+)") @@ -4113,7 +4113,7 @@ def _read(self, filename: PathLike, parsed_kpoints: set[tuple[Kpoint]] | None = # total and x,y,z) for each band, while non-SOC have only 1 list of projections: tot_count = 0 band_count = 0 - for line in file: + for line in file_handle: if total_expr.match(line): tot_count += 1 elif band_expr.match(line): @@ -4121,7 +4121,7 @@ def _read(self, filename: PathLike, parsed_kpoints: set[tuple[Kpoint]] | None = if band_count == 2: break - file.seek(0) # reset file handle to beginning + file_handle.seek(0) # reset file handle to beginning if tot_count == 1: is_soc = False elif tot_count == 4: @@ -4138,7 +4138,7 @@ def _read(self, filename: PathLike, parsed_kpoints: set[tuple[Kpoint]] | None = skipping_kpoint = False # true when skipping projections for a previously-parsed kpoint ion_line_count = 0 # printed twice when phase factors present proj_data_parsed_for_band = 0 # 0 for non-SOC, 1-4 for SOC/phase factors - for line in file: + for line in file_handle: line = line.strip() if ion_expr.match(line): ion_line_count += 1 @@ -4373,8 +4373,8 @@ def smart_convert(header: str, num: float | str) -> float | str: electronic_pattern = re.compile(r"\s*\w+\s*:(.*)") header: list = [] - with zopen(filename, mode="rt", encoding="utf-8") as file: - for line in file: + with zopen(filename, mode="rt") as fid: + for line in fid: if match := electronic_pattern.match(line.strip()): tokens = match[1].split() data = {header[idx]: smart_convert(header[idx], tokens[idx]) for idx in range(len(tokens))} @@ -4516,10 +4516,10 @@ def __init__( if ionicstep_end is not None and ionicstep_end < 1: raise ValueError("End ionic step cannot be less than 1") - file_len = sum(1 for _ in zopen(filename, mode="rt", encoding="utf-8")) + file_len = sum(1 for _ in zopen(filename, mode="rt")) ionicstep_cnt = 1 ionicstep_start = ionicstep_start or 0 - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: title = None for iline, line in enumerate(file): line = line.strip() @@ -4615,7 +4615,7 @@ def concatenate( raise ValueError("End ionic step cannot be less than 1") ionicstep_cnt = 1 - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: for line in file: line = line.strip() if preamble is None: @@ -4703,7 +4703,7 @@ def write_file(self, filename: PathLike, **kwargs) -> None: **kwargs: The same as those for the Xdatcar.get_str method and are passed through directly. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(self.get_str(**kwargs)) @@ -4725,7 +4725,7 @@ def __init__(self, filename: PathLike) -> None: Args: filename: Name of file containing DYNMAT. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: lines = list(clean_lines(file.readlines())) self._nspecs, self._natoms, self._ndisps = map(int, lines[0].split()) self._masses = map(float, lines[1].split()) @@ -5420,7 +5420,7 @@ def __init__( self.occu_tol = occu_tol self.separate_spins = separate_spins - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="r") as file: self.ispin = int(file.readline().split()[-1]) # Remove useless header information @@ -5571,7 +5571,7 @@ def from_formatted(cls, filename: PathLike) -> Self: Returns: A Waveder object. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: nspin, nkpts, nbands = file.readline().split() # 1 and 4 are the eigenvalues of the bands (this data is missing in the WAVEDER file) # 6:12 are the complex matrix elements in each cartesian direction. diff --git a/src/pymatgen/io/xr.py b/src/pymatgen/io/xr.py index e86554a83be..834b8d6d797 100644 --- a/src/pymatgen/io/xr.py +++ b/src/pymatgen/io/xr.py @@ -68,7 +68,7 @@ def write_file(self, filename: str | Path) -> None: Args: filename (str): name of the file to write to. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(str(self) + "\n") @classmethod @@ -155,5 +155,5 @@ def from_file(cls, filename: str | Path, use_cores: bool = True, thresh: float = xr (Xr): Xr object corresponding to the input file. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls.from_str(file.read(), use_cores=use_cores, thresh=thresh) diff --git a/src/pymatgen/io/xyz.py b/src/pymatgen/io/xyz.py index 4f44f66e552..98078baa63c 100644 --- a/src/pymatgen/io/xyz.py +++ b/src/pymatgen/io/xyz.py @@ -111,7 +111,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: XYZ object """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: return cls.from_str(file.read()) def as_dataframe(self): @@ -151,5 +151,5 @@ def write_file(self, filename: str) -> None: Args: filename (str): File name of output file. """ - with zopen(filename, mode="wt", encoding="utf-8") as file: + with zopen(filename, mode="wt") as file: file.write(str(self)) diff --git a/src/pymatgen/io/zeopp.py b/src/pymatgen/io/zeopp.py index 64f1557f4d1..55a1506dc44 100644 --- a/src/pymatgen/io/zeopp.py +++ b/src/pymatgen/io/zeopp.py @@ -146,7 +146,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: ZeoCssr object. """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="r") as file: return cls.from_str(file.read()) @@ -200,7 +200,7 @@ def from_file(cls, filename: str | Path) -> Self: Returns: XYZ object """ - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename) as file: return cls.from_str(file.read()) def __str__(self) -> str: diff --git a/src/pymatgen/transformations/site_transformations.py b/src/pymatgen/transformations/site_transformations.py index 6b1bede1f2a..4672bad1183 100644 --- a/src/pymatgen/transformations/site_transformations.py +++ b/src/pymatgen/transformations/site_transformations.py @@ -431,7 +431,9 @@ def apply_transformation(self, structure: Structure, return_ranked_list: bool | n_to_remove = round(n_to_remove) num_remove_dict[tuple(idx)] = n_to_remove n = len(idx) - total_combos += round(math.factorial(n) / math.factorial(n_to_remove) / math.factorial(n - n_to_remove)) + total_combos += int( + round(math.factorial(n) / math.factorial(n_to_remove) / math.factorial(n - n_to_remove)) + ) self.logger.debug(f"Total combinations = {total_combos}") diff --git a/src/pymatgen/util/io_utils.py b/src/pymatgen/util/io_utils.py index 7bf4efdc2f4..f8c7d268f43 100644 --- a/src/pymatgen/util/io_utils.py +++ b/src/pymatgen/util/io_utils.py @@ -81,7 +81,7 @@ def micro_pyawk(filename, search, results=None, debug=None, postdebug=None): for entry in search: entry[0] = re.compile(entry[0]) - with zopen(filename, mode="rt", encoding="utf-8") as file: + with zopen(filename, mode="rt") as file: for line in file: for entry in search: match = re.search(entry[0], line) diff --git a/tests/electronic_structure/test_dos.py b/tests/electronic_structure/test_dos.py index edf2c6957ae..a7c8dec41b1 100644 --- a/tests/electronic_structure/test_dos.py +++ b/tests/electronic_structure/test_dos.py @@ -108,7 +108,7 @@ class TestCompleteDos(TestCase): def setUp(self): with open(f"{TEST_DIR}/complete_dos.json") as file: self.dos = CompleteDos.from_dict(json.load(file)) - with zopen(f"{TEST_DIR}/pdag3_complete_dos.json.gz", mode="rt", encoding="utf-8") as file: + with zopen(f"{TEST_DIR}/pdag3_complete_dos.json.gz") as file: self.dos_pdag3 = CompleteDos.from_dict(json.load(file)) def test_get_gap(self): diff --git a/tests/io/aims/conftest.py b/tests/io/aims/conftest.py index b2ac971c904..3cabe2b3ca8 100644 --- a/tests/io/aims/conftest.py +++ b/tests/io/aims/conftest.py @@ -150,7 +150,7 @@ def compare_single_files(ref_file: PathLike, test_file: PathLike) -> None: with open(test_file) as tf: test_lines = tf.readlines()[5:] - with zopen(f"{ref_file}.gz", mode="rt", encoding="utf-8") as rf: + with zopen(f"{ref_file}.gz", mode="rt") as rf: ref_lines = rf.readlines()[5:] for test_line, ref_line in zip(test_lines, ref_lines, strict=True): diff --git a/tests/io/pwmat/test_inputs.py b/tests/io/pwmat/test_inputs.py index e1438482623..7618e7ecdf6 100644 --- a/tests/io/pwmat/test_inputs.py +++ b/tests/io/pwmat/test_inputs.py @@ -47,7 +47,7 @@ class TestACstrExtractor(PymatgenTest): def test_extract(self): filepath = f"{TEST_DIR}/atom.config" ac_extractor = ACExtractor(file_path=filepath) - with zopen(filepath, mode="rt", encoding="utf-8") as file: + with zopen(filepath, mode="rt") as file: ac_str_extractor = ACstrExtractor(atom_config_str="".join(file.readlines())) assert ac_extractor.n_atoms == ac_str_extractor.get_n_atoms() for idx in range(9): @@ -102,7 +102,7 @@ def test_write_file(self): tmp_file = f"{self.tmp_path}/gen.kpt.testing.lzma" gen_kpt.write_file(tmp_file) tmp_gen_kpt_str = "" - with zopen(tmp_file, mode="rt", encoding="utf-8") as file: + with zopen(tmp_file, mode="rt") as file: tmp_gen_kpt_str = file.read() assert gen_kpt.get_str() == tmp_gen_kpt_str @@ -128,7 +128,7 @@ def test_write_file(self): tmp_filepath = f"{self.tmp_path}/HIGH_SYMMETRY_POINTS.testing.lzma" high_symmetry_points.write_file(tmp_filepath) tmp_high_symmetry_points_str = "" - with zopen(tmp_filepath, "rt", encoding="utf-8") as file: + with zopen(tmp_filepath, "rt") as file: tmp_high_symmetry_points_str = file.read() assert tmp_high_symmetry_points_str == high_symmetry_points.get_str() diff --git a/tests/io/vasp/test_inputs.py b/tests/io/vasp/test_inputs.py index 5b572d44acd..1616ff4e5e7 100644 --- a/tests/io/vasp/test_inputs.py +++ b/tests/io/vasp/test_inputs.py @@ -1560,7 +1560,7 @@ def test_from_file(self): } def test_potcar_map(self): - fe_potcar = zopen(f"{FAKE_POTCAR_DIR}/POT_GGA_PAW_PBE/POTCAR.Fe_pv.gz", mode="rt", encoding="utf-8").read() + fe_potcar = zopen(f"{FAKE_POTCAR_DIR}/POT_GGA_PAW_PBE/POTCAR.Fe_pv.gz").read().decode("utf-8") # specify V instead of Fe - this makes sure the test won't pass if the # code just grabs the POTCAR from the config file (the config file would # grab the V POTCAR) diff --git a/tests/io/vasp/test_outputs.py b/tests/io/vasp/test_outputs.py index 8044a130e55..837884c22f0 100644 --- a/tests/io/vasp/test_outputs.py +++ b/tests/io/vasp/test_outputs.py @@ -2164,7 +2164,7 @@ def test_consistency(self): wder_ref = np.loadtxt(f"{VASP_OUT_DIR}/WAVEDERF.Si.gz", skiprows=1) def _check(wder): - with zopen(f"{VASP_OUT_DIR}/WAVEDERF.Si.gz", mode="rt", encoding="utf-8") as file: + with zopen(f"{VASP_OUT_DIR}/WAVEDERF.Si.gz") as file: first_line = [int(a) for a in file.readline().split()] assert wder.nkpoints == first_line[1] assert wder.nbands == first_line[2] From b28b4c24d92830327f2aaf199804b726d680437c Mon Sep 17 00:00:00 2001 From: Shyue Ping Ong Date: Tue, 10 Dec 2024 18:24:15 -0800 Subject: [PATCH 11/12] Remove offending linting issue. --- src/pymatgen/io/cp2k/sets.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/pymatgen/io/cp2k/sets.py b/src/pymatgen/io/cp2k/sets.py index b03506c3ba1..cbd857891dd 100644 --- a/src/pymatgen/io/cp2k/sets.py +++ b/src/pymatgen/io/cp2k/sets.py @@ -371,7 +371,7 @@ def __init__( def get_basis_and_potential( structure: Structure, basis_and_potential: dict[str, dict[str, Any]], - cp2k_data_dir: str | Path | None = None, + cp2k_data_dir: str | None = None, ) -> dict[str, dict[str, Any]]: """Get a dictionary of basis and potential info for constructing the input file. From 362bf543541a30468d51946508f33c553ce15147 Mon Sep 17 00:00:00 2001 From: "Haoyu (Daniel) YANG" Date: Wed, 11 Dec 2024 22:03:55 +0800 Subject: [PATCH 12/12] Fix CI failure: bump `netcdf4`, replace `Namespace` with standard `dict` for `io.abinit.pseudos` (#4223) --- pyproject.toml | 3 +-- src/pymatgen/io/abinit/pseudos.py | 6 ++++-- 2 files changed, 5 insertions(+), 4 deletions(-) diff --git a/pyproject.toml b/pyproject.toml index 7fba2e4c661..db6f4b52ee6 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -87,8 +87,7 @@ Issues = "https://github.com/materialsproject/pymatgen/issues" Pypi = "https://pypi.org/project/pymatgen" [project.optional-dependencies] -# PR4128: netcdf4 1.7.[0/1] yanked, 1.7.1.post[1/2]/1.7.2 cause CI error -abinit = ["netcdf4>=1.6.5,!=1.7.1.post1,!=1.7.1.post2,!=1.7.2"] +abinit = ["netcdf4>=1.7.2"] ase = ["ase>=3.23.0"] ci = ["pytest-cov>=4", "pytest-split>=0.8", "pytest>=8"] docs = ["invoke", "sphinx", "sphinx_markdown_builder", "sphinx_rtd_theme"] diff --git a/src/pymatgen/io/abinit/pseudos.py b/src/pymatgen/io/abinit/pseudos.py index db9a6d2bb08..3c08a28ded6 100644 --- a/src/pymatgen/io/abinit/pseudos.py +++ b/src/pymatgen/io/abinit/pseudos.py @@ -19,7 +19,7 @@ from xml.etree import ElementTree as ET import numpy as np -from monty.collections import AttrDict, Namespace +from monty.collections import AttrDict from monty.functools import lazy_property from monty.itertools import iterator_from_slice from monty.json import MontyDecoder, MSONable @@ -601,7 +601,9 @@ def _dict_from_lines(lines, key_nums, sep=None) -> dict: if len(lines) != len(key_nums): raise ValueError(f"{lines = }\n{key_nums = }") - kwargs = Namespace() + # TODO: PR 4223: kwargs was using `monty.collections.Namespace`, + # revert to original implementation if needed + kwargs: dict = {} for idx, nk in enumerate(key_nums): if nk == 0: