Skip to content
Snippets Groups Projects
Code owners
Assign users and groups as approvers for specific file changes. Learn more.
AtomsData.py 3.62 KiB
from __future__ import (absolute_import, division, print_function)
import numpy as np
import six
import AbinsModules


class AtomsDaTa(AbinsModules.GeneralData):

    def __init__(self, num_atoms=None):
        super(AtomsDaTa, self).__init__()
        if not isinstance(num_atoms, int):
            raise ValueError("Invalid number of atoms.")
        if num_atoms < 0:
            raise ValueError("Number of atoms cannot be negative.")

        self._num_atoms = num_atoms
        self._num_atom = 0
        self._data = {}

    def _append(self, item=None):
        """
        Adds one elements to the collection of atoms data.
        @param item: element to be added
        """

        if not isinstance(item, dict):
            raise ValueError("Every element of AtomsData has  a form of the dictionary.")

        if not sorted(item.keys()) == sorted(AbinsModules.AbinsConstants.ALL_KEYWORDS_ATOMS_DATA):
            raise ValueError("Invalid structure of the dictionary to be added.")

        # "symbol"
        if not item["symbol"] in AbinsModules.AbinsConstants.ALL_SYMBOLS:
            raise ValueError("Invalid value of symbol.")

        # "fract_coord"
        fract_coord = item["fract_coord"]
        if not isinstance(fract_coord, np.ndarray):
            raise ValueError("Coordinates of an atom should have a form of a numpy array.")
        if len(fract_coord.shape) != 1:
            raise ValueError("Coordinates should have a form of 1D numpy array.")
        if fract_coord.shape[0] != 3:
            raise ValueError("Coordinates should have a form of numpy array with three elements.")
        if fract_coord.dtype.num != AbinsModules.AbinsConstants.FLOAT_ID:
            raise ValueError("All coordinates should be real numbers.")

        # "sort"
        sort = item["sort"]
        if not (isinstance(sort, six.integer_types) or np.issubdtype(sort.dtype, np.integer)):
            raise ValueError("Parameter sort  should be integer.")
        if sort < 0:
            raise ValueError("Parameter sort cannot be negative.")
        if sort >= self._num_atoms:  # here = because we count from 0
            raise ValueError("Parameter sort cannot be larger than the total number of atoms.")

        # "mass"
        mass = item["mass"]
        if not isinstance(mass, float):
            raise ValueError("Mass of atom should be a real number.")
        if mass < 0:
            raise ValueError("Mass of atom cannot be negative.")

        if self._num_atom == self._num_atoms:
            raise ValueError("Number of atom cannot be larger than total number of atoms in the system.")

        self._data.update({"atom_%s" % self._num_atom: item})
        self._num_atom += 1

    def set(self, items=None):

        if len(items) != self._num_atoms:
            raise ValueError("Inconsistent size of new data and number of atoms. (%s != %s)" %
                             (len(items), self._num_atoms))

        atoms_list = ["atom_%s" % atom for atom in range(self._num_atoms)]
        for atom in atoms_list:
            if atom not in items:
                raise ValueError("Missing data for atom number %s." % atom)

        if isinstance(items, dict):
            self._data = {}
            self._num_atom = 0
            size = len(items)
            for atom in range(size):
                self._append(item=items["atom_%s" % atom])
        else:
            raise ValueError("Invalid type of data.")

    def extract(self):
        if len(self._data) == self._num_atoms:
            return self._data
        else:
            raise ValueError("Size of AtomsData and number of atoms is inconsistent.")

    def __str__(self):
        return "Atoms data"