Source code for aiida.orm.nodes.data.code.installed

###########################################################################
# Copyright (c), The AiiDA team. All rights reserved.                     #
# This file is part of the AiiDA code.                                    #
#                                                                         #
# The code is hosted on GitHub at https://github.com/aiidateam/aiida-core #
# For further information on the license, see the LICENSE.txt file        #
# For further information please visit http://www.aiida.net               #
###########################################################################
"""Data plugin representing an executable code on a remote computer.

This plugin should be used if an executable is pre-installed on a computer. The ``InstalledCode`` represents the code by
storing the absolute filepath of the relevant executable and the computer on which it is installed. The computer is
represented by an instance of :class:`aiida.orm.computers.Computer`. Each time a :class:`aiida.engine.CalcJob` is run
using an ``InstalledCode``, it will run its executable on the associated computer.
"""

from __future__ import annotations

import pathlib

from pydantic import field_validator, model_validator

from aiida.common import exceptions
from aiida.common.lang import type_check
from aiida.common.log import override_log_level
from aiida.common.pydantic import MetadataField
from aiida.orm import Computer
from aiida.orm.entities import from_backend_entity

from .abstract import AbstractCode
from .legacy import Code

__all__ = ('InstalledCode',)


[docs] class InstalledCode(Code): """Data plugin representing an executable code on a remote computer.""" _KEY_ATTRIBUTE_FILEPATH_EXECUTABLE: str = 'filepath_executable'
[docs] class Model(AbstractCode.Model): """Model describing required information to create an instance.""" computer: str = MetadataField( ..., title='Computer', description='The remote computer on which the executable resides.', short_name='-Y', priority=2, ) filepath_executable: str = MetadataField( ..., title='Filepath executable', description='Filepath of the executable on the remote computer.', short_name='-X', priority=1, )
[docs] @field_validator('label') @classmethod def validate_label_uniqueness(cls, value: str) -> str: """Override the validator for the ``label`` of the base class since uniqueness is defined on full label.""" return value
[docs] @field_validator('computer') @classmethod def validate_computer(cls, value: str) -> Computer: """Override the validator for the ``label`` of the base class since uniqueness is defined on full label.""" from aiida.orm import load_computer try: return load_computer(value) except exceptions.NotExistent as exception: raise ValueError(exception) from exception
[docs] @model_validator(mode='after') # type: ignore[misc] def validate_full_label_uniqueness(self) -> AbstractCode.Model: """Validate that the full label does not already exist.""" from aiida.orm import load_code full_label = f'{self.label}@{self.computer.label}' # type: ignore[attr-defined] try: load_code(full_label) except exceptions.NotExistent: return self except exceptions.MultipleObjectsError as exception: raise ValueError(f'Multiple codes with the label `{full_label}` already exist.') from exception else: raise ValueError(f'A code with the label `{full_label}` already exists.')
[docs] def __init__(self, computer: Computer, filepath_executable: str, **kwargs): """Construct a new instance. :param computer: The remote computer on which the executable is located. :param filepath_executable: The absolute filepath of the executable on the remote computer. """ super().__init__(**kwargs) self.computer = computer self.filepath_executable = filepath_executable # type: ignore[assignment]
[docs] def _validate(self): """Validate the instance by checking that a computer has been defined. :raises :class:`aiida.common.exceptions.ValidationError`: If the state of the node is invalid. """ super(Code, self)._validate() # Change to ``super()._validate()`` once deprecated ``Code`` class is removed. if not self.computer: # type: ignore[truthy-bool] raise exceptions.ValidationError('The `computer` is undefined.') try: self.filepath_executable except TypeError as exception: raise exceptions.ValidationError('The `filepath_executable` is not set.') from exception
[docs] def validate_filepath_executable(self): """Validate the ``filepath_executable`` attribute. Checks whether the executable exists on the remote computer if a transport can be opened to it. This method is intentionally not called in ``_validate`` as to allow the creation of ``Code`` instances whose computers can not yet be connected to and as to not require the overhead of opening transports in storing a new code. .. note:: If the ``filepath_executable`` is not an absolute path, the check is skipped. :raises `~aiida.common.exceptions.ValidationError`: if no transport could be opened or if the defined executable does not exist on the remote computer. """ if not self.filepath_executable.is_absolute(): return try: with override_log_level(): # Temporarily suppress noisy logging with self.computer.get_transport() as transport: file_exists = transport.isfile(str(self.filepath_executable)) except Exception as exception: raise exceptions.ValidationError( 'Could not connect to the configured computer to determine whether the specified executable exists.' ) from exception if not file_exists: raise exceptions.ValidationError( f'The provided remote absolute path `{self.filepath_executable}` does not exist on the computer.' )
[docs] def can_run_on_computer(self, computer: Computer) -> bool: """Return whether the code can run on a given computer. :param computer: The computer. :return: ``True`` if the provided computer is the same as the one configured for this code. """ type_check(computer, Computer) return computer.pk == self.computer.pk
[docs] def get_executable(self) -> pathlib.PurePosixPath: """Return the executable that the submission script should execute to run the code. :return: The executable to be called in the submission script. """ return self.filepath_executable
@property # type: ignore[override] def computer(self) -> Computer: """Return the computer of this code.""" assert self.backend_entity.computer is not None return from_backend_entity(Computer, self.backend_entity.computer) @computer.setter def computer(self, computer: Computer) -> None: """Set the computer of this code. :param computer: A `Computer`. """ if self.is_stored: raise exceptions.ModificationNotAllowed('cannot set the computer on a stored node') type_check(computer, Computer, allow_none=False) self.backend_entity.computer = computer.backend_entity @property def full_label(self) -> str: """Return the full label of this code. The full label can be just the label itself but it can be something else. However, it at the very least has to include the label of the code. :return: The full label of the code. """ return f'{self.label}@{self.computer.label}' @property def filepath_executable(self) -> pathlib.PurePosixPath: """Return the absolute filepath of the executable that this code represents. :return: The absolute filepath of the executable. """ return pathlib.PurePosixPath(self.base.attributes.get(self._KEY_ATTRIBUTE_FILEPATH_EXECUTABLE)) @filepath_executable.setter def filepath_executable(self, value: str) -> None: """Set the absolute filepath of the executable that this code represents. :param value: The absolute filepath of the executable. """ type_check(value, str) self.base.attributes.set(self._KEY_ATTRIBUTE_FILEPATH_EXECUTABLE, value)