Skip to content

Commit

Permalink
add spacegroup determination to geometry optimisation
Browse files Browse the repository at this point in the history
  • Loading branch information
alinelena committed May 24, 2024
1 parent f12a5c7 commit 6251767
Show file tree
Hide file tree
Showing 5 changed files with 115 additions and 2 deletions.
22 changes: 21 additions & 1 deletion janus_core/calculations/geom_opt.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,13 +17,15 @@

from janus_core.helpers.janus_types import ASEOptArgs, ASEWriteArgs
from janus_core.helpers.log import config_logger
from janus_core.helpers.utils import none_to_dict
from janus_core.helpers.utils import none_to_dict, spacegroup


def optimize( # pylint: disable=too-many-arguments,too-many-locals,too-many-branches
struct: Atoms,
fmax: float = 0.1,
steps: int = 1000,
symmetry_precision=0.001,
angle_precision=-1,
filter_func: Optional[Callable] = DefaultFilter,
filter_kwargs: Optional[dict[str, Any]] = None,
optimizer: Callable = LBFGS,
Expand All @@ -45,6 +47,12 @@ def optimize( # pylint: disable=too-many-arguments,too-many-locals,too-many-bra
Default is 0.1.
steps : int
Set maximum number of optimization steps to run. Default is 1000.
symmetry_precision : float
Atom displacement tolerance for spglib symmetry determination.
Default is 0.0001 Å.
angle_precision : float
Set angle precision crieria for spacegroup determination.
Default is -1.0 in degrees.
filter_func : Optional[callable]
Apply constraints to atoms through ASE filter function.
Default is `FrechetCellFilter` if available otherwise `ExpCellFilter`.
Expand Down Expand Up @@ -93,6 +101,13 @@ def optimize( # pylint: disable=too-many-arguments,too-many-locals,too-many-bra
log_kwargs.setdefault("name", __name__)
logger = config_logger(**log_kwargs)

s_grp = spacegroup(struct, symmetry_precision, angle_precision)
message = f"Before optimisation spacegroup {s_grp}"
struct.info["initial_spacegroup"] = s_grp

if logger:
logger.info(message)

if filter_func is not None:
filtered_struct = filter_func(struct, **filter_kwargs)
dyn = optimizer(filtered_struct, **opt_kwargs)
Expand All @@ -114,13 +129,18 @@ def optimize( # pylint: disable=too-many-arguments,too-many-locals,too-many-bra

converged = dyn.run(fmax=fmax, steps=steps)

s_grp = spacegroup(struct, symmetry_precision, angle_precision)
message = f"After optimisation spacegroup {s_grp}"
struct.info["final_spacegroup"] = s_grp

# Calculate current maximum force
if filter_func is not None:
max_force = linalg.norm(filtered_struct.get_forces(), axis=1).max()
else:
max_force = linalg.norm(struct.get_forces(), axis=1).max()

if logger:
logger.info(message)
logger.info("Max force: %.6f", max_force)

if not converged:
Expand Down
36 changes: 36 additions & 0 deletions janus_core/helpers/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,42 @@
from pathlib import Path
from typing import Optional

from ase import Atoms
from spglib import get_spacegroup


def spacegroup(
struct: Atoms, sym_precision: float = 0.001, angle_precision: float = -1.0
) -> str:
"""
Determine the spacegroup for a structure.
Parameters
----------
struct : Atoms
Structure as an ase Atoms object.
sym_precision : float
Atom displacement tolerance for spglib symmetry determination.
Default 0.001 Å.
angle_precision : float
Angle precision for spglib symmetry determination.
Default: -1.0 in degrees.
Returns
-------
str
Spacegroup name.
"""
return get_spacegroup(
cell=(
struct.get_cell(),
struct.get_scaled_positions(),
struct.get_atomic_numbers(),
),
symprec=sym_precision,
angle_tolerance=angle_precision,
)


def none_to_dict(dictionaries: list[Optional[dict]]) -> list[dict]:
"""
Expand Down
2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ mace-torch = "^0.3.4"
pyyaml = "^6.0.1"
typer = "^0.9.0"
typer-config = "^1.4.0"
spglib = "^2.3.0"

[tool.poetry.group.dev.dependencies]
coverage = {extras = ["toml"], version = "^7.4.1"}
Expand Down Expand Up @@ -65,7 +66,6 @@ optional = true
dgl = "^1.1.3"
chgnet = "^0.3.4"
matgl = "^1.0.0"
spglib = "<=2.2.0"
torch-dftd = "^0.4.0"

[build-system]
Expand Down
33 changes: 33 additions & 0 deletions tests/data/NaCl-sg.cif
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
data_image0
_chemical_formula_structural NaClNaClNaClNaCl
_chemical_formula_sum "Na4 Cl4"
_cell_length_a 5.65
_cell_length_b 5.64
_cell_length_c 5.64
_cell_angle_alpha 90.0
_cell_angle_beta 90.0
_cell_angle_gamma 90.0

_space_group_name_H-M_alt "P 1"
_space_group_IT_number 1

loop_
_space_group_symop_operation_xyz
'x, y, z'

loop_
_atom_site_type_symbol
_atom_site_label
_atom_site_symmetry_multiplicity
_atom_site_fract_x
_atom_site_fract_y
_atom_site_fract_z
_atom_site_occupancy
Na Na1 1.0 0.0 0.0 0.0 1.0000
Cl Cl1 1.0 0.5 0.0 0.0 1.0000
Na Na2 1.0 0.0 0.5 0.5 1.0000
Cl Cl2 1.0 0.5 0.5 0.5 1.0000
Na Na3 1.0 0.5 0.0 0.5 1.0000
Cl Cl3 1.0 0.0 0.0 0.5 1.0000
Na Na4 1.0 0.5 0.5 0.0 1.0000
Cl Cl4 1.0 0.0 0.5 0.0 1.0000
24 changes: 24 additions & 0 deletions tests/test_geom_opt.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,11 @@
except ImportError:
from ase.constraints import UnitCellFilter

try:
from ase.filters import FrechetCellFilter as DefaultFilter
except ImportError:
from ase.constraints import ExpCellFilter as DefaultFilter

from ase.io import read
import pytest

Expand Down Expand Up @@ -187,3 +192,22 @@ def test_restart(tmp_path):
)
final_energy = single_point.run("energy")["energy"]
assert final_energy < intermediate_energy


def test_space_group():
"""Test spacegroup of the structure."""

single_point = SinglePoint(
struct_path=DATA_PATH / "NaCl-sg.cif",
architecture="mace_mp",
calc_kwargs={"model": MODEL_PATH},
)

optimize(
single_point.struct,
fmax=0.001,
filter_func=DefaultFilter,
)

assert single_point.struct.info["initial_spacegroup"] == "I4/mmm (139)"
assert single_point.struct.info["final_spacegroup"] == "Fm-3m (225)"

0 comments on commit 6251767

Please sign in to comment.