-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathvalidation.py
226 lines (198 loc) · 7.3 KB
/
validation.py
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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
"""Validation functions for values.
Based on validation module of pyrk (https://github.com/pyrk).
.. moduleauthor:: Kyle Niemeyer <[email protected]>
"""
# Python 2 compatibility
from __future__ import print_function
from __future__ import division
import sys
if sys.version_info > (3,):
long = int
import pint
# Local imports
from .utils import units
def validate_geq(value_name, value, low_lim):
"""Raise error if value lower than specified lower limit or wrong type.
Parameters
---------
value_name : str
Name of value being tested
value : int, float, numpy.ndarray, pint.Quantity
Value to be tested
low_lim : type(value)
Lowest acceptable limit of ``value``
Returns
-------
value : type(value)
The original value
"""
try:
if validate_num(value_name, value) < low_lim:
msg = (value_name + ' must be greater than or equal to ' +
str(low_lim) + '.\n'
'Value provided was: ' + str(value)
)
# RuntimeError used to avoid being caught by Pint comparison error.
# Pint should really raise TypeError (or something) rather than
# ValueError.
raise RuntimeError(msg)
else:
return value
except ValueError:
if isinstance(value, units.Quantity):
msg = ('\n' + value_name + ' given with units, when variable '
'should be dimensionless.'
)
raise pint.DimensionalityError(value.units, None,
extra_msg=msg
)
else:
msg = ('\n' + value_name + ' not given in units. '
'Correct units share dimensionality with: ' +
str(low_lim.units)
)
raise pint.DimensionalityError(None, low_lim.units,
extra_msg=msg
)
except pint.DimensionalityError:
msg = ('\n' + value_name + ' given in incompatible units. '
'Correct units share dimensionality with: ' +
str(low_lim.units)
)
raise pint.DimensionalityError(value.units, low_lim.units,
extra_msg=msg
)
except:
raise
def validate_gt(value_name, value, low_lim):
"""Raise error if value not greater than lower limit or wrong type.
Parameters
---------
value_name : str
Name of value being tested
value : int, float, numpy.ndarray, pint.Quantity
Value to be tested
low_lim : type(value)
``value`` must be greater than this limit
Returns
-------
value : type(value)
The original value
"""
try:
if not validate_num(value_name, value) > low_lim:
msg = (value_name + ' must be greater than ' +
str(low_lim) + '.\n'
'Value provided was: ' + str(value)
)
# RuntimeError used to avoid being caught by Pint comparison error.
# Pint should really raise TypeError (or something) rather than
# ValueError.
raise RuntimeError(msg)
else:
return value
except ValueError:
if isinstance(value, units.Quantity):
msg = ('\n' + value_name + ' given with units, when variable '
'should be dimensionless.'
)
raise pint.DimensionalityError(value.units, None,
extra_msg=msg
)
else:
msg = ('\n' + value_name + ' not given in units. '
'Correct units share dimensionality with: ' +
str(low_lim.units)
)
raise pint.DimensionalityError(None, low_lim.units,
extra_msg=msg
)
except pint.DimensionalityError:
msg = ('\n' + value_name + ' given in incompatible units. '
'Correct units share dimensionality with: ' +
str(low_lim.units)
)
raise pint.DimensionalityError(value.units, low_lim.units,
extra_msg=msg
)
except:
raise
def validate_leq(value_name, value, upp_lim):
"""Raise error if value greater than specified upper limit or wrong type.
Parameters
---------
value_name : str
Name of value being tested
value : int, float, numpy.ndarray, pint.Quantity
Value to be tested
upp_lim : type(value)
Highest acceptable limit of ``value``
Returns
-------
value : type(value)
The original value
"""
try:
if validate_num(value_name, value) > upp_lim:
msg = (value_name + ' must be less than or equal to ' +
str(upp_lim) + '.\n'
'Value provided was: ' + str(value)
)
# RuntimeError used to avoid being caught by Pint comparison error.
# Pint should really raise TypeError (or something) rather than
# ValueError.
raise RuntimeError(msg)
else:
return value
except ValueError:
if isinstance(value, units.Quantity):
msg = ('\n' + value_name + ' given with units, when variable '
'should be dimensionless.'
)
raise pint.DimensionalityError(value.units, None,
extra_msg=msg
)
else:
msg = ('\n' + value_name + ' not given in units. '
'Correct units share dimensionality with: ' +
str(upp_lim.units)
)
raise pint.DimensionalityError(None, upp_lim.units,
extra_msg=msg
)
except pint.DimensionalityError:
msg = ('\n' + value_name + ' given in incompatible units. '
'Correct units share dimensionality with: ' +
str(upp_lim.units)
)
raise pint.DimensionalityError(value.units, upp_lim.units,
extra_msg=msg
)
except:
raise
def validate_num(value_name, value):
"""Raise error if value is not a number.
Parameters
---------
value_name : str
Name of value being tested
value : int, float, numpy.ndarray, pint.Quantity
Value to be tested
Returns
-------
value : type(value)
The original value
"""
if isinstance(value, (int, long, float, units.Quantity)):
return value
else:
try:
if isinstance(value.magnitude, (int, long, float, units.Quantity)):
return value
except AttributeError:
pass
msg = (value_name + ' must be an integer, long, float, or Quantity. \n'
'The value provided was of type ' + str(type(value)) + ' and '
'value ' + str(value)
)
raise TypeError(msg)