Skip to content

Commit

Permalink
Add support for electron-phonon calculations
Browse files Browse the repository at this point in the history
Make several changes in the plugins for `ph.x`, `q2r.x` and `matdyn.x` to provide
support for electron-phonon calculations:

* Adapt the `prepare_for_submission` scripts to allow remote copying of
the `elph_dir` for both plugins in case `la2F` is set to true.
* Add support to the `MatDynCalculation` plugin for setting `dos` to
true, and in this case converting the `kpoints` input to the `nkX`
tags instead of providing them as a list.
* For `matdyn.x` calculations where `dos` is true, parse the phonon DOS
instead of the phonon bands, and provide this as an output.
* For the `MatdynCalculation` plugin, providing the `parent_folder` input
currently only makes sense for electron-phonon calculations, so a
validator is added to check this. The remote copy/symlink list is also
overriden by the `elph_dir` to avoid adding it to the default `out`
directory of the `NamelistsCalculation`, which is typically no longer
required for the `matdyn.x` step.
  • Loading branch information
mbercx committed Apr 15, 2023
1 parent 74d25d1 commit d674b7b
Show file tree
Hide file tree
Showing 4 changed files with 173 additions and 42 deletions.
77 changes: 66 additions & 11 deletions src/aiida_quantumespresso/calculations/matdyn.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,12 @@
# -*- coding: utf-8 -*-
"""`CalcJob` implementation for the matdyn.x code of Quantum ESPRESSO."""
from pathlib import Path

from aiida import orm

from aiida_quantumespresso.calculations import _uppercase_dict
from aiida_quantumespresso.calculations.namelists import NamelistsCalculation
from aiida_quantumespresso.calculations.ph import PhCalculation
from aiida_quantumespresso.data.force_constants import ForceConstantsData


Expand Down Expand Up @@ -31,10 +35,12 @@ def define(cls, spec):
super().define(spec)
spec.input('force_constants', valid_type=ForceConstantsData, required=True)
spec.input('kpoints', valid_type=orm.KpointsData, help='Kpoints on which to calculate the phonon frequencies.')
spec.input('parent_folder', valid_type=orm.RemoteData, required=False)
spec.inputs.validator = cls._validate_inputs

spec.output('output_parameters', valid_type=orm.Dict)
spec.output('output_phonon_bands', valid_type=orm.BandsData)
spec.output('output_phonon_bands', valid_type=orm.BandsData, required=False)
spec.output('output_phonon_dos', valid_type=orm.XyData, required=False)
spec.default_output_node = 'output_parameters'

