Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

MP2 psi4 #315

Merged
merged 84 commits into from
Jun 21, 2023
Merged
Show file tree
Hide file tree
Changes from 82 commits
Commits
Show all changes
84 commits
Select commit Hold shift + click to select a range
cea6b32
added essolver base class and separated pyscf
JamesB-1qbit Apr 18, 2023
613ee74
added comments
JamesB-1qbit Apr 18, 2023
678f478
fixed failed tests except for fci semi_empirical oniom dmet
JamesB-1qbit Apr 19, 2023
23e9536
fixes for remaining tests
JamesB-1qbit Apr 19, 2023
60c31bf
small test for dummy solver
JamesB-1qbit Apr 19, 2023
cfbceb7
test of no pyscf
JamesB-1qbit May 3, 2023
d2f1e58
fixed import error
JamesB-1qbit May 4, 2023
cc2cf5b
updated github testing
JamesB-1qbit May 5, 2023
9b0b73b
fix upload location
JamesB-1qbit May 8, 2023
a1db904
added psi4
JamesB-1qbit May 9, 2023
6c29ccd
update actions
JamesB-1qbit May 9, 2023
7fbe9b9
init conda
JamesB-1qbit May 9, 2023
6b8c813
psi4 fix
JamesB-1qbit May 9, 2023
1ffe514
added shell
JamesB-1qbit May 9, 2023
8c3caf7
added init shell
JamesB-1qbit May 9, 2023
d9ceb0d
add conda info
JamesB-1qbit May 9, 2023
02337f9
updated activate
JamesB-1qbit May 9, 2023
49e7f78
install qulacs
JamesB-1qbit May 9, 2023
4fb79a0
add missing file
JamesB-1qbit May 9, 2023
aee3e59
remove python=3.8
JamesB-1qbit May 9, 2023
1dae806
use setup-miniconda
JamesB-1qbit May 9, 2023
eabf909
v3 -> v2
JamesB-1qbit May 9, 2023
2b50e11
remove environment
JamesB-1qbit May 9, 2023
26b2596
add activate to each
JamesB-1qbit May 9, 2023
15e265d
remove nopyscf test
JamesB-1qbit May 9, 2023
4954136
use environment variable to ignore pyscf
JamesB-1qbit May 18, 2023
56dacbe
ONIOM test and molecule_library functionality
JamesB-1qbit May 18, 2023
9c352e9
fixes for errors
JamesB-1qbit May 18, 2023
b32c380
insure that pyscf is not installed before no_pyscf tests
JamesB-1qbit May 18, 2023
995e96e
fix IntegralSolver preference ordering
JamesB-1qbit May 18, 2023
f5f4a31
codestyle improvements
JamesB-1qbit May 26, 2023
8b91187
first review
JamesB-1qbit May 29, 2023
4d2234b
further improvements
JamesB-1qbit May 29, 2023
37799c0
added IntegralSolverEmpty
JamesB-1qbit May 29, 2023
8525981
missing newline
JamesB-1qbit May 30, 2023
c9db409
revert attribute docstrings, change function call
JamesB-1qbit May 31, 2023
e0ba0d9
added uhf support for psi4
JamesB-1qbit May 31, 2023
77a4d67
minao is not allowed in psi4
JamesB-1qbit May 31, 2023
83cd5b6
further improvements
JamesB-1qbit Jun 1, 2023
527f813
fix file locations
JamesB-1qbit Jun 1, 2023
7fbb9e7
moved chem tests to molecular_computation tests
JamesB-1qbit Jun 2, 2023
9327ab7
fix for failing tests
JamesB-1qbit Jun 2, 2023
2a6c5b4
first steps for fci psi4
JamesB-1qbit Jun 5, 2023
3fb4814
explicit orbital rotation
JamesB-1qbit Jun 5, 2023
5adad5d
next attempt
JamesB-1qbit Jun 5, 2023
27e702e
further attempts
JamesB-1qbit Jun 5, 2023
9c02b0a
FCI psi4 functioning, symmetry labels in psi4 IntegralSolver
JamesB-1qbit Jun 12, 2023
f1c0f7d
merged develop by fixing conflicts
JamesB-1qbit Jun 12, 2023
2eeea51
functioning fci solver for psi4
JamesB-1qbit Jun 13, 2023
7c63c5f
updated docstrings
JamesB-1qbit Jun 13, 2023
3be7063
pr review
JamesB-1qbit Jun 14, 2023
2089cbb
use sympy Permutation
JamesB-1qbit Jun 14, 2023
b3bf13f
code cleanup
JamesB-1qbit Jun 15, 2023
c1c735f
Changed FCISolver to class with solver attribute
JamesB-1qbit Jun 15, 2023
42dc939
first attempt
JamesB-1qbit Jun 14, 2023
6c1e2e7
passing tests
JamesB-1qbit Jun 15, 2023
92767b1
docstring fixes
JamesB-1qbit Jun 15, 2023
9ef29d1
ciwfn -> ccwfn
JamesB-1qbit Jun 15, 2023
73a4a1c
added ccsd testing
JamesB-1qbit Jun 15, 2023
55a218f
move to getswaps
JamesB-1qbit Jun 16, 2023
3147795
change order of swap operations
JamesB-1qbit Jun 16, 2023
9edb1c6
remove unused import
JamesB-1qbit Jun 16, 2023
251c891
removed another unused import
JamesB-1qbit Jun 16, 2023
60c8e0c
Change multi-controlled CNOT to multi-controlled CX (#308)
JamesB-1qbit Jun 6, 2023
914088d
Braket connection (#312)
ValentinS4t1qbit Jun 15, 2023
1c3c88c
Support for UMP2 initial parameters (#310)
alexfleury-sb Jun 15, 2023
0be9085
Checkfile for IntegralSolverPySCF (#311)
alexfleury-sb Jun 15, 2023
9f603ad
first implementation of psi4 mp2
JamesB-1qbit Jun 16, 2023
d797f1c
skip test
JamesB-1qbit Jun 16, 2023
eae3923
small fixes
JamesB-1qbit Jun 16, 2023
a9d5238
add mp2 to psi4 tests
JamesB-1qbit Jun 16, 2023
986a706
removed uneccessary code
JamesB-1qbit Jun 16, 2023
552270b
loosen test bounds
JamesB-1qbit Jun 16, 2023
9692e80
skip more tests
JamesB-1qbit Jun 16, 2023
732722c
fixed merge conflicts
JamesB-1qbit Jun 19, 2023
05b750f
updated test
JamesB-1qbit Jun 19, 2023
9d1ebac
fix merge conflict
JamesB-1qbit Jun 19, 2023
016c36f
fix merge conflict
JamesB-1qbit Jun 19, 2023
8c0a5b6
add get_mp2_amplitudes
JamesB-1qbit Jun 19, 2023
5c6404b
merge conflicts
JamesB-1qbit Jun 20, 2023
1e7cb1a
added docstrings changed psi4 test to encompass all classical solvers
JamesB-1qbit Jun 20, 2023
ee66425
conformance fixes
JamesB-1qbit Jun 20, 2023
c9ed7e8
codestyle improvements
JamesB-1qbit Jun 21, 2023
f03f955
fixed module docstring
JamesB-1qbit Jun 21, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 1 addition & 2 deletions .github/workflows/run_psi4_test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -65,8 +65,7 @@ jobs:
run: |
cd tangelo/algorithms/classical/tests
conda activate p4env
pytest --doctest-modules --junitxml=junit/psi4-classical-test-results.xml test_fci_solver.py
pytest --doctest-modules --junitxml=junit/psi4-classical-cc-test-results.xml test_ccsd_solver.py
pytest --doctest-modules --junitxml=junit/psi4-classical-test-results.xml
if: always()

- name: Upload psi4 test results
Expand Down
197 changes: 192 additions & 5 deletions tangelo/algorithms/classical/mp2_solver.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,18 +12,33 @@
# See the License for the specific language governing permissions and
# limitations under the License.

"""Class performing electronic structure calculation employing the Moller-Plesset perturbation theory (MP2) method.
"""Define electronic structure solver employing the full configuration
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved
interaction (CI) method.
"""
from typing import Union, Type
from itertools import combinations, product
from math import ceil

import numpy as np

from itertools import combinations, product
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved
from math import ceil

from tangelo.algorithms.electronic_structure_solver import ElectronicStructureSolver
from tangelo.helpers.utils import is_package_installed
from tangelo.toolboxes.molecular_computation.molecule import SecondQuantizedMolecule
from tangelo.toolboxes.molecular_computation import IntegralSolverPsi4, IntegralSolverPySCF
from tangelo.helpers.utils import installed_chem_backends, is_package_installed
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved
from tangelo.toolboxes.ansatz_generator._unitary_cc_openshell import uccsd_openshell_get_packed_amplitudes

if 'pyscf' in installed_chem_backends:
default_mp2_solver = 'pyscf'
elif 'psi4' in installed_chem_backends:
default_mp2_solver = 'psi4'
else:
default_mp2_solver = None

class MP2Solver(ElectronicStructureSolver):

class MP2SolverPySCF(ElectronicStructureSolver):
"""Uses the Second-order Moller-Plesset perturbation theory (MP2) method to solve the electronic structure problem,
through pyscf.

Expand Down Expand Up @@ -132,12 +147,184 @@ def get_mp2_amplitudes(self):

# Get singles and doubles amplitudes associated with one spatial occupied-virtual pair
doubles_1 = [-self.mp2_t2[q, q, p, p]/2. if (abs(-self.mp2_t2[q, q, p, p]/2.) > 1e-15) else 0.
for p, q in product(range(self.n_virtual), range(self.n_occupied))]
for p, q in product(range(self.n_virtual), range(self.n_occupied))]

# Get doubles amplitudes associated with two spatial occupied-virtual pairs
doubles_2 = [-self.mp2_t2[q, s, p, r] for (p, q), (r, s)
in combinations(product(range(self.n_virtual), range(self.n_occupied)), 2)]
in combinations(product(range(self.n_virtual), range(self.n_occupied)), 2)]

mp2_params = singles + doubles_1 + doubles_2

return mp2_params


class MP2SolverPsi4(ElectronicStructureSolver):
""" Uses the MP2 method to solve the electronic structure problem,
through Psi4.
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved

Only supports frozen core (active) orbitals sequentially from bottom (top) of energy ordering.

Args:
molecule (SecondQuantizedMolecule): The molecule to simulate.

Attributes:
mp2wfn (psi4.core.Wavefunction): The Psi4 Wavefunction returned from an mp2 calculation.
backend (psi4): The psi4 module
molecule (SecondQuantizedMolecule): The molecule with symmetry=False
"""

def __init__(self, molecule: SecondQuantizedMolecule):
if not is_package_installed("psi4"):
raise ModuleNotFoundError(f"Using {self.__class__.__name__} requires the installation of the Psi4 package")

import psi4
self.backend = psi4
self.backend.core.clean_options()
self.backend.core.clean()
self.backend.core.clean_variables()
self.mp2wfn = None

self.molecule = SecondQuantizedMolecule(xyz=molecule.xyz, q=molecule.q, spin=molecule.spin,
solver=IntegralSolverPsi4(),
basis=molecule.basis,
ecp=molecule.ecp,
symmetry=False,
uhf=molecule.uhf,
frozen_orbitals=molecule.frozen_orbitals)
self.basis = molecule.basis

def simulate(self):
"""Perform the simulation (energy calculation) for the molecule.

Returns:
float: Total MP2 energy.
"""
n_frozen_vir = len(self.molecule.frozen_virtual)
n_frozen_occ = len(self.molecule.frozen_occupied)
if n_frozen_occ or n_frozen_vir:
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved
if self.molecule.uhf:
if (set(self.molecule.frozen_occupied[0]) != set(self.molecule.frozen_occupied[1]) or
set(self.molecule.frozen_virtual[0]) != set(self.molecule.frozen_virtual)):
raise ValueError(f"Only identical frozen orbitals for alpha and beta are supported in {self.__class__.__name__}")
focc = np.array(self.molecule.frozen_occupied)
fvir = np.array(self.molecule.frozen_virtual)
if np.any(focc > n_frozen_occ-1) or np.any(fvir < self.molecule.n_mos-n_frozen_vir):
raise ValueError(f"{self.__class__.__name__} does not support freezing interior orbitals")

if not self.molecule.uhf:
ref = 'rhf' if self.molecule.spin == 0 else 'rohf'
else:
ref = 'uhf'
self.backend.set_options({'basis': self.basis, 'mcscf_maxiter': 300, 'mcscf_diis_start': 20,
'opdm': True, 'tpdm': True, 'frozen_docc': [n_frozen_occ], 'frozen_uocc': [n_frozen_vir],
'reference': ref})

energy, self.mp2wfn = self.backend.energy('mp2', molecule=self.molecule.solver.mol,
basis=self.basis, return_wfn=True)
return energy

def get_rdm(self):
"""Calculate the 1- and 2-particle reduced density matrices.

Obtaining MP2 rdms from Psi4 is not supported in Tangelo.
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved

Raises:
RuntimeError: Not implemented at this time"""
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved
raise RuntimeError("Returning MP2 rdms from Psi4 is not currently supported in Tangelo")

def get_mp2_amplitudes(self):
"""Compute the double amplitudes from the MP2 perturbative method, and
then reorder the elements into a dense list. The single (T1) amplitudes
are set to a small non-zero value. The ordering is single, double
(diagonal), double (non-diagonal).

Returning MP2 amplitudes from Psi4 is not supported in Tangelo
ValentinS4t1qbit marked this conversation as resolved.
Show resolved Hide resolved

Raises:
NotImplementedError: Not implemented at this time"""
raise RuntimeError("Returning MP2 amplitudes from Psi4 is not currently supported in Tangelo")
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved


available_mp2_solvers = {'pyscf': MP2SolverPySCF, 'psi4': MP2SolverPsi4}


def get_mp2_solver(molecule: SecondQuantizedMolecule, solver: Union[None, str, Type[ElectronicStructureSolver]] = default_mp2_solver, **solver_kwargs):
"""Return requested target MP2SolverName object.

Args:
molecule (SecondQuantizedMolecule) : Molecule
solver (string or Type[ElectronicStructureSolver] or None): Supported string identifiers can be found in
available_mp2_solvers (from tangelo.algorithms.classical.mp2_solver). Can also provide a user-defined MP2 implementation
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved
(child to ElectronicStructureSolver class)
solver_kwargs: Other arguments that could be passed to a target. Examples are solver type (e.g. mcscf, mp2), Convergence options etc.
"""
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved

if solver is None:
if isinstance(molecule.solver, IntegralSolverPySCF):
solver = MP2SolverPySCF
elif isinstance(molecule.solver, IntegralSolverPsi4):
solver = MP2SolverPsi4
elif default_mp2_solver is not None:
solver = default_mp2_solver
else:
raise ModuleNotFoundError(f"One of the backends for {list(available_mp2_solvers.keys())} needs to be installed to use MP2Solver"
"without providing a user-defined implementation.")

# If target is a string use target_dict to return built-in backend
elif isinstance(solver, str):
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved
try:
solver = available_mp2_solvers[solver.lower()]
except KeyError:
raise ValueError(f"Error: backend {solver} not supported. Available built-in options: {list(available_mp2_solvers.keys())}")
elif not issubclass(solver, ElectronicStructureSolver):
raise TypeError(f"Target must be a str or a subclass of ElectronicStructureSolver but received class {type(solver).__name__}")

return solver(molecule, **solver_kwargs)


class MP2Solver(ElectronicStructureSolver):
"""Uses the MP2 method to solve the electronic structure problem.

Args:
molecule (SecondQuantizedMolecule) : Molecule
solver (string or Type[ElectronicStructureSolver] or None): Supported string identifiers can be found in
available_mp2_solvers (from tangelo.algorithms.classical.mp2_solver). Can also provide a user-defined MP2 implementation
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved
(child to ElectronicStructureSolver class)
solver_kwargs: Other arguments that could be passed to a target. Examples are solver type (e.g. dfmp2, mp2), Convergence options etc.

Attributes:
solver (Type[ElectronicStructureSolver]): The solver that is used for obtaining the MP2 solution.
"""
def __init__(self, molecule: SecondQuantizedMolecule, solver: Union[None, str, Type[ElectronicStructureSolver]] = default_mp2_solver, **solver_kwargs):
self.solver = get_mp2_solver(molecule, solver, **solver_kwargs)

def simulate(self):
"""Perform the simulation (energy calculation) for the molecule.

Returns:
float: Total MP2 energy.
"""
return self.solver.simulate()

def get_rdm(self):
"""Compute the Full CI 1- and 2-particle reduced density matrices.

Returns:
numpy.array: One-particle RDM.
numpy.array: Two-particle RDM.

Raises:
RuntimeError: If method "simulate" hasn't been run.
JamesB-1qbit marked this conversation as resolved.
Show resolved Hide resolved
"""
return self.solver.get_rdm()

def get_mp2_amplitudes(self):
"""Compute the double amplitudes from the MP2 perturbative method, and
then reorder the elements into a dense list. The single (T1) amplitudes
are set to a small non-zero value. The ordering is single, double
(diagonal), double (non-diagonal).

Returns:
list of float: The electronic excitation amplitudes.
"""
return self.solver.get_mp2_amplitudes()
36 changes: 32 additions & 4 deletions tangelo/algorithms/classical/tests/test_mp2_solver.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@

import numpy as np

from tangelo.algorithms.classical import MP2Solver
from tangelo.algorithms.classical.mp2_solver import MP2Solver, default_mp2_solver
from tangelo.molecule_library import mol_H2_321g, mol_Be_321g, mol_H2_sto3g, mol_H2_sto3g_uhf


Expand All @@ -28,14 +28,15 @@ def test_h2(self):
solver = MP2Solver(mol_H2_321g)
energy = solver.simulate()

self.assertAlmostEqual(energy, -1.14025452, places=6)
self.assertAlmostEqual(energy, -1.14025452, places=3)

@unittest.skipIf("pyscf" != default_mp2_solver, "Test Skipped: Only functions for pyscf \n")
def test_be(self):
"""Test MP2Solver against result from reference implementation (Be)."""

solver = MP2Solver(mol_Be_321g)
energy = solver.simulate()
self.assertAlmostEqual(energy, -14.51026131, places=6)
self.assertAlmostEqual(energy, -14.51026131, places=3)

# Assert energy calculated from RDMs and MP2 calculation are the same.
one_rdm, two_rdm = solver.get_rdm()
Expand All @@ -59,8 +60,9 @@ def test_be_frozen_core(self):
solver = MP2Solver(mol_Be_321g_freeze1)
energy = solver.simulate()

self.assertAlmostEqual(energy, -14.5092873, places=6)
self.assertAlmostEqual(energy, -14.5092873, places=3)

@unittest.skipIf("pyscf" != default_mp2_solver, "Test Skipped: Only functions for pyscf \n")
def test_get_mp2_params_restricted(self):
"""Test the packing of RMP2 amplitudes as initial parameters for coupled
cluster based methods.
Expand All @@ -73,6 +75,32 @@ def test_get_mp2_params_restricted(self):

np.testing.assert_array_almost_equal(ref_params, solver.get_mp2_amplitudes())

@unittest.skipIf("pyscf" != default_mp2_solver, "Test Skipped: Only functions for pyscf \n")
def test_get_mp2_params_unrestricted(self):
"""Test the packing of UMP2 amplitudes as initial parameters for coupled
cluster based methods.
"""

solver = MP2Solver(mol_H2_sto3g_uhf)
solver.simulate()
ref_params = [0., 0., 0.030736]

np.testing.assert_array_almost_equal(ref_params, solver.get_mp2_amplitudes())

@unittest.skipIf("pyscf" != default_mp2_solver, "Test Skipped: Only functions for pyscf \n")
def test_get_mp2_params_restricted(self):
"""Test the packing of RMP2 amplitudes as initial parameters for coupled
cluster based methods.
"""

solver = MP2Solver(mol_H2_sto3g)
solver.simulate()

ref_params = [2.e-05, 3.632537e-02]

np.testing.assert_array_almost_equal(ref_params, solver.get_mp2_amplitudes())

@unittest.skipIf("pyscf" != default_mp2_solver, "Test Skipped: Only functions for pyscf \n")
def test_get_mp2_params_unrestricted(self):
"""Test the packing of UMP2 amplitudes as initial parameters for coupled
cluster based methods.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,8 @@

class SemiEmpiricalSolverTest(unittest.TestCase):

@unittest.skipIf(not is_package_installed("pyscf.semiempirical"), "Test Skipped: pyscf.semiempirical module not available \n")
@unittest.skipIf(not is_package_installed("pyscf") or not is_package_installed("pyscf.semiempirical"),
"Test Skipped: pyscf.semiempirical module not available \n")
def test_mindo3_energy(self):
"""Test MINDO3Solver with pyridine. Validated with:
- MINDO/3-derived geometries and energies of alkylpyridines and the
Expand Down