..
    : 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 <http://www.gnu.org/licenses/>
    :
    : --

.. _user_posthf_intro:

General Remarks concerning Post-Hartree-Fock Calculations
#########################################################

All post-Hartree-Fock modules in PyBEST have a similar input structure.
To optimize the wave function of any post-Hartree-Fock method, the corresponding
module requires a Hamiltonian and some orbitals
(either an AO/MO coefficient matrix or an MO/MO coefficient matrix)
including the overlap matrix (for restart purposes) as input arguments.
In the following, we will discuss some general aspects on how to perform
post-Hartree-Fock calculations in PyBEST.
We recommend that all users read the previous sections of the User
Documentation to be familiar with the basic input structure and the variable
names used throughout the PyBEST Documentation
to label Hamiltonians, orbitals, occupation models, etc.

.. note::

    See :ref:`user_hamiltonian` for how to define the molecular geometry,
    basis set, or the Hamiltonian.

.. note::

    See :ref:`user_orbitals_occupations` for how to define orbitals and
    the occupation models (which also determines the total charge)

Typically, a post-Hartree-Fock calculation requires a Hartree-Fock calculation
as initial step (exceptions are, for instance, restarts).  

.. note::

    See :ref:`user_hf` for how to perform a Hartree-Fock calculation.

PyBEST uses fixed labels/names for the one- and two-electron integrals,
orbitals, density matrices, etc., which are summarized in :ref:`user_iodata`.
For reasons of consistency, we recommend not to change those labels/names
if you intend to directly manipulate the :py:class:`~pybest.io.iodata.IOData`
container. However, if you manipulate data outside the
:py:class:`~pybest.io.iodata.IOData` container, you are free to choose
names/labels as you want.


General input structure
=======================

To set up a post-Hartree-Fock calculation, you have to perform similar steps
as in a simple Hartree-Fock calculation.
If some Hamiltonian (see for instance
:ref:`user_molecularham_matrix_elements` or :ref:`modphysham`
or :ref:`hamiltonian_io_fcidump`) and some initial orbitals are present
(for instance from a previous Hartree-Fock calculation, see :ref:`user_scf_hf`),
a post-Hartree-Fock calculations can be initialized as follows

.. code-block:: python

    # Create an instance of a post-Hartree-Fock method (here called PostHF)
    posthf = PostHF(lf, occ_model)
    # Start the calculation through a function call using the results of
    # a previous RHF calculation stored in hf_output
    posthf_output = posthf(kin, ne, eri, hf_output)

In the above example, ``PostHF`` is some post-Hartree-Fock flavor (see the
next sections for more details). The arguments are described in
:ref:`user_scf_preliminaries`.
Thus, the input structure is similar to a Hartree-Fock calculation, except that
only the Hamiltonian elements have to be passed explicitly.
The orbitals and overlap (and the external energy term)
are passed using the :py:class:`~pybest.io.iodata.IOData` container ``hf_output``,
which stores the results of a previous wave function calculation like RHF (as in this
example).

.. note::

    The :py:class:`~pybest.io.iodata.IOData` container has lowest precedence and (mostly)
    all wave function information can be overwritten using arguments or keyword
    arguments. For reasons of robustness, however, we recommend this option only for
    experienced users. Thus, stick to the :py:class:`~pybest.io.iodata.IOData` container
    whenever possible.

If external one- and two-electron integrals are read from a file (assuming the
FCIDUMP file format), all one-electron integrals are combined to one single
term (see :ref:`hamiltonian_io_fcidump`). Furthermore, both the overlap integrals
and the orbitals are stored in the :py:class:`~pybest.io.iodata.IOData` container.
The function call of a post-Hartree-Fock method changes then to

.. code-block:: python

    # Read external integrals stored in FCIDUMP
    ints = IOData.from_file('FCIDUMP')

    # Create an instance of a post-Hartree-Fock method (here called PostHF)
    posthf = PostHF(lf, occ_model)
    # Start the calculation through a function call 
    posthf_output = posthf(ints.one, ints.two, ints)

Since the FCIDUMP file represents the integrals in an orthonormal basis, the
overlap matrix and molecular orbitals that need to be passed to the function
call are unity matrices (see also next section).

.. note::

    The Hamiltonian, here labeled as ``one`` and ``two``, has to be passed explicitly.


Defining the orbitals for post-Hartree-Fock calculations
========================================================

