.. : PyBEST: Pythonic Black-box Electronic Structure Tool : Copyright (C) 2016-- The PyBEST Development Team : : This file is part of PyBEST. : : PyBEST is free software; you can redistribute it and/or : modify it under the terms of the GNU General Public License : as published by the Free Software Foundation; either version 3 : of the License, or (at your option) any later version. : : PyBEST is distributed in the hope that it will be useful, : but WITHOUT ANY WARRANTY; without even the implied warranty of : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the : GNU General Public License for more details. : : You should have received a copy of the GNU General Public License : along with this program; if not, see : -- .. _hamiltonian_activespace: Defining an Active Space Hamiltonian #################################### PyBEST also supports the feature of generating an active space Hamiltonian, that is, an electronic Hamiltonian defined for a specific subsets of molecular orbitals. The :py:func:`~pybest.orbital_utils.split_core_active` function performs several operations: 1. Transform AO integrals into the MO basis (if required) 2. Calculcate active space Hamiltonian When calculating an active-space Hamitlonian, the core energy as well as the one-electron integrals need to be updated, while the two-electron integrals are simply 'sliced'. Specifically, the one-electron integrals for the active space :math:`\tilde{h}_{pq}` are calculated from .. math:: \tilde{h}_{pq} = h_{pq} + \sum_{i \in \textrm{ncore}} ( 2 \langle pi \vert qi \rangle - \langle pi \vert iq \rangle), where :math:`h_{pq}` is an element of the original one-electron integrals and :math:`\langle pi \vert qi \rangle` are the original two-electron integral in physicist's notation. The core energy of the active space is determined from .. math:: E_\text{core} = E_\text{nn} + 2\sum_{i \in \textrm{ncore}} t_{ii} + \sum_{i, j \in \textrm{ncore}} (2 \langle ij \vert ij \rangle - \langle ij \vert ji \rangle) where :math:`E_\text{nn}` is the original core energy (for instance the nuclear-nuclear repulsion term). The sums over the one- and two-electron integrals run over the frozen core orbitals only. The updated two-electron integrals :math:`\langle pq \vert rs \rangle` contain only the elements corresponding to the active orbital indices :math:`p,q,r,s`. Defining the molecular Hamiltonian for an active orbital space -------------------------------------------------------------- An active space can be defined as follows, where the integrals are transformed into the molecular orbital basis, .. code-block:: python # Define an active space with ncore=10 frozen core orbitals and nactive=24 # active orbitals cas = split_core_active(kin, ne, er, orb, e_core=external, ncore=10, nactive=24) # all one-electron integrals one = cas.one # all two-electron integrals two = cas.two # the core energy e_core = cas.e_core where ``kin``, ``ne``, and ``er`` are the original one- and two-electron integrals, respectively (see also :ref:`hamiltonian_transformation`), ``external`` is the original core energy (see also :ref:`user_molecularham_matrix_elements`), ``orb`` are the molecular orbitals that transform the AO integrals into the MO basis, and ``ncore`` (``nactive``) is the number of **frozen core** (**active**) orbitals. The active space Hamiltonian is stored/returned as an instance of :py:class:`~pybest.io.iodata.IOData`. .. note:: The core energy ``e_core`` as well as number of core and active orbitals has to be passed as a **keyword** argument using ``e_core``, ``ncore``, or ``nactive``, respectively. If you want to calculate the active space Hamiltonian for a specific method (that is, for a specific set of orbitals and method specific core energy), you have to pass the corresponding method output (an instance of the :py:class:`~pybest.io.iodata.IOData` class) to the :py:func:`~pybest.orbital_utils.split_core_active` function (in addition to the Hamiltonian). PyBEST will assign all quantities automatically, .. code-block:: python # Define an active space with ncore=10 frozen core orbitals and nactive=24 # active orbitals from a pCCD wave function stored in pccd_out cas = split_core_active(kin, ne, er, pccd_out, ncore=10, nactive=24) # all one-electron integrals one = cas.one # all two-electron integrals two = cas.two # the core energy e_core = cas.e_core Thus, only all one- and two-electron integrals as well as the return value of the desired QC method are required. Note that the :py:class:`~pybest.io.iodata.IOData` container has to contain an ``e_core`` attribute. By default, all QC methods implemented in PyBEST return their corresponding core energy. The :py:class:`~pybest.io.iodata.IOData` container can be overwritten using arguments or keyword arguments. However, we recommend this option for experienced users only. .. note:: The order of the arguments does not matter. PyBEST will automatically recognize all Hamiltonian terms, orbitals, and external energy contributions. If the AO/MO transformation of the one- and two-electron integrals should be skipped, do no pass the orbitals as argument, .. code-block:: python # Define an active space with 10 frozen core orbitals and 24 active orbitals # Skip the AO/MO transformation step cas = split_core_active(kin, ne, er, e_core=external, ncore=10, nactive=24) .. note:: The :py:func:`~pybest.orbital_utils.split_core_active` function supports only restricted orbitals. Dumping the active space Hamiltonian to disk -------------------------------------------- Once the active space Hamiltonian has been defined, it can be dump to disk (see :ref:`hamiltonian_io` for more information). In PyBEST, the active space integrals can be either stored using the internal file format (binary) .. code-block:: python # Dump active space integrals stored in "cas" to disk using .h5 format cas.to_file('cas_hamiltonian.h5') or the FCIDUMP format (ASCII) .. code-block:: python # Dump active space integrals stored in "cas" to disk using the FCIDUMP format cas.to_file('cas_hamiltonian.FCIDUMP')