test_ase.py 3.88 KB
Newer Older
Gao, Xiang's avatar
Gao, Xiang committed
1
2
from ase.lattice.cubic import Diamond
from ase.md.langevin import Langevin
3
from ase.md.nptberendsen import NPTBerendsen
4
from ase import units
5
from ase.io import read
Gao, Xiang's avatar
Gao, Xiang committed
6
from ase.calculators.test import numeric_force
Gao, Xiang's avatar
Gao, Xiang committed
7
import numpy as np
Gao, Xiang's avatar
Gao, Xiang committed
8
9
10
import torch
import torchani
import unittest
11
12
13
import os

path = os.path.dirname(os.path.realpath(__file__))
Gao, Xiang's avatar
Gao, Xiang committed
14
15
16


def get_numeric_force(atoms, eps):
17
    fn = torch.zeros((len(atoms), 3), dtype=torch.double)
Gao, Xiang's avatar
Gao, Xiang committed
18
19
20
21
22
23
24
25
    for i in range(len(atoms)):
        for j in range(3):
            fn[i, j] = numeric_force(atoms, i, j, eps)
    return fn


class TestASE(unittest.TestCase):

26
    def setUp(self):
27
        self.model = torchani.models.ANI1x(model_index=0).double()
28

29
    def testWithNumericalForceWithPBCEnabled(self):
Gao, Xiang's avatar
Gao, Xiang committed
30
31
32
33
        # Run a Langevin thermostat dynamic for 100 steps and after the dynamic
        # check once that the numerical and analytical force agree to a given
        # relative tolerance
        relative_tolerance = 0.1
34
        atoms = Diamond(symbol="C", pbc=True)
35
        calculator = self.model.ase()
Gao, Xiang's avatar
Gao, Xiang committed
36
37
38
39
40
41
42
        atoms.set_calculator(calculator)
        dyn = Langevin(atoms, 5 * units.fs, 30000000 * units.kB, 0.002)
        dyn.run(100)
        f = torch.from_numpy(atoms.get_forces())
        fn = get_numeric_force(atoms, 0.001)
        df = (f - fn).abs().max()
        avgf = f.abs().mean()
Gao, Xiang's avatar
Gao, Xiang committed
43
        if avgf > 0:
Gao, Xiang's avatar
Gao, Xiang committed
44
            self.assertLess(df / avgf, relative_tolerance)
Gao, Xiang's avatar
Gao, Xiang committed
45

46
    def testWithNumericalStressWithPBCEnabled(self):
Gao, Xiang's avatar
Gao, Xiang committed
47
48
49
50
51
        # Run NPT dynamics for some steps and periodically check that the
        # numerical and analytical stresses agree up to a given
        # absolute difference
        tolerance = 1e-5
        filename = os.path.join(path, '../tools/generate-unit-test-expect/others/Benzene.json')
52
        benzene = read(filename)
Gao, Xiang's avatar
Gao, Xiang committed
53
54
55
56
57
58
        # set velocities to a very small value to avoid division by zero
        # warning due to initial zero temperature.
        #
        # Note that there are 4 benzene molecules, thus, 48 atoms in
        # Benzene.json
        benzene.set_velocities(np.full((48, 3), 1e-15))
59
        calculator = self.model.ase()
60
61
62
63
64
65
66
67
68
69
        benzene.set_calculator(calculator)
        dyn = NPTBerendsen(benzene, timestep=0.1 * units.fs,
                           temperature=300 * units.kB,
                           taut=0.1 * 1000 * units.fs, pressure=1.01325,
                           taup=1.0 * 1000 * units.fs, compressibility=4.57e-5)

        def test_stress():
            stress = benzene.get_stress()
            numerical_stress = calculator.calculate_numerical_stress(benzene)
            diff = torch.from_numpy(stress - numerical_stress).abs().max().item()
Gao, Xiang's avatar
Gao, Xiang committed
70
71
72
            self.assertLess(diff, tolerance)
        dyn.attach(test_stress, interval=5)
        dyn.run(20)
73

Gao, Xiang's avatar
Gao, Xiang committed
74

75
class TestASEWithPTI(unittest.TestCase):
Gao, Xiang's avatar
Gao, Xiang committed
76
77
78
    # Tests that the values obtained by wrapping a BuiltinModel or
    # BuiltinEnsemble with a calculator are the same with and without
    # periodic_table_index
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102

    def setUp(self):
        self.model_pti = torchani.models.ANI1x(periodic_table_index=True).double()
        self.model = torchani.models.ANI1x().double()

    def testEqualEnsemblePTI(self):
        calculator_pti = self.model_pti.ase()
        calculator = self.model.ase()
        atoms = Diamond(symbol="C", pbc=True)
        atoms_pti = Diamond(symbol="C", pbc=True)
        atoms.set_calculator(calculator)
        atoms_pti.set_calculator(calculator_pti)
        self.assertEqual(atoms.get_potential_energy(), atoms_pti.get_potential_energy())

    def testEqualOneModelPTI(self):
        calculator_pti = self.model_pti[0].ase()
        calculator = self.model[0].ase()
        atoms = Diamond(symbol="C", pbc=True)
        atoms_pti = Diamond(symbol="C", pbc=True)
        atoms.set_calculator(calculator)
        atoms_pti.set_calculator(calculator_pti)
        self.assertEqual(atoms.get_potential_energy(), atoms_pti.get_potential_energy())


Gao, Xiang's avatar
Gao, Xiang committed
103
104
if __name__ == '__main__':
    unittest.main()