Source code for aiida.backends.tests.test_dataclasses

# -*- coding: utf-8 -*-
###########################################################################
# 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               #
###########################################################################
"""
Tests for specific subclasses of Data
"""
from __future__ import print_function
from __future__ import absolute_import
from __future__ import division

import io
import os
import tempfile
import unittest

from six.moves import range

from aiida.backends.testbase import AiidaTestCase
from aiida.common.exceptions import ModificationNotAllowed
from aiida.common.utils import Capturing
from aiida.orm import load_node
from aiida.orm import CifData, SinglefileData, StructureData, KpointsData, BandsData, ArrayData, TrajectoryData, Dict
from aiida.orm.nodes.data.structure import Kind, Site


[docs]def has_seekpath(): """ Check if there is the seekpath dependency :return: True if seekpath is installed, False otherwise """ try: import seekpath return True except ImportError: return False
[docs]def to_list_of_lists(lofl): """ Converts an iterable of iterables to a list of lists, needed for some tests (e.g. when one has a tuple of lists, a list of tuples, ...) :param lofl: an iterable of iterables :return: a list of lists """ return [[el for el in l] for l in lofl]
[docs]def simplify(string): """ Takes a string, strips spaces in each line and returns it Useful to compare strings when different versions of a code give different spaces. """ return "\n".join(s.strip() for s in string.split())
[docs]class TestSinglefileData(AiidaTestCase): """ Test the SinglefileData class. """
[docs] def test_reload_singlefiledata(self): file_content = 'some text ABCDE' with tempfile.NamedTemporaryFile(mode='w+') as tmpf: filename = tmpf.name basename = os.path.split(filename)[1] tmpf.write(file_content) tmpf.flush() a = SinglefileData(filepath=filename) the_uuid = a.uuid self.assertEquals(a.list_object_names(), [basename]) with a.open() as fhandle: self.assertEquals(fhandle.read(), file_content) a.store() with a.open() as fhandle: self.assertEquals(fhandle.read(), file_content) self.assertEquals(a.list_object_names(), [basename]) b = load_node(the_uuid) # I check the retrieved object self.assertTrue(isinstance(b, SinglefileData)) self.assertEquals(b.list_object_names(), [basename]) with b.open() as fhandle: self.assertEquals(fhandle.read(), file_content)
[docs]class TestCifData(AiidaTestCase): """ Tests for CifData class. """ from distutils.version import StrictVersion from aiida.orm.nodes.data.cif import has_pycifrw from aiida.orm.nodes.data.structure import has_ase, has_pymatgen, has_spglib, get_pymatgen_version valid_sample_cif_str = ''' data_test _cell_length_a 10 _cell_length_b 10 _cell_length_c 10 _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 90 _chemical_formula_sum 'C O2' loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z _atom_site_attached_hydrogens C 0 0 0 0 O 0.5 0.5 0.5 . H 0.75 0.75 0.75 0 ''' valid_sample_cif_str_2 = ''' data_test _cell_length_a 10 _cell_length_b 10 _cell_length_c 10 _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 90 _chemical_formula_sum 'C O' loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z _atom_site_attached_hydrogens C 0 0 0 0 O 0.5 0.5 0.5 . '''
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_reload_cifdata(self): file_content = "data_test _cell_length_a 10(1)" with tempfile.NamedTemporaryFile(mode='w+') as tmpf: filename = tmpf.name basename = os.path.split(filename)[1] tmpf.write(file_content) tmpf.flush() a = CifData(filepath=filename, source={'version': '1234', 'db_name': 'COD', 'id': '0000001'}) # Key 'db_kind' is not allowed in source description: with self.assertRaises(KeyError): a.source = {'db_kind': 'small molecule'} the_uuid = a.uuid self.assertEquals(a.list_object_names(), [basename]) with a.open() as fhandle: self.assertEquals(fhandle.read(), file_content) a.store() self.assertEquals(a.source, { 'db_name': 'COD', 'id': '0000001', 'version': '1234', }) with a.open() as fhandle: self.assertEquals(fhandle.read(), file_content) self.assertEquals(a.list_object_names(), [basename]) b = load_node(the_uuid) # I check the retrieved object self.assertTrue(isinstance(b, CifData)) self.assertEquals(b.list_object_names(), [basename]) with b.open() as fhandle: self.assertEquals(fhandle.read(), file_content) # Checking the get_or_create() method: with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(file_content) tmpf.flush() c, created = CifData.get_or_create(tmpf.name, store_cif=False) self.assertTrue(isinstance(c, CifData)) self.assertTrue(not created) self.assertEquals(c.get_content(), file_content) other_content = "data_test _cell_length_b 10(1)" with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(other_content) tmpf.flush() c, created = CifData.get_or_create(tmpf.name, store_cif=False) self.assertTrue(isinstance(c, CifData)) self.assertTrue(created) self.assertEquals(c.get_content(), other_content)
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_parse_cifdata(self): file_content = "data_test _cell_length_a 10(1)" with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(file_content) tmpf.flush() a = CifData(filepath=tmpf.name) self.assertEquals(list(a.values.keys()), ['test'])
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_change_cifdata_file(self): file_content_1 = "data_test _cell_length_a 10(1)" file_content_2 = "data_test _cell_length_a 11(1)" with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(file_content_1) tmpf.flush() a = CifData(filepath=tmpf.name) self.assertEquals(a.values['test']['_cell_length_a'], '10(1)') with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(file_content_2) tmpf.flush() a.put_object_from_file(tmpf.name) self.assertEquals(a.values['test']['_cell_length_a'], '11(1)')
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_get_structure(self): with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(''' data_test _cell_length_a 10 _cell_length_b 10 _cell_length_c 10 _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 90 loop_ _symmetry_equiv_pos_site_id _symmetry_equiv_pos_as_xyz 1 +x,+y,+z loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z C 0 0 0 O 0.5 0.5 0.5 ''') tmpf.flush() a = CifData(filepath=tmpf.name) with self.assertRaises(ValueError): a.get_structure(converter='none') c = a.get_structure() self.assertEquals(c.get_kind_names(), ['C', 'O'])
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_ase_primitive_and_conventional_cells_ase(self): """ Checking the number of atoms per primitive/conventional cell returned by ASE ase.io.read() method. Test input is adapted from http://www.crystallography.net/cod/9012064.cif@120115 """ import ase with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(''' data_9012064 _space_group_IT_number 166 _symmetry_space_group_name_H-M 'R -3 m :H' _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 120 _cell_length_a 4.395 _cell_length_b 4.395 _cell_length_c 30.440 _cod_database_code 9012064 loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z _atom_site_U_iso_or_equiv Bi 0.00000 0.00000 0.40046 0.02330 Te1 0.00000 0.00000 0.00000 0.01748 Te2 0.00000 0.00000 0.79030 0.01912 ''') tmpf.flush() c = CifData(filepath=tmpf.name) ase = c.get_structure(converter='ase', primitive_cell=False).get_ase() self.assertEquals(ase.get_number_of_atoms(), 15) ase = c.get_structure(converter='ase').get_ase() self.assertEquals(ase.get_number_of_atoms(), 15) ase = c.get_structure(converter='ase', primitive_cell=True, subtrans_included=False).get_ase() self.assertEquals(ase.get_number_of_atoms(), 5)
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_ase_primitive_and_conventional_cells_pymatgen(self): """ Checking the number of atoms per primitive/conventional cell returned by ASE ase.io.read() method. Test input is adapted from http://www.crystallography.net/cod/9012064.cif@120115 """ with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(''' data_9012064 _space_group_IT_number 166 _symmetry_space_group_name_H-M 'R -3 m :H' _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 120 _cell_length_a 4.395 _cell_length_b 4.395 _cell_length_c 30.440 _cod_database_code 9012064 loop_ _symmetry_equiv_pos_as_xyz x,y,z 2/3+x,1/3+y,1/3+z 1/3+x,2/3+y,2/3+z x,x-y,z 2/3+x,1/3+x-y,1/3+z 1/3+x,2/3+x-y,2/3+z y,x,-z 2/3+y,1/3+x,1/3-z 1/3+y,2/3+x,2/3-z -x+y,y,z loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z _atom_site_U_iso_or_equiv Bi 0.00000 0.00000 0.40046 0.02330 Te1 0.00000 0.00000 0.00000 0.01748 Te2 0.00000 0.00000 0.79030 0.01912 ''') tmpf.flush() c = CifData(filepath=tmpf.name) ase = c.get_structure(converter='pymatgen', primitive_cell=False).get_ase() self.assertEquals(ase.get_number_of_atoms(), 15) ase = c.get_structure(converter='pymatgen').get_ase() self.assertEquals(ase.get_number_of_atoms(), 15) ase = c.get_structure(converter='pymatgen', primitive_cell=True).get_ase() self.assertEquals(ase.get_number_of_atoms(), 5)
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_pycifrw_from_datablocks(self): """ Tests CifData.pycifrw_from_cif() """ from aiida.orm.nodes.data.cif import pycifrw_from_cif import re datablocks = [{ '_atom_site_label': ['A', 'B', 'C'], '_atom_site_occupancy': [1.0, 0.5, 0.5], '_publ_section_title': 'Test CIF' }] with Capturing(): lines = pycifrw_from_cif(datablocks).WriteOut().split('\n') non_comments = [] for line in lines: if not re.search('^#', line): non_comments.append(line) self.assertEquals( simplify("\n".join(non_comments)), simplify(''' data_0 loop_ _atom_site_label A B C loop_ _atom_site_occupancy 1.0 0.5 0.5 _publ_section_title 'Test CIF' ''')) loops = {'_atom_site': ['_atom_site_label', '_atom_site_occupancy']} with Capturing(): lines = pycifrw_from_cif(datablocks, loops).WriteOut().split('\n') non_comments = [] for line in lines: if not re.search('^#', line): non_comments.append(line) self.assertEquals( simplify("\n".join(non_comments)), simplify(''' data_0 loop_ _atom_site_label _atom_site_occupancy A 1.0 B 0.5 C 0.5 _publ_section_title 'Test CIF' '''))
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_pycifrw_syntax(self): """ Tests CifData.pycifrw_from_cif() - check syntax pb in PyCifRW 3.6 """ from aiida.orm.nodes.data.cif import pycifrw_from_cif import re datablocks = [{ '_tag': '[value]', }] with Capturing(): lines = pycifrw_from_cif(datablocks).WriteOut().split('\n') non_comments = [] for line in lines: if not re.search('^#', line): non_comments.append(line) self.assertEquals( simplify("\n".join(non_comments)), simplify(''' data_0 _tag '[value]' '''))
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_cif_with_long_line(self): """ Tests CifData - check that long lines (longer than 2048 characters) are supported. Should not raise any error. """ with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(''' data_0 _tag {} '''.format('a' * 5000)) tmpf.flush() _ = CifData(filepath=tmpf.name)
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_cif_roundtrip(self): with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(''' data_test _cell_length_a 10 _cell_length_b 10 _cell_length_c 10 _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 90 loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z C 0 0 0 O 0.5 0.5 0.5 _cod_database_code 0000001 _[local]_flags '' ''') tmpf.flush() a = CifData(filepath=tmpf.name) b = CifData(values=a.values) c = CifData(values=b.values) self.assertEquals(b._prepare_cif(), c._prepare_cif()) b = CifData(ase=a.ase) c = CifData(ase=b.ase) self.assertEquals(b._prepare_cif(), c._prepare_cif())
[docs] def test_symop_string_from_symop_matrix_tr(self): from aiida.tools.data.cif import symop_string_from_symop_matrix_tr self.assertEquals(symop_string_from_symop_matrix_tr([[1, 0, 0], [0, 1, 0], [0, 0, 1]]), "x,y,z") self.assertEquals(symop_string_from_symop_matrix_tr([[1, 0, 0], [0, -1, 0], [0, 1, 1]]), "x,-y,y+z") self.assertEquals( symop_string_from_symop_matrix_tr([[-1, 0, 0], [0, 1, 0], [0, 0, 1]], [1, -1, 0]), "-x+1,y-1,z")
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_attached_hydrogens(self): with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(''' data_test _cell_length_a 10 _cell_length_b 10 _cell_length_c 10 _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 90 loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z _atom_site_attached_hydrogens C 0 0 0 ? O 0.5 0.5 0.5 . H 0.75 0.75 0.75 0 ''') tmpf.flush() a = CifData(filepath=tmpf.name) self.assertEqual(a.has_attached_hydrogens, False) with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(''' data_test _cell_length_a 10 _cell_length_b 10 _cell_length_c 10 _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 90 loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z _atom_site_attached_hydrogens C 0 0 0 ? O 0.5 0.5 0.5 1 H 0.75 0.75 0.75 0 ''') tmpf.flush() a = CifData(filepath=tmpf.name) self.assertEqual(a.has_attached_hydrogens, True)
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") @unittest.skipIf(not has_spglib(), "Unable to import spglib") def test_refine(self): """ Test case for refinement (space group determination) for a CifData object. """ from aiida.tools.data.cif import refine_inline with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(''' data_test _cell_length_a 10 _cell_length_b 10 _cell_length_c 10 _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 90 loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z C 0.5 0.5 0.5 O 0.25 0.5 0.5 O 0.75 0.5 0.5 ''') tmpf.flush() a = CifData(filepath=tmpf.name) ret_dict = refine_inline(a) b = ret_dict['cif'] self.assertEqual(list(b.values.keys()), ['test']) self.assertEqual(b.values['test']['_chemical_formula_sum'], 'C O2') self.assertEqual(b.values['test']['_symmetry_equiv_pos_as_xyz'], [ 'x,y,z', '-x,-y,-z', '-y,x,z', 'y,-x,-z', '-x,-y,z', 'x,y,-z', 'y,-x,z', '-y,x,-z', 'x,-y,-z', '-x,y,z', '-y,-x,-z', 'y,x,z', '-x,y,-z', 'x,-y,z', 'y,x,-z', '-y,-x,z' ]) with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(''' data_a data_b ''') tmpf.flush() c = CifData(filepath=tmpf.name) with self.assertRaises(ValueError): ret_dict = refine_inline(c)
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") @unittest.skipIf(not has_spglib(), "Unable to import spglib") def test_parse_formula(self): from aiida.orm.nodes.data.cif import parse_formula self.assertEqual(parse_formula("C H"), {'C': 1, 'H': 1}) self.assertEqual(parse_formula("C5 H1"), {'C': 5, 'H': 1}) self.assertEqual(parse_formula("Ca5 Ho"), {'Ca': 5, 'Ho': 1}) self.assertEqual(parse_formula("H0.5 O"), {'H': 0.5, 'O': 1}) self.assertEqual(parse_formula("C0 O0"), {'C': 0, 'O': 0}) self.assertEqual(parse_formula("C1 H1 "), {'C': 1, 'H': 1}) # Trailing spaces should be accepted self.assertEqual(parse_formula(" C1 H1"), {'C': 1, 'H': 1}) # Leading spaces should be accepted # Invalid literal for float() with self.assertRaises(ValueError): parse_formula("H0.5.2 O")
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_scan_type(self): """ Check that different scan_types of PyCifRW produce the same result. """ with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(self.valid_sample_cif_str) tmpf.flush() default = CifData(filepath=tmpf.name) default2 = CifData(filepath=tmpf.name, scan_type='standard') self.assertEquals(default._prepare_cif(), default2._prepare_cif()) flex = CifData(filepath=tmpf.name, scan_type='flex') self.assertEquals(default._prepare_cif(), flex._prepare_cif())
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_empty_cif(self): """ Test empty CifData Note: This test does not need PyCifRW. """ with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(self.valid_sample_cif_str) tmpf.flush() # empty cifdata should be possible a = CifData() # but it does not have a file with self.assertRaises(AttributeError): a.filename #now it has a.put_object_from_file(tmpf.name) a.filename a.store()
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_parse_policy(self): """ Test that loading of CIF file occurs as defined by parse_policy. """ with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(self.valid_sample_cif_str) tmpf.flush() # this will parse the cif eager = CifData(filepath=tmpf.name, parse_policy='eager') self.assertIsNot(eager._values, None) # this should not parse the cif lazy = CifData(filepath=tmpf.name, parse_policy='lazy') self.assertIs(lazy._values, None) # also lazy-loaded nodes should be storable lazy.store() # this should parse the cif lazy.values self.assertIsNot(lazy._values, None)
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_put_object_from_file(self): """ Test that setting a new file clears formulae and spacegroups. """ with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(self.valid_sample_cif_str) tmpf.flush() a = CifData(filepath=tmpf.name) f1 = a.get_formulae() self.assertIsNot(f1, None) with tempfile.NamedTemporaryFile(mode='w+') as tmpf: tmpf.write(self.valid_sample_cif_str_2) tmpf.flush() # this should reset formulae and spacegroup_numbers a.put_object_from_file(tmpf.name) self.assertIs(a.get_attribute('formulae'), None) self.assertIs(a.get_attribute('spacegroup_numbers'), None) # this should populate formulae a.parse() f2 = a.get_formulae() self.assertIsNot(f2, None) # empty cifdata should be possible a = CifData() # but it does not have a file with self.assertRaises(AttributeError): a.filename # now it has a.put_object_from_file(tmpf.name) a.parse() a.filename self.assertNotEquals(f1, f2)
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_has_partial_occupancies(self): tests = [ # Unreadable occupations should not count as a partial occupancy ('O 0.5 0.5(1) 0.5 ?', False), # The default epsilon for deviation of unity for an occupation to be considered partial is 1E-6 ('O 0.5 0.5(1) 0.5 1.0(000000001)', False), # Partial occupancies should be able to deal with parentheses in the value ('O 0.5 0.5(1) 0.5 1.0(000132)', True), # Partial occupancies should be able to deal with parentheses in the value ('O 0.5 0.5(1) 0.5 0.9(0000132)', True), ] for test_string, result in tests: with tempfile.NamedTemporaryFile(mode='w+') as handle: handle.write(""" data_test loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z _atom_site_occupancy {} """.format(test_string)) handle.flush() cif = CifData(filepath=handle.name) self.assertEqual(cif.has_partial_occupancies, result)
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_has_unknown_species(self): tests = [ ('H2 O', False), # No unknown species ('OsAx', True), # Ax is an unknown specie ('UX', True), # X counts as unknown specie despite being defined in aiida.common.constants.elements ('', None), # If no chemical formula is defined, None should be returned ] for formula, result in tests: with tempfile.NamedTemporaryFile(mode='w+') as handle: formula_string = "_chemical_formula_sum '{}'".format(formula) if formula else '\n' handle.write("""data_test\n{}\n""".format(formula_string)) handle.flush() cif = CifData(filepath=handle.name) self.assertEqual(cif.has_unknown_species, result, formula_string)
[docs] @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_has_undefined_atomic_sites(self): tests = [ ('C 0.0 0.0 0.0', False), # Should return False because all sites have valid coordinates ('C 0.0 0.0 ?', True), # Should return True because one site has an undefined coordinate ('', True), # Should return True if no sites defined at all ] for test_string, result in tests: with tempfile.NamedTemporaryFile(mode='w+') as handle: base = 'loop_\n_atom_site_label\n_atom_site_fract_x\n_atom_site_fract_y\n_atom_site_fract_z' atomic_site_string = '{}\n{}'.format(base, test_string) if test_string else '' handle.write("""data_test\n{}\n""".format(atomic_site_string)) handle.flush() cif = CifData(filepath=handle.name) self.assertEqual(cif.has_undefined_atomic_sites, result)
[docs]class TestKindValidSymbols(AiidaTestCase): """ Tests the symbol validation of the aiida.orm.nodes.data.structure.Kind class. """
[docs] def test_bad_symbol(self): """ Should not accept a non-existing symbol. """ with self.assertRaises(ValueError): Kind(symbols='Hxx')
[docs] def test_empty_list_symbols(self): """ Should not accept an empty list """ with self.assertRaises(ValueError): Kind(symbols=[])
[docs] def test_valid_list(self): """ Should not raise any error. """ Kind(symbols=['H', 'He'], weights=[0.5, 0.5])
[docs] def test_unknown_symbol(self): """ Should test if symbol X is valid and defined in the elements dictionary. """ Kind(symbols=['X'])
[docs]class TestSiteValidWeights(AiidaTestCase): """ Tests valid weight lists. """
[docs] def test_isnot_list(self): """ Should not accept a non-list, non-number weight """ with self.assertRaises(ValueError): Kind(symbols='Ba', weights='aaa')
[docs] def test_empty_list_weights(self): """ Should not accept an empty list """ with self.assertRaises(ValueError): Kind(symbols='Ba', weights=[])
[docs] def test_symbol_weight_mismatch(self): """ Should not accept a size mismatch of the symbols and weights list. """ with self.assertRaises(ValueError): Kind(symbols=['Ba', 'C'], weights=[1.]) with self.assertRaises(ValueError): Kind(symbols=['Ba'], weights=[0.1, 0.2])
[docs] def test_negative_value(self): """ Should not accept a negative weight """ with self.assertRaises(ValueError): Kind(symbols=['Ba', 'C'], weights=[-0.1, 0.3])
[docs] def test_sum_greater_one(self): """ Should not accept a sum of weights larger than one """ with self.assertRaises(ValueError): Kind(symbols=['Ba', 'C'], weights=[0.5, 0.6])
[docs] def test_sum_one_weights(self): """ Should accept a sum equal to one """ Kind(symbols=['Ba', 'C'], weights=[1. / 3., 2. / 3.])
[docs] def test_sum_less_one_weights(self): """ Should accept a sum equal less than one """ Kind(symbols=['Ba', 'C'], weights=[1. / 3., 1. / 3.])
[docs] def test_none(self): """ Should accept None. """ Kind(symbols='Ba', weights=None)
[docs]class TestKindTestGeneral(AiidaTestCase): """ Tests the creation of Kind objects and their methods. """
[docs] def test_sum_one_general(self): """ Should accept a sum equal to one """ a = Kind(symbols=['Ba', 'C'], weights=[1. / 3., 2. / 3.]) self.assertTrue(a.is_alloy) self.assertFalse(a.has_vacancies)
[docs] def test_sum_less_one_general(self): """ Should accept a sum equal less than one """ a = Kind(symbols=['Ba', 'C'], weights=[1. / 3., 1. / 3.]) self.assertTrue(a.is_alloy) self.assertTrue(a.has_vacancies)
[docs] def test_no_position(self): """ Should not accept a 'positions' parameter """ with self.assertRaises(ValueError): Kind(position=[0., 0., 0.], symbols=['Ba'], weights=[1.])
[docs] def test_simple(self): """ Should recognize a simple element. """ a = Kind(symbols='Ba') self.assertFalse(a.is_alloy) self.assertFalse(a.has_vacancies) b = Kind(symbols='Ba', weights=1.) self.assertFalse(b.is_alloy) self.assertFalse(b.has_vacancies) c = Kind(symbols='Ba', weights=None) self.assertFalse(c.is_alloy) self.assertFalse(c.has_vacancies)
[docs] def test_automatic_name(self): """ Check the automatic name generator. """ a = Kind(symbols='Ba') self.assertEqual(a.name, 'Ba') a = Kind(symbols='X') self.assertEqual(a.name, 'X') a = Kind(symbols=('Si', 'Ge'), weights=(1. / 3., 2. / 3.)) self.assertEqual(a.name, 'GeSi') a = Kind(symbols=('Si', 'X'), weights=(1. / 3., 2. / 3.)) self.assertEqual(a.name, 'SiX') a = Kind(symbols=('Si', 'Ge'), weights=(0.4, 0.5)) self.assertEqual(a.name, 'GeSiX') a = Kind(symbols=('Si', 'X'), weights=(0.4, 0.5)) self.assertEqual(a.name, 'SiXX') # Manually setting the name of the species a.name = 'newstring' self.assertEqual(a.name, 'newstring')
[docs]class TestKindTestMasses(AiidaTestCase): """ Tests the management of masses during the creation of Kind objects. """
[docs] def test_auto_mass_one(self): """ mass for elements with sum one """ from aiida.orm.nodes.data.structure import _atomic_masses a = Kind(symbols=['Ba', 'C'], weights=[1. / 3., 2. / 3.]) self.assertAlmostEqual(a.mass, (_atomic_masses['Ba'] + 2. * _atomic_masses['C']) / 3.)
[docs] def test_sum_less_one_masses(self): """ mass for elements with sum less than one """ from aiida.orm.nodes.data.structure import _atomic_masses a = Kind(symbols=['Ba', 'C'], weights=[1. / 3., 1. / 3.]) self.assertAlmostEqual(a.mass, (_atomic_masses['Ba'] + _atomic_masses['C']) / 2.)
[docs] def test_sum_less_one_singleelem(self): """ mass for a single element """ from aiida.orm.nodes.data.structure import _atomic_masses a = Kind(symbols=['Ba']) self.assertAlmostEqual(a.mass, _atomic_masses['Ba'])
[docs] def test_manual_mass(self): """ mass set manually """ a = Kind(symbols=['Ba', 'C'], weights=[1. / 3., 1. / 3.], mass=1000.) self.assertAlmostEqual(a.mass, 1000.)
[docs]class TestStructureDataInit(AiidaTestCase): """ Tests the creation of StructureData objects (cell and pbc). """
[docs] def test_cell_wrong_size_1(self): """ Wrong cell size (not 3x3) """ with self.assertRaises(ValueError): StructureData(cell=((1., 2., 3.),))
[docs] def test_cell_wrong_size_2(self): """ Wrong cell size (not 3x3) """ with self.assertRaises(ValueError): StructureData(cell=((1., 0., 0.), (0., 0., 3.), (0., 3.)))
[docs] def test_cell_zero_vector(self): """ Wrong cell (one vector has zero length) """ with self.assertRaises(ValueError): StructureData(cell=((0., 0., 0.), (0., 1., 0.), (0., 0., 1.)))
[docs] def test_cell_zero_volume(self): """ Wrong cell (volume is zero) """ with self.assertRaises(ValueError): StructureData(cell=((1., 0., 0.), (0., 1., 0.), (1., 1., 0.)))
[docs] def test_cell_ok_init(self): """ Correct cell """ cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) a = StructureData(cell=cell) out_cell = a.cell for i in range(3): for j in range(3): self.assertAlmostEqual(cell[i][j], out_cell[i][j])
[docs] def test_volume(self): """ Check the volume calculation """ a = StructureData(cell=((1., 0., 0.), (0., 2., 0.), (0., 0., 3.))) self.assertAlmostEqual(a.get_cell_volume(), 6.)
[docs] def test_wrong_pbc_1(self): """ Wrong pbc parameter (not bool or iterable) """ with self.assertRaises(ValueError): cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) StructureData(cell=cell, pbc=1)
[docs] def test_wrong_pbc_2(self): """ Wrong pbc parameter (iterable but with wrong len) """ with self.assertRaises(ValueError): cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) StructureData(cell=cell, pbc=[True, True])
[docs] def test_wrong_pbc_3(self): """ Wrong pbc parameter (iterable but with wrong len) """ with self.assertRaises(ValueError): cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) StructureData(cell=cell, pbc=[])
[docs] def test_ok_pbc_1(self): """ Single pbc value """ cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) a = StructureData(cell=cell, pbc=True) self.assertEquals(a.pbc, tuple([True, True, True])) a = StructureData(cell=cell, pbc=False) self.assertEquals(a.pbc, tuple([False, False, False]))
[docs] def test_ok_pbc_2(self): """ One-element list """ cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) a = StructureData(cell=cell, pbc=[True]) self.assertEqual(a.pbc, tuple([True, True, True])) a = StructureData(cell=cell, pbc=[False]) self.assertEqual(a.pbc, tuple([False, False, False]))
[docs] def test_ok_pbc_3(self): """ Three-element list """ cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) a = StructureData(cell=cell, pbc=[True, False, True]) self.assertEqual(a.pbc, tuple([True, False, True]))
[docs]class TestStructureData(AiidaTestCase): """ Tests the creation of StructureData objects (cell and pbc). """ from aiida.orm.nodes.data.structure import has_ase, has_spglib from aiida.orm.nodes.data.cif import has_pycifrw
[docs] def test_cell_ok_and_atoms(self): """ Test the creation of a cell and the appending of atoms """ cell = [[2., 0., 0.], [0., 2., 0.], [0., 0., 2.]] a = StructureData(cell=cell) out_cell = a.cell self.assertAlmostEquals(cell, out_cell) a.append_atom(position=(0., 0., 0.), symbols=['Ba']) a.append_atom(position=(1., 1., 1.), symbols=['Ti']) a.append_atom(position=(1.2, 1.4, 1.6), symbols=['Ti']) self.assertFalse(a.is_alloy) self.assertFalse(a.has_vacancies) # There should be only two kinds! (two atoms of kind Ti should # belong to the same kind) self.assertEquals(len(a.kinds), 2) a.append_atom(position=(0.5, 1., 1.5), symbols=['O', 'C'], weights=[0.5, 0.5]) self.assertTrue(a.is_alloy) self.assertFalse(a.has_vacancies) a.append_atom(position=(0.5, 1., 1.5), symbols=['O'], weights=[0.5]) self.assertTrue(a.is_alloy) self.assertTrue(a.has_vacancies) a.clear_kinds() a.append_atom(position=(0.5, 1., 1.5), symbols=['O'], weights=[0.5]) self.assertFalse(a.is_alloy) self.assertTrue(a.has_vacancies)
[docs] def test_cell_ok_and_unknown_atoms(self): """ Test the creation of a cell and the appending of atoms, including the unknown entry. """ cell = [[2., 0., 0.], [0., 2., 0.], [0., 0., 2.]] a = StructureData(cell=cell) out_cell = a.cell self.assertAlmostEquals(cell, out_cell) a.append_atom(position=(0., 0., 0.), symbols=['Ba']) a.append_atom(position=(1., 1., 1.), symbols=['X']) a.append_atom(position=(1.2, 1.4, 1.6), symbols=['X']) self.assertFalse(a.is_alloy) self.assertFalse(a.has_vacancies) # There should be only two kinds! (two atoms of kind X should # belong to the same kind) self.assertEquals(len(a.kinds), 2) a.append_atom(position=(0.5, 1., 1.5), symbols=['O', 'C'], weights=[0.5, 0.5]) self.assertTrue(a.is_alloy) self.assertFalse(a.has_vacancies) a.append_atom(position=(0.5, 1., 1.5), symbols=['O'], weights=[0.5]) self.assertTrue(a.is_alloy) self.assertTrue(a.has_vacancies) a.clear_kinds() a.append_atom(position=(0.5, 1., 1.5), symbols=['X'], weights=[0.5]) self.assertFalse(a.is_alloy) self.assertTrue(a.has_vacancies)
[docs] def test_kind_1(self): """ Test the management of kinds (automatic detection of kind of simple atoms). """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols=['Ba']) a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Ba']) a.append_atom(position=(1., 1., 1.), symbols=['Ti']) self.assertEqual(len(a.kinds), 2) # I should only have two types # I check for the default names of kinds self.assertEqual(set(k.name for k in a.kinds), set(('Ba', 'Ti')))
[docs] def test_kind_1_unknown(self): """ Test the management of kinds (automatic detection of kind of simple atoms), inluding the unknown entry. """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols=['X']) a.append_atom(position=(0.5, 0.5, 0.5), symbols=['X']) a.append_atom(position=(1., 1., 1.), symbols=['Ti']) self.assertEqual(len(a.kinds), 2) # I should only have two types # I check for the default names of kinds self.assertEqual(set(k.name for k in a.kinds), set(('X', 'Ti')))
[docs] def test_kind_2(self): """ Test the management of kinds (manual specification of kind name). """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols=['Ba'], name='Ba1') a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Ba'], name='Ba2') a.append_atom(position=(1., 1., 1.), symbols=['Ti']) kind_list = a.kinds self.assertEqual(len(kind_list), 3) # I should have now three kinds self.assertEqual(set(k.name for k in kind_list), set(('Ba1', 'Ba2', 'Ti')))
[docs] def test_kind_2_unknown(self): """ Test the management of kinds (manual specification of kind name), including the unknown entry. """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols=['X'], name='X1') a.append_atom(position=(0.5, 0.5, 0.5), symbols=['X'], name='X2') a.append_atom(position=(1., 1., 1.), symbols=['Ti']) kind_list = a.kinds self.assertEqual(len(kind_list), 3) # I should have now three kinds self.assertEqual(set(k.name for k in kind_list), set(('X1', 'X2', 'Ti')))
[docs] def test_kind_3(self): """ Test the management of kinds (adding an atom with different mass). """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols=['Ba'], mass=100.) with self.assertRaises(ValueError): # Shouldn't allow, I am adding two sites with the same name 'Ba' a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Ba'], mass=101., name='Ba') # now it should work because I am using a different kind name a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Ba'], mass=101., name='Ba2') a.append_atom(position=(1., 1., 1.), symbols=['Ti']) self.assertEqual(len(a.kinds), 3) # I should have now three types self.assertEqual(len(a.sites), 3) # and 3 sites self.assertEqual(set(k.name for k in a.kinds), set(('Ba', 'Ba2', 'Ti')))
[docs] def test_kind_3_unknown(self): """ Test the management of kinds (adding an atom with different mass), including the unknown entry. """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols=['X'], mass=100.) with self.assertRaises(ValueError): # Shouldn't allow, I am adding two sites with the same name 'Ba' a.append_atom(position=(0.5, 0.5, 0.5), symbols=['X'], mass=101., name='X') # now it should work because I am using a different kind name a.append_atom(position=(0.5, 0.5, 0.5), symbols=['X'], mass=101., name='X2') a.append_atom(position=(1., 1., 1.), symbols=['Ti']) self.assertEqual(len(a.kinds), 3) # I should have now three types self.assertEqual(len(a.sites), 3) # and 3 sites self.assertEqual(set(k.name for k in a.kinds), set(('X', 'X2', 'Ti')))
[docs] def test_kind_4(self): """ Test the management of kind (adding an atom with different symbols or weights). """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols=['Ba', 'Ti'], weights=(1., 0.), name='mytype') with self.assertRaises(ValueError): # Shouldn't allow, different weights a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Ba', 'Ti'], weights=(0.9, 0.1), name='mytype') with self.assertRaises(ValueError): # Shouldn't allow, different weights (with vacancy) a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Ba', 'Ti'], weights=(0.8, 0.1), name='mytype') with self.assertRaises(ValueError): # Shouldn't allow, different symbols list a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Ba'], name='mytype') with self.assertRaises(ValueError): # Shouldn't allow, different symbols list a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Si', 'Ti'], weights=(1., 0.), name='mytype') # should allow because every property is identical a.append_atom(position=(0., 0., 0.), symbols=['Ba', 'Ti'], weights=(1., 0.), name='mytype') self.assertEquals(len(a.kinds), 1)
[docs] def test_kind_4_unknown(self): """ Test the management of kind (adding an atom with different symbols or weights), including the unknown entry. """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols=['X', 'Ti'], weights=(1., 0.), name='mytype') with self.assertRaises(ValueError): # Shouldn't allow, different weights a.append_atom(position=(0.5, 0.5, 0.5), symbols=['X', 'Ti'], weights=(0.9, 0.1), name='mytype') with self.assertRaises(ValueError): # Shouldn't allow, different weights (with vacancy) a.append_atom(position=(0.5, 0.5, 0.5), symbols=['X', 'Ti'], weights=(0.8, 0.1), name='mytype') with self.assertRaises(ValueError): # Shouldn't allow, different symbols list a.append_atom(position=(0.5, 0.5, 0.5), symbols=['X'], name='mytype') with self.assertRaises(ValueError): # Shouldn't allow, different symbols list a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Si', 'Ti'], weights=(1., 0.), name='mytype') # should allow because every property is identical a.append_atom(position=(0., 0., 0.), symbols=['X', 'Ti'], weights=(1., 0.), name='mytype') self.assertEquals(len(a.kinds), 1)
[docs] def test_kind_5(self): """ Test the management of kinds (automatic creation of new kind if name is not specified and properties are different). """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols='Ba', mass=100.) a.append_atom(position=(0., 0., 0.), symbols='Ti') # The name does not exist a.append_atom(position=(0., 0., 0.), symbols='Ti', name='Ti2') # The name already exists, but the properties are identical => OK a.append_atom(position=(1., 1., 1.), symbols='Ti', name='Ti2') # The name already exists, but the properties are different! with self.assertRaises(ValueError): a.append_atom(position=(1., 1., 1.), symbols='Ti', mass=100., name='Ti2') # Should not complain, should create a new type a.append_atom(position=(0., 0., 0.), symbols='Ba', mass=150.) # There should be 4 kinds, the automatic name for the last one # should be Ba1 self.assertEquals([k.name for k in a.kinds], ['Ba', 'Ti', 'Ti2', 'Ba1']) self.assertEquals(len(a.sites), 5)
[docs] def test_kind_5_unknown(self): """ Test the management of kinds (automatic creation of new kind if name is not specified and properties are different), including the unknown entry. """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols='X', mass=100.) a.append_atom(position=(0., 0., 0.), symbols='Ti') # The name does not exist a.append_atom(position=(0., 0., 0.), symbols='Ti', name='Ti2') # The name already exists, but the properties are identical => OK a.append_atom(position=(1., 1., 1.), symbols='Ti', name='Ti2') # The name already exists, but the properties are different! with self.assertRaises(ValueError): a.append_atom(position=(1., 1., 1.), symbols='Ti', mass=100., name='Ti2') # Should not complain, should create a new type a.append_atom(position=(0., 0., 0.), symbols='X', mass=150.) # There should be 4 kinds, the automatic name for the last one # should be Ba1 self.assertEquals([k.name for k in a.kinds], ['X', 'Ti', 'Ti2', 'X1']) self.assertEquals(len(a.sites), 5)
[docs] def test_kind_5_bis(self): """ Test the management of kinds (automatic creation of new kind if name is not specified and properties are different). This test was failing in, e.g., commit f6a8f4b. """ from aiida.common.constants import elements s = StructureData(cell=((6., 0., 0.), (0., 6., 0.), (0., 0., 6.))) s.append_atom(symbols='Fe', position=[0, 0, 0], mass=12) s.append_atom(symbols='Fe', position=[1, 0, 0], mass=12) s.append_atom(symbols='Fe', position=[2, 0, 0], mass=12) s.append_atom(symbols='Fe', position=[2, 0, 0]) s.append_atom(symbols='Fe', position=[4, 0, 0]) # I expect only two species, the first one with name 'Fe', mass 12, # and referencing the first three atoms; the second with name # 'Fe1', mass = elements[26]['mass'], and referencing the last two atoms self.assertEquals(set([(k.name, k.mass) for k in s.kinds]), set([('Fe', 12.0), ('Fe1', elements[26]['mass'])])) kind_of_each_site = [site.kind_name for site in s.sites] self.assertEquals(kind_of_each_site, ['Fe', 'Fe', 'Fe', 'Fe1', 'Fe1'])
[docs] def test_kind_5_bis_unknown(self): """ Test the management of kinds (automatic creation of new kind if name is not specified and properties are different). This test was failing in, e.g., commit f6a8f4b. This also includes the unknown entry. """ from aiida.common.constants import elements s = StructureData(cell=((6., 0., 0.), (0., 6., 0.), (0., 0., 6.))) s.append_atom(symbols='X', position=[0, 0, 0], mass=12) s.append_atom(symbols='X', position=[1, 0, 0], mass=12) s.append_atom(symbols='X', position=[2, 0, 0], mass=12) s.append_atom(symbols='X', position=[2, 0, 0]) s.append_atom(symbols='X', position=[4, 0, 0]) # I expect only two species, the first one with name 'X', mass 12, # and referencing the first three atoms; the second with name # 'X', mass = elements[0]['mass'], and referencing the last two atoms self.assertEquals(set([(k.name, k.mass) for k in s.kinds]), set([('X', 12.0), ('X1', elements[0]['mass'])])) kind_of_each_site = [site.kind_name for site in s.sites] self.assertEquals(kind_of_each_site, ['X', 'X', 'X', 'X1', 'X1'])
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") def test_kind_5_bis_ase(self): """ Same test as test_kind_5_bis, but using ase """ import ase asecell = ase.Atoms('Fe5', cell=((6., 0., 0.), (0., 6., 0.), (0., 0., 6.))) asecell.set_positions([ [0, 0, 0], [1, 0, 0], [2, 0, 0], [3, 0, 0], [4, 0, 0], ]) asecell[0].mass = 12. asecell[1].mass = 12. asecell[2].mass = 12. s = StructureData(ase=asecell) # I expect only two species, the first one with name 'Fe', mass 12, # and referencing the first three atoms; the second with name # 'Fe1', mass = elements[26]['mass'], and referencing the last two atoms self.assertEquals(set([(k.name, k.mass) for k in s.kinds]), set([('Fe', 12.0), ('Fe1', asecell[3].mass)])) kind_of_each_site = [site.kind_name for site in s.sites] self.assertEquals(kind_of_each_site, ['Fe', 'Fe', 'Fe', 'Fe1', 'Fe1'])
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") def test_kind_5_bis_ase_unknown(self): """ Same test as test_kind_5_bis_unknown, but using ase """ import ase asecell = ase.Atoms('X5', cell=((6., 0., 0.), (0., 6., 0.), (0., 0., 6.))) asecell.set_positions([ [0, 0, 0], [1, 0, 0], [2, 0, 0], [3, 0, 0], [4, 0, 0], ]) asecell[0].mass = 12. asecell[1].mass = 12. asecell[2].mass = 12. s = StructureData(ase=asecell) # I expect only two species, the first one with name 'X', mass 12, # and referencing the first three atoms; the second with name # 'X1', mass = elements[26]['mass'], and referencing the last two atoms self.assertEquals(set([(k.name, k.mass) for k in s.kinds]), set([('X', 12.0), ('X1', asecell[3].mass)])) kind_of_each_site = [site.kind_name for site in s.sites] self.assertEquals(kind_of_each_site, ['X', 'X', 'X', 'X1', 'X1'])
[docs] def test_kind_6(self): """ Test the returning of kinds from the string name (most of the code copied from :py:meth:`.test_kind_5`). """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols='Ba', mass=100.) a.append_atom(position=(0., 0., 0.), symbols='Ti') # The name does not exist a.append_atom(position=(0., 0., 0.), symbols='Ti', name='Ti2') # The name already exists, but the properties are identical => OK a.append_atom(position=(1., 1., 1.), symbols='Ti', name='Ti2') # Should not complain, should create a new type a.append_atom(position=(0., 0., 0.), symbols='Ba', mass=150.) # There should be 4 kinds, the automatic name for the last one # should be Ba1 (same check of test_kind_5 self.assertEquals([k.name for k in a.kinds], ['Ba', 'Ti', 'Ti2', 'Ba1']) ############################# # Here I start the real tests # No such kind with self.assertRaises(ValueError): a.get_kind('Ti3') k = a.get_kind('Ba1') self.assertEqual(k.symbols, ('Ba',)) self.assertAlmostEqual(k.mass, 150.)
[docs] def test_kind_6_unknown(self): """ Test the returning of kinds from the string name (most of the code copied from :py:meth:`.test_kind_5`), including the unknown entry. """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols='X', mass=100.) a.append_atom(position=(0., 0., 0.), symbols='Ti') # The name does not exist a.append_atom(position=(0., 0., 0.), symbols='Ti', name='Ti2') # The name already exists, but the properties are identical => OK a.append_atom(position=(1., 1., 1.), symbols='Ti', name='Ti2') # Should not complain, should create a new type a.append_atom(position=(0., 0., 0.), symbols='X', mass=150.) # There should be 4 kinds, the automatic name for the last one # should be Ba1 (same check of test_kind_5 self.assertEquals([k.name for k in a.kinds], ['X', 'Ti', 'Ti2', 'X1']) ############################# # Here I start the real tests # No such kind with self.assertRaises(ValueError): a.get_kind('Ti3') k = a.get_kind('X1') self.assertEqual(k.symbols, ('X',)) self.assertAlmostEqual(k.mass, 150.)
[docs] def test_kind_7(self): """ Test the functions returning the list of kinds, symbols, ... """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols='Ba', mass=100.) a.append_atom(position=(0., 0., 0.), symbols='Ti') # The name does not exist a.append_atom(position=(0., 0., 0.), symbols='Ti', name='Ti2') # The name already exists, but the properties are identical => OK a.append_atom(position=(0., 0., 0.), symbols=['O', 'H'], weights=[0.9, 0.1], mass=15.) self.assertEquals(a.get_symbols_set(), set(['Ba', 'Ti', 'O', 'H']))
[docs] def test_kind_7_unknown(self): """ Test the functions returning the list of kinds, symbols, ... including the unknown entry. """ a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols='Ba', mass=100.) a.append_atom(position=(0., 0., 0.), symbols='X') # The name does not exist a.append_atom(position=(0., 0., 0.), symbols='X', name='X2') # The name already exists, but the properties are identical => OK a.append_atom(position=(0., 0., 0.), symbols=['O', 'H'], weights=[0.9, 0.1], mass=15.) self.assertEquals(a.get_symbols_set(), set(['Ba', 'X', 'O', 'H']))
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_spglib(), "Unable to import spglib") def test_kind_8(self): """ Test the ase_refine_cell() function """ from aiida.orm.nodes.data.structure import ase_refine_cell import ase import math import numpy a = ase.Atoms(cell=[10, 10, 10]) a.append(ase.Atom('C', [0, 0, 0])) a.append(ase.Atom('C', [5, 0, 0])) b, sym = ase_refine_cell(a) sym.pop('rotations') sym.pop('translations') self.assertEquals(b.get_chemical_symbols(), ['C']) self.assertEquals(b.cell.tolist(), [[10, 0, 0], [0, 10, 0], [0, 0, 5]]) self.assertEquals(sym, {'hall': '-P 4 2', 'hm': 'P4/mmm', 'tables': 123}) a = ase.Atoms(cell=[10, 2 * math.sqrt(75), 10]) a.append(ase.Atom('C', [0, 0, 0])) a.append(ase.Atom('C', [5, math.sqrt(75), 0])) b, sym = ase_refine_cell(a) sym.pop('rotations') sym.pop('translations') self.assertEquals(b.get_chemical_symbols(), ['C']) self.assertEquals(numpy.round(b.cell, 2).tolist(), [[10, 0, 0], [-5, 8.66, 0], [0, 0, 10]]) self.assertEquals(sym, {'hall': '-P 6 2', 'hm': 'P6/mmm', 'tables': 191}) a = ase.Atoms(cell=[[10, 0, 0], [-10, 10, 0], [0, 0, 10]]) a.append(ase.Atom('C', [5, 5, 5])) a.append(ase.Atom('F', [0, 0, 0])) b, sym = ase_refine_cell(a) sym.pop('rotations') sym.pop('translations') self.assertEquals(b.get_chemical_symbols(), ['C', 'F']) self.assertEquals(b.cell.tolist(), [[10, 0, 0], [0, 10, 0], [0, 0, 10]]) self.assertEquals(b.get_scaled_positions().tolist(), [[0.5, 0.5, 0.5], [0, 0, 0]]) self.assertEquals(sym, {'hall': '-P 4 2 3', 'hm': 'Pm-3m', 'tables': 221}) a = ase.Atoms(cell=[[10, 0, 0], [-10, 10, 0], [0, 0, 10]]) a.append(ase.Atom('C', [0, 0, 0])) a.append(ase.Atom('F', [5, 5, 5])) b, sym = ase_refine_cell(a) sym.pop('rotations') sym.pop('translations') self.assertEquals(b.get_chemical_symbols(), ['C', 'F']) self.assertEquals(b.cell.tolist(), [[10, 0, 0], [0, 10, 0], [0, 0, 10]]) self.assertEquals(b.get_scaled_positions().tolist(), [[0, 0, 0], [0.5, 0.5, 0.5]]) self.assertEquals(sym, {'hall': '-P 4 2 3', 'hm': 'Pm-3m', 'tables': 221}) a = ase.Atoms(cell=[[12.132, 0, 0], [0, 6.0606, 0], [0, 0, 8.0956]]) a.append(ase.Atom('Ba', [1.5334848, 1.3999986, 2.00042276])) b, sym = ase_refine_cell(a) sym.pop('rotations') sym.pop('translations') self.assertEquals(b.cell.tolist(), [[6.0606, 0, 0], [0, 8.0956, 0], [0, 0, 12.132]]) self.assertEquals(b.get_scaled_positions().tolist(), [[0, 0, 0]]) a = ase.Atoms(cell=[10, 10, 10]) a.append(ase.Atom('C', [5, 5, 5])) a.append(ase.Atom('O', [2.5, 5, 5])) a.append(ase.Atom('O', [7.5, 5, 5])) b, sym = ase_refine_cell(a) sym.pop('rotations') sym.pop('translations') self.assertEquals(b.get_chemical_symbols(), ['C', 'O']) self.assertEquals(sym, {'hall': '-P 4 2', 'hm': 'P4/mmm', 'tables': 123}) # Generated from COD entry 1507756 # (http://www.crystallography.net/cod/1507756.cif@87343) from ase.spacegroup import crystal a = crystal( ['Ba', 'Ti', 'O', 'O'], [[0, 0, 0], [0.5, 0.5, 0.482], [0.5, 0.5, 0.016], [0.5, 0, 0.515]], cell=[3.9999, 3.9999, 4.0170], spacegroup=99) b, sym = ase_refine_cell(a) sym.pop('rotations') sym.pop('translations') self.assertEquals(b.get_chemical_symbols(), ['Ba', 'Ti', 'O', 'O']) self.assertEquals(sym, {'hall': 'P 4 -2', 'hm': 'P4mm', 'tables': 99})
[docs] def test_get_formula(self): """ Tests the generation of formula """ from aiida.orm.nodes.data.structure import get_formula self.assertEquals(get_formula(['Ba', 'Ti'] + ['O'] * 3), 'BaO3Ti') self.assertEquals(get_formula(['Ba', 'Ti', 'C'] + ['O'] * 3, separator=" "), 'C Ba O3 Ti') self.assertEquals(get_formula(['H'] * 6 + ['C'] * 6), 'C6H6') self.assertEquals(get_formula(['H'] * 6 + ['C'] * 6, mode="hill_compact"), 'CH') self.assertEquals(get_formula((['Ba', 'Ti'] + ['O'] * 3) * 2 + \ ['Ba'] + ['Ti'] * 2 + ['O'] * 3, mode="group"), '(BaTiO3)2BaTi2O3') self.assertEquals(get_formula((['Ba', 'Ti'] + ['O'] * 3) * 2 + \ ['Ba'] + ['Ti'] * 2 + ['O'] * 3, mode="group", separator=" "), '(Ba Ti O3)2 Ba Ti2 O3') self.assertEquals(get_formula((['Ba', 'Ti'] + ['O'] * 3) * 2 + \ ['Ba'] + ['Ti'] * 2 + ['O'] * 3, mode="reduce"), 'BaTiO3BaTiO3BaTi2O3') self.assertEquals(get_formula((['Ba', 'Ti'] + ['O'] * 3) * 2 + \ ['Ba'] + ['Ti'] * 2 + ['O'] * 3, mode="reduce", separator=", "), 'Ba, Ti, O3, Ba, Ti, O3, Ba, Ti2, O3') self.assertEquals(get_formula((['Ba', 'Ti'] + ['O'] * 3) * 2, mode="count"), 'Ba2Ti2O6') self.assertEquals(get_formula((['Ba', 'Ti'] + ['O'] * 3) * 2, mode="count_compact"), 'BaTiO3')
[docs] def test_get_formula_unknown(self): """ Tests the generation of formula, including unknown entry. """ from aiida.orm.nodes.data.structure import get_formula self.assertEquals(get_formula(['Ba', 'Ti'] + ['X'] * 3), 'BaTiX3') self.assertEquals(get_formula(['Ba', 'Ti', 'C'] + ['X'] * 3, separator=" "), 'C Ba Ti X3') self.assertEquals(get_formula(['X'] * 6 + ['C'] * 6), 'C6X6') self.assertEquals(get_formula(['X'] * 6 + ['C'] * 6, mode="hill_compact"), 'CX') self.assertEquals(get_formula((['Ba', 'Ti'] + ['X'] * 3) * 2 + \ ['Ba'] + ['X'] * 2 + ['O'] * 3, mode="group"), '(BaTiX3)2BaX2O3') self.assertEquals(get_formula((['Ba', 'Ti'] + ['X'] * 3) * 2 + \ ['Ba'] + ['X'] * 2 + ['O'] * 3, mode="group", separator=" "), '(Ba Ti X3)2 Ba X2 O3') self.assertEquals(get_formula((['Ba', 'Ti'] + ['X'] * 3) * 2 + \ ['Ba'] + ['Ti'] * 2 + ['X'] * 3, mode="reduce"), 'BaTiX3BaTiX3BaTi2X3') self.assertEquals(get_formula((['Ba', 'Ti'] + ['X'] * 3) * 2 + \ ['Ba'] + ['Ti'] * 2 + ['X'] * 3, mode="reduce", separator=", "), 'Ba, Ti, X3, Ba, Ti, X3, Ba, Ti2, X3') self.assertEquals(get_formula((['Ba', 'Ti'] + ['O'] * 3) * 2, mode="count"), 'Ba2Ti2O6') self.assertEquals(get_formula((['Ba', 'Ti'] + ['X'] * 3) * 2, mode="count_compact"), 'BaTiX3')
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pycifrw(), "Unable to import PyCifRW") def test_get_cif(self): """ Tests the conversion to CifData """ import re a = StructureData(cell=((2., 0., 0.), (0., 2., 0.), (0., 0., 2.))) a.append_atom(position=(0., 0., 0.), symbols=['Ba']) a.append_atom(position=(0.5, 0.5, 0.5), symbols=['Ba']) a.append_atom(position=(1., 1., 1.), symbols=['Ti']) c = a.get_cif() lines = c._prepare_cif()[0].decode('utf-8').split('\n') non_comments = [] for line in lines: if not re.search('^#', line): non_comments.append(line) self.assertEquals( simplify("\n".join(non_comments)), simplify(""" data_0 loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z _atom_site_type_symbol Ba1 0.0 0.0 0.0 Ba Ba2 0.25 0.25 0.25 Ba Ti1 0.5 0.5 0.5 Ti _cell_angle_alpha 90.0 _cell_angle_beta 90.0 _cell_angle_gamma 90.0 _cell_length_a 2.0 _cell_length_b 2.0 _cell_length_c 2.0 loop_ _symmetry_equiv_pos_as_xyz 'x, y, z' _symmetry_Int_Tables_number 1 _symmetry_space_group_name_H-M 'P 1' _symmetry_space_group_name_Hall 'P 1' _cell_formula_units_Z 1 _chemical_formula_sum 'Ba2 Ti' """))
[docs] def test_xyz_parser(self): from aiida.orm.nodes.data.structure import StructureData import numpy as np xyz_string1 = """ 3 Li 0.00000000 0.00000000 0.00000000 6.94100000 3 Si 4.39194796 0.00000000 10.10068356 28.08550000 14 Si 4.39194796 0.00000000 3.79747116 28.08550000 14 """ xyz_string2 = """ 2 Silver dimer; Ag 0 0 0 Ag 0 0 2.5335 """ xyz_string3 = """ 2 Shifted Silver dimer; Ag 0 0 -.5 Ag 0 0 2.0335 """ for xyz_string in (xyz_string1, xyz_string2, xyz_string3): s = StructureData() # Parsing the string: s._parse_xyz(xyz_string) # Making sure that the periodic boundary condition are not True # because I cannot parse a cell! self.assertTrue(not (any(s.pbc))) # Making sure that the structure has sites, kinds and a cell self.assertTrue(s.sites) self.assertTrue(s.kinds) self.assertTrue(s.cell) # The default cell is given in these cases: self.assertEqual(s.cell, np.diag([1, 1, 1]).tolist()) # Testing a case where 1 xyz_string4 = """ 1 Li 0.00000000 0.00000000 0.00000000 6.94100000 3 Si 4.39194796 0.00000000 10.10068356 28.08550000 14 Si 4.39194796 0.00000000 3.79747116 28.08550000 14 """ xyz_string5 = """ 10 Li 0.00000000 0.00000000 0.00000000 6.94100000 3 Si 4.39194796 0.00000000 10.10068356 28.08550000 14 Si 4.39194796 0.00000000 3.79747116 28.08550000 14 """ xyz_string6 = """ 2 Shifted Silver dimer; Ag 0 0 -.5 Ag 0 0 """ # The above cases have to fail because the number of atoms is wrong for xyz_string in (xyz_string4, xyz_string5, xyz_string6): with self.assertRaises(TypeError): StructureData()._parse_xyz(xyz_string)
[docs]class TestStructureDataLock(AiidaTestCase): """ Tests that the structure is locked after storage """
[docs] def test_lock(self): """ Start from a StructureData object, convert to raw and then back """ cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) a = StructureData(cell=cell) a.pbc = [False, True, True] k = Kind(symbols='Ba', name='Ba') s = Site(position=(0., 0., 0.), kind_name='Ba') a.append_kind(k) a.append_site(s) a.append_atom(symbols='Ti', position=[0., 0., 0.]) a.store() k2 = Kind(symbols='Ba', name='Ba') # Nothing should be changed after store() with self.assertRaises(ModificationNotAllowed): a.append_kind(k2) with self.assertRaises(ModificationNotAllowed): a.append_site(s) with self.assertRaises(ModificationNotAllowed): a.clear_sites() with self.assertRaises(ModificationNotAllowed): a.clear_kinds() with self.assertRaises(ModificationNotAllowed): a.cell = cell with self.assertRaises(ModificationNotAllowed): a.pbc = [True, True, True] _ = a.get_cell_volume() _ = a.is_alloy _ = a.has_vacancies b = a.clone() # I check that clone returned an unstored copy and so can be altered b.append_site(s) b.clear_sites() # I check that the original did not change self.assertNotEquals(len(a.sites), 0) b.cell = cell b.pbc = [True, True, True]
[docs]class TestStructureDataReload(AiidaTestCase): """ Tests the creation of StructureData, converting it to a raw format and converting it back. """
[docs] def test_reload(self): """ Start from a StructureData object, convert to raw and then back """ cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) a = StructureData(cell=cell) a.pbc = [False, True, True] a.append_atom(position=(0., 0., 0.), symbols=['Ba']) a.append_atom(position=(1., 1., 1.), symbols=['Ti']) a.store() b = load_node(uuid=a.uuid) for i in range(3): for j in range(3): self.assertAlmostEqual(cell[i][j], b.cell[i][j]) self.assertEqual(b.pbc, (False, True, True)) self.assertEqual(len(b.sites), 2) self.assertEqual(b.kinds[0].symbols[0], 'Ba') self.assertEqual(b.kinds[1].symbols[0], 'Ti') for i in range(3): self.assertAlmostEqual(b.sites[0].position[i], 0.) for i in range(3): self.assertAlmostEqual(b.sites[1].position[i], 1.) # Fully reload from UUID b = load_node(a.uuid, sub_classes=(StructureData,)) for i in range(3): for j in range(3): self.assertAlmostEqual(cell[i][j], b.cell[i][j]) self.assertEqual(b.pbc, (False, True, True)) self.assertEqual(len(b.sites), 2) self.assertEqual(b.kinds[0].symbols[0], 'Ba') self.assertEqual(b.kinds[1].symbols[0], 'Ti') for i in range(3): self.assertAlmostEqual(b.sites[0].position[i], 0.) for i in range(3): self.assertAlmostEqual(b.sites[1].position[i], 1.)
[docs] def test_clone(self): """ Start from a StructureData object, clone it and see if it is preserved """ cell = ((1., 0., 0.), (0., 2., 0.), (0., 0., 3.)) a = StructureData(cell=cell) a.pbc = [False, True, True] a.append_atom(position=(0., 0., 0.), symbols=['Ba']) a.append_atom(position=(1., 1., 1.), symbols=['Ti']) b = a.clone() for i in range(3): for j in range(3): self.assertAlmostEqual(cell[i][j], b.cell[i][j]) self.assertEqual(b.pbc, (False, True, True)) self.assertEqual(len(b.kinds), 2) self.assertEqual(len(b.sites), 2) self.assertEqual(b.kinds[0].symbols[0], 'Ba') self.assertEqual(b.kinds[1].symbols[0], 'Ti') for i in range(3): self.assertAlmostEqual(b.sites[0].position[i], 0.) for i in range(3): self.assertAlmostEqual(b.sites[1].position[i], 1.) a.store() # Clone after store() c = a.clone() for i in range(3): for j in range(3): self.assertAlmostEqual(cell[i][j], c.cell[i][j]) self.assertEqual(c.pbc, (False, True, True)) self.assertEqual(len(c.kinds), 2) self.assertEqual(len(c.sites), 2) self.assertEqual(c.kinds[0].symbols[0], 'Ba') self.assertEqual(c.kinds[1].symbols[0], 'Ti') for i in range(3): self.assertAlmostEqual(c.sites[0].position[i], 0.) for i in range(3): self.assertAlmostEqual(c.sites[1].position[i], 1.)
[docs]class TestStructureDataFromAse(AiidaTestCase): """ Tests the creation of Sites from/to a ASE object. """ from aiida.orm.nodes.data.structure import has_ase
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") def test_ase(self): """ Tests roundtrip ASE -> StructureData -> ASE """ from aiida.orm.nodes.data.structure import StructureData import ase a = ase.Atoms('SiGe', cell=(1., 2., 3.), pbc=(True, False, False)) a.set_positions(( (0., 0., 0.), (0.5, 0.7, 0.9), )) a[1].mass = 110.2 b = StructureData(ase=a) c = b.get_ase() self.assertEqual(a[0].symbol, c[0].symbol) self.assertEqual(a[1].symbol, c[1].symbol) for i in range(3): self.assertAlmostEqual(a[0].position[i], c[0].position[i]) for i in range(3): for j in range(3): self.assertAlmostEqual(a.cell[i][j], c.cell[i][j]) self.assertAlmostEqual(c[1].mass, 110.2)
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") def test_conversion_of_types_1(self): """ Tests roundtrip ASE -> StructureData -> ASE, with tags """ import ase a = ase.Atoms('Si4Ge4', cell=(1., 2., 3.), pbc=(True, False, False)) a.set_positions(( (0.0, 0.0, 0.0), (0.1, 0.1, 0.1), (0.2, 0.2, 0.2), (0.3, 0.3, 0.3), (0.4, 0.4, 0.4), (0.5, 0.5, 0.5), (0.6, 0.6, 0.6), (0.7, 0.7, 0.7), )) a.set_tags((0, 1, 2, 3, 4, 5, 6, 7)) b = StructureData(ase=a) self.assertEquals([k.name for k in b.kinds], ["Si", "Si1", "Si2", "Si3", "Ge4", "Ge5", "Ge6", "Ge7"]) c = b.get_ase() a_tags = list(a.get_tags()) c_tags = list(c.get_tags()) self.assertEqual(a_tags, c_tags)
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") def test_conversion_of_types_2(self): """ Tests roundtrip ASE -> StructureData -> ASE, with tags, and changing the atomic masses """ import ase a = ase.Atoms('Si4', cell=(1., 2., 3.), pbc=(True, False, False)) a.set_positions(( (0.0, 0.0, 0.0), (0.1, 0.1, 0.1), (0.2, 0.2, 0.2), (0.3, 0.3, 0.3), )) a.set_tags((0, 1, 0, 1)) a[2].mass = 100. a[3].mass = 300. b = StructureData(ase=a) # This will give funny names to the kinds, because I am using # both tags and different properties (mass). I just check to have # 4 kinds self.assertEquals(len(b.kinds), 4) # Do I get the same tags after one full iteration back and forth? c = b.get_ase() d = StructureData(ase=c) e = d.get_ase() c_tags = list(c.get_tags()) e_tags = list(e.get_tags()) self.assertEqual(c_tags, e_tags)
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") def test_conversion_of_types_3(self): """ Tests StructureData -> ASE, with all sorts of kind names """ a = StructureData() a.append_atom(position=(0., 0., 0.), symbols='Ba', name='Ba') a.append_atom(position=(0., 0., 0.), symbols='Ba', name='Ba1') a.append_atom(position=(0., 0., 0.), symbols='Cu', name='Cu') # continues with a number a.append_atom(position=(0., 0., 0.), symbols='Cu', name='Cu2') # does not continue with a number a.append_atom(position=(0., 0., 0.), symbols='Cu', name='Cu_my') # random string a.append_atom(position=(0., 0., 0.), symbols='Cu', name='a_name') # a name of another chemical symbol a.append_atom(position=(0., 0., 0.), symbols='Cu', name='Fe') # lowercase! as if it were a random string a.append_atom(position=(0., 0., 0.), symbols='Cu', name='cu1') # Just to be sure that the species were saved with the correct name # in the first place self.assertEquals([k.name for k in a.kinds], ['Ba', 'Ba1', 'Cu', 'Cu2', 'Cu_my', 'a_name', 'Fe', 'cu1']) b = a.get_ase() self.assertEquals(b.get_chemical_symbols(), ['Ba', 'Ba', 'Cu', 'Cu', 'Cu', 'Cu', 'Cu', 'Cu']) self.assertEquals(list(b.get_tags()), [0, 1, 0, 2, 3, 4, 5, 6])
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") def test_conversion_of_types_4(self): """ Tests ASE -> StructureData -> ASE, in particular conversion tags / kind names """ import ase atoms = ase.Atoms('Fe5') atoms[2].tag = 1 atoms[3].tag = 1 atoms[4].tag = 4 atoms.set_cell([1,1,1]) s = StructureData(ase=atoms) kindnames = set([k.name for k in s.kinds]) self.assertEquals(kindnames, set(['Fe', 'Fe1', 'Fe4'])) # check roundtrip ASE -> StructureData -> ASE atoms2 = s.get_ase() self.assertEquals(list(atoms2.get_tags()), list(atoms.get_tags())) self.assertEquals(list(atoms2.get_chemical_symbols()), list(atoms.get_chemical_symbols())) self.assertEquals(atoms2.get_chemical_formula(), 'Fe5')
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") def test_conversion_of_types_5(self): """ Tests ASE -> StructureData -> ASE, in particular conversion tags / kind names (subtle variation of test_conversion_of_types_4) """ import ase atoms = ase.Atoms('Fe5') atoms[0].tag = 1 atoms[2].tag = 1 atoms[3].tag = 4 atoms.set_cell([1, 1, 1]) s = StructureData(ase=atoms) kindnames = set([k.name for k in s.kinds]) self.assertEquals(kindnames, set(['Fe', 'Fe1', 'Fe4'])) # check roundtrip ASE -> StructureData -> ASE atoms2 = s.get_ase() self.assertEquals(list(atoms2.get_tags()), list(atoms.get_tags())) self.assertEquals(list(atoms2.get_chemical_symbols()), list(atoms.get_chemical_symbols())) self.assertEquals(atoms2.get_chemical_formula(), 'Fe5')
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") def test_conversion_of_types_6(self): """ Tests roundtrip StructureData -> ASE -> StructureData, with tags/kind names """ a = StructureData(cell=[[4, 0, 0], [0, 4, 0], [0, 0, 4]]) a.append_atom(position=(0, 0, 0), symbols='Ni', name='Ni1') a.append_atom(position=(2, 2, 2), symbols='Ni', name='Ni2') a.append_atom(position=(1, 0, 1), symbols='Cl', name='Cl') a.append_atom(position=(1, 3, 1), symbols='Cl', name='Cl') b = a.get_ase() self.assertEquals(b.get_chemical_symbols(), ['Ni', 'Ni', 'Cl', 'Cl']) self.assertEquals(list(b.get_tags()), [1, 2, 0, 0]) c = StructureData(ase=b) self.assertEquals(c.get_site_kindnames(), ['Ni1', 'Ni2', 'Cl', 'Cl']) self.assertEquals([k.symbol for k in c.kinds], ['Ni', 'Ni', 'Cl']) self.assertEquals([s.position for s in c.sites], [(0., 0., 0.), (2., 2., 2.), (1., 0., 1.), (1., 3., 1.)])
[docs]class TestStructureDataFromPymatgen(AiidaTestCase): """ Tests the creation of StructureData from a pymatgen Structure and Molecule objects. """ from aiida.orm.nodes.data.structure import has_pymatgen, get_pymatgen_version
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_1(self): """ Tests roundtrip pymatgen -> StructureData -> pymatgen Test's input is derived from COD entry 9011963, processed with cif_mark_disorder (from cod-tools) and abbreviated. """ from pymatgen.io.cif import CifParser with tempfile.NamedTemporaryFile(mode='w+', suffix=".cif") as tmpf: tmpf.write("""data_9011963 _space_group_IT_number 166 _symmetry_space_group_name_Hall '-R 3 2"' _symmetry_space_group_name_H-M 'R -3 m :H' _cell_angle_alpha 90 _cell_angle_beta 90 _cell_angle_gamma 120 _cell_length_a 4.298 _cell_length_b 4.298 _cell_length_c 29.774 loop_ _atom_site_label _atom_site_fract_x _atom_site_fract_y _atom_site_fract_z _atom_site_occupancy _atom_site_U_iso_or_equiv _atom_site_disorder_assembly _atom_site_disorder_group Bi 0.00000 0.00000 0.39580 1.00000 0.02343 . . Te1 0.00000 0.00000 0.00000 0.66667 0.02343 A 1 Se1 0.00000 0.00000 0.00000 0.33333 0.02343 A 2 Te2 0.00000 0.00000 0.21180 0.66667 0.02343 B 1 Se2 0.00000 0.00000 0.21180 0.33333 0.02343 B 2 """) tmpf.flush() pymatgen_parser = CifParser(tmpf.name) pymatgen_struct = pymatgen_parser.get_structures()[0] structs_to_test = [StructureData(pymatgen=pymatgen_struct), StructureData(pymatgen_structure=pymatgen_struct)] for struct in structs_to_test: self.assertEquals(struct.get_site_kindnames(), ['Bi', 'Bi', 'SeTe', 'SeTe', 'SeTe']) # Pymatgen's Composition does not guarantee any particular ordering of the kinds, # see the definition of its internal datatype at # pymatgen/core/composition.py#L135 (d4fe64c18a52949a4e22bfcf7b45de5b87242c51) self.assertEquals([sorted(x.symbols) for x in struct.kinds], [[ 'Bi', ], ['Se', 'Te']]) self.assertEquals([sorted(x.weights) for x in struct.kinds], [[ 1.0, ], [0.33333, 0.66667]]) struct = StructureData(pymatgen_structure=pymatgen_struct) # Testing pymatgen Structure -> StructureData -> pymatgen Structure # roundtrip. pymatgen_struct_roundtrip = struct.get_pymatgen_structure() dict1 = pymatgen_struct.as_dict() dict2 = pymatgen_struct_roundtrip.as_dict() for i in dict1['sites']: i['abc'] = [round(j, 2) for j in i['abc']] for i in dict2['sites']: i['abc'] = [round(j, 2) for j in i['abc']] self.assertEquals(dict1, dict2)
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_2(self): """ Tests xyz -> pymatgen -> StructureData Input source: http://pymatgen.org/_static/Molecule.html """ from pymatgen.io.xyz import XYZ with tempfile.NamedTemporaryFile(mode='w+', suffix=".xyz") as tmpf: tmpf.write("""5 H4 C1 C 0.000000 0.000000 0.000000 H 0.000000 0.000000 1.089000 H 1.026719 0.000000 -0.363000 H -0.513360 -0.889165 -0.363000 H -0.513360 0.889165 -0.363000""") tmpf.flush() pymatgen_xyz = XYZ.from_file(tmpf.name) pymatgen_mol = pymatgen_xyz.molecule for struct in [StructureData(pymatgen=pymatgen_mol), StructureData(pymatgen_molecule=pymatgen_mol)]: self.assertEquals(struct.get_site_kindnames(), ['H', 'H', 'H', 'H', 'C']) self.assertEquals(struct.pbc, (False, False, False)) self.assertEquals([round(x, 2) for x in list(struct.sites[0].position)], [5.77, 5.89, 6.81]) self.assertEquals([round(x, 2) for x in list(struct.sites[1].position)], [6.8, 5.89, 5.36]) self.assertEquals([round(x, 2) for x in list(struct.sites[2].position)], [5.26, 5.0, 5.36]) self.assertEquals([round(x, 2) for x in list(struct.sites[3].position)], [5.26, 6.78, 5.36]) self.assertEquals([round(x, 2) for x in list(struct.sites[4].position)], [5.77, 5.89, 5.73])
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_partial_occ_and_spin(self): """ Tests pymatgen -> StructureData, with partial occupancies and spins. This should raise a ValueError. """ import pymatgen Fe_spin_up = pymatgen.structure.Specie('Fe', 0, properties={'spin': 1}) Mn_spin_up = pymatgen.structure.Specie('Mn', 0, properties={'spin': 1}) Fe_spin_down = pymatgen.structure.Specie('Fe', 0, properties={'spin': -1}) Mn_spin_down = pymatgen.structure.Specie('Mn', 0, properties={'spin': -1}) FeMn1 = pymatgen.Composition({Fe_spin_up: 0.5, Mn_spin_up: 0.5}) FeMn2 = pymatgen.Composition({Fe_spin_down: 0.5, Mn_spin_down: 0.5}) a = pymatgen.structure.Structure( lattice=[[4, 0, 0], [0, 4, 0], [0, 0, 4]], species=[FeMn1, FeMn2], coords=[[0, 0, 0], [0.5, 0.5, 0.5]]) with self.assertRaises(ValueError): StructureData(pymatgen=a) # same, with vacancies Fe1 = pymatgen.Composition({Fe_spin_up: 0.5}) Fe2 = pymatgen.Composition({Fe_spin_down: 0.5}) a = pymatgen.structure.Structure( lattice=[[4, 0, 0], [0, 4, 0], [0, 0, 4]], species=[Fe1, Fe2], coords=[[0, 0, 0], [0.5, 0.5, 0.5]]) with self.assertRaises(ValueError): StructureData(pymatgen=a)
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_multiple_kinds_partial_occupancies(self): """ Tests that a structure with multiple sites with the same element but different partial occupancies, get their own unique kind name """ import pymatgen Mg1 = pymatgen.structure.Composition({'Mg': 0.50}) Mg2 = pymatgen.structure.Composition({'Mg': 0.25}) a = pymatgen.structure.Structure( lattice=[[4, 0, 0], [0, 4, 0], [0, 0, 4]], species=[Mg1, Mg2], coords=[[0, 0, 0], [0.5, 0.5, 0.5]]) structure = StructureData(pymatgen=a)
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_multiple_kinds_alloy(self): """ Tests that a structure with multiple sites with the same alloy symbols but different weights, get their own unique kind name """ import pymatgen alloy_one = pymatgen.structure.Composition({'Mg': 0.25, 'Al': 0.75}) alloy_two = pymatgen.structure.Composition({'Mg': 0.45, 'Al': 0.55}) a = pymatgen.structure.Structure( lattice=[[4, 0, 0], [0, 4, 0], [0, 0, 4]], species=[alloy_one, alloy_two], coords=[[0, 0, 0], [0.5, 0.5, 0.5]]) structure = StructureData(pymatgen=a)
[docs]class TestPymatgenFromStructureData(AiidaTestCase): """ Tests the creation of pymatgen Structure and Molecule objects from StructureData. """ from aiida.orm.nodes.data.structure import has_ase, has_pymatgen, get_pymatgen_version
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_1(self): """ Tests the check of periodic boundary conditions. """ from aiida.orm.nodes.data.structure import StructureData struct = StructureData() struct.pbc = [True, True, True] pmg_struct = struct.get_pymatgen_structure() struct.pbc = [True, True, False] with self.assertRaises(ValueError): pmg_struct = struct.get_pymatgen_structure()
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_2(self): """ Tests ASE -> StructureData -> pymatgen """ import ase aseatoms = ase.Atoms('Si4', cell=(1., 2., 3.), pbc=(True, True, True)) aseatoms.set_scaled_positions(( (0.0, 0.0, 0.0), (0.1, 0.1, 0.1), (0.2, 0.2, 0.2), (0.3, 0.3, 0.3), )) a_struct = StructureData(ase=aseatoms) p_struct = a_struct.get_pymatgen_structure() p_struct_dict = p_struct.as_dict() coord_array = [x['abc'] for x in p_struct_dict['sites']] for i in range(len(coord_array)): coord_array[i] = [round(x, 2) for x in coord_array[i]] self.assertEquals(coord_array, [[0.0, 0.0, 0.0], [0.1, 0.1, 0.1], [0.2, 0.2, 0.2], [0.3, 0.3, 0.3]])
[docs] @unittest.skipIf(not has_ase(), "Unable to import ase") @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_3(self): """ Tests the conversion of StructureData to pymatgen's Molecule (ASE -> StructureData -> pymatgen) """ import ase aseatoms = ase.Atoms('Si4', cell=(10, 10, 10), pbc=(True, True, True)) aseatoms.set_scaled_positions(( (0.0, 0.0, 0.0), (0.1, 0.1, 0.1), (0.2, 0.2, 0.2), (0.3, 0.3, 0.3), )) a_struct = StructureData(ase=aseatoms) p_mol = a_struct.get_pymatgen_molecule() p_mol_dict = p_mol.as_dict() self.assertEquals([x['xyz'] for x in p_mol_dict['sites']], [[0.0, 0.0, 0.0], [1.0, 1.0, 1.0], [2.0, 2.0, 2.0], [3.0, 3.0, 3.0]])
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_roundtrip(self): """ Tests roundtrip StructureData -> pymatgen -> StructureData (no spins) """ a = StructureData(cell=[[5.6, 0, 0], [0, 5.6, 0], [0, 0, 5.6]]) a.append_atom(position=(0, 0, 0), symbols='Cl') a.append_atom(position=(2.8, 0, 2.8), symbols='Cl') a.append_atom(position=(0, 2.8, 2.8), symbols='Cl') a.append_atom(position=(2.8, 2.8, 0), symbols='Cl') a.append_atom(position=(2.8, 2.8, 2.8), symbols='Na') a.append_atom(position=(2.8, 0, 0), symbols='Na') a.append_atom(position=(0, 2.8, 0), symbols='Na') a.append_atom(position=(0, 0, 2.8), symbols='Na') b = a.get_pymatgen() c = StructureData(pymatgen=b) self.assertEquals(c.get_site_kindnames(), ['Cl', 'Cl', 'Cl', 'Cl', 'Na', 'Na', 'Na', 'Na']) self.assertEquals([k.symbol for k in c.kinds], ['Cl', 'Na']) self.assertEquals([s.position for s in c.sites], [(0., 0., 0.), (2.8, 0, 2.8), (0, 2.8, 2.8), (2.8, 2.8, 0), (2.8, 2.8, 2.8), (2.8, 0, 0), (0, 2.8, 0), (0, 0, 2.8)])
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_roundtrip_kindnames(self): """ Tests roundtrip StructureData -> pymatgen -> StructureData (no spins, but with all kind of kind names) """ a = StructureData(cell=[[5.6, 0, 0], [0, 5.6, 0], [0, 0, 5.6]]) a.append_atom(position=(0, 0, 0), symbols='Cl', name='Cl') a.append_atom(position=(2.8, 0, 2.8), symbols='Cl', name='Cl10') a.append_atom(position=(0, 2.8, 2.8), symbols='Cl', name='Cla') a.append_atom(position=(2.8, 2.8, 0), symbols='Cl', name='cl_x') a.append_atom(position=(2.8, 2.8, 2.8), symbols='Na', name='Na1') a.append_atom(position=(2.8, 0, 0), symbols='Na', name='Na2') a.append_atom(position=(0, 2.8, 0), symbols='Na', name='Na_Na') a.append_atom(position=(0, 0, 2.8), symbols='Na', name='Na4') b = a.get_pymatgen() self.assertEquals([site.properties['kind_name'] for site in b.sites], ['Cl', 'Cl10', 'Cla', 'cl_x', 'Na1', 'Na2', 'Na_Na', 'Na4']) c = StructureData(pymatgen=b) self.assertEquals(c.get_site_kindnames(), ['Cl', 'Cl10', 'Cla', 'cl_x', 'Na1', 'Na2', 'Na_Na', 'Na4']) self.assertEquals(c.get_symbols_set(), set(['Cl', 'Na'])) self.assertEquals([s.position for s in c.sites], [(0., 0., 0.), (2.8, 0, 2.8), (0, 2.8, 2.8), (2.8, 2.8, 0), (2.8, 2.8, 2.8), (2.8, 0, 0), (0, 2.8, 0), (0, 0, 2.8)])
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_roundtrip_spins(self): """ Tests roundtrip StructureData -> pymatgen -> StructureData (with spins) """ a = StructureData(cell=[[5.6, 0, 0], [0, 5.6, 0], [0, 0, 5.6]]) a.append_atom(position=(0, 0, 0), symbols='Mn', name='Mn1') a.append_atom(position=(2.8, 0, 2.8), symbols='Mn', name='Mn1') a.append_atom(position=(0, 2.8, 2.8), symbols='Mn', name='Mn1') a.append_atom(position=(2.8, 2.8, 0), symbols='Mn', name='Mn1') a.append_atom(position=(2.8, 2.8, 2.8), symbols='Mn', name='Mn2') a.append_atom(position=(2.8, 0, 0), symbols='Mn', name='Mn2') a.append_atom(position=(0, 2.8, 0), symbols='Mn', name='Mn2') a.append_atom(position=(0, 0, 2.8), symbols='Mn', name='Mn2') b = a.get_pymatgen(add_spin=True) # check the spins self.assertEquals([s.as_dict()['properties']['spin'] for s in b.species], [-1, -1, -1, -1, 1, 1, 1, 1]) # back to StructureData c = StructureData(pymatgen=b) self.assertEquals(c.get_site_kindnames(), ['Mn1', 'Mn1', 'Mn1', 'Mn1', 'Mn2', 'Mn2', 'Mn2', 'Mn2']) self.assertEquals([k.symbol for k in c.kinds], ['Mn', 'Mn']) self.assertEquals([s.position for s in c.sites], [(0., 0., 0.), (2.8, 0, 2.8), (0, 2.8, 2.8), (2.8, 2.8, 0), (2.8, 2.8, 2.8), (2.8, 0, 0), (0, 2.8, 0), (0, 0, 2.8)])
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_roundtrip_partial_occ(self): """ Tests roundtrip StructureData -> pymatgen -> StructureData (with partial occupancies). """ a = StructureData(cell=[[4.0, 0.0, 0.0], [-2., 3.5, 0.0], [0.0, 0.0, 16.]]) a.append_atom(position=(0.0, 0.0, 13.5), symbols='Mn') a.append_atom(position=(0.0, 0.0, 2.6), symbols='Mn') a.append_atom(position=(0.0, 0.0, 5.5), symbols='Mn') a.append_atom(position=(0.0, 0.0, 11.), symbols='Mn') a.append_atom(position=(2., 1., 12.), symbols='Mn', weights=0.8) a.append_atom(position=(0.0, 2.2, 4.), symbols='Mn', weights=0.8) a.append_atom(position=(0.0, 2.2, 12.), symbols='Si') a.append_atom(position=(2., 1., 4.), symbols='Si') a.append_atom(position=(2., 1., 15.), symbols='N') a.append_atom(position=(0.0, 2.2, 1.5), symbols='N') a.append_atom(position=(0.0, 2.2, 7.), symbols='N') a.append_atom(position=(2., 1., 9.5), symbols='N') # a few checks on the structure kinds and symbols self.assertEquals(a.get_symbols_set(), set(['Mn', 'Si', 'N'])) self.assertEquals(a.get_site_kindnames(), ['Mn', 'Mn', 'Mn', 'Mn', 'MnX', 'MnX', 'Si', 'Si', 'N', 'N', 'N', 'N']) self.assertEquals(a.get_formula(), 'Mn4N4Si2{Mn0.80X0.20}2') b = a.get_pymatgen() # check the partial occupancies self.assertEquals([s.as_dict() for s in b.species_and_occu], [{ 'Mn': 1.0 }, { 'Mn': 1.0 }, { 'Mn': 1.0 }, { 'Mn': 1.0 }, { 'Mn': 0.8 }, { 'Mn': 0.8 }, { 'Si': 1.0 }, { 'Si': 1.0 }, { 'N': 1.0 }, { 'N': 1.0 }, { 'N': 1.0 }, { 'N': 1.0 }]) # back to StructureData c = StructureData(pymatgen=b) self.assertEquals(c.cell, [[4., 0.0, 0.0], [-2., 3.5, 0.0], [0.0, 0.0, 16.]]) self.assertEquals(c.get_symbols_set(), set(['Mn', 'Si', 'N'])) self.assertEquals(c.get_site_kindnames(), ['Mn', 'Mn', 'Mn', 'Mn', 'MnX', 'MnX', 'Si', 'Si', 'N', 'N', 'N', 'N']) self.assertEquals(c.get_formula(), 'Mn4N4Si2{Mn0.80X0.20}2') self.assertEquals([s.position for s in c.sites], [(0.0, 0.0, 13.5), (0.0, 0.0, 2.6), (0.0, 0.0, 5.5), (0.0, 0.0, 11.), (2., 1., 12.), (0.0, 2.2, 4.), (0.0, 2.2, 12.), (2., 1., 4.), (2., 1., 15.), (0.0, 2.2, 1.5), (0.0, 2.2, 7.), (2., 1., 9.5)])
[docs] @unittest.skipIf(not has_pymatgen(), "Unable to import pymatgen") def test_partial_occ_and_spin(self): """ Tests StructureData -> pymatgen, with partial occupancies and spins. This should raise a ValueError. """ a = StructureData(cell=[[4, 0, 0], [0, 4, 0], [0, 0, 4]]) a.append_atom(position=(0, 0, 0), symbols=('Fe', 'Al'), weights=(0.8, 0.2), name='FeAl1') a.append_atom(position=(2, 2, 2), symbols=('Fe', 'Al'), weights=(0.8, 0.2), name='FeAl2') # a few checks on the structure kinds and symbols self.assertEquals(a.get_symbols_set(), set(['Fe', 'Al'])) self.assertEquals(a.get_site_kindnames(), ['FeAl1', 'FeAl2']) self.assertEquals(a.get_formula(), '{Al0.20Fe0.80}2') with self.assertRaises(ValueError): a.get_pymatgen(add_spin=True) # same, with vacancies a = StructureData(cell=[[4, 0, 0], [0, 4, 0], [0, 0, 4]]) a.append_atom(position=(0, 0, 0), symbols='Fe', weights=0.8, name='FeX1') a.append_atom(position=(2, 2, 2), symbols='Fe', weights=0.8, name='FeX2') # a few checks on the structure kinds and symbols self.assertEquals(a.get_symbols_set(), set(['Fe'])) self.assertEquals(a.get_site_kindnames(), ['FeX1', 'FeX2']) self.assertEquals(a.get_formula(), '{Fe0.80X0.20}2') with self.assertRaises(ValueError): a.get_pymatgen(add_spin=True)
[docs]class TestArrayData(AiidaTestCase): """ Tests the ArrayData objects. """
[docs] def test_creation(self): """ Check the methods to add, remove, modify, and get arrays and array shapes. """ import numpy # Create a node with two arrays n = ArrayData() first = numpy.random.rand(2, 3, 4) n.set_array('first', first) second = numpy.arange(10) n.set_array('second', second) third = numpy.random.rand(6, 6) n.set_array('third', third) # Check if the arrays are there self.assertEquals(set(['first', 'second', 'third']), set(n.get_arraynames())) self.assertAlmostEquals(abs(first - n.get_array('first')).max(), 0.) self.assertAlmostEquals(abs(second - n.get_array('second')).max(), 0.) self.assertAlmostEquals(abs(third - n.get_array('third')).max(), 0.) self.assertEquals(first.shape, n.get_shape('first')) self.assertEquals(second.shape, n.get_shape('second')) self.assertEquals(third.shape, n.get_shape('third')) with self.assertRaises(KeyError): n.get_array('nonexistent_array') # Delete an array, and try to delete a non-existing one n.delete_array('third') with self.assertRaises(KeyError): n.delete_array('nonexistent_array') # Overwrite an array first = numpy.random.rand(4, 5, 6) n.set_array('first', first) # Check if the arrays are there, and if I am getting the new one self.assertEquals(set(['first', 'second']), set(n.get_arraynames())) self.assertAlmostEquals(abs(first - n.get_array('first')).max(), 0.) self.assertAlmostEquals(abs(second - n.get_array('second')).max(), 0.) self.assertEquals(first.shape, n.get_shape('first')) self.assertEquals(second.shape, n.get_shape('second')) n.store() # Same checks, after storing self.assertEquals(set(['first', 'second']), set(n.get_arraynames())) self.assertAlmostEquals(abs(first - n.get_array('first')).max(), 0.) self.assertAlmostEquals(abs(second - n.get_array('second')).max(), 0.) self.assertEquals(first.shape, n.get_shape('first')) self.assertEquals(second.shape, n.get_shape('second')) # Same checks, again (this is checking the caching features) self.assertEquals(set(['first', 'second']), set(n.get_arraynames())) self.assertAlmostEquals(abs(first - n.get_array('first')).max(), 0.) self.assertAlmostEquals(abs(second - n.get_array('second')).max(), 0.) self.assertEquals(first.shape, n.get_shape('first')) self.assertEquals(second.shape, n.get_shape('second')) # Same checks, after reloading n2 = load_node(uuid=n.uuid) self.assertEquals(set(['first', 'second']), set(n2.get_arraynames())) self.assertAlmostEquals(abs(first - n2.get_array('first')).max(), 0.) self.assertAlmostEquals(abs(second - n2.get_array('second')).max(), 0.) self.assertEquals(first.shape, n2.get_shape('first')) self.assertEquals(second.shape, n2.get_shape('second')) # Same checks, after reloading with UUID n2 = load_node(n.uuid, sub_classes=(ArrayData,)) self.assertEquals(set(['first', 'second']), set(n2.get_arraynames())) self.assertAlmostEquals(abs(first - n2.get_array('first')).max(), 0.) self.assertAlmostEquals(abs(second - n2.get_array('second')).max(), 0.) self.assertEquals(first.shape, n2.get_shape('first')) self.assertEquals(second.shape, n2.get_shape('second')) # Check that I cannot modify the node after storing with self.assertRaises(ModificationNotAllowed): n.delete_array('first') with self.assertRaises(ModificationNotAllowed): n.set_array('second', first) # Again same checks, to verify that the attempts to delete/overwrite # arrays did not damage the node content self.assertEquals(set(['first', 'second']), set(n.get_arraynames())) self.assertAlmostEquals(abs(first - n.get_array('first')).max(), 0.) self.assertAlmostEquals(abs(second - n.get_array('second')).max(), 0.) self.assertEquals(first.shape, n.get_shape('first')) self.assertEquals(second.shape, n.get_shape('second'))
[docs] def test_iteration(self): """ Check the functionality of the get_iterarrays() iterator """ import numpy # Create a node with two arrays n = ArrayData() first = numpy.random.rand(2, 3, 4) n.set_array('first', first) second = numpy.arange(10) n.set_array('second', second) third = numpy.random.rand(6, 6) n.set_array('third', third) for name, array in n.get_iterarrays(): if name == 'first': self.assertAlmostEquals(abs(first - array).max(), 0.) if name == 'second': self.assertAlmostEquals(abs(second - array).max(), 0.) if name == 'third': self.assertAlmostEquals(abs(third - array).max(), 0.)
[docs]class TestTrajectoryData(AiidaTestCase): """ Tests the TrajectoryData objects. """
[docs] def test_creation(self): """ Check the methods to set and retrieve a trajectory. """ import numpy # Create a node with two arrays n = TrajectoryData() # I create sample data stepids = numpy.array([60, 70]) times = stepids * 0.01 cells = numpy.array([[[ 2., 0., 0., ], [ 0., 2., 0., ], [ 0., 0., 2., ]], [[ 3., 0., 0., ], [ 0., 3., 0., ], [ 0., 0., 3., ]]]) symbols = ['H', 'O', 'C'] positions = numpy.array([[[0., 0., 0.], [0.5, 0.5, 0.5], [1.5, 1.5, 1.5]], [[0., 0., 0.], [0.5, 0.5, 0.5], [1.5, 1.5, 1.5]]]) velocities = numpy.array([[[0., 0., 0.], [0., 0., 0.], [0., 0., 0.]], [[0.5, 0.5, 0.5], [0.5, 0.5, 0.5], [-0.5, -0.5, -0.5]]]) # I set the node n.set_trajectory( stepids=stepids, cells=cells, symbols=symbols, positions=positions, times=times, velocities=velocities) # Generic checks self.assertEqual(n.numsites, 3) self.assertEqual(n.numsteps, 2) self.assertAlmostEqual(abs(stepids - n.get_stepids()).sum(), 0.) self.assertAlmostEqual(abs(times - n.get_times()).sum(), 0.) self.assertAlmostEqual(abs(cells - n.get_cells()).sum(), 0.) self.assertEqual(symbols, n.symbols) self.assertAlmostEqual(abs(positions - n.get_positions()).sum(), 0.) self.assertAlmostEqual(abs(velocities - n.get_velocities()).sum(), 0.) # get_step_data function check data = n.get_step_data(1) self.assertEqual(data[0], stepids[1]) self.assertAlmostEqual(data[1], times[1]) self.assertAlmostEqual(abs(cells[1] - data[2]).sum(), 0.) self.assertEqual(symbols, data[3]) self.assertAlmostEqual(abs(data[4] - positions[1]).sum(), 0.) self.assertAlmostEqual(abs(data[5] - velocities[1]).sum(), 0.) # Step 70 has index 1 self.assertEqual(1, n.get_index_from_stepid(70)) with self.assertRaises(ValueError): # Step 66 does not exist n.get_index_from_stepid(66) ######################################################## # I set the node, this time without times or velocities (the same node) n.set_trajectory(stepids=stepids, cells=cells, symbols=symbols, positions=positions) # Generic checks self.assertEqual(n.numsites, 3) self.assertEqual(n.numsteps, 2) self.assertAlmostEqual(abs(stepids - n.get_stepids()).sum(), 0.) self.assertIsNone(n.get_times()) self.assertAlmostEqual(abs(cells - n.get_cells()).sum(), 0.) self.assertEqual(symbols, n.symbols) self.assertAlmostEqual(abs(positions - n.get_positions()).sum(), 0.) self.assertIsNone(n.get_velocities()) # Same thing, but for a new node n = TrajectoryData() n.set_trajectory(stepids=stepids, cells=cells, symbols=symbols, positions=positions) # Generic checks self.assertEqual(n.numsites, 3) self.assertEqual(n.numsteps, 2) self.assertAlmostEqual(abs(stepids - n.get_stepids()).sum(), 0.) self.assertIsNone(n.get_times()) self.assertAlmostEqual(abs(cells - n.get_cells()).sum(), 0.) self.assertEqual(symbols, n.symbols) self.assertAlmostEqual(abs(positions - n.get_positions()).sum(), 0.) self.assertIsNone(n.get_velocities()) ######################################################## # I set the node, this time without velocities (the same node) n.set_trajectory(stepids=stepids, cells=cells, symbols=symbols, positions=positions, times=times) # Generic checks self.assertEqual(n.numsites, 3) self.assertEqual(n.numsteps, 2) self.assertAlmostEqual(abs(stepids - n.get_stepids()).sum(), 0.) self.assertAlmostEqual(abs(times - n.get_times()).sum(), 0.) self.assertAlmostEqual(abs(cells - n.get_cells()).sum(), 0.) self.assertEqual(symbols, n.symbols) self.assertAlmostEqual(abs(positions - n.get_positions()).sum(), 0.) self.assertIsNone(n.get_velocities()) # Same thing, but for a new node n = TrajectoryData() n.set_trajectory(stepids=stepids, cells=cells, symbols=symbols, positions=positions, times=times) # Generic checks self.assertEqual(n.numsites, 3) self.assertEqual(n.numsteps, 2) self.assertAlmostEqual(abs(stepids - n.get_stepids()).sum(), 0.) self.assertAlmostEqual(abs(times - n.get_times()).sum(), 0.) self.assertAlmostEqual(abs(cells - n.get_cells()).sum(), 0.) self.assertEqual(symbols, n.symbols) self.assertAlmostEqual(abs(positions - n.get_positions()).sum(), 0.) self.assertIsNone(n.get_velocities()) n.store() # Again same checks, but after storing # Generic checks self.assertEqual(n.numsites, 3) self.assertEqual(n.numsteps, 2) self.assertAlmostEqual(abs(stepids - n.get_stepids()).sum(), 0.) self.assertAlmostEqual(abs(times - n.get_times()).sum(), 0.) self.assertAlmostEqual(abs(cells - n.get_cells()).sum(), 0.) self.assertEqual(symbols, n.symbols) self.assertAlmostEqual(abs(positions - n.get_positions()).sum(), 0.) self.assertIsNone(n.get_velocities()) # get_step_data function check data = n.get_step_data(1) self.assertEqual(data[0], stepids[1]) self.assertAlmostEqual(data[1], times[1]) self.assertAlmostEqual(abs(cells[1] - data[2]).sum(), 0.) self.assertEqual(symbols, data[3]) self.assertAlmostEqual(abs(data[4] - positions[1]).sum(), 0.) self.assertIsNone(data[5]) # Step 70 has index 1 self.assertEqual(1, n.get_index_from_stepid(70)) with self.assertRaises(ValueError): # Step 66 does not exist n.get_index_from_stepid(66) ############################################################## # Again, but after reloading from uuid n = load_node(n.uuid, sub_classes=(TrajectoryData,)) # Generic checks self.assertEqual(n.numsites, 3) self.assertEqual(n.numsteps, 2) self.assertAlmostEqual(abs(stepids - n.get_stepids()).sum(), 0.) self.assertAlmostEqual(abs(times - n.get_times()).sum(), 0.) self.assertAlmostEqual(abs(cells - n.get_cells()).sum(), 0.) self.assertEqual(symbols, n.symbols) self.assertAlmostEqual(abs(positions - n.get_positions()).sum(), 0.) self.assertIsNone(n.get_velocities()) # get_step_data function check data = n.get_step_data(1) self.assertEqual(data[0], stepids[1]) self.assertAlmostEqual(data[1], times[1]) self.assertAlmostEqual(abs(cells[1] - data[2]).sum(), 0.) self.assertEqual(symbols, data[3]) self.assertAlmostEqual(abs(data[4] - positions[1]).sum(), 0.) self.assertIsNone(data[5]) # Step 70 has index 1 self.assertEqual(1, n.get_index_from_stepid(70)) with self.assertRaises(ValueError): # Step 66 does not exist n.get_index_from_stepid(66)
[docs] def test_conversion_to_structure(self): """ Check the methods to export a given time step to a StructureData node. """ import numpy # Create a node with two arrays n = TrajectoryData() # I create sample data stepids = numpy.array([60, 70]) times = stepids * 0.01 cells = numpy.array([[[ 2., 0., 0., ], [ 0., 2., 0., ], [ 0., 0., 2., ]], [[ 3., 0., 0., ], [ 0., 3., 0., ], [ 0., 0., 3., ]]]) symbols = ['H', 'O', 'C'] positions = numpy.array([[[0., 0., 0.], [0.5, 0.5, 0.5], [1.5, 1.5, 1.5]], [[0., 0., 0.], [0.5, 0.5, 0.5], [1.5, 1.5, 1.5]]]) velocities = numpy.array([[[0., 0., 0.], [0., 0., 0.], [0., 0., 0.]], [[0.5, 0.5, 0.5], [0.5, 0.5, 0.5], [-0.5, -0.5, -0.5]]]) # I set the node n.set_trajectory( stepids=stepids, cells=cells, symbols=symbols, positions=positions, times=times, velocities=velocities) from_step = n.get_step_structure(1) from_get_structure = n.get_structure(index=1) for struc in [from_step, from_get_structure]: self.assertEqual(len(struc.sites), 3) # 3 sites self.assertAlmostEqual(abs(numpy.array(struc.cell) - cells[1]).sum(), 0) newpos = numpy.array([s.position for s in struc.sites]) self.assertAlmostEqual(abs(newpos - positions[1]).sum(), 0) newkinds = [s.kind_name for s in struc.sites] self.assertEqual(newkinds, symbols) # Weird assignments (nobody should ever do this, but it is possible in # principle and we want to check k1 = Kind(name='C', symbols='Cu') k2 = Kind(name='H', symbols='He') k3 = Kind(name='O', symbols='Os', mass=100.) k4 = Kind(name='Ge', symbols='Ge') with self.assertRaises(ValueError): # Not enough kinds struc = n.get_step_structure(1, custom_kinds=[k1, k2]) with self.assertRaises(ValueError): # Too many kinds struc = n.get_step_structure(1, custom_kinds=[k1, k2, k3, k4]) with self.assertRaises(ValueError): # Wrong kinds struc = n.get_step_structure(1, custom_kinds=[k1, k2, k4]) with self.assertRaises(ValueError): # Two kinds with the same name struc = n.get_step_structure(1, custom_kinds=[k1, k2, k3, k3]) # Correct kinds struc = n.get_step_structure(1, custom_kinds=[k1, k2, k3]) # Checks self.assertEqual(len(struc.sites), 3) # 3 sites self.assertAlmostEqual(abs(numpy.array(struc.cell) - cells[1]).sum(), 0) newpos = numpy.array([s.position for s in struc.sites]) self.assertAlmostEqual(abs(newpos - positions[1]).sum(), 0) newkinds = [s.kind_name for s in struc.sites] # Kinds are in the same order as given in the custm_kinds list self.assertEqual(newkinds, symbols) newatomtypes = [struc.get_kind(s.kind_name).symbols[0] for s in struc.sites] # Atoms remain in the same order as given in the positions list self.assertEqual(newatomtypes, ['He', 'Os', 'Cu']) # Check the mass of the kind of the second atom ('O' _> symbol Os, mass 100) self.assertAlmostEqual(struc.get_kind(struc.sites[1].kind_name).mass, 100.)
[docs] def test_conversion_from_structurelist(self): """ Check the method to create a TrajectoryData from list of AiiDA structures. """ cells = [[[ 2., 0., 0., ], [ 0., 2., 0., ], [ 0., 0., 2., ]], [[ 3., 0., 0., ], [ 0., 3., 0., ], [ 0., 0., 3., ]]] symbols = [['H', 'O', 'C'], ['H', 'O', 'C']] positions = [[[0., 0., 0.], [0.5, 0.5, 0.5], [1.5, 1.5, 1.5]], [[0., 0., 0.], [0.75, 0.75, 0.75], [1.25, 1.25, 1.25]]] structurelist = [] for i in range(0, 2): struct = StructureData(cell=cells[i]) for j, symbol in enumerate(symbols[i]): struct.append_atom(symbols=symbol, position=positions[i][j]) structurelist.append(struct) td = TrajectoryData(structurelist=structurelist) self.assertEqual(td.get_cells().tolist(), cells) self.assertEqual(td.symbols, symbols[0]) self.assertEqual(td.get_positions().tolist(), positions) symbols = [['H', 'O', 'C'], ['H', 'O', 'P']] structurelist = [] for i in range(0, 2): struct = StructureData(cell=cells[i]) for j, symbol in enumerate(symbols[i]): struct.append_atom(symbols=symbol, position=positions[i][j]) structurelist.append(struct) with self.assertRaises(ValueError): td = TrajectoryData(structurelist=structurelist)
[docs] def test_export_to_file(self): """ Export the band structure on a file, check if it is working """ import numpy import os from aiida.orm.nodes.data.cif import has_pycifrw n = TrajectoryData() # I create sample data stepids = numpy.array([60, 70]) times = stepids * 0.01 cells = numpy.array([[[ 2., 0., 0., ], [ 0., 2., 0., ], [ 0., 0., 2., ]], [[ 3., 0., 0., ], [ 0., 3., 0., ], [ 0., 0., 3., ]]]) symbols = ['H', 'O', 'C'] positions = numpy.array([[[0., 0., 0.], [0.5, 0.5, 0.5], [1.5, 1.5, 1.5]], [[0., 0., 0.], [0.5, 0.5, 0.5], [1.5, 1.5, 1.5]]]) velocities = numpy.array([[[0., 0., 0.], [0., 0., 0.], [0., 0., 0.]], [[0.5, 0.5, 0.5], [0.5, 0.5, 0.5], [-0.5, -0.5, -0.5]]]) # I set the node n.set_trajectory( stepids=stepids, cells=cells, symbols=symbols, positions=positions, times=times, velocities=velocities) # define a cell alat = 4. cell = numpy.array([ [alat, 0., 0.], [0., alat, 0.], [0., 0., alat], ]) # It is not obvious how to check that the bands are correct. # I just check, for a few formats, that the file is correctly # created, at this stage ## I use this to get a file. I then close it and ask the .export() function ## to create it again. I have to remember to delete everything at the end. handle, filename = tempfile.mkstemp() os.close(handle) os.remove(filename) if has_pycifrw(): formats_to_test = ['cif', 'xsf'] else: formats_to_test = ['xsf'] for format in formats_to_test: files_created = [] # In case there is an exception try: files_created = n.export(filename, fileformat=format) with io.open(filename, encoding='utf8') as fhandle: filedata = fhandle.read() finally: for file in files_created: if os.path.exists(file): os.remove(file)
[docs]class TestKpointsData(AiidaTestCase): """ Tests the KpointsData objects. """
[docs] def test_set_kpoints_path_legacy(self): """ Regression test for the deprecated KpointsData.set_kpoints_path method. For certain formats of a direct kpoint list, it is not necessary to have defined a cell. """ import numpy from aiida.orm.nodes.data.array.kpoints import KpointsData # Create a node with two arrays kpoints_01 = KpointsData() kpoints_02 = KpointsData() kpoints_03 = KpointsData() kpoints_04 = KpointsData() # The various allowed formats format_01 = [('G', 'M')] format_02 = [('G', 'M', 30)] format_03 = [('G', (0, 0, 0), 'M', (1, 1, 1))] format_04 = [('G', (0, 0, 0), 'M', (1, 1, 1), 30)] # Without a cell defined, the first two should fail, the last two should work with self.assertRaises(ValueError): kpoints_01.set_kpoints_path(format_01) with self.assertRaises(ValueError): kpoints_02.set_kpoints_path(format_02) kpoints_03.set_kpoints_path(format_03) kpoints_04.set_kpoints_path(format_04) # Define a cell and settings it enable the usage of formats 1 and 2 alat = 4. cell = numpy.array([ [alat, 0., 0.], [0., alat, 0.], [0., 0., alat], ]) kpoints_01.set_cell(cell) kpoints_02.set_cell(cell) kpoints_01.set_kpoints_path(format_01) kpoints_02.set_kpoints_path(format_02) kpoints_03.set_kpoints_path(format_03) kpoints_04.set_kpoints_path(format_04)
[docs] def test_mesh(self): """ Check the methods to set and retrieve a mesh. """ # Create a node with two arrays k = KpointsData() # check whether the mesh can be set properly input_mesh = [4, 4, 4] k.set_kpoints_mesh(input_mesh) mesh, offset = k.get_kpoints_mesh() self.assertEqual(mesh, list(input_mesh)) self.assertEqual(offset, [0., 0., 0.]) # must be a tuple of three 0 by default # a too long list should fail with self.assertRaises(ValueError): k.set_kpoints_mesh([4, 4, 4, 4]) # now try to put explicitely an offset input_offset = [0.5, 0.5, 0.5] k.set_kpoints_mesh(input_mesh, input_offset) mesh, offset = k.get_kpoints_mesh() self.assertEqual(mesh, list(input_mesh)) self.assertEqual(offset, list(input_offset)) # verify the same but after storing k.store() self.assertEqual(mesh, list(input_mesh)) self.assertEqual(offset, list(input_offset)) # cannot modify it after storage with self.assertRaises(ModificationNotAllowed): k.set_kpoints_mesh(input_mesh)
[docs] def test_list(self): """ Test the method to set and retrieve a kpoint list. """ import numpy k = KpointsData() input_klist = numpy.array([ (0.0, 0.0, 0.0), (0.2, 0.0, 0.0), (0.0, 0.2, 0.0), (0.0, 0.0, 0.2), ]) # set kpoints list k.set_kpoints(input_klist) klist = k.get_kpoints() # try to get the same self.assertTrue(numpy.array_equal(input_klist, klist)) # if no cell is set, cannot convert into cartesian with self.assertRaises(AttributeError): _ = k.get_kpoints(cartesian=True) # try to set also weights # should fail if the weights length do not match kpoints input_weights = numpy.ones(6) with self.assertRaises(ValueError): k.set_kpoints(input_klist, weights=input_weights) # try a right one input_weights = numpy.ones(4) k.set_kpoints(input_klist, weights=input_weights) klist, weights = k.get_kpoints(also_weights=True) self.assertTrue(numpy.array_equal(weights, input_weights)) self.assertTrue(numpy.array_equal(klist, input_klist)) # verify the same, but after storing k.store() klist, weights = k.get_kpoints(also_weights=True) self.assertTrue(numpy.array_equal(weights, input_weights)) self.assertTrue(numpy.array_equal(klist, input_klist)) # cannot modify it after storage with self.assertRaises(ModificationNotAllowed): k.set_kpoints(input_klist)
[docs] def test_kpoints_to_cartesian(self): """ Test how the list of kpoints is converted to cartesian coordinates """ import numpy k = KpointsData() input_klist = numpy.array([ (0.0, 0.0, 0.0), (0.2, 0.0, 0.0), (0.0, 0.2, 0.0), (0.0, 0.0, 0.2), ]) # define a cell alat = 4. cell = numpy.array([ [alat, 0., 0.], [0., alat, 0.], [0., 0., alat], ]) k.set_cell(cell) # set kpoints list k.set_kpoints(input_klist) # verify that it is not the same of the input # (at least I check that there something has been done) klist = k.get_kpoints(cartesian=True) self.assertFalse(numpy.array_equal(klist, input_klist)) # put the kpoints in cartesian and get them back, they should be equal # internally it is doing two matrix transforms k.set_kpoints(input_klist, cartesian=True) klist = k.get_kpoints(cartesian=True) self.assertTrue(numpy.allclose(klist, input_klist, atol=1e-16))
[docs] def test_path(self): """ Test the methods to generate automatically a list of kpoints """ import numpy k = KpointsData() # shouldn't get anything wiothout having set the cell with self.assertRaises(ValueError): k.set_kpoints_path() # define a cell alat = 4. cell = numpy.array([ [alat, 0., 0.], [0., alat, 0.], [0., 0., alat], ]) k.set_cell(cell) k.set_kpoints_path() # something should be retrieved klist = k.get_kpoints() # test the various formats for specifying the path k.set_kpoints_path([ ('G', 'M'), ]) k.set_kpoints_path([ ('G', 'M', 30), ]) k.set_kpoints_path([ ('G', (0., 0., 0.), 'M', (1., 1., 1.)), ]) k.set_kpoints_path([ ('G', (0., 0., 0.), 'M', (1., 1., 1.), 30), ]) # at least 2 points per segment with self.assertRaises(ValueError): k.set_kpoints_path([ ('G', 'M', 1), ]) with self.assertRaises(ValueError): k.set_kpoints_path([ ('G', (0., 0., 0.), 'M', (1., 1., 1.), 1), ]) # try to set points with a spacing k.set_kpoints_path(kpoint_distance=0.1) # try to modify after storage k.store() with self.assertRaises(ModificationNotAllowed): k.set_kpoints_path()
[docs] def test_path_wrapper_legacy(self): """ This is a clone of the test_path test but instead it goes through the new wrapper calling the deprecated legacy implementation. This tests that the wrapper maintains the same behavior of the old implementation """ import numpy from aiida.tools.data.array.kpoints import get_explicit_kpoints_path # Shouldn't get anything without having set the cell with self.assertRaises(AttributeError): get_explicit_kpoints_path(None) # Define a cell alat = 4. cell = numpy.array([ [alat, 0., 0.], [0., alat, 0.], [0., 0., alat], ]) structure = StructureData(cell=cell) # test the various formats for specifying the path get_explicit_kpoints_path( structure, method='legacy', value=[ ('G', 'M'), ]) get_explicit_kpoints_path( structure, method='legacy', value=[ ('G', 'M', 30), ]) get_explicit_kpoints_path( structure, method='legacy', value=[ ('G', (0., 0., 0.), 'M', (1., 1., 1.)), ]) get_explicit_kpoints_path( structure, method='legacy', value=[ ('G', (0., 0., 0.), 'M', (1., 1., 1.), 30), ]) # at least 2 points per segment with self.assertRaises(ValueError): get_explicit_kpoints_path( structure, method='legacy', value=[ ('G', 'M', 1), ]) with self.assertRaises(ValueError): get_explicit_kpoints_path( structure, method='legacy', value=[ ('G', (0., 0., 0.), 'M', (1., 1., 1.), 1), ]) # try to set points with a spacing get_explicit_kpoints_path(structure, method='legacy', kpoint_distance=0.1)
[docs] def test_tetra_x(self): """ testing tetragonal cells with axis along X """ import numpy from aiida.plugins import DataFactory alat = 1.5 cell_x = [[alat, 0, 0], [0, 1, 0], [0, 0, 1]] K = DataFactory('array.kpoints') k = K() k.set_cell(cell_x) points = k.get_special_points(cartesian=True) self.assertAlmostEqual(points[0]['Z'][0], numpy.pi / alat) self.assertAlmostEqual(points[0]['Z'][1], 0.)
[docs] def test_tetra_z(self): """ testing tetragonal cells with axis along X """ import numpy from aiida.plugins import DataFactory alat = 1.5 cell_x = [[1, 0, 0], [0, 1, 0], [0, 0, alat]] K = DataFactory('array.kpoints') k = K() k.set_cell(cell_x) points = k.get_special_points(cartesian=True) self.assertAlmostEqual(points[0]['Z'][2], numpy.pi / alat) self.assertAlmostEqual(points[0]['Z'][0], 0.)
[docs] def test_tetra_z_wrapper_legacy(self): """ This is a clone of the test_tetra_z test but instead it goes through the new wrapper calling the deprecated legacy implementation. This tests that the wrapper maintains the same behavior of the old implementation """ import numpy from aiida.tools.data.array.kpoints import get_kpoints_path alat = 1.5 cell_x = [[1, 0, 0], [0, 1, 0], [0, 0, alat]] s = StructureData(cell=cell_x) result = get_kpoints_path(s, method='legacy', cartesian=True) self.assertIsInstance(result['parameters'], Dict) point_coords = result['parameters'].dict.point_coords path = result['parameters'].dict.path self.assertAlmostEqual(point_coords['Z'][2], numpy.pi / alat) self.assertAlmostEqual(point_coords['Z'][0], 0.)
[docs]class TestSpglibTupleConversion(AiidaTestCase):
[docs] def test_simple_to_aiida(self): """ Test conversion of a simple tuple to an AiiDA structure """ import numpy as np from aiida.tools import spglib_tuple_to_structure cell = np.array([[4., 1., 0.], [0., 4., 0.], [0., 0., 4.]]) relcoords = np.array([[0.09493671, 0., 0.], [0.59493671, 0.5, 0.5], [0.59493671, 0.5, 0.], [0.59493671, 0., 0.5], [0.09493671, 0.5, 0.5]]) abscoords = np.dot(cell.T, relcoords.T).T numbers = [56, 22, 8, 8, 8] struc = spglib_tuple_to_structure((cell, relcoords, numbers)) self.assertAlmostEqual(np.sum(np.abs(np.array(struc.cell) - np.array(cell))), 0.) self.assertAlmostEqual( np.sum(np.abs(np.array([site.position for site in struc.sites]) - np.array(abscoords))), 0.) self.assertEqual([site.kind_name for site in struc.sites], ['Ba', 'Ti', 'O', 'O', 'O'])
[docs] def test_complex1_to_aiida(self): """ Test conversion of a tuple to an AiiDA structure when passing also information on the kinds """ import numpy as np from aiida.tools import spglib_tuple_to_structure cell = np.array([[4., 1., 0.], [0., 4., 0.], [0., 0., 4.]]) relcoords = np.array([[0.09493671, 0., 0.], [0.59493671, 0.5, 0.5], [0.59493671, 0.5, 0.], [0.59493671, 0., 0.5], [0.09493671, 0.5, 0.5], [0.09493671, 0.5, 0.5], [0.09493671, 0.5, 0.5], [0.09493671, 0.5, 0.5], [0.09493671, 0.5, 0.5]]) abscoords = np.dot(cell.T, relcoords.T).T numbers = [56, 22, 8, 8, 8, 56000, 200000, 200001, 56001] kind_info = {'Ba': 56, 'Ba2': 56000, 'Ba3': 56001, 'BaTi': 200000, 'BaTi2': 200001, 'O': 8, 'Ti': 22} kind_info_wrong = {'Ba': 56, 'Ba2': 56000, 'Ba3': 56002, 'BaTi': 200000, 'BaTi2': 200001, 'O': 8, 'Ti': 22} kinds = [ Kind(name='Ba', symbols="Ba"), Kind(name='Ti', symbols="Ti"), Kind(name='O', symbols="O"), Kind(name='Ba2', symbols="Ba", mass=100.), Kind(name='BaTi', symbols=("Ba", "Ti"), weights=(0.5, 0.5), mass=100.), Kind(name='BaTi2', symbols=("Ba", "Ti"), weights=(0.4, 0.6), mass=100.), Kind(name='Ba3', symbols="Ba", mass=100.) ] kinds_wrong = [ Kind(name='Ba', symbols="Ba"), Kind(name='Ti', symbols="Ti"), Kind(name='O', symbols="O"), Kind(name='Ba2', symbols="Ba", mass=100.), Kind(name='BaTi', symbols=("Ba", "Ti"), weights=(0.5, 0.5), mass=100.), Kind(name='BaTi2', symbols=("Ba", "Ti"), weights=(0.4, 0.6), mass=100.), Kind(name='Ba4', symbols="Ba", mass=100.) ] # Must specify also kind_info and kinds with self.assertRaises(ValueError): struc = spglib_tuple_to_structure((cell, relcoords, numbers),) # There is no kind_info for one of the numbers with self.assertRaises(ValueError): struc = spglib_tuple_to_structure((cell, relcoords, numbers), kind_info=kind_info_wrong, kinds=kinds) # There is no kind in the kinds for one of the labels # specified in kind_info with self.assertRaises(ValueError): struc = spglib_tuple_to_structure((cell, relcoords, numbers), kind_info=kind_info, kinds=kinds_wrong) struc = spglib_tuple_to_structure((cell, relcoords, numbers), kind_info=kind_info, kinds=kinds) self.assertAlmostEqual(np.sum(np.abs(np.array(struc.cell) - np.array(cell))), 0.) self.assertAlmostEqual( np.sum(np.abs(np.array([site.position for site in struc.sites]) - np.array(abscoords))), 0.) self.assertEqual([site.kind_name for site in struc.sites], ['Ba', 'Ti', 'O', 'O', 'O', 'Ba2', 'BaTi', 'BaTi2', 'Ba3'])
[docs] def test_from_aiida(self): """ Test conversion of an AiiDA structure to a spglib tuple """ import numpy as np from aiida.tools import structure_to_spglib_tuple cell = np.array([[4., 1., 0.], [0., 4., 0.], [0., 0., 4.]]) struc = StructureData(cell=cell) struc.append_atom(symbols='Ba', position=(0, 0, 0)) struc.append_atom(symbols='Ti', position=(1, 2, 3)) struc.append_atom(symbols='O', position=(-1, -2, -4)) struc.append_kind(Kind(name='Ba2', symbols="Ba", mass=100.)) struc.append_site(Site(kind_name='Ba2', position=[3, 2, 1])) struc.append_kind(Kind(name='Test', symbols=["Ba", "Ti"], weights=[0.2, 0.4], mass=120.)) struc.append_site(Site(kind_name='Test', position=[3, 5, 1])) struc_tuple, kind_info, kinds = structure_to_spglib_tuple(struc) abscoords = np.array([_.position for _ in struc.sites]) struc_relpos = np.dot(np.linalg.inv(cell.T), abscoords.T).T self.assertAlmostEqual(np.sum(np.abs(np.array(struc.cell) - np.array(struc_tuple[0]))), 0.) self.assertAlmostEqual(np.sum(np.abs(np.array(struc_tuple[1]) - struc_relpos)), 0.) expected_kind_info = [kind_info[site.kind_name] for site in struc.sites] self.assertEqual(struc_tuple[2], expected_kind_info)
[docs] def test_aiida_roundtrip(self): """ Convert an AiiDA structure to a tuple and go back to see if we get the same results """ import numpy as np from aiida.tools import structure_to_spglib_tuple, spglib_tuple_to_structure cell = np.array([[4., 1., 0.], [0., 4., 0.], [0., 0., 4.]]) struc = StructureData(cell=cell) struc.append_atom(symbols='Ba', position=(0, 0, 0)) struc.append_atom(symbols='Ti', position=(1, 2, 3)) struc.append_atom(symbols='O', position=(-1, -2, -4)) struc.append_kind(Kind(name='Ba2', symbols="Ba", mass=100.)) struc.append_site(Site(kind_name='Ba2', position=[3, 2, 1])) struc.append_kind(Kind(name='Test', symbols=["Ba", "Ti"], weights=[0.2, 0.4], mass=120.)) struc.append_site(Site(kind_name='Test', position=[3, 5, 1])) struc_tuple, kind_info, kinds = structure_to_spglib_tuple(struc) roundtrip_struc = spglib_tuple_to_structure(struc_tuple, kind_info, kinds) self.assertAlmostEqual(np.sum(np.abs(np.array(struc.cell) - np.array(roundtrip_struc.cell))), 0.) self.assertEqual(struc.get_attribute('kinds'), roundtrip_struc.get_attribute('kinds')) self.assertEqual([_.kind_name for _ in struc.sites], [_.kind_name for _ in roundtrip_struc.sites]) self.assertEqual( np.sum( np.abs( np.array([_.position for _ in struc.sites]) - np.array([_.position for _ in roundtrip_struc.sites]))), 0.)
[docs]class TestSeekpathExplicitPath(AiidaTestCase):
[docs] @unittest.skipIf(not has_seekpath(), "No seekpath available") def test_simple(self): import numpy as np from aiida.plugins import DataFactory from aiida.tools import get_explicit_kpoints_path structure = DataFactory('structure')(cell=[[4, 0, 0], [0, 4, 0], [0, 0, 6]]) structure.append_atom(symbols='Ba', position=[0, 0, 0]) structure.append_atom(symbols='Ti', position=[2, 2, 3]) structure.append_atom(symbols='O', position=[2, 2, 0]) structure.append_atom(symbols='O', position=[2, 0, 3]) structure.append_atom(symbols='O', position=[0, 2, 3]) params = {'with_time_reversal': True, 'reference_distance': 0.025, 'recipe': 'hpkot', 'threshold': 1.e-7} return_value = get_explicit_kpoints_path(structure, method='seekpath', **params) retdict = return_value['parameters'].get_dict() self.assertTrue(retdict['has_inversion_symmetry']) self.assertFalse(retdict['augmented_path']) self.assertAlmostEqual(retdict['volume_original_wrt_prim'], 1.0) self.assertEqual( to_list_of_lists(retdict['explicit_segments']), [[0, 31], [30, 61], [60, 104], [103, 123], [122, 153], [152, 183], [182, 226], [226, 246], [246, 266]]) ret_k = return_value['explicit_kpoints'] self.assertEqual( to_list_of_lists(ret_k.labels), [[0, 'GAMMA'], [30, 'X'], [60, 'M'], [103, 'GAMMA'], [122, 'Z'], [152, 'R'], [182, 'A'], [225, 'Z'], [226, 'X'], [245, 'R'], [246, 'M'], [265, 'A']]) kpts = ret_k.get_kpoints(cartesian=False) highsympoints_relcoords = [kpts[idx] for idx, label in ret_k.labels] self.assertAlmostEqual( np.sum( np.abs( np.array([ [0., 0., 0.], # Gamma [0., 0.5, 0.], # X [0.5, 0.5, 0.], # M [0., 0., 0.], # Gamma [0., 0., 0.5], # Z [0., 0.5, 0.5], # R [0.5, 0.5, 0.5], # A [0., 0., 0.5], # Z [0., 0.5, 0.], # X [0., 0.5, 0.5], # R [0.5, 0.5, 0.], # M [0.5, 0.5, 0.5], # A ]) - np.array(highsympoints_relcoords))), 0.) ret_prims = return_value['primitive_structure'] ret_convs = return_value['conv_structure'] # The primitive structure should be the same as the one I input self.assertAlmostEqual(np.sum(np.abs(np.array(structure.cell) - np.array(ret_prims.cell))), 0.) self.assertEqual([_.kind_name for _ in structure.sites], [_.kind_name for _ in ret_prims.sites]) self.assertEqual( np.sum( np.abs( np.array([_.position for _ in structure.sites]) - np.array([_.position for _ in ret_prims.sites]))), 0.) # Also the conventional structure should be the same as the one I input self.assertAlmostEqual(np.sum(np.abs(np.array(structure.cell) - np.array(ret_convs.cell))), 0.) self.assertEqual([_.kind_name for _ in structure.sites], [_.kind_name for _ in ret_convs.sites]) self.assertEqual( np.sum( np.abs( np.array([_.position for _ in structure.sites]) - np.array([_.position for _ in ret_convs.sites]))), 0.)
[docs]class TestSeekpathPath(AiidaTestCase):
[docs] @unittest.skipIf(not has_seekpath(), "No seekpath available") def test_simple(self): import numpy as np from aiida.plugins import DataFactory from aiida.tools import get_kpoints_path structure = DataFactory('structure')(cell=[[4, 0, 0], [0, 4, 0], [0, 0, 6]]) structure.append_atom(symbols='Ba', position=[0, 0, 0]) structure.append_atom(symbols='Ti', position=[2, 2, 3]) structure.append_atom(symbols='O', position=[2, 2, 0]) structure.append_atom(symbols='O', position=[2, 0, 3]) structure.append_atom(symbols='O', position=[0, 2, 3]) params = {'with_time_reversal': True, 'recipe': 'hpkot', 'threshold': 1.e-7} return_value = get_kpoints_path(structure, method='seekpath', **params) retdict = return_value['parameters'].get_dict() self.assertTrue(retdict['has_inversion_symmetry']) self.assertFalse(retdict['augmented_path']) self.assertAlmostEqual(retdict['volume_original_wrt_prim'], 1.0) self.assertAlmostEqual(retdict['volume_original_wrt_conv'], 1.0) self.assertEqual(retdict['bravais_lattice'], 'tP') self.assertEqual(retdict['bravais_lattice_extended'], 'tP1') self.assertEqual( to_list_of_lists(retdict['path']), [['GAMMA', 'X'], ['X', 'M'], ['M', 'GAMMA'], ['GAMMA', 'Z'], ['Z', 'R'], ['R', 'A'], ['A', 'Z'], ['X', 'R'], ['M', 'A']]) self.assertEqual( retdict['point_coords'], { 'A': [0.5, 0.5, 0.5], 'M': [0.5, 0.5, 0.0], 'R': [0.0, 0.5, 0.5], 'X': [0.0, 0.5, 0.0], 'Z': [0.0, 0.0, 0.5], 'GAMMA': [0.0, 0.0, 0.0] }) self.assertAlmostEqual( np.sum( np.abs( np.array(retdict['inverse_primitive_transformation_matrix']) - np.array([[1, 0, 0], [0, 1, 0], [0, 0, 1]]))), 0.) ret_prims = return_value['primitive_structure'] ret_convs = return_value['conv_structure'] # The primitive structure should be the same as the one I input self.assertAlmostEqual(np.sum(np.abs(np.array(structure.cell) - np.array(ret_prims.cell))), 0.) self.assertEqual([_.kind_name for _ in structure.sites], [_.kind_name for _ in ret_prims.sites]) self.assertEqual( np.sum( np.abs( np.array([_.position for _ in structure.sites]) - np.array([_.position for _ in ret_prims.sites]))), 0.) # Also the conventional structure should be the same as the one I input self.assertAlmostEqual(np.sum(np.abs(np.array(structure.cell) - np.array(ret_convs.cell))), 0.) self.assertEqual([_.kind_name for _ in structure.sites], [_.kind_name for _ in ret_convs.sites]) self.assertEqual( np.sum( np.abs( np.array([_.position for _ in structure.sites]) - np.array([_.position for _ in ret_convs.sites]))), 0.)
[docs]class TestBandsData(AiidaTestCase): """ Tests the BandsData objects. """
[docs] def test_band(self): """ Check the methods to set and retrieve a mesh. """ import numpy # define a cell alat = 4. cell = numpy.array([ [alat, 0., 0.], [0., alat, 0.], [0., 0., alat], ]) k = KpointsData() k.set_cell(cell) k.set_kpoints_path() b = BandsData() b.set_kpointsdata(k) self.assertTrue(numpy.array_equal(b.cell, k.cell)) input_bands = numpy.array([numpy.ones(4) for i in range(k.get_kpoints().shape[0])]) input_occupations = input_bands b.set_bands(input_bands, occupations=input_occupations, units='ev') b.set_bands(input_bands, units='ev') b.set_bands(input_bands, occupations=input_occupations) with self.assertRaises(TypeError): b.set_bands(occupations=input_occupations, units='ev') b.set_bands(input_bands, occupations=input_occupations, units='ev') bands, occupations = b.get_bands(also_occupations=True) self.assertTrue(numpy.array_equal(bands, input_bands)) self.assertTrue(numpy.array_equal(occupations, input_occupations)) self.assertTrue(b.units == 'ev') b.store() with self.assertRaises(ModificationNotAllowed): b.set_bands(bands)
[docs] def test_export_to_file(self): """ Export the band structure on a file, check if it is working """ import numpy from aiida.orm import BandsData, KpointsData # define a cell alat = 4. cell = numpy.array([ [alat, 0., 0.], [0., alat, 0.], [0., 0., alat], ]) k = KpointsData() k.set_cell(cell) k.set_kpoints_path() b = BandsData() b.set_kpointsdata(k) # 4 bands with linearly increasing energies, it does not make sense # but is good for testing input_bands = numpy.array([numpy.ones(4) * i for i in range(k.get_kpoints().shape[0])]) b.set_bands(input_bands, units='eV') # It is not obvious how to check that the bands are correct. # I just check, for a few formats, that the file is correctly # created, at this stage # I use this to get a file. I then close it and ask the .export() function # to create it again. I have to remember to delete everything at the end. handle, filename = tempfile.mkstemp() os.close(handle) os.remove(filename) for format in ['agr', 'agr_batch', 'json', 'mpl_singlefile', 'dat_blocks', 'dat_multicolumn']: files_created = [] # In case there is an exception try: files_created = b.export(filename, fileformat=format) with io.open(filename, encoding='utf8') as fhandle: filedata = fhandle.read() finally: for file in files_created: if os.path.exists(file): os.remove(file)