spec.exit_code(310, 'ERROR_OUTPUT_STDOUT_READ',
Expand All @@ -43,6 +49,8 @@ def define(cls, spec):
message='The stdout output file was incomplete probably because the calculation got interrupted.')
spec.exit_code(330, 'ERROR_OUTPUT_FREQUENCIES',
message='The output frequencies file could not be read from the retrieved folder.')
spec.exit_code(330, 'ERROR_OUTPUT_DOS',
message='The output DOS file could not be read from the retrieved folder.')
spec.exit_code(410, 'ERROR_OUTPUT_KPOINTS_MISSING',
message='Number of kpoints not found in the output data')
spec.exit_code(411, 'ERROR_OUTPUT_KPOINTS_INCOMMENSURATE',
Expand All @@ -60,6 +68,12 @@ def _validate_inputs(value, _):
if parameters.get('INPUT', {}).get('flfrc', None) is not None:
return '`INPUT.flfrc` is set automatically from the `force_constants` input.'

if 'parent_folder' in value and not parameters.get('INPUT').get('la2F', False):
return (
'The `parent_folder` input is only used to calculate the el-ph coefficients but `la2F` is not set '
'to `.true.`'
)

def generate_input_file(self, parameters): # pylint: disable=arguments-differ
"""Generate namelist input_file content given a dict of parameters.
Expand All @@ -68,21 +82,31 @@ def generate_input_file(self, parameters): # pylint: disable=arguments-differ
:return: 'str' containing the input_file content a plain text.
"""
kpoints = self.inputs.kpoints
append_string = ''

parameters.setdefault('INPUT', {})['flfrc'] = self.inputs.force_constants.filename
file_content = super().generate_input_file(parameters)

try:
kpoints_list = kpoints.get_kpoints()
except AttributeError:
kpoints_list = kpoints.get_kpoints_mesh(print_list=True)
# Calculating DOS requires (nk1,nk2,nk3), see
# https://gitlab.com/QEF/q-e/-/blob/b231a0d0174ad1853f191160389029aa14fba6e9/PHonon/PH/matdyn.f90#L82
if parameters['INPUT'].get('dos', False):
kpoints_mesh = kpoints.get_kpoints_mesh()[0]
parameters['INPUT']['nk1'] = kpoints_mesh[0]
parameters['INPUT']['nk2'] = kpoints_mesh[1]
parameters['INPUT']['nk3'] = kpoints_mesh[2]
else:
try:
kpoints_list = kpoints.get_kpoints()
except AttributeError:
kpoints_list = kpoints.get_kpoints_mesh(print_list=True)

kpoints_string = [f'{len(kpoints_list)}']
for kpoint in kpoints_list:
kpoints_string.append('{:18.10f} {:18.10f} {:18.10f}'.format(*kpoint)) # pylint: disable=consider-using-f-string
kpoints_string = [f'{len(kpoints_list)}']
for kpoint in kpoints_list:
kpoints_string.append('{:18.10f} {:18.10f} {:18.10f}'.format(*kpoint)) # pylint: disable=consider-using-f-string
append_string = '\n'.join(kpoints_string) + '\n'

file_content += '\n'.join(kpoints_string) + '\n'
file_content = super().generate_input_file(parameters)

return file_content
return file_content + append_string

def prepare_for_submission(self, folder):
"""Prepare the calculation job for submission by transforming input nodes into input files.
Expand All @@ -91,6 +115,10 @@ def prepare_for_submission(self, folder):
contains lists of files that need to be copied to the remote machine before job submission, as well as file
lists that are to be retrieved after job completion.
After calling the method of the parent `NamelistsCalculation` class, the input parameters are checked to see
if the `la2F` tag is set to true. In this case the remote symlink or copy list is set to the electron-phonon
directory, depending on the settings.
:param folder: a sandbox folder to temporarily write files on disk.
:return: :class:`~aiida.common.datastructures.CalcInfo` instance.
"""
Expand All @@ -99,4 +127,31 @@ def prepare_for_submission(self, folder):
force_constants = self.inputs.force_constants
calcinfo.local_copy_list.append((force_constants.uuid, force_constants.filename, force_constants.filename))

if 'settings' in self.inputs:
settings = _uppercase_dict(self.inputs.settings.get_dict(), dict_name='settings')
else:
settings = {}

if 'parameters' in self.inputs:
parameters = _uppercase_dict(self.inputs.parameters.get_dict(), dict_name='parameters')
else:
parameters = {}

source = self.inputs.get('parent_folder', None)

if source is not None and parameters.get('INPUT').get('la2F', False):

# pylint: disable=protected-access
dirpath = Path(source.get_remote_path()) / PhCalculation._FOLDER_ELECTRON_PHONON
remote_list = [(source.computer.uuid, str(dirpath), PhCalculation._FOLDER_ELECTRON_PHONON)]

# For el-ph calculations, _only_ the `elph_dir` should be copied from the parent folder
if settings.pop('PARENT_FOLDER_SYMLINK', False):
calcinfo.remote_symlink_list = remote_list
else:
calcinfo.remote_copy_list = remote_list

calcinfo.retrieve_list += [f'a2F.dos{i}' for i in range(1, 11)]
calcinfo.retrieve_list.append('lambda')

return calcinfo
52 changes: 32 additions & 20 deletions src/aiida_quantumespresso/calculations/ph.py
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ class PhCalculation(CalcJob):
_DVSCF_PREFIX = 'dvscf'
_DRHO_STAR_EXT = 'drho_rot'
_FOLDER_DYNAMICAL_MATRIX = 'DYN_MAT'
_FOLDER_ELECTRON_PHONON = 'elph_dir'
_VERBOSITY = 'high'
_OUTPUT_DYNAMICAL_MATRIX_PREFIX = os.path.join(_FOLDER_DYNAMICAL_MATRIX, 'dynamical-matrix-')

Expand Down Expand Up @@ -228,25 +229,6 @@ def prepare_for_submission(self, folder):
if not restart_flag: # if it is a restart, it will be copied over
folder.get_subfolder(self._FOLDER_DYNAMICAL_MATRIX, create=True)

with folder.open(self.metadata.options.input_filename, 'w') as infile:
for namelist_name in namelists_toprint:
infile.write(f'&{namelist_name}\n')
# namelist content; set to {} if not present, so that we leave an empty namelist
namelist = parameters.pop(namelist_name, {})
for key, value in sorted(namelist.items()):
infile.write(convert_input_to_namelist_entry(key, value))
infile.write('/\n')

# add list of qpoints if required
if postpend_text is not None:
infile.write(postpend_text)

if parameters:
raise exceptions.InputValidationError(
'The following namelists are specified in parameters, but are not valid namelists for the current type '
f'of calculation: {",".join(list(parameters.keys()))}'
)

# copy the parent scratch
symlink = settings.pop('PARENT_FOLDER_SYMLINK', self._default_symlink_usage) # a boolean
if symlink:
Expand Down Expand Up @@ -284,14 +266,44 @@ def prepare_for_submission(self, folder):
os.path.join(parent_folder.get_remote_path(),
self._FOLDER_DYNAMICAL_MATRIX), self._FOLDER_DYNAMICAL_MATRIX
))

