-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathgeomopt.py
141 lines (120 loc) · 4.39 KB
/
geomopt.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
"""Class to run geom opt calculations."""
from aiida.common import datastructures
import aiida.common.folders
from aiida.engine import CalcJobProcessSpec
import aiida.engine.processes
from aiida.orm import (
Bool,
Dict,
Float,
Int,
SinglefileData,
Str,
StructureData,
TrajectoryData,
)
from aiida_mlip.calculations.singlepoint import Singlepoint
class GeomOpt(Singlepoint): # numpydoc ignore=PR01
"""
Calcjob implementation to run geometry optimisation calculations using mlips.
Methods
-------
define(spec: CalcJobProcessSpec) -> None:
Define the process specification, its inputs, outputs and exit codes.
prepare_for_submission(folder: Folder) -> CalcInfo:
Create the input files for the `CalcJob`.
"""
DEFAULT_TRAJ_FILE = "aiida-traj.xyz"
@classmethod
def define(cls, spec: CalcJobProcessSpec) -> None:
"""
Define the process specification, its inputs, outputs and exit codes.
Parameters
----------
spec : `aiida.engine.CalcJobProcessSpec`
The calculation job process spec to define.
"""
super().define(spec)
# Additional inputs for geometry optimisation
spec.input(
"traj",
valid_type=Str,
required=False,
default=lambda: Str(cls.DEFAULT_TRAJ_FILE),
help="Path to save optimisation frames to",
)
spec.input(
"opt_cell_fully",
valid_type=Bool,
required=False,
help="Fully optimise the cell vectors, angles, and atomic positions",
)
spec.input(
"opt_cell_lengths",
valid_type=Bool,
required=False,
help="Optimise cell vectors, as well as atomic positions",
)
spec.input(
"fmax",
valid_type=Float,
required=False,
help="Maximum force for convergence",
)
spec.input(
"steps",
valid_type=Int,
required=False,
help="Number of optimisation steps",
)
spec.input(
"opt_kwargs",
valid_type=Dict,
required=False,
help="Other optimisation keywords",
)
spec.inputs["metadata"]["options"]["parser_name"].default = "mlip.opt_parser"
spec.output("traj_file", valid_type=SinglefileData)
spec.output("traj_output", valid_type=TrajectoryData)
spec.output("final_structure", valid_type=StructureData)
def prepare_for_submission(
self, folder: aiida.common.folders.Folder
) -> datastructures.CalcInfo:
"""
Create the input files for the `Calcjob`.
Parameters
----------
folder : aiida.common.folders.Folder
Folder where the calculation is run.
Returns
-------
aiida.common.datastructures.CalcInfo
An instance of `aiida.common.datastructures.CalcInfo`.
"""
# Call the parent class method to prepare common inputs
calcinfo = super().prepare_for_submission(folder)
codeinfo = calcinfo.codes_info[0]
geom_opt_cmdline = {"traj": self.inputs.traj.value}
if "opt_kwargs" in self.inputs:
opt_kwargs = self.inputs.opt_kwargs.get_dict()
geom_opt_cmdline["opt-kwargs"] = opt_kwargs
if "opt_cell_fully" in self.inputs:
geom_opt_cmdline["opt-cell-fully"] = self.inputs.opt_cell_fully.value
if "opt_cell_lengths" in self.inputs:
geom_opt_cmdline["opt-cell-lengths"] = self.inputs.opt_cell_lengths.value
if "fmax" in self.inputs:
geom_opt_cmdline["fmax"] = self.inputs.fmax.value
if "steps" in self.inputs:
geom_opt_cmdline["steps"] = self.inputs.steps.value
# Adding command line params for when we run janus
# 'geomopt' is overwriting the placeholder "calculation" from the base.py file
codeinfo.cmdline_params[0] = "geomopt"
for flag, value in geom_opt_cmdline.items():
if isinstance(value, bool):
# Add boolean flags without value if True
if value:
codeinfo.cmdline_params.append(f"--{flag}")
else:
codeinfo.cmdline_params += [f"--{flag}", value]
calcinfo.retrieve_list.append(self.inputs.traj.value)
return calcinfo