In all post-Hartree-Fock calculations, the molecular orbitals have to be passed
as the optimization of the wave function is performed in the molecular orbital
basis.
The AO/MO transformation is handled internally and the user does **not** need
to transform the integrals explicitly (see :ref:`hamiltonian_transformation` for
more information).

Typically, previously optimized Hartree-Fock orbitals are a convenient choice
for any post-Hartree-Fock calculation (see :ref:`user_hf`).
Other choices, however, are also possible. In the following, some examples are
listed.

    1. Orbitals from a previous SCF optimization (:ref:`user_hf`)
    2. Localized orbitals (:ref:`user_orbitals_localization`)
    3. External orbitals read from file (:ref:`user_orbitals_import`)
    4. Rotated or swapped orbitals (:ref:`orbitals_rotations` and :ref:`orbitals_swapping`)
    5. Molecular orbitals corresponding to some external Hamiltonian (read from
       the :ref:`hamiltonian_io_fcidump`)

In the last case, all integrals are expressed in an orthonormal basis. The
corresponding molecular orbitals and overlap matrix that need to be passed
to the post-Hartree-Fock method are equivalent to a unitary matrix. They are
created when reading in the FCIDUMP file and can be accessed as attributes of
the :py:class:`~pybest.io.iodata.IOData` container,

.. code-block:: python

    # Read external integrals stored in FCIDUMP
    ints = IOData.from_file('FCIDUMP')

    # Access the orbitals and overlap matrix as attributes
    orb_a = ints.orb_a
    olp = ints.olp

.. note::

    Rotating or swapping pairs of orbitals is only recommended if the orbitals
    are optimized in the post-Hartree-Fock method.


.. _user_posthf_iodata:

The return value of a post-Hartree-Fock calculation
===================================================

All post-Hartree-Fock methods return an instance of the :py:class:`~pybest.io.iodata.IOData`
container class (similar to the RHF/UHF modules).
Its attributes are the most important results of a calculation and may include
the orbitals (including the overlap matrix), N-particle reduced density matrices,
total and correlation energies, etc.
This attributes are labeled according the :py:class:`~pybest.io.iodata.IOData`
standard (see also :ref:`user_iodata`). In the following, we will mention some
examples, assuming ``posthf_output`` to be the :py:class:`~pybest.io.iodata.IOData`
container.

    :posthf_output.e_tot:    The total energy (including all external terms)
    :posthf_output.e_corr:   The total correlation energy
    :posthf_output.e_corr_s: The correlation energy corresponding to single excitations
    :posthf_output.e_corr_d: The correlation energy corresponding to double excitations
    :posthf_output.orb_a:    The molecular orbitals used in the post-Hartree-Fock method
                             (they represent a deep copy of the initial orbitals)
    :posthf_output.olp:      The overlap integrals (stored for restart purposes)
    :posthf_output.dm_1:     The 1-particle reduced density matrix (dm_1_a/dm_1_b are
                             used solely by the unrestricted SCF module)
    :posthf_output.dm_2:     The 2-particle reduced density matrix
    :posthf_output.t_1:      The wave function amplitudes for single excitations 
    :posthf_output.t_p:      The wave function amplitudes for pair excitations 
    :posthf_output.t_2:      The wave function amplitudes for (all) double excitations 
    :posthf_output.l_1:      The :math:`\lambda` amplitudes for single (de-)excitations 
    :posthf_output.l_p:      The :math:`\lambda` amplitudes for pair (de-)excitations 
    :posthf_output.l_2:      The :math:`\lambda` amplitudes for (all) double (de-)excitations 


.. _user_posthf_checkpoint:

Checkpoint files of a post-Hartree-Fock calculation
===================================================

All post-Hartree-Fock methods dump their own checkpoint files to disk.
PyBEST stores all restartable files to its own results directory called
**pybest-results**.
Those checkpoint files contain the most recent results of the corresponding
post-Hartree-Fock method.
They can be used for later restarts or post-processing.
They are typically named as ``checkpoint_*someposthf*.h5``.

.. note::

    You can change the default result directory on-the-fly when executing a PyBEST
    script. To update this directory, the ``filemanager`` instance of the
    :py:class:`~pybest.filemanager.FileManager` class has to be used:

    .. code-block:: python

        # Change result directory to 'my_results'
        filemanger.result_dir = my_results

        [do some stuff]

        # Change result directory to 'my_new_results'
        filemanger.result_dir = my_new_results