.. : 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 : : -- .. _user_wroking_with_orbitals: Working with Orbitals in PyBEST ################################ The (molecular) orbitals in PyBEST contain information on the AO/MO coefficient matrix, the orbital occupation numbers, and the orbital energies. Internally, the orbitals are an instance of the :py:class:`~pybest.linalg.dense.DenseOrbital` class. Specifically, the AO/MO coefficients, occupation numbers, and energies are attributes of the :py:class:`~pybest.linalg.dense.DenseOrbital` class. In the following, some simple orbital manipulations are explained that might be useful for users. Creating an empty set of orbitals ================================= Prior to any wave function calculation, the orbitals need to be created as an instance of the :py:class:`~pybest.linalg.dense.DenseOrbital` class .. code-block:: python # Create an empty set of orbitals orb_a = lf.create_orbital() where ``lf`` is an instance of the :py:class:`~pybest.linalg.LinalgFactory` class, where all linear algebra and tensor contraction operations are defined. Printing orbital information ============================ The AO/MO coefficient matrix, the orbital occupation numbers, and the orbital energies can be accessed as follows (assuming ``orb`` to be an instance of :py:class:`~pybest.linalg.dense.DenseOrbital`) .. code-block:: python # Print AO/MO coefficients print(orb.coeffs) # Print occupations print(orb.occupations) # Print orbital energies print(orb.energies) The AO/MO coefficients can also be printed in a more readable format using the :py:meth:`~pybest.orbital_utils.print_ao_mo_coeffs` function .. code-block:: python # Print all AO/MO coefficients in a more readable format print_ao_mo_coeffs(gobasis, orb) where ``gobasis`` is an instance of the ``GOBasis`` class. If only a subset of orbitals is required, this can be achieved using optional arguments .. code-block:: python # Print AO/MO coefficients in a more readable format for orbitals # no. 4 to no. 31 (including) print_ao_mo_coeffs(gobasis, orb, begin=4, end=31) where ``begin`` and ``end`` are the first and last index, respectively, of the orbital to be printed. Note that the orbitals are indexed using Fortran indexing, that is, beginning with 1. The order of the AOs is according to the list of atoms in the geometry input. In the output, each column corresponds to one molecular orbital. .. note:: In all electronic structure methods that optimize the orbitals, the current orbitals are always overwritten with the most recent ones. To save a copy of the current orbitals during the execution of the PyBEST code, use the :py:meth:`~pybest.linalg.DenseOrbital.copy` method of :py:class:`~pybest.linalg.dense.DenseOrbital`, .. code-block:: python # Create a true copy of the current orbitals orb_output = orb.copy() Writing orbitals to a checkpoint file ===================================== In PyBEST, all electronic structure methods save their own copy of the molecular orbitals and dump them in their method-dependent checkpoint files. In case, you want to write the molecular orbitals to a separate file at any stage of the program (represented in PyBEST's internal format), you can use the :py:class:`~pybest.io.iodata.IOData` container (see also :ref:`user_iodata` for more details on how to use the container), .. code-block:: python # Dump orbitals for restart mol = IOData(olp=olp, orb_a=orb) mol.to_file('checkpoint.h5') where the overlap matrix is stored in ``olp`` and the orbitals in ``orb_a``. The molecular orbitals (for restricted orbitals or alpha spin) are always associated with ``orb_a``, while the overlap matrix is associated with ``olp``. In case of unrestricted orbitals, PyBEST uses the attribute name ``orb_b``. If the user chooses a different naming convention, PyBEST might run into unexpected errors (for instance, during restarts). See also :ref:`user_iodata` for more information on how to use the IOData container and all default variable names. .. note:: To use previously dumped orbitals for restart purposes, the overlap matrix has to be written to file as well. .. _user_orbitals_import: Reading orbitals from a checkpoint file ======================================= Most electronic structure methods in PyBEST feature a restart option, which reads and projects some previously stored orbitals automatically. The documentation below is mentioned for reasons of completeness to allow the user to manually read in orbitals from disk. .. _orbitals_restart_pybest_format: PyBEST's internal format ------------------------- Similar to the recipe of dumping molecular orbitals to file, they can be read from disk using the :py:class:`~pybest.io.iodata.IOData` container functionality. First, some previously stored PyBEST checkpoint file that contains the molecular orbitals and overlap integrals has to be read in, .. code-block:: python # Read some PyBEST checkpoint file and store the corresponding data in 'old' old = IOData.from_file('checkpoint.h5') The molecular orbitals and overlap matrix can be accessed from ``old`` as ``old.orb_a`` and ``old.olp``, respectively. If the molecular geometry of the restart orbitals differs from the current molecular geometry, the molecular orbitals have to be projected against the current molecular orbitals ``orb`` and current overlap matrix ``olp``, .. code-block:: python # Project orbitals read from file against current orbitals project_orbitals(old.olp, olp, old.orb_a, orb) The :py:meth:`~pybest.orbital_utils.project_orbitals` function overwrites ``orb`` and ``olp`` with the projected restart orbitals and overlap matrix, respectively. .. note:: If the restart orbitals are obtained for a different molecular geometry, the :py:meth:`~pybest.orbital_utils.project_orbitals` function has to be used, otherwise all electronic structure calculations will be corrupted. External file format -------------------- Orbital information can also be imported from external programs. Supported file formats are ``.mkl`` and ``.molden``. Similar to the procedure mentioned above (:ref:`orbitals_restart_pybest_format`), the :py:class:`~pybest.io.iodata.IOData` container is used to read in some external file. The file extension is handled automatically. .. code-block:: python # Load molden file file mol = IOData.from_file('water.molden') # Print the total number of molecular orbitals print(mol.orb_a.nfn) Similarly, if the orbitals are not projected onto a new basis set, the Hamiltonian has to be recomputed for the external basis set stored as ``mol.gobasis``. The mol instance has the following attributes :mol.gobasis: A :py:class:`~pybest.gbasis.pybasis.Basis` instance. It contains all basis set information. :mol.orb_a: A :py:class:`~pybest.linalg.dense.DenseOrbital` instance containing the molecular orbitals for alpha spin :mol.orb_b: A :py:class:`~pybest.linalg.dense.DenseOrbital` instance containing the molecular orbitals for beta spin (if present) :mol.coordinates: A np.array containing the xyz coordinates :mol.atom: A list of strings with the element names .. _orbitals_rotations: Rotating orbitals ================= The molecular orbitals can be perturbed by various rotations. The user has the choice between (a series) of 2-by-2 rotations or a random rotation of all molecular orbitals. Givens rotation (2-by-2 rotations) ---------------------------------- Two molecular orbitals can be rotated using the :py:meth:`~pybest.linalg.dense.DenseOrbital.rotate_2orbitals` method. Depending on the angle of rotation, the selected orbitals are either mixed or swapped. By default, the function rotates the HOMO and LUMO orbitals by 45 degrees. .. code-block:: python # Mix HOMO and LUMO orbitals orb.rotate_2orbitals() If a different pair of orbitals is to be rotated, the corresponding orbital indices can be specified as optional arguments, .. code-block:: python # Rotate 7th and 12th orbital by 60 deg; note zero-based indexing orb.rotate_2orbitals(60*deg, 6, 11) Keep in mind that PyBEST uses radians as unit for angles, that is, ``60*deg == 60*np.pi/180``. Since :py:meth:`~pybest.linalg.dense.DenseOrbital.rotate_2orbitals` is a class method (defined in :py:class:`~pybest.linalg.dense.DenseOrbital`), all quantities use Python-based indexing, starting with 0. The :py:meth:`~pybest.linalg.dense.DenseOrbital.rotate_2orbitals` method can be applied in sequence if several orbitals are to be rotated with each other, .. code-block:: python # Perform series of rotations: 7/12 by 60 degree, 8/14 by 20 degree, 9/26 # by 80 degree orb.rotate_2orbitals(60*deg, 6, 11) orb.rotate_2orbitals(20*deg, 7, 13) orb.rotate_2orbitals(80*deg, 8, 25) Note that an angle of 90 degrees is equivalent to swapping the corresponding orbitals. This functionality can also be achieved using the :py:meth:`~pybest.linalg.dense.DenseOrbital.swap_orbitals` method (see :ref:`orbitals_swapping` below). Random orbital rotations ------------------------ All molecular orbitals can be perturbed simultaneously using the :py:meth:`~pybest.linalg.dense.DenseOrbital.rotate_random` method. This function method performs a *random unitary rotation* to **all** molecular orbitals. .. code-block:: python # perform random unitary rotation of orbitals orb orb.rotate_random() .. note:: All orbital rotation operations leave the occupation numbers and orbital energies unchanged. Only the MO coefficients are perturbed. The AOs remain always untouched. .. _orbitals_swapping: Swapping orbitals ================= Two molecular orbitals can be swapped using the :py:meth:`~pybest.linalg.dense.DenseOrbital.swap_orbitals` method. The list of orbital swaps is passed as a 2-dimensional numpy array, where each row corresponds to one transposition of orbitals. Since :py:meth:`~pybest.linalg.dense.DenseOrbital.swap_orbitals` is again a class method, the indexing starts with 0. .. code-block:: python # Swap some orbitals; note zero-based indexing swaps = np.array([[0, 6], [5, 9]]) orb.swap_orbitals(swaps) By default, the orbital occupations and orbital energies are swapped as well. In some cases (like in SCF calculations), the swapping of orbital occupations and/or orbital energies should be suppressed to provide a perturbed guess for the molecular orbitals. These features can be switched on using optional arguments, .. code-block:: python # Swap only AO/MO coeffients and leave occupations and energies unchanged; # note zero-based indexing swaps = np.array([[0, 6], [5, 9]]) orb.swap_orbitals(swaps, skip_occs=True, skip_energies=True) .. _user_orbitals_localization: Orbital localization ==================== The current version of PyBEST supports only Pipek-Mezey localization. For more details on Pipek-Mezey localization please refer to :ref:`localization`. Due to reasons of completeness, we will only briefly explain the syntax on how to perform a localization of molecular orbitals. In the Pipek-Mezey localization algorithm, the occupied and virtual block of the (canoncial Hartree-Fock) orbitals are localized using Mulliken projectors. In order to perform a localization, the Mulliken projectors need to be determined. To do so, an instance of the :py:class:`~pybest.gbasis.pybasis.Basis` class (here ``gobasis``) and of the :py:class:`~pybest.linalg.base.LinalgFactory` class (here ``lf``) are required. .. literalinclude:: ../src/pybest/data/examples/localization/water_pm.py :lines: 46-53 As soon as the Mulliken projectors are calculated, they can be passed to the :py:class:`~pybest.localization.localizaton.PipekMezey` class. The occupied and virtual blocks can be localized individually by two separate function calls, .. literalinclude:: ../src/pybest/data/examples/localization/water_pm.py :lines: 54-61 Example Python scripts ---------------------- The following, we list a complete example script for a restricted Hartree-Fock calculation of the water molecule followed by orbital localization. .. literalinclude:: ../src/pybest/data/examples/localization/water_pm.py :caption: data/examples/localization/water_pm.py :lines: 3-45 .. _user_orbitals_molden: Generating molden files ======================= PyBEST uses libint2 to dump molden files. As all I/O operations, the generation of molden files is handled by the :py:class:`~pybest.io.iodata.IOData` container class. The corresponding class instance has to contain an instance of the orbitals, for which the molden file has to be generated, as well as an instance of the :py:class:`~pybest.gbasis.pybasis.Basis` class. The latter contains all necessary information to dump the molden file. .. literalinclude:: ../src/pybest/data/examples/localization/water_pm.py :lines: 46- Specifying the file ending ``molden`` tells the :py:meth:`~pybest.io.iodata.IOData.to_file` method to dump a molden file to disk. In the case of unrestricted orbitals, both alpha and beta orbitals have to be passed to the :py:class:`~pybest.io.iodata.IOData` container, .. code-block:: python molden = IOData(gobasis=gobasis, orb_a=orb_a, orb_b=orb_b )