File: checkresult.py

package info (click to toggle)
neuron 8.2.6-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 34,760 kB
  • sloc: cpp: 149,571; python: 58,465; ansic: 50,329; sh: 3,510; xml: 213; pascal: 51; makefile: 35; sed: 5
file content (95 lines) | stat: -rw-r--r-- 3,405 bytes parent folder | download | duplicates (2)
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
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
import json
import math
from neuron import h


class Chk:
    """A test helper that compares stored results with test results
    chk = Chk(filename)

    chk("name", value) normally asserts that the value == dict[name]
    But, if dict[name] does not exist, the value is added to the dict.
    """

    def __init__(self, fname, must_exist=True):
        """JSON with name, fname, must be a dict.
        Assert file exists by default.
        A temporary must_exist=False provides a simple way for the test
        author to bootstrap the file.
        """
        import os

        self.fname = fname
        self.d = {}
        self.modified = False
        if must_exist:
            assert os.path.exists(fname)
        else:
            if not os.path.exists(fname):
                return

        with open(fname, "rb") as f:
            self.d = json.load(f)
        assert type(self.d) is dict

    def __call__(self, key, value, tol=0.0):
        """Assert value == dict[key] unless value is a hoc_Vector.
        For the Vector case, assert the maximum relative element difference is <= tol.
        If the key does not exist add {key:value} to the dict.
        """

        if key in self.d:
            if type(value) == type(h.Vector):  # actually hoc.HocObject
                # Convert to list to keep the `equal` method below simple
                value = list(value)

            # Hand-rolled comparison that uses `tol` for arithmetic values
            # buried inside lists of lists.
            def equal(a, b):
                assert type(a) == type(b)
                if type(a) in (float, int):
                    match = math.isclose(a, b, rel_tol=tol)
                    if not match:
                        print(a, b, "diff", abs(a - b) / max(abs(a), abs(b)), ">", tol)
                    return match
                elif type(a) == str:
                    return a == b
                elif type(a) == list:
                    # List comprehension avoids short-circuit, so the "diff"
                    # message just above is printed for all elements
                    return all([equal(aa, bb) for aa, bb in zip(a, b)])
                raise Exception(
                    "Don't know how to compare objects of type " + str(type(a))
                )

            match = equal(value, self.d[key])
            if not match:
                print(key, "difference")
                print("std = ", self.d[key])
                print("val = ", value)
            assert match
        else:
            print("{} added {}".format(self, key))
            if type(value) == type(h.Vector):  # actually hoc.HocObject
                self.d[key] = value.to_python()
            else:
                self.d[key] = value
            self.modified = True

    def rm(self, key):
        """Remove key from dict.
        Only needed temporarily when a key value needs updating.
        """
        if key in self.d:
            del self.d[key]
            self.modified = True

    def save(self):
        """If dict has been modified, save back to original filename
        This is typically called at the end of the test and normally
        does nothing unless the author adds a new chk(...) to the test.
        """
        if self.modified:
            with open(self.fname, "w") as f:
                json.dump(self.d, f, indent=2)
                f.write("\n")