# -*- 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, 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 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(file=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(file=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(file=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.set_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(file=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(file=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(file=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(file=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(file=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(file=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(file=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(file=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(file=tmpf.name)
with self.assertRaises(ValueError):
ret_dict = refine_inline(c)
[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(file=tmpf.name)
default2 = CifData(file=tmpf.name, scan_type='standard')
self.assertEquals(default._prepare_cif(), default2._prepare_cif())
flex = CifData(file=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.set_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(file=tmpf.name, parse_policy='eager')
self.assertIsNot(eager._values, None)
# this should not parse the cif
lazy = CifData(file=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_set_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(file=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.set_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.set_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(file=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(file=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(file=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] @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_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, 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, input_mesh)
self.assertEqual(offset, input_offset)
# verify the same but after storing
k.store()
self.assertEqual(mesh, input_mesh)
self.assertEqual(offset, 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_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_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([[0.,0.,0.],[0.1,0.1,0.1]])
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([[0.,0.,0.],[0.1,0.1,0.1]])
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)