if parameters['INPUTPH'].get('electron_phonon', None) is not None:
remote_symlink_list.append((
parent_folder.computer.uuid,
os.path.join(parent_folder.get_remote_path(),
self._FOLDER_ELECTRON_PHONON), self._FOLDER_ELECTRON_PHONON
))
else:
# copy the dynamical matrices
# no need to copy the _ph0, since I copied already the whole ./out folder
remote_copy_list.append((
parent_folder.computer.uuid,
os.path.join(parent_folder.get_remote_path(), self._FOLDER_DYNAMICAL_MATRIX), '.'
))
if parameters['INPUTPH'].get('electron_phonon', None) is not None:
remote_copy_list.append((
parent_folder.computer.uuid,
os.path.join(parent_folder.get_remote_path(),
self._FOLDER_ELECTRON_PHONON), self._FOLDER_ELECTRON_PHONON
))

with folder.open(self.metadata.options.input_filename, 'w') as infile:
for namelist_name in namelists_toprint:
infile.write(f'&{namelist_name}\n')
# namelist content; set to {} if not present, so that we leave an empty namelist
namelist = parameters.pop(namelist_name, {})
for key, value in sorted(namelist.items()):
infile.write(convert_input_to_namelist_entry(key, value))
infile.write('/\n')

# add list of qpoints if required
if postpend_text is not None:
infile.write(postpend_text)

if parameters:
raise exceptions.InputValidationError(
'The following namelists are specified in parameters, but are not valid namelists for the current type '
f'of calculation: {",".join(list(parameters.keys()))}'
)

# Create an `.EXIT` file if `only_initialization` flag in `settings` is set to `True`
if settings.pop('ONLY_INITIALIZATION', False):
Expand Down
39 changes: 39 additions & 0 deletions src/aiida_quantumespresso/calculations/q2r.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
# -*- coding: utf-8 -*-
"""`CalcJob` implementation for the q2r.x code of Quantum ESPRESSO."""
from pathlib import Path

from aiida import orm

from aiida_quantumespresso.calculations import _uppercase_dict
from aiida_quantumespresso.calculations.namelists import NamelistsCalculation
from aiida_quantumespresso.calculations.ph import PhCalculation
from aiida_quantumespresso.data.force_constants import ForceConstantsData
Expand Down Expand Up @@ -39,3 +41,40 @@ def define(cls, spec):
spec.exit_code(330, 'ERROR_READING_FORCE_CONSTANTS_FILE',
message='The force constants file could not be read.')
# yapf: enable

def prepare_for_submission(self, folder):
"""Prepare the calculation job for submission by transforming input nodes into input files.
In addition to the input files being written to the sandbox folder, a `CalcInfo` instance will be returned that
contains lists of files that need to be copied to the remote machine before job submission, as well as file
lists that are to be retrieved after job completion.
After calling the method of the parent `NamelistsCalculation` class, the input parameters are checked to see
if the `la2F` tag is set to true. In this case the electron-phonon directory is added to the remote symlink or
copy list, depending on the settings.
:param folder: a sandbox folder to temporarily write files on disk.
:return: :py:`~aiida.common.datastructures.CalcInfo` instance.
"""
calcinfo = super().prepare_for_submission(folder)

if 'settings' in self.inputs:
settings = _uppercase_dict(self.inputs.settings.get_dict(), dict_name='settings')
else:
settings = {}

