1 |
#!/usr/bin/env python |
2 |
# |
3 |
# Copyright (C) 2002, 2003 by Intevation GmbH |
4 |
# Authors: |
5 |
# Thomas Koester <[email protected]> |
6 |
# |
7 |
# This program is free software under the GPL (>=v2) |
8 |
# Read the file COPYING coming with the software for details. |
9 |
|
10 |
""" |
11 |
Range class for Scientific Parameter |
12 |
""" |
13 |
|
14 |
__version__ = "$Revision$" |
15 |
# $Source$ |
16 |
# $Id$ |
17 |
|
18 |
import re |
19 |
import types |
20 |
|
21 |
_inf = float('1e1000') # FIXME: hack for infinite |
22 |
|
23 |
class Range: |
24 |
|
25 |
number_re = '(?P<%s>-?(\d*\.?\d*([eE][\-+]?\d+)?|oo))' |
26 |
brace_re = '(?P<%s>[][])' |
27 |
range_re = '^' + brace_re % 'left' + number_re % 'begin' + \ |
28 |
';' + number_re % 'end' + brace_re % 'right' + '$' |
29 |
parse_range = re.compile(range_re) |
30 |
|
31 |
def __init__(self, range=None): |
32 |
self._SetRange(range) |
33 |
|
34 |
def _SetRange(self, range): |
35 |
if isinstance(range, Range): |
36 |
self._SetRange(range.GetRange()) |
37 |
elif range in [None, '']: |
38 |
self._SetRange(']-oo;oo[') |
39 |
elif type(range) == types.TupleType: |
40 |
if (len(range) == 4 and range[0] in ['[', ']'] |
41 |
and range[3] in ['[', ']']): |
42 |
self._left = range[0] |
43 |
self._begin = self.float(range[1]) |
44 |
self._end = self.float(range[2]) |
45 |
self._right = range[3] |
46 |
if (self._begin > self._end or |
47 |
(self._begin == self._end and |
48 |
(self._left != '[' or self._right != ']'))): |
49 |
raise ValueError("illegal range: %s" % (range,)) |
50 |
else: |
51 |
raise ValueError("can't parse range: %r" % (range,)) |
52 |
else: |
53 |
self._range = range |
54 |
match = self.parse_range.match(self._range) |
55 |
if match: |
56 |
self._SetRange((match.group('left'), |
57 |
match.group('begin'), |
58 |
match.group('end'), |
59 |
match.group('right'))) |
60 |
else: |
61 |
raise ValueError("can't parse range: %s" % (range,)) |
62 |
|
63 |
def GetRange(self): |
64 |
"""return internal representation of range |
65 |
|
66 |
4-tuple ('[' or ']', begin(float), end(float), '[' or ']') |
67 |
|
68 |
""" |
69 |
return (self._left, self._begin, self._end, self._right) |
70 |
|
71 |
def float(self, value): |
72 |
"""convert string or number to float""" |
73 |
if value == 'oo': |
74 |
return _inf |
75 |
elif value == '-oo': |
76 |
return -_inf |
77 |
else: |
78 |
return float(value) |
79 |
|
80 |
def _float2string(self, value): |
81 |
"""convert float value to string |
82 |
|
83 |
(minus) infinity will be converted to (-)oo, |
84 |
scientific notation will be used if necessary. |
85 |
|
86 |
""" |
87 |
if value == _inf: |
88 |
return 'oo' |
89 |
elif value == -_inf: |
90 |
return '-oo' |
91 |
else: |
92 |
return "%g" % (value,) |
93 |
|
94 |
def string(self, range): |
95 |
"""convert internal representation to string""" |
96 |
left, begin, end, right = range |
97 |
return "%s%s;%s%s" % (left, self._float2string(begin), |
98 |
self._float2string(end), right) |
99 |
|
100 |
def __contains__(self, value): |
101 |
if self._left == ']': |
102 |
contains = value > self._begin |
103 |
else: |
104 |
contains = value >= self._begin |
105 |
if self._right == '[': |
106 |
contains = contains and (value < self._end) |
107 |
else: |
108 |
contains = contains and (value <= self._end) |
109 |
return contains |
110 |
|
111 |
def __eq__(self, other): |
112 |
return (self.GetRange() == other.GetRange()) |
113 |
|
114 |
def __ne__(self, other): |
115 |
return not self.__eq__(other) |
116 |
|
117 |
def __str__(self): |
118 |
return self.string(self.GetRange()) |
119 |
|
120 |
|
121 |
def _test(): |
122 |
range1 = Range(']0;99]') |
123 |
print 'range1 =', range1, range1.GetRange() |
124 |
for i in [-0.1, 0, 0.1, 9.9, 99, 99.9]: |
125 |
print '%4.1f in range1 =' % i, i in range1 |
126 |
range2 = Range(']-oo;10[') |
127 |
print 'range2 =', range2, range2.GetRange() |
128 |
for i in [-0.1, 0, 0.1, 9.9, 10, 10.1]: |
129 |
print '%4.1f not in range2 =' % i, i not in range2 |
130 |
range3 = Range(']1e-1;1E2]') |
131 |
print 'range3 =', range3, range3.GetRange() |
132 |
for i in [0, 0.1, 0.11, 10, 100, 101]: |
133 |
print '%4.1f not in range3 =' % i, i not in range3 |
134 |
print 'range3 != range2 =', range3 != range2 |
135 |
print 'range3 != Range("]1e-1;1E2]") =', range3 != Range("]1e-1;1E2]") |
136 |
|
137 |
range4 = Range('') |
138 |
print 'range4 =', range4, range4.GetRange() |
139 |
|
140 |
range5 = Range(']0;99E+00]') |
141 |
print 'range5 =', range5, range5.GetRange() |
142 |
range6 = Range(']0;99E+01]') |
143 |
print 'range6 =', range6, range6.GetRange() |
144 |
range7 = Range(']0;99E-01]') |
145 |
print 'range7 =', range7, range7.GetRange() |
146 |
|
147 |
if __name__ == "__main__": |
148 |
_test() |