Source code for tcutility.analysis.pyfrag

import os
from typing import List, Union

import numpy as np
from scm import plams

from tcutility.cache import cache
from tcutility.environment import requires_optional_package
from tcutility.geometry import parameter
from tcutility.pathfunc import match


[docs] def get_pyfrag_results(path): return PyFragResult(path)
[docs] class PyFragResult: def __init__(self, path, step_prefix: str = "Step."): self.path = path self._frag_results = {} self._step_results = [] self._order = [] self._mask = [] self._properties = {} self.step_prefix = step_prefix self._load() def _load(self): for frag_path, info in match(self.path, "frag_{frag}").items(): self._frag_results[info.frag] = get_pyfrag_results(frag_path) for step_path, info in match(self.path, self.step_prefix + "{step}", sort_by="step").items(): self._step_results.append({}) for dir_name in os.listdir(step_path): p = os.path.join(step_path, dir_name) res = get_pyfrag_results(p) self._step_results[-1][dir_name] = res self._order.append(int(info.step.removeprefix("0"))) self._mask.append(True) self._mask = np.array(self._mask)
[docs] def get_property(self, key: str, calc: str = "complex"): if key in self._properties: return self._properties[key][self._order][self._mask[self._order]] p = np.array([res[calc].properties.get_multi_key(key) for res in self._step_results]) return p[self._order][self._mask[self._order]]
@property def fragments(self): return list(self._frag_results.keys()) def __len__(self): return sum(self._mask)
[docs] def get_geometry(self, *args, **kwargs): calc = kwargs.pop("calc", "complex") g = np.array([parameter(res[calc].molecule.input, *args, **kwargs) for res in self._step_results]) return g[self._order][self._mask[self._order]]
[docs] def get_molecules(self, calc: str = "complex") -> List[plams.Molecule]: mols = [res[calc].molecule.input for res in self._step_results] return mols return [mols[int(i)] for i in np.array(self._order)[self._mask[self._order]]]
[docs] def sort_by(self, val: Union[str, List], calc: str = "complex"): if isinstance(val, str): val = self.get_property(val, calc=calc) self._order = np.argsort(val)
[docs] def set_mask(self, mask: list): self._mask = mask
[docs] def set_coord(self, coord: list): self._coord = coord
[docs] def coord(self): return self._coord[self._order][self._mask[self._order]]
[docs] def set_property(self, name: str, vals): self._properties[name] = vals
@property def ts_idx(self): energies = self.total_energy() for i in range(len(self) - 2, 1, -1): if energies[i - 1] < energies[i] and energies[i] < energies[i + 1]: return i + 1
[docs] def total_energy(self): return self.interaction_energy() + self.strain_energy()
[docs] def interaction_energy(self): return self.get_property("energy.bond", "complex")
[docs] def strain_energy(self, fragment: str = ""): if fragment is None: t = 0 for frag in self.fragments: t += self.strain_energy(frag) return t return self.get_property("energy.bond", f"frag_{fragment}") - self._frag_results[fragment].properties.energy.bond
[docs] @requires_optional_package("pyfmo") def overlap(self, orb1, orb2, calc: str = "complex", absolute: bool = True): S = np.array([orb.sfos[orb1] @ orb.sfos[orb2] for orb in self.orbs(calc)]) if absolute: S = abs(S) return S[self._order][self._mask[self._order]]
[docs] @requires_optional_package("pyfmo") def orbital_energy_gap(self, orb1, orb2, calc: str = "complex"): dE = np.array([abs(orb.sfos[orb1].energy - orb.sfos[orb2].energy) for orb in self.orbs(calc)]) return dE[self._order][self._mask[self._order]]
[docs] @requires_optional_package("pyfmo") def sfo_coefficient(self, sfo, mo, calc: str = "complex"): C = np.array([orb.sfos[sfo].coefficient(orb.mos[mo]) for orb in self.orbs(calc)]) return C[self._order][self._mask[self._order]]
[docs] @cache @requires_optional_package("pyfmo") def orbs(self, calc: str = "complex"): import pyfmo return [pyfmo.Orbitals(res[calc].files["adf.rkf"]) for res in self._step_results]
if __name__ == "__main__": import matplotlib.pyplot as plt res_C = get_pyfrag_results("/Users/yumanhordijk/PhD/Projects/RadicalAdditionASMEDA/data/DFT/TS_C_O/PyFrag_OLYP_TZ2P") res_X = get_pyfrag_results("/Users/yumanhordijk/PhD/Projects/RadicalAdditionASMEDA/data/DFT/TS_X_O/PyFrag_OLYP_TZ2P") res_C.set_coord(res_C.get_geometry(0, 1)) res_X.set_coord(res_X.get_geometry(1, 2)) res_C.sort_by(res_C.coord()) res_X.sort_by(res_X.coord()) plt.figure() plt.plot(res_C.coord(), res_C.overlap("Methyl(SOMO)_A", "Substrate(LUMO)_A"), label="C-addition") plt.plot(res_X.coord(), res_X.overlap("Methyl(SOMO)_A", "Substrate(LUMO)_A"), label="X-addition") plt.ylabel(r"$\langle SOMO | LUMO \rangle$") plt.legend() plt.xlim(3, 2.2) plt.figure() plt.plot(res_C.coord(), res_C.orbital_energy_gap("Methyl(SOMO)_A", "Substrate(LUMO)_A"), label="C-addition") plt.plot(res_X.coord(), res_X.orbital_energy_gap("Methyl(SOMO)_A", "Substrate(LUMO)_A"), label="X-addition") plt.ylabel(r"$|\epsilon_{SOMO} - \epsilon_{LUMO}|$") plt.legend() plt.xlim(3, 2.2) plt.figure() plt.plot(res_C.coord(), res_C.overlap("Methyl(LUMO)_B", "Substrate(7A_B)_B"), label="C-addition") plt.plot(res_X.coord(), res_X.overlap("Methyl(LUMO)_B", "Substrate(7A_B)_B"), label="X-addition") plt.ylabel(r"$\langle SUMO | HOMO \rangle$") plt.legend() plt.xlim(3, 2.2) plt.figure() plt.plot(res_C.coord(), res_C.orbital_energy_gap("Methyl(LUMO)_B", "Substrate(7A_B)_B"), label="C-addition") plt.plot(res_X.coord(), res_X.orbital_energy_gap("Methyl(LUMO)_B", "Substrate(7A_B)_B"), label="X-addition") plt.ylabel(r"$|\epsilon_{SUMO} - \epsilon_{HOMO}|$") plt.legend() plt.xlim(3, 2.2) plt.show()