Skip to content
Snippets Groups Projects
model.py 3.24 KiB
Newer Older
import numpy as np

Kostas Vilkelis's avatar
Kostas Vilkelis committed
from pymf.mf import (
    construct_density_matrix,
from pymf.tb.tb import add_tb, _tb_type
def _check_hermiticity(h):
    for vector in h.keys():
        op_vector = tuple(-1 * np.array(vector))
        op_vector = tuple(-1 * np.array(vector))
        if not np.allclose(h[vector], h[op_vector].conj().T):
            raise ValueError("Tight-binding dictionary must be hermitian.")


def _tb_type_check(tb):
    for count, key in enumerate(tb):
        if not isinstance(tb[key], np.ndarray):
            raise ValueError(
                "Values of the tight-binding dictionary must be numpy arrays"
            )
        shape = tb[key].shape
        if count == 0:
            size = shape[0]
        if not len(shape) == 2:
            raise ValueError(
                "Values of the tight-binding dictionary must be square matrices"
            )
        if not size == shape[0]:
            raise ValueError(
                "Values of the tight-binding dictionary must have consistent shape"
            )
class Model:
    Data class which defines the interacting tight-binding problem.

    Parameters
    ----------
    h_0 :
        Non-interacting hermitian Hamiltonian tight-binding dictionary.
        Interaction hermitian Hamiltonian tight-binding dictionary.
    filling :
        Number of particles in a unit cell.
        Used to determine the Fermi level.

    Notes
    -----
    The interaction h_int must be of density-density type.
    For example in 1D system with ndof internal degrees of freedom,
    h_int[(2,)] = U * np.ones((ndof, ndof)) is a Coulomb repulsion interaction
    with strength U between unit cells separated by 2 lattice vectors, where
    the interaction is the same between all internal degrees of freedom.
    def __init__(self, h_0: _tb_type, h_int: _tb_type, filling: float) -> None:
Kostas Vilkelis's avatar
Kostas Vilkelis committed
        _tb_type_check(h_0)
Kostas Vilkelis's avatar
Kostas Vilkelis committed
        _tb_type_check(h_int)
Kostas Vilkelis's avatar
Kostas Vilkelis committed
        if not isinstance(filling, (float, int)):
            raise ValueError("Filling must be a float or an integer")
        if not filling > 0:
            raise ValueError("Filling must be a positive value")
        self.filling = filling
        _first_key = list(h_0)[0]
        self._ndim = len(_first_key)
Kostas Vilkelis's avatar
Kostas Vilkelis committed
        self._ndof = h_0[_first_key].shape[0]
        self._local_key = tuple(np.zeros((self._ndim,), dtype=int))
        _check_hermiticity(h_0)
        _check_hermiticity(h_int)
    def mfield(self, mf: _tb_type, nk: int = 20) -> _tb_type:
        """Computes a new mean-field correction from a given one.

        Parameters
        ----------
        mf :
            Initial mean-field correction tight-binding dictionary.
            Number of k-points in a grid to sample the Brillouin zone along each dimension.
            If the system is 0-dimensional (finite), this parameter is ignored.
            new mean-field correction tight-binding dictionary.
        rho, fermi_energy = construct_density_matrix(
            add_tb(self.h_0, mf), self.filling, nk
Kostas Vilkelis's avatar
Kostas Vilkelis committed
            meanfield(rho, self.h_int),
Kostas Vilkelis's avatar
Kostas Vilkelis committed
            {self._local_key: -fermi_energy * np.eye(self._ndof)},