File: test_optimizers.py

package info (click to toggle)
python-ase 3.21.1-2
  • links: PTS, VCS
  • area: main
  • in suites: bullseye
  • size: 13,936 kB
  • sloc: python: 122,428; xml: 946; makefile: 111; javascript: 47
file content (76 lines) | stat: -rw-r--r-- 2,277 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
from functools import partial

import pytest

from ase.calculators.emt import EMT
from ase.optimize import (MDMin, FIRE, LBFGS, LBFGSLineSearch, BFGSLineSearch,
                          BFGS, GoodOldQuasiNewton, GPMin, Berny)
from ase.optimize.sciopt import SciPyFminCG, SciPyFminBFGS
from ase.optimize.precon import PreconFIRE, PreconLBFGS
from ase.cluster import Icosahedron
from ase.build import bulk


@pytest.fixture(scope='module')
def ref_atoms():
    atoms = bulk('Au')
    atoms.calc = EMT()
    atoms.get_potential_energy()
    return atoms


def atoms_no_pbc():
    ref_atoms = Icosahedron('Ag', 2, 3.82975)
    ref_atoms.calc = EMT()
    atoms = ref_atoms.copy()
    atoms.calc = EMT()
    atoms.rattle(stdev=0.1, seed=7)
    e_unopt = atoms.get_potential_energy()
    assert e_unopt > 7  # it's 7.318 as of writing this test
    return atoms, ref_atoms


@pytest.fixture
def atoms(ref_atoms):
    atoms = ref_atoms * (2, 2, 2)
    atoms.rattle(stdev=0.1, seed=7)
    atoms.calc = EMT()
    e_unopt = atoms.get_potential_energy()
    assert e_unopt > 0.45  # it's 0.499 as of writing this test
    return atoms


optclasses = [
    MDMin, FIRE, LBFGS, LBFGSLineSearch, BFGSLineSearch,
    BFGS, GoodOldQuasiNewton, GPMin, SciPyFminCG, SciPyFminBFGS,
    PreconLBFGS, PreconFIRE, Berny,
]


@pytest.mark.parametrize('optcls', optclasses)
def test_optimize(optcls, atoms, ref_atoms):
    if optcls is Berny:
        pytest.importorskip('berny')  # check if pyberny installed
        optcls = partial(optcls, dihedral=False)
        optcls.__name__ = Berny.__name__
        atoms, ref_atoms = atoms_no_pbc()
    kw = {}
    if optcls is PreconLBFGS:
        kw['precon'] = None

    fmax = 0.01
    with optcls(atoms, logfile='opt.log', **kw) as opt:
        opt.run(fmax=fmax)

    forces = atoms.get_forces()
    final_fmax = max((forces**2).sum(axis=1)**0.5)
    ref_energy = ref_atoms.get_potential_energy()
    e_opt = atoms.get_potential_energy() * len(ref_atoms) / len(atoms)
    e_err = abs(e_opt - ref_energy)
    print()
    print('{:>20}: fmax={:.05f} eopt={:.06f}, err={:06e}'
          .format(optcls.__name__, final_fmax, e_opt, e_err))
    return final_fmax, e_err

    assert final_fmax < fmax
    assert e_err < 1e-5  # (This tolerance is arbitrary)