parameters = self.inputs.parameters.get_dict()
source = self.inputs.get('parent_folder', None)

if source is not None:

if parameters.get('INPUT').get('la2F', False):

symlink = settings.pop('PARENT_FOLDER_SYMLINK', False)
remote_list = calcinfo.remote_symlink_list if symlink else calcinfo.remote_copy_list

# pylint: disable=protected-access
dirpath = Path(source.get_remote_path()) / PhCalculation._FOLDER_ELECTRON_PHONON
remote_list.append((source.computer.uuid, str(dirpath), PhCalculation._FOLDER_ELECTRON_PHONON))

return calcinfo
47 changes: 36 additions & 11 deletions src/aiida_quantumespresso/parsers/matdyn.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
# -*- coding: utf-8 -*-
from aiida import orm
import numpy
from qe_tools import CONSTANTS

from aiida_quantumespresso.calculations import _uppercase_dict
from aiida_quantumespresso.calculations.matdyn import MatdynCalculation

from .base import Parser
Expand All @@ -15,6 +17,7 @@ def parse(self, **kwargs):
retrieved = self.retrieved
filename_stdout = self.node.get_option('output_filename')
filename_frequencies = MatdynCalculation._PHONON_FREQUENCIES_NAME
filename_dos = MatdynCalculation._PHONON_DOS_NAME

if filename_stdout not in retrieved.base.repository.list_object_names():
return self.exit(self.exit_codes.ERROR_OUTPUT_STDOUT_READ)
Expand All @@ -23,7 +26,10 @@ def parse(self, **kwargs):
return self.exit(self.exit_codes.ERROR_OUTPUT_STDOUT_INCOMPLETE)

if filename_frequencies not in retrieved.base.repository.list_object_names():
return self.exit(self.exit_codes.ERROR_OUTPUT_STDOUT_READ)
return self.exit(self.exit_codes.ERROR_OUTPUT_FREQUENCIES)

if filename_dos not in retrieved.base.repository.list_object_names():
return self.exit(self.exit_codes.ERROR_OUTPUT_DOS)

# Extract the kpoints from the input data and create the `KpointsData` for the `BandsData`
try:
Expand All @@ -36,23 +42,42 @@ def parse(self, **kwargs):

parsed_data = parse_raw_matdyn_phonon_file(retrieved.base.repository.get_object_content(filename_frequencies))

try:
num_kpoints = parsed_data.pop('num_kpoints')
except KeyError:
return self.exit(self.exit_codes.ERROR_OUTPUT_KPOINTS_MISSING)
if 'parameters' in self.node.inputs:
parameters = _uppercase_dict(self.node.inputs.parameters.get_dict(), dict_name='parameters')
else:
parameters = {}

if parameters.get('INPUT', {}).get('dos', False):
parsed_data.pop('phonon_bands', None)

with retrieved.open(filename_dos) as handle:
dos_array = numpy.genfromtxt(handle)

output_dos = orm.XyData()
output_dos.set_x(dos_array[:, 0], 'frequency', 'cm^(-1)')
output_dos.set_y(dos_array[:, 1], 'dos', 'states * cm')

self.out('output_phonon_dos', output_dos)

else:
if num_kpoints != kpoints.shape[0]:
return self.exit(self.exit_codes.ERROR_OUTPUT_KPOINTS_INCOMMENSURATE)

try:
num_kpoints = parsed_data.pop('num_kpoints')
except KeyError:
return self.exit(self.exit_codes.ERROR_OUTPUT_KPOINTS_MISSING)

if num_kpoints != kpoints.shape[0]:
return self.exit(self.exit_codes.ERROR_OUTPUT_KPOINTS_INCOMMENSURATE)
output_bands = orm.BandsData()
output_bands.set_kpointsdata(kpoints_for_bands)
output_bands.set_bands(parsed_data.pop('phonon_bands'), units='THz')

output_bands = orm.BandsData()
output_bands.set_kpointsdata(kpoints_for_bands)
output_bands.set_bands(parsed_data.pop('phonon_bands'), units='THz')
self.out('output_phonon_bands', output_bands)

for message in parsed_data['warnings']:
self.logger.error(message)

self.out('output_parameters', orm.Dict(parsed_data))
self.out('output_phonon_bands', output_bands)

return

Expand Down

0 comments on commit d674b7b

Please sign in to comment.