Source code for pennylane.qchem.molecule
# Copyright 2018-2021 Xanadu Quantum Technologies Inc.
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
# http://www.apache.org/licenses/LICENSE-2.0
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""
This module contains functions and classes to create a
:class:`~pennylane.qchem.molecule.Molecule` object. This object stores all
the necessary information to perform a Hartree-Fock calculation for a given molecule.
"""
import collections
# pylint: disable=too-few-public-methods, too-many-arguments, too-many-instance-attributes
import itertools
import warnings
import pennylane as qml
from .basis_data import atomic_numbers
from .basis_set import BasisFunction, mol_basis_data
from .integrals import contracted_norm, primitive_norm
# Bohr-Angstrom correlation coefficient (https://physics.nist.gov/cgi-bin/cuu/Value?bohrrada0)
bohr_angs = 0.529177210903
[docs]class Molecule:
r"""Create a molecule object that stores molecular information and default basis set parameters.
The molecule object can be passed to functions that perform a Hartree-Fock calculation.
Args:
symbols (list[str]): Symbols of the atomic species in the molecule. Currently, atoms with
atomic numbers 1-10 are supported.
coordinates (array[float]): 1D array with the atomic positions in Cartesian coordinates. The
coordinates must be given in atomic units and the size of the array should be ``3*N``
where ``N`` is the number of atoms.
charge (int): net charge of the molecule
mult (int): Spin multiplicity :math:`\mathrm{mult}=N_\mathrm{unpaired} + 1` for
:math:`N_\mathrm{unpaired}` unpaired electrons occupying the HF orbitals.
basis_name (str): Atomic basis set used to represent the molecular orbitals. Currently, the
only supported basis sets are ``STO-3G``, ``6-31G``, ``6-311G`` and ``CC-PVDZ``. Other
basis sets can be loaded from the basis-set-exchange library using ``load_data``.
load_data (bool): flag to load data from the basis-set-exchange library
l (tuple[int]): angular momentum quantum numbers of the basis function
alpha (array[float]): exponents of the primitive Gaussian functions
coeff (array[float]): coefficients of the contracted Gaussian functions
r (array[float]): positions of the Gaussian functions
normalize (bool): if True, the basis functions get normalized
unit (str): unit of atomic coordinates. Available options are ``unit="bohr"`` and ``unit="angstrom"``.
.. note::
:class:`~.qchem.Molecule` is not currently compatible with :func:`~.qjit` and ``jax.jit``.
**Example**
Import necessary modules:
>>> from pennylane import numpy as np
>>> from pennylane.qchem import Molecule
Define molecular symbols and geometry:
>>> symbols = ['H', 'H']
>>> geometry = np.array([[0.0, 0.0, -0.694349],
... [0.0, 0.0, 0.694349]], requires_grad = True)
>>> mol = Molecule(symbols, geometry)
>>> print(mol.n_electrons)
2
"""
def __init__(
self,
symbols,
coordinates,
charge=0,
mult=1,
basis_name="sto-3g",
name="molecule",
load_data=False,
l=None,
alpha=None,
coeff=None,
normalize=True,
unit="bohr",
):
if (
basis_name.lower()
not in [
"sto-3g",
"6-31g",
"6-311g",
"cc-pvdz",
]
and load_data is False
):
raise ValueError(
"Currently, the only supported basis sets are 'sto-3g', '6-31g', '6-311g' and "
"'cc-pvdz'. Please consider using `load_data=True` to download the basis set from "
"an external library that can be installed with: pip install basis-set-exchange."
)
if set(symbols) - set(atomic_numbers):
raise ValueError(f"Atoms in {set(symbols) - set(atomic_numbers)} are not supported.")
self.symbols = symbols
self.coordinates = coordinates
self.unit = unit.strip().lower()
self.charge = charge
self.mult = mult
self.basis_name = basis_name.lower()
self.name = name
self.load_data = load_data
self.n_basis, self.basis_data = mol_basis_data(self.basis_name, self.symbols, load_data)
if self.unit not in ("angstrom", "bohr"):
raise ValueError(
f"The provided unit '{unit}' is not supported. "
f"Please set 'unit' to 'bohr' or 'angstrom'."
)
if self.unit == "angstrom":
self.coordinates = self.coordinates / bohr_angs
self.nuclear_charges = [atomic_numbers[s] for s in self.symbols]
self.n_electrons = sum(self.nuclear_charges) - self.charge
if l is None:
l = [i[0] for i in self.basis_data]
use_jax = any(qml.math.get_interface(x) == "jax" for x in [coordinates, alpha, coeff])
interface_args = [{"like": "autograd", "requires_grad": False}, {"like": "jax"}][use_jax]
if alpha is None:
alpha = [qml.math.array(i[1], **interface_args) for i in self.basis_data]
if coeff is None:
coeff = [qml.math.array(i[2], **interface_args) for i in self.basis_data]
if normalize:
coeff = [
qml.math.array(c * primitive_norm(l[i], alpha[i]), **interface_args)
for i, c in enumerate(coeff)
]
if (
len(
{
qml.math.get_deep_interface(x)
for x in [coordinates, alpha, coeff]
if qml.math.get_deep_interface(x) != "numpy"
}
)
> 1
):
warnings.warn(
"The parameters coordinates, coeff, and alpha are not of the same interface. Please use the same interface for all 3 or there may be unintended behavior.",
UserWarning,
)
r = list(
itertools.chain(
*[[self.coordinates[i]] * self.n_basis[i] for i in range(len(self.n_basis))]
)
)
self.l = l
self.alpha = alpha
self.coeff = coeff
self.r = r
self.basis_set = [
BasisFunction(self.l[i], self.alpha[i], self.coeff[i], self.r[i]) for i in range(len(l))
]
self.n_orbitals = len(self.l)
self.mo_coefficients = None
self.normalize = normalize
def __repr__(self):
"""Returns the molecule representation in string format"""
elements, counter, flags = set(self.symbols), collections.Counter(self.symbols), []
if counter["C"]: # Hill Notation
flags = ["C", "H"] if counter["H"] else ["C"]
ordered_elems = flags + list(sorted(elements.difference(set(flags))))
formula = "".join([x + str(counter[x]) if counter[x] > 1 else x for x in ordered_elems])
return f"<Molecule = {formula}, Charge: {self.charge}, Basis: {self.basis_name.upper()}, Orbitals: {self.n_orbitals}, Electrons: {self.n_electrons}>"
[docs] def atomic_orbital(self, index):
r"""Return a function that evaluates an atomic orbital at a given position.
Args:
index (int): index of the atomic orbital, order follwos the order of atomic symbols
Returns:
function: function that computes the value of the orbital at a given position
**Example**
>>> symbols = ['H', 'H']
>>> geometry = np.array([[0.0, 0.0, 0.0], [0.0, 0.0, 1.0]], requires_grad = False)
>>> mol = qml.qchem.Molecule(symbols, geometry)
>>> ao = mol.atomic_orbital(0)
>>> ao(0.0, 0.0, 0.0)
0.62824688
"""
l = self.basis_set[index].l
alpha = self.basis_set[index].alpha
coeff = self.basis_set[index].coeff
r = self.basis_set[index].r
coeff = coeff * contracted_norm(l, alpha, coeff)
lx, ly, lz = l
def orbital(x, y, z):
r"""Evaluate a basis function at a given position.
Args:
x (float): x component of the position
y (float): y component of the position
z (float): z component of the position
Returns:
array[float]: value of a basis function
"""
c = ((x - r[0]) ** lx) * ((y - r[1]) ** ly) * ((z - r[2]) ** lz)
e = qml.math.exp(
qml.math.tensordot(
-alpha, (x - r[0]) ** 2 + (y - r[1]) ** 2 + (z - r[2]) ** 2, axes=0
)
)
return c * qml.math.dot(coeff, e)
return orbital
[docs] def molecular_orbital(self, index):
r"""Return a function that evaluates a molecular orbital at a given position.
Args:
index (int): index of the molecular orbital
Returns:
function: function to evaluate the molecular orbital
**Example**
>>> symbols = ['H', 'H']
>>> geometry = np.array([[0.0, 0.0, 0.0], [0.0, 0.0, 1.0]], requires_grad = False)
>>> mol = qml.qchem.Molecule(symbols, geometry)
>>> qml.qchem.scf(mol)() # run scf to obtain the optimized molecular orbitals
>>> mo = mol.molecular_orbital(1)
>>> mo(0.0, 0.0, 0.0)
0.01825128
"""
# molecular coefficients are set by other modules
c = self.mo_coefficients[index] # pylint:disable=unsubscriptable-object
def orbital(x, y, z):
r"""Evaluate a molecular orbital at a given position.
Args:
x (float): x component of the position
y (float): y component of the position
z (float): z component of the position
Returns:
array[float]: value of a molecular orbital
"""
m = 0.0
for i in range(self.n_orbitals):
m = m + c[i] * self.atomic_orbital(i)(x, y, z)
return m
return orbital
_modules/pennylane/qchem/molecule
Download Python script
Download Notebook
View on GitHub