# coding: utf-8
"""History file with structural relaxation results."""
from __future__ import annotations
import os
import numpy as np
import pandas as pd
import pymatgen.core.units as units
import abipy.core.abinit_units as abu
from collections import OrderedDict
from monty.functools import lazy_property
from monty.collections import AttrDict
from monty.string import marquee, list_strings
from pymatgen.core.periodic_table import Element
from pymatgen.analysis.structure_analyzer import RelaxationAnalyzer
from abipy.tools.plotting import add_fig_kwargs, get_ax_fig_plt, get_axarray_fig_plt, set_visible, get_figs_plotly, \
get_fig_plotly, add_plotly_fig_kwargs, plotlyfigs_to_browser, push_to_chart_studio, PlotlyRowColDesc, plotly_set_lims, \
latex_greek_2unicode
from abipy.core.structure import Structure
from abipy.core.mixins import AbinitNcFile, NotebookWriter
from abipy.abio.robots import Robot
from abipy.iotools import ETSF_Reader
from abipy.tools.typing import Figure
[docs]
class HistFile(AbinitNcFile, NotebookWriter):
"""
File with the history of a structural relaxation or molecular dynamics calculation.
Usage example:
.. code-block:: python
with HistFile("foo_HIST") as hist:
hist.plot()
.. rubric:: Inheritance Diagram
.. inheritance-diagram:: HistFile
"""
[docs]
@classmethod
def from_file(cls, filepath: str) -> HistFile:
"""Initialize the object from a netcdf_ file"""
return cls(filepath)
def __init__(self, filepath: str):
super().__init__(filepath)
self.reader = self.r = HistReader(filepath)
[docs]
def close(self) -> None:
"""Close the file."""
self.r.close()
[docs]
@lazy_property
def params(self) -> dict:
"""dict with parameters that might be subject to convergence studies."""
return {}
def __str__(self) -> str:
return self.to_string()
#def read_structures(self, index: str):
# TODO: Add more metadata.
#@lazy_property
#def nsppol(self):
# """Number of independent spins."""
# return self.r.read_dimvalue("nsppol")
#@lazy_property
#def nspden(self):
# """Number of independent spin densities."""
# return self.r.read_dimvalue("nspden")
#@lazy_property
#def nspinor(self):
# """Number of spinor components."""
# return self.r.read_dimvalue("nspinor")
[docs]
@lazy_property
def final_energy(self) -> float:
"""Total energy in eV of the last iteration."""
return self.etotals[-1]
[docs]
@lazy_property
def final_pressure(self) -> float:
"""Final pressure in Gpa."""
cart_stress_tensors, pressures = self.r.read_cart_stress_tensors()
return pressures[-1]
#@lazy_property
#def final_max_force(self):
[docs]
def get_fstats_dict(self, step) -> AttrDict:
"""
Return |AttrDict| with stats on the forces at the given ``step``.
"""
# [time, natom, 3]
var = self.r.read_variable("fcart")
forces = units.ArrayWithUnit(var[step], "Ha bohr^-1").to("eV ang^-1")
fmods = np.array([np.linalg.norm(force) for force in forces])
return AttrDict(
fmin=fmods.min(),
fmax=fmods.max(),
fmean=fmods.mean(),
fstd=fmods.std(),
drift=np.linalg.norm(forces.sum(axis=0)),
)
[docs]
def to_string(self, verbose=0, title=None) -> str:
"""String representation."""
lines = []; app = lines.append
if title is not None: app(marquee(title, mark="="))
app(marquee("File Info", mark="="))
app(self.filestat(as_string=True))
app("")
app(self.initial_structure.to_string(verbose=verbose, title="Initial Structure"))
app("")
app("Number of relaxation steps performed: %d" % self.num_steps)
app(self.final_structure.to_string(verbose=verbose, title="Final structure"))
app("")
an = self.get_relaxation_analyzer()
app("Volume change in percentage: %.2f%%" % (an.get_percentage_volume_change() * 100))
d = an.get_percentage_lattice_parameter_changes()
vals = tuple(d[k] * 100 for k in ("a", "b", "c"))
app("Percentage lattice parameter changes:\n\ta: %.2f%%, b: %.2f%%, c: %2.f%%" % vals)
#an.get_percentage_bond_dist_changes(max_radius=3.0)
app("")
cart_stress_tensors, pressures = self.r.read_cart_stress_tensors()
app("Stress tensor (Cartesian coordinates in GPa):\n%s" % cart_stress_tensors[-1])
app("Pressure: %.3f [GPa]" % pressures[-1])
return "\n".join(lines)
@property
def num_steps(self) -> int:
"""Number of iterations performed."""
return self.r.num_steps
[docs]
@lazy_property
def steps(self) -> list:
"""Step indices."""
return list(range(self.num_steps))
@property
def initial_structure(self) -> Structure:
"""The initial |Structure|."""
return self.structures[0]
@property
def final_structure(self) -> Structure:
"""The |Structure| of the last iteration."""
return self.structures[-1]
[docs]
@lazy_property
def structures(self) -> list[Structure]:
"""List of |Structure| objects at the different steps."""
return self.r.read_all_structures()
[docs]
@lazy_property
def etotals(self) -> np.ndarray:
"""|numpy-array| with total energies in eV at the different steps."""
return self.r.read_eterms().etotals
[docs]
def get_relaxation_analyzer(self) -> RelaxationAnalyzer:
"""
Return a pymatgen :class:`RelaxationAnalyzer` object to analyze the relaxation in a calculation.
"""
return RelaxationAnalyzer(self.initial_structure, self.final_structure)
[docs]
def to_xdatcar(self, filepath=None, groupby_type=True, to_unit_cell=False, **kwargs): #-> Xdatcar:
"""
Return Xdatcar pymatgen object. See write_xdatcar for the meaning of arguments.
Args:
to_unit_cell (bool): Whether to translate sites into the unit cell.
kwargs: keywords arguments passed to Xdatcar constructor.
"""
from pymatgen.io.vasp.outputs import Xdatcar
filepath = self.write_xdatcar(filepath=filepath, groupby_type=groupby_type,
to_unit_cell=to_unit_cell, overwrite=True)
return Xdatcar(filepath, **kwargs)
[docs]
def write_xdatcar(self, filepath="XDATCAR", groupby_type=True, overwrite=False, to_unit_cell=False) -> str:
"""
Write Xdatcar file with unit cell and atomic positions to file ``filepath``.
Args:
filepath: Xdatcar filename. If None, a temporary file is created.
groupby_type: If True, atoms are grouped by type. Note that this option
may change the order of the atoms. This option is needed because
there are post-processing tools (e.g. ovito) that do not work as expected
if the atoms in the structure are not grouped by type.
overwrite: raise RuntimeError, if False and filepath exists.
to_unit_cell (bool): Whether to translate sites into the unit cell.
Return:
path to Xdatcar file.
"""
# This library takes 13s to import on HPC (07/02/24) so moved to class method instead of header
from pymatgen.io.vasp.outputs import Xdatcar
if filepath is not None and os.path.exists(filepath) and not overwrite:
raise RuntimeError("Cannot overwrite pre-existing file `%s`" % filepath)
if filepath is None:
import tempfile
fd, filepath = tempfile.mkstemp(text=True, suffix="_XDATCAR")
# int typat[natom], double znucl[npsp]
# NB: typat is double in the HIST.nc file
typat = self.r.read_value("typat").astype(int)
znucl = self.r.read_value("znucl")
ntypat = self.r.read_dimvalue("ntypat")
num_pseudos = self.r.read_dimvalue("npsp")
if num_pseudos != ntypat:
raise NotImplementedError("Alchemical mixing is not supported, num_pseudos != ntypat")
#print("znucl:", znucl, "\ntypat:", typat)
symb2pos = OrderedDict()
symbols_atom = []
for iatom, itype in enumerate(typat):
itype = itype - 1
symbol = Element.from_Z(int(znucl[itype])).symbol
if symbol not in symb2pos: symb2pos[symbol] = []
symb2pos[symbol].append(iatom)
symbols_atom.append(symbol)
if not groupby_type:
group_ids = np.arange(self.r.natom)
else:
group_ids = []
for pos_list in symb2pos.values():
group_ids.extend(pos_list)
group_ids = np.array(group_ids, dtype=int)
comment = " %s\n" % self.initial_structure.formula
with open(filepath, "wt") as fh:
# comment line + scaling factor set to 1.0
fh.write(comment)
fh.write("1.0\n")
for vec in self.initial_structure.lattice.matrix:
fh.write("%.12f %.12f %.12f\n" % (vec[0], vec[1], vec[2]))
if not groupby_type:
fh.write(" ".join(symbols_atom) + "\n")
fh.write("1 " * len(symbols_atom) + "\n")
else:
fh.write(" ".join(symb2pos.keys()) + "\n")
fh.write(" ".join(str(len(p)) for p in symb2pos.values()) + "\n")
# Write atomic positions in reduced coordinates.
xred_list = self.r.read_value("xred")
if to_unit_cell:
xred_list = xred_list % 1
for step in range(self.num_steps):
fh.write("Direct configuration= %d\n" % (step + 1))
frac_coords = xred_list[step, group_ids]
for fs in frac_coords:
fh.write("%.12f %.12f %.12f\n" % (fs[0], fs[1], fs[2]))
return filepath
[docs]
def visualize(self, appname="ovito", to_unit_cell=False): # pragma: no cover
"""
Visualize the crystalline structure with visualizer.
See :class:`Visualizer` for the list of applications and formats supported.
Args:
to_unit_cell (bool): Whether to translate sites into the unit cell.
"""
if appname == "mayavi": return self.mayaview()
# Get the Visualizer subclass from the string.
from abipy.iotools import Visualizer
visu = Visualizer.from_name(appname)
if visu.name != "ovito":
raise NotImplementedError("visualizer: %s" % visu.name)
filepath = self.write_xdatcar(filepath=None, groupby_type=True, to_unit_cell=to_unit_cell)
return visu(filepath)()
#if options.trajectories:
# hist.mvplot_trajectories()
#else:
# hist.mvanimate()
[docs]
def plot_ax(self, ax, what, fontsize=8, **kwargs) -> None:
"""
Helper function to plot quantity ``what`` on axis ``ax`` with matplotlib.
Args:
fontsize: fontsize for legend.
kwargs are passed to matplotlib plot method.
"""
label = None
if what == "energy":
# Total energy in eV.
marker = kwargs.pop("marker", "o")
label = kwargs.pop("label", "Energy")
ax.plot(self.steps, self.etotals, label=label, marker=marker, **kwargs)
ax.set_ylabel('Energy (eV)')
elif what == "abc":
# Lattice parameters.
mark = kwargs.pop("marker", None)
markers = ["o", "^", "v"] if mark is None else 3 * [mark]
for i, label in enumerate(["a", "b", "c"]):
ax.plot(self.steps, [s.lattice.abc[i] for s in self.structures], label=label,
marker=markers[i], **kwargs)
ax.set_ylabel("abc (A)")
elif what in ("a", "b", "c"):
i = ("a", "b", "c").index(what)
marker = kwargs.pop("marker", None)
if marker is None:
marker = {"a": "o", "b": "^", "c": "v"}[what]
label = kwargs.pop("label", what)
ax.plot(self.steps, [s.lattice.abc[i] for s in self.structures], label=label,
marker=marker, **kwargs)
ax.set_ylabel('%s (A)' % what)
elif what == "angles":
# Lattice Angles
mark = kwargs.pop("marker", None)
markers = ["o", "^", "v"] if mark is None else 3 * [mark]
for i, label in enumerate(["alpha", "beta", "gamma"]):
ax.plot(self.steps, [s.lattice.angles[i] for s in self.structures], label=label,
marker=markers[i], **kwargs)
ax.set_ylabel(r"$\alpha\beta\gamma$ (degree)")
elif what in ("alpha", "beta", "gamma"):
i = ("alpha", "beta", "gamma").index(what)
marker = kwargs.pop("marker", None)
if marker is None:
marker = {"alpha": "o", "beta": "^", "gamma": "v"}[what]
label = kwargs.pop("label", what)
ax.plot(self.steps, [s.lattice.angles[i] for s in self.structures], label=label,
marker=marker, **kwargs)
ax.set_ylabel(r"$\%s$ (degree)" % what)
elif what == "volume":
marker = kwargs.pop("marker", "o")
ax.plot(self.steps, [s.lattice.volume for s in self.structures], marker=marker, **kwargs)
ax.set_ylabel(r'$V\, (A^3)$')
elif what == "pressure":
stress_cart_tensors, pressures = self.r.read_cart_stress_tensors()
marker = kwargs.pop("marker", "o")
label = kwargs.pop("label", "P")
ax.plot(self.steps, pressures, label=label, marker=marker, **kwargs)
ax.set_ylabel('P (GPa)')
elif what == "forces":
forces_hist = self.r.read_cart_forces()
fmin_steps, fmax_steps, fmean_steps, fstd_steps = [], [], [], []
for step in range(self.num_steps):
forces = forces_hist[step]
fmods = np.sqrt([np.dot(force, force) for force in forces])
fmean_steps.append(fmods.mean())
fstd_steps.append(fmods.std())
fmin_steps.append(fmods.min())
fmax_steps.append(fmods.max())
mark = kwargs.pop("marker", None)
markers = ["o", "^", "v", "X"] if mark is None else 4 * [mark]
ax.plot(self.steps, fmin_steps, label="min |F|", marker=markers[0], **kwargs)
ax.plot(self.steps, fmax_steps, label="max |F|", marker=markers[1], **kwargs)
ax.plot(self.steps, fmean_steps, label="mean |F|", marker=markers[2], **kwargs)
ax.plot(self.steps, fstd_steps, label="std |F|", marker=markers[3], **kwargs)
label = "std |F|"
ax.set_ylabel('F stats (eV/A)')
else:
raise ValueError(f"Invalid value for {what=}")
ax.set_xlabel('Step')
ax.grid(True)
if label is not None:
ax.legend(loc='best', fontsize=fontsize, shadow=True)
[docs]
def plotly_traces(self, fig, what, rcd=None, fontsize=8, showlegend=False, **kwargs):
"""
Helper function to plot quantity ``what`` on figure ``fig`` with plotly.
Args:
rcd: If ``fig`` has subplots, ``rcd`` is used to add traces on these subplots.
fontsize: fontsize for legend.
kwargs are passed to fig.add_scatter method.
"""
rcd = PlotlyRowColDesc.from_object(rcd)
ply_row, ply_col = rcd.ply_row, rcd.ply_col
label = None
if what == "energy":
# Total energy in eV.
marker = kwargs.pop("marker", 0)
label = kwargs.pop("label", "Energy")
fig.add_scatter(x=self.steps, y=self.etotals, mode='lines+markers', name=label, marker_symbol=marker,
row=ply_row, col=ply_col, **kwargs)
fig.layout['yaxis%u' % rcd.iax].title.text = 'Energy (eV)'
elif what == "abc":
# Lattice parameters.
mark = kwargs.pop("marker", None)
markers = [0, 5, 6] if mark is None else 3 * [mark]
for i, label in enumerate(["a", "b", "c"]):
fig.add_scatter(x=self.steps, y=[s.lattice.abc[i] for s in self.structures], mode='lines+markers',
name=label, marker_symbol=markers[i], row=ply_row, col=ply_col, **kwargs)
fig.layout['yaxis%u' % rcd.iax].title.text = "abc (A)"
elif what in ("a", "b", "c"):
i = ("a", "b", "c").index(what)
marker = kwargs.pop("marker", None)
if marker is None:
marker = {"a": 0, "b": 5, "c": 6}[what]
label = kwargs.pop("label", what)
fig.add_scatter(x=self.steps, y=[s.lattice.abc[i] for s in self.structures], mode='lines+markers',
name=label, marker_symbol=marker, row=ply_row, col=ply_col, **kwargs)
fig.layout['yaxis%u' % rcd.iax].title.text = '%s (A)' % what
elif what == "angles":
# Lattice Angles
mark = kwargs.pop("marker", None)
markers = [0, 5, 6] if mark is None else 3 * [mark]
for i, label in enumerate(["α ", "β ", "ɣ"]):
fig.add_scatter(x=self.steps, y=[s.lattice.angles[i] for s in self.structures], mode='lines+markers',
name=label, marker_symbol=markers[i], row=ply_row, col=ply_col, **kwargs)
fig.layout['yaxis%u' % rcd.iax].title.text = "αβɣ (degree)" + " "
fig.layout['yaxis%u' % rcd.iax].tickformat = ".3r"
elif what in ("alpha", "beta", "gamma"):
i = ("alpha", "beta", "gamma").index(what)
marker = kwargs.pop("marker", None)
if marker is None:
marker = {"alpha": 0, "beta": 5, "gamma": 6}[what]
label = kwargs.pop("label", what)
fig.add_scatter(x=self.steps, y=[s.lattice.angles[i] for s in self.structures], mode='lines+markers',
name=label, marker_symbol=marker, row=ply_row, col=ply_col, **kwargs)
fig.layout['yaxis%u' % rcd.iax].title.text = r"%s (degree)" % latex_greek_2unicode(what)
fig.layout['yaxis%u' % rcd.iax].tickformat = ".3r"
elif what == "volume":
marker = kwargs.pop("marker", 0)
label = kwargs.pop("label", "Volume")
fig.add_scatter(x=self.steps, y=[s.lattice.volume for s in self.structures], mode='lines+markers',
name=label, marker_symbol=marker, row=ply_row, col=ply_col, **kwargs)
fig.layout['yaxis%u' % rcd.iax].title.text = 'V (A³)'
elif what == "pressure":
stress_cart_tensors, pressures = self.r.read_cart_stress_tensors()
marker = kwargs.pop("marker", 0)
label = kwargs.pop("label", "P")
fig.add_scatter(x=self.steps, y=pressures, mode='lines+markers',
name=label, marker_symbol=marker, row=ply_row, col=ply_col, **kwargs)
fig.layout['yaxis%u' % rcd.iax].title.text = 'P (GPa)'
elif what == "forces":
forces_hist = self.r.read_cart_forces()
fmin_steps, fmax_steps, fmean_steps, fstd_steps = [], [], [], []
for step in range(self.num_steps):
forces = forces_hist[step]
fmods = np.sqrt([np.dot(force, force) for force in forces])
fmean_steps.append(fmods.mean())
fstd_steps.append(fmods.std())
fmin_steps.append(fmods.min())
fmax_steps.append(fmods.max())
mark = kwargs.pop("marker", None)
markers = [0, 5, 6, 4] if mark is None else 4 * [mark]
fig.add_scatter(x=self.steps, y=fmin_steps, mode='lines+markers',
name="min |F|", marker_symbol=markers[0], row=ply_row, col=ply_col, **kwargs)
fig.add_scatter(x=self.steps, y=fmax_steps, mode='lines+markers',
name="max |F|", marker_symbol=markers[1], row=ply_row, col=ply_col, **kwargs)
fig.add_scatter(x=self.steps, y=fmean_steps, mode='lines+markers',
name="mean |F|", marker_symbol=markers[2], row=ply_row, col=ply_col, **kwargs)
fig.add_scatter(x=self.steps, y=fstd_steps, mode='lines+markers',
name="std |F|", marker_symbol=markers[3], row=ply_row, col=ply_col, **kwargs)
label = "std |F|"
fig.layout['yaxis%u' % rcd.iax].title.text = 'F stats (eV/A)'
else:
raise ValueError("Invalid value for what: `%s`" % str(what))
fig.layout.legend.font.size = fontsize
[docs]
@add_fig_kwargs
def plot(self, what_list=None, ax_list=None, fontsize=8, **kwargs) -> Figure:
"""
Plot the evolution of structural parameters (lattice lengths, angles and volume)
as well as pressure, info on forces and total energy with matplotlib.
Args:
what_list:
ax_list: List of |matplotlib-Axes|. If None, a new figure is created.
fontsize: fontsize for legend
Returns: |matplotlib-Figure|
"""
if what_list is None:
what_list = ["abc", "angles", "volume", "pressure", "forces", "energy"]
else:
what_list = list_strings(what_list)
nplots = len(what_list)
nrows, ncols = 1, 1
if nplots > 1:
ncols = 2
nrows = nplots // ncols + nplots % ncols
ax_list, fig, plt = get_axarray_fig_plt(ax_list, nrows=nrows, ncols=ncols,
sharex=True, sharey=False, squeeze=False)
ax_list = ax_list.ravel()
# don't show the last ax if nplots is odd.
if nplots % ncols != 0: ax_list[-1].axis("off")
for what, ax in zip(what_list, ax_list):
self.plot_ax(ax, what, fontsize=fontsize, marker="o")
return fig
[docs]
@add_plotly_fig_kwargs
def plotly(self, what_list=None, fig=None, fontsize=12, **kwargs):
"""
Plot the evolution of structural parameters (lattice lengths, angles and volume)
as well as pressure, info on forces and total energy with plotly.
Args:
what_list:
fig: The fig for plot and the DOS plot. If None, a new figure is created.
fontsize: fontsize for legend
Returns: |plotly.graph_objects.Figure|
"""
if what_list is None:
what_list = ["abc", "angles", "volume", "pressure", "forces", "energy"]
else:
what_list = list_strings(what_list)
nplots = len(what_list)
nrows, ncols = 1, 1
if nplots > 1:
ncols = 2
nrows = nplots // ncols + nplots % ncols
if fig is None:
fig, _ = get_figs_plotly(nrows=nrows, ncols=ncols, subplot_titles=[], sharex=True, sharey=False,
vertical_spacing=0.05)
for i, what in enumerate(what_list):
rcd = PlotlyRowColDesc(i // ncols, i % ncols, nrows, ncols)
self.plotly_traces(fig, what, rcd=rcd, fontsize=fontsize, marker=0)
fig.layout['xaxis%u' % rcd.iax].title.text = 'Step'
if nplots > 1:
fig.layout['xaxis%s' % str(rcd.iax-1)].title.text = 'Step'
return fig
[docs]
@add_plotly_fig_kwargs
def plotly_energies(self, fig=None, fontsize=12, **kwargs):
"""
Plot the total energies as function of the iteration step with plotly.
"""
return self.plotly(what_list="energy", fig=fig, fontsize=fontsize, show=False, **kwargs)
[docs]
@add_fig_kwargs
def plot_energies(self, ax=None, fontsize=8, **kwargs) -> Figure:
"""
Plot the total energies as function of the iteration step with matplotlib.
Args:
ax: |matplotlib-Axes| or None if a new figure should be created.
fontsize: Legend and title fontsize.
Returns: |matplotlib-Figure|
"""
# TODO max force and pressure
ax, fig, plt = get_ax_fig_plt(ax=ax)
terms = self.r.read_eterms()
for key, values in terms.items():
if np.all(values == 0.0): continue
ax.plot(self.steps, values, marker="o", label=key)
ax.set_xlabel('Step')
ax.set_ylabel('Energies (eV)')
ax.grid(True)
ax.legend(loc='best', fontsize=fontsize, shadow=True)
return fig
[docs]
def yield_figs(self, **kwargs): # pragma: no cover
"""
This function *generates* a predefined list of matplotlib figures with minimal input from the user.
"""
yield self.plot(show=False)
yield self.plot_energies(show=False)
[docs]
def yield_plotly_figs(self, **kwargs): # pragma: no cover
"""
This function *generates* a predefined list of matplotlib figures with minimal input from the user.
"""
yield self.plotly(show=False)
yield self.plotly_energies(show=False)
[docs]
def mvplot_trajectories(self, colormap="hot", sampling=1, figure=None, show=True,
with_forces=True, **kwargs): # pragma: no cover
"""
Call mayavi_ to plot atomic trajectories and the variation of the unit cell.
"""
from abipy.display import mvtk
figure, mlab = mvtk.get_fig_mlab(figure=figure)
style = "labels"
line_width = 100
mvtk.plot_structure(self.initial_structure, style=style, unit_cell_color=(1, 0, 0), figure=figure)
mvtk.plot_structure(self.final_structure, style=style, unit_cell_color=(0, 0, 0), figure=figure)
steps = np.arange(start=0, stop=self.num_steps, step=sampling)
xcart_list = self.r.read_value("xcart") * units.bohr_to_ang
for iatom in range(self.r.natom):
x, y, z = xcart_list[::sampling, iatom, :].T
#for i in zip(x, y, z): print(i)
trajectory = mlab.plot3d(x, y, z, steps, colormap=colormap, tube_radius=None,
line_width=line_width, figure=figure)
mlab.colorbar(trajectory, title='Iteration', orientation='vertical')
if with_forces:
fcart_list = self.r.read_cart_forces(unit="eV ang^-1")
for iatom in range(self.r.natom):
x, y, z = xcart_list[::sampling, iatom, :].T
u, v, w = fcart_list[::sampling, iatom, :].T
q = mlab.quiver3d(x, y, z, u, v, w, figure=figure, colormap=colormap,
line_width=line_width, scale_factor=10)
#mlab.colorbar(q, title='Forces [eV/Ang]', orientation='vertical')
if show: mlab.show()
return figure
[docs]
def mvanimate(self, delay=500): # pragma: no cover
from abipy.display import mvtk
figure, mlab = mvtk.get_fig_mlab(figure=None)
style = "points"
#mvtk.plot_structure(self.initial_structure, style=style, figure=figure)
#mvtk.plot_structure(self.final_structure, style=style, figure=figure)
xcart_list = self.r.read_value("xcart") * units.bohr_to_ang
#t = np.arange(self.num_steps)
#line_width = 2
#for iatom in range(self.r.natom):
# x, y, z = xcart_list[:, iatom, :].T
# trajectory = mlab.plot3d(x, y, z, t, colormap=colormap, tube_radius=None, line_width=line_width, figure=figure)
#mlab.colorbar(trajectory, title='Iteration', orientation='vertical')
#x, y, z = xcart_list[0, :, :].T
#nodes = mlab.points3d(x, y, z)
#nodes.glyph.scale_mode = 'scale_by_vector'
#this sets the vectors to be a 3x5000 vector showing some random scalars
#nodes.mlab_source.dataset.point_data.vectors = np.tile( np.random.random((5000,)), (3,1))
#nodes.mlab_source.dataset.point_data.scalars = np.random.random((5000,))
@mlab.show
@mlab.animate(delay=delay, ui=True)
def anim():
"""Animate."""
#for it in range(self.num_steps):
for it, structure in enumerate(self.structures):
print('Updating scene for iteration:', it)
#mlab.clf(figure=figure)
mvtk.plot_structure(structure, style=style, figure=figure)
#x, y, z = xcart_list[it, :, :].T
#nodes.mlab_source.set(x=x, y=y, z=z)
#figure.scene.render()
mlab.draw(figure=figure)
yield
anim()
[docs]
def get_panel(self, **kwargs):
"""
Build panel with widgets to interact with the |HistFile| either in a notebook or in panel app.
"""
from abipy.panels.hist import HistFilePanel
return HistFilePanel(self).get_panel(**kwargs)
[docs]
def write_notebook(self, nbpath=None) -> str:
"""
Write a jupyter_ notebook to ``nbpath``. If nbpath is None, a temporay file in the current
working directory is created. Return path to the notebook.
"""
nbformat, nbv, nb = self.get_nbformat_nbv_nb(title=None)
nb.cells.extend([
#nbv.new_markdown_cell("# This is a markdown cell"),
nbv.new_code_cell("hist = abilab.abiopen('%s')" % self.filepath),
nbv.new_code_cell("print(hist)"),
nbv.new_code_cell("hist.plot_energies();"),
nbv.new_code_cell("hist.plot();"),
])
return self._write_nb_nbpath(nb, nbpath)
[docs]
class HistRobot(Robot):
"""
This robot analyzes the results contained in multiple HIST.nc_ files.
.. rubric:: Inheritance Diagram
.. inheritance-diagram:: HistRobot
"""
EXT = "HIST"
[docs]
def to_string(self, verbose: int = 0) -> str:
"""String representation with verbosity level ``verbose``."""
s = ""
if verbose:
s = super().to_string(verbose=0)
df = self.get_dataframe()
s_df = "Table with final structures, pressures in GPa and force stats in eV/Ang:\n\n%s" % str(df)
if s:
return "\n".join([s, str(s_df)])
else:
return str(s_df)
[docs]
def get_dataframe(self, with_geo=True, index=None, abspath=False,
with_spglib=True, funcs=None, **kwargs) -> pd.DataFrame:
"""
Return a |pandas-DataFrame| with the most important final results and the filenames as index.
Args:
with_geo: True if structure info should be added to the dataframe
abspath: True if paths in index should be absolute. Default: Relative to getcwd().
index: Index of the dataframe, if None, robot labels are used
with_spglib: If True, spglib_ is invoked to get the space group symbol and number
kwargs:
attrs:
List of additional attributes of the |HistFile| to add to the |pandas-DataFrame|.
funcs: Function or list of functions to execute to add more data to the DataFrame.
Each function receives a |HistFile| object and returns a tuple (key, value)
where key is a string with the name of column and value is the value to be inserted.
"""
# Add attributes specified by the users
attrs = [
"num_steps", "final_energy", "final_pressure",
"final_fmin", "final_fmax", "final_fmean", "final_fstd", "final_drift",
"initial_fmin", "initial_fmax", "initial_fmean", "initial_fstd", "initial_drift",
# TODO add more columns but must update HIST file
#"nsppol", "nspinor", "nspden",
#"ecut", "pawecutdg", "tsmear", "nkpt",
] + kwargs.pop("attrs", [])
rows, row_names = [], []
for label, hist in self.items():
row_names.append(label)
d = OrderedDict()
initial_fstas_dict = hist.get_fstats_dict(step=0)
final_fstas_dict = hist.get_fstats_dict(step=-1)
# Add info on structure.
if with_geo:
d.update(hist.final_structure.get_dict4pandas(with_spglib=with_spglib))
for aname in attrs:
if aname in ("final_fmin", "final_fmax", "final_fmean", "final_fstd", "final_drift",):
value = final_fstas_dict[aname.replace("final_", "")]
elif aname in ("initial_fmin", "initial_fmax", "initial_fmean", "initial_fstd", "initial_drift"):
value = initial_fstas_dict[aname.replace("initial_", "")]
else:
value = getattr(hist, aname, None)
d[aname] = value
# Execute functions
if funcs is not None: d.update(self._exec_funcs(funcs, hist))
rows.append(d)
row_names = row_names if not abspath else self._to_relpaths(row_names)
index = row_names if index is None else index
return pd.DataFrame(rows, index=index, columns=list(rows[0].keys()))
@property
def what_list(self) -> list[str]:
"""List with all quantities that can be plotted (what_list)."""
return ["energy", "abc", "angles", "volume", "pressure", "forces"]
[docs]
@add_fig_kwargs
def gridplot(self, what_list=None, sharex="row", sharey="row", fontsize=8, **kwargs) -> Figure:
"""
Plot the ``what`` value extracted from multiple HIST.nc_ files on a grid.
Args:
what_list: List of quantities to plot.
Must be in ["energy", "abc", "angles", "volume", "pressure", "forces"]
sharex: True if xaxis should be shared.
sharey: True if yaxis should be shared.
fontsize: fontsize for legend.
Returns: |matplotlib-Figure|
"""
what_list = list_strings(what_list) if what_list is not None else self.what_list
# Build grid of plots.
nrows, ncols = len(what_list), len(self)
ax_mat, fig, plt = get_axarray_fig_plt(None, nrows=nrows, ncols=ncols,
sharex=sharex, sharey=sharey, squeeze=False)
ax_mat = np.reshape(ax_mat, (nrows, ncols))
for irow, what in enumerate(what_list):
for icol, hist in enumerate(self.abifiles):
ax = ax_mat[irow, icol]
ax.grid(True)
hist.plot_ax(ax_mat[irow, icol], what, fontsize=fontsize, marker="o")
if irow == 0:
ax.set_title(hist.relpath, fontsize=fontsize)
if irow != nrows - 1:
set_visible(ax, False, "xlabel")
if icol != 0:
set_visible(ax, False, "ylabel")
return fig
[docs]
@add_fig_kwargs
def combiplot(self, what_list=None, colormap="jet", fontsize=6, **kwargs) -> Figure:
"""
Plot multiple HIST.nc_ files on a grid. One plot for each ``what`` value.
Args:
what_list: List of strings with the quantities to plot. If None, all quanties are plotted.
colormap: matplotlib color map.
fontsize: fontisize for legend.
Returns: |matplotlib-Figure|.
"""
what_list = (list_strings(what_list) if what_list is not None
else ["energy", "a", "b", "c", "alpha", "beta", "gamma", "volume", "pressure"])
num_plots, ncols, nrows = len(what_list), 1, 1
if num_plots > 1:
ncols = 2
nrows = (num_plots // ncols) + (num_plots % ncols)
ax_list, fig, plt = get_axarray_fig_plt(None, nrows=nrows, ncols=ncols,
sharex=True, sharey=False, squeeze=False)
ax_list = ax_list.ravel()
cmap = plt.get_cmap(colormap)
for i, (ax, what) in enumerate(zip(ax_list, what_list)):
for ih, hist in enumerate(self.abifiles):
label = None if i != 0 else hist.relpath
hist.plot_ax(ax, what, color=cmap(ih / len(self)), label=label, fontsize=fontsize)
if label is not None:
ax.legend(loc="best", fontsize=fontsize, shadow=True)
if i == len(ax_list) - 1:
ax.set_xlabel("Step")
else:
ax.set_xlabel("")
# Get around a bug in matplotlib.
if num_plots % ncols != 0: ax_list[-1].axis('off')
return fig
[docs]
def yield_figs(self, **kwargs): # pragma: no cover
"""
This function *generates* a predefined list of matplotlib figures with minimal input from the user.
"""
yield self.gridplot(show=False)
yield self.combiplot(show=False)
[docs]
def write_notebook(self, nbpath=None) -> str:
"""
Write a jupyter_ notebook to nbpath. If nbpath is None, a temporay file in the current
working directory is created. Return path to the notebook.
"""
nbformat, nbv, nb = self.get_nbformat_nbv_nb(title=None)
args = [(l, f.filepath) for l, f in self.items()]
nb.cells.extend([
#nbv.new_markdown_cell("# This is a markdown cell"),
nbv.new_code_cell("robot = abilab.HistRobot(*%s)\nrobot.trim_paths()\nrobot" % str(args)),
nbv.new_code_cell("robot.get_dataframe()"),
nbv.new_code_cell("for what in robot.what_list: robot.gridplot(what=what, tight_layout=True);"),
])
# Mixins
#nb.cells.extend(self.get_baserobot_code_cells())
return self._write_nb_nbpath(nb, nbpath)
[docs]
class HistReader(ETSF_Reader):
"""
This object reads data from the HIST file.
.. rubric:: Inheritance Diagram
.. inheritance-diagram:: HistReader
"""
[docs]
@lazy_property
def num_steps(self) -> int:
"""Number of iterations present in the HIST.nc_ file."""
return self.read_dimvalue("time")
[docs]
@lazy_property
def natom(self) -> int:
"""Number of atoms un the unit cell."""
return self.read_dimvalue("natom")
[docs]
def read_all_structures(self) -> list[Structure]:
"""Return the list of structures at the different iteration steps."""
rprimd_list = self.read_value("rprimd")
xred_list = self.read_value("xred")
# Alchemical mixing is not supported.
num_pseudos = self.read_dimvalue("npsp")
ntypat = self.read_dimvalue("ntypat")
if num_pseudos != ntypat:
raise NotImplementedError("Alchemical mixing is not supported, num_pseudos != ntypat")
znucl, typat = self.read_value("znucl"), self.read_value("typat").astype(int)
#print(znucl.dtype, typat)
cart_forces_step = self.read_cart_forces(unit="eV ang^-1")
structures = []
#print("typat", type(typat))
for step in range(self.num_steps):
s = Structure.from_abivars(
xred=xred_list[step],
rprim=rprimd_list[step],
acell=3 * [1.0],
# FIXME ntypat, typat, znucl are missing!
znucl=znucl,
typat=typat,
)
s.add_site_property("cartesian_forces", cart_forces_step[step])
structures.append(s)
return structures
[docs]
def read_eterms(self, unit: str = "eV") -> AttrDict:
"""|AttrDict| with the decomposition of the total energy in units ``unit``"""
return AttrDict(
etotals=units.EnergyArray(self.read_value("etotal"), "Ha").to(unit),
kinetic_terms=units.EnergyArray(self.read_value("ekin"), "Ha").to(unit),
entropies=units.EnergyArray(self.read_value("entropy"), "Ha").to(unit),
)
[docs]
def read_cart_forces(self, unit: str = "eV ang^-1") -> np.ndarray:
"""
Read and return a |numpy-array| with the cartesian forces in unit ``unit``.
Shape (num_steps, natom, 3)
"""
return units.ArrayWithUnit(self.read_value("fcart"), "Ha bohr^-1").to(unit)
[docs]
def read_reduced_forces(self) -> np.ndarray:
"""
Read and return a |numpy-array| with the forces in reduced coordinates
Shape (num_steps, natom, 3)
"""
return self.read_value("fred")
[docs]
def read_cart_stress_tensors(self) -> tuple[np.ndarray, np.ndarray]:
"""
Return the stress tensors (nstep x 3 x 3) in cartesian coordinates (GPa)
and the list of pressures in GPa unit.
"""
# Abinit stores 6 unique components of this symmetric 3x3 tensor:
# Given in order (1,1), (2,2), (3,3), (3,2), (3,1), (2,1).
c = self.read_value("strten")
tensors = np.empty((self.num_steps, 3, 3), dtype=float)
for step in range(self.num_steps):
for i in range(3): tensors[step, i, i] = c[step, i]
for p, (i, j) in enumerate(((2, 1), (2, 0), (1, 0))):
tensors[step, i, j] = c[step, 3+p]
tensors[step, j, i] = c[step, 3+p]
tensors *= abu.HaBohr3_GPa
pressures = np.empty(self.num_steps)
for step, tensor in enumerate(tensors):
pressures[step] = - tensor.trace() / 3
return tensors, pressures