forked from ladybug-tools/honeybee-core
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtyping.py
334 lines (272 loc) · 12.4 KB
/
typing.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
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
"""Collection of methods for type input checking."""
import re
import os
import math
import uuid
try:
INFPOS = math.inf
INFNEG = -1 * math.inf
except AttributeError:
# python 2
INFPOS = float('inf')
INFNEG = float('-inf')
def valid_string(value, input_name=''):
"""Check that a string is valid for both Radiance and EnergyPlus.
This is used for honeybee geometry object names.
"""
try:
illegal_match = re.search(r'[^.A-Za-z0-9_-]', value)
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
assert illegal_match is None, 'Illegal character "{}" found in {}'.format(
illegal_match.group(0), input_name)
assert len(value) > 0, 'Input {} "{}" contains no characters.'.format(
input_name, value)
assert len(value) <= 100, 'Input {} "{}" must be less than 100 characters.'.format(
input_name, value)
return value
def valid_rad_string(value, input_name=''):
"""Check that a string is valid for Radiance.
This is used for radiance modifier names, etc.
"""
try:
illegal_match = re.search(r'[^.A-Za-z0-9_-]', value)
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
assert illegal_match is None, 'Illegal character "{}" found in {}'.format(
illegal_match.group(0), input_name)
assert len(value) > 0, 'Input {} "{}" contains no characters.'.format(
input_name, value)
return value
def valid_ep_string(value, input_name=''):
"""Check that a string is valid for EnergyPlus.
This is used for energy material names, schedule names, etc.
"""
try:
non_ascii = tuple(i for i in value if ord(i) >= 128)
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
assert non_ascii == (), 'Illegal characters {} found in {}'.format(
non_ascii, input_name)
illegal_match = re.search(r'[,;!\n\t]', value)
assert illegal_match is None, 'Illegal character "{}" found in {}'.format(
illegal_match.group(0), input_name)
assert len(value) > 0, 'Input {} "{}" contains no characters.'.format(
input_name, value)
assert len(value) <= 100, 'Input {} "{}" must be less than 100 characters.'.format(
input_name, value)
return value
def _number_check(value, input_name):
"""Check if value is a number."""
try:
number = float(value)
except (ValueError, TypeError):
raise TypeError('Input {} must be a number. Got {}: {}.'.format(
input_name, type(value), value))
return number
def float_in_range(value, mi=INFNEG, ma=INFPOS, input_name=''):
"""Check a float value to be between minimum and maximum."""
number = _number_check(value, input_name)
assert mi <= number <= ma, 'Input number {} must be between {} and {}. ' \
'Got {}'.format(input_name, mi, ma, value)
return number
def float_in_range_excl(value, mi=INFNEG, ma=INFPOS, input_name=''):
"""Check a float value to be greater than minimum and less than maximum."""
number = _number_check(value, input_name)
assert mi < number < ma, 'Input number {} must be greater than {} ' \
'and less than {}. Got {}'.format(input_name, mi, ma, value)
return number
def float_in_range_excl_incl(value, mi=INFNEG, ma=INFPOS, input_name=''):
"""Check a float value to be greater than minimum and less than/equal to maximum."""
number = _number_check(value, input_name)
assert mi < number <= ma, 'Input number {} must be greater than {} and less than ' \
'or equal to {}. Got {}'.format(input_name, mi, ma, value)
return number
def float_in_range_incl_excl(value, mi=INFNEG, ma=INFPOS, input_name=''):
"""Check a float value to be greater than/equal to minimum and less than maximum."""
number = _number_check(value, input_name)
assert mi <= number < ma, 'Input number {} must be greater than or equal to {} ' \
'and less than {}. Got {}'.format(input_name, mi, ma, value)
return number
def int_in_range(value, mi=INFNEG, ma=INFPOS, input_name=''):
"""Check an integer value to be between minimum and maximum."""
try:
number = int(value)
except ValueError:
# try to convert to float and then digit if possible
try:
number = int(float(value))
except (ValueError, TypeError):
raise TypeError('Input {} must be an integer. Got {}: {}.'.format(
input_name, type(value), value))
except (ValueError, TypeError):
raise TypeError('Input {} must be an integer. Got {}: {}.'.format(
input_name, type(value), value))
assert mi <= number <= ma, 'Input integer {} must be between {} and {}. ' \
'Got {}.'.format(input_name, mi, ma, value)
return number
def float_positive(value, input_name=''):
"""Check a float value to be positive."""
return float_in_range(value, 0, INFPOS, input_name)
def int_positive(value, input_name=''):
"""Check if an integer value is positive."""
return int_in_range(value, 0, INFPOS, input_name)
def tuple_with_length(value, length=3, item_type=float, input_name=''):
"""Try to create a tuple with a certain value."""
try:
value = tuple(item_type(v) for v in value)
except (ValueError, TypeError):
raise TypeError('Input {} must be a {}.'.format(
input_name, item_type))
assert len(value) == length, 'Input {} length must be {} not {}'.format(
input_name, length, len(value))
return value
def list_with_length(value, length=3, item_type=float, input_name=''):
"""Try to create a list with a certain value."""
try:
value = [item_type(v) for v in value]
except (ValueError, TypeError):
raise TypeError('Input {} must be a {}.'.format(
input_name, item_type))
assert len(value) == length, 'Input {} length must be {} not {}'.format(
input_name, length, len(value))
return value
def clean_string(value, input_name=''):
"""Clean a string so that it is valid for both Radiance and EnergyPlus.
This will strip out spaces and special characters and raise an error if the
string is empty after stripping or has more than 100 characters.
"""
try:
val = re.sub(r'[^.A-Za-z0-9_-]', '', value)
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
assert len(val) > 0, 'Input {} "{}" contains no valid characters.'.format(
input_name, value)
assert len(val) <= 100, 'Input {} "{}" must be less than 100 characters.'.format(
input_name, value)
return val
def clean_rad_string(value, input_name=''):
"""Clean a string for Radiance that can be used for rad material names.
This includes stripping out illegal characters and white spaces as well as
raising an error if no legal characters are found.
"""
try:
val = re.sub(r'[^.A-Za-z0-9_-]', '', value)
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
assert len(val) > 0, 'Input {} "{}" contains no valid characters.'.format(
input_name, value)
return val
def clean_ep_string(value, input_name=''):
"""Clean a string for EnergyPlus that can be used for energy material names.
This includes stripping out all illegal characters, removing trailing spaces,
and rasing an error if the name is not longer than 100 characters or no legal
characters found.
"""
try:
val = ''.join(i for i in value if ord(i) < 128) # strip out non-ascii
val = re.sub(r'[,;!\n\t]', '', val) # strip out E+ special characters
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
val = val.strip()
assert len(val) > 0, 'Input {} "{}" contains no valid characters.'.format(
input_name, value)
assert len(val) <= 100, 'Input {} "{}" must be less than 100 characters.'.format(
input_name, value)
return val
def clean_and_id_string(value, input_name=''):
"""Clean a string and add 8 unique characters to it to make it unique.
Strings longer than 50 characters will be truncated before adding the ID.
The resulting string will be valid for both Radiance and EnergyPlus.
"""
try:
val = re.sub(r'[^.A-Za-z0-9_-]', '', value)
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
if len(val) > 50:
val = val[:50]
return val + '_' + str(uuid.uuid4())[:8]
def clean_and_id_rad_string(value, input_name=''):
"""Clean a string and add 8 unique characters to it to make it unique for Radiance.
This includes stripping out illegal characters and white spaces.
"""
try:
val = re.sub(r'[^.A-Za-z0-9_-]', '', value)
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
return val + '_' + str(uuid.uuid4())[:8]
def clean_and_id_ep_string(value, input_name=''):
"""Clean a string and add 8 unique characters to it to make it unique for EnergyPlus.
This includes stripping out all illegal characters and removing trailing white spaces.
Strings longer than 50 characters will be truncated before adding the ID.
"""
try:
val = ''.join(i for i in value if ord(i) < 128) # strip out non-ascii
val = re.sub(r'[,;!\n\t]', '', val) # strip out E+ special characters
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
val = val.strip()
if len(val) > 50:
val = val[:50]
return val + '_' + str(uuid.uuid4())[:8]
def truncate_and_id_string(value, truncate_len=32, uuid_len=0, input_name=''):
"""Truncate a string to a length with an option to add unique characters at the end.
Note that all outputs will always be the truncate_len or less and the uuid_len
just specifies the number of characters to replace at the end with unique ones.
The result will be valid for EnergyPlus, Radiance, and likely many more engines
with different types of character restrictions.
"""
try:
val = re.sub(r'[^.A-Za-z0-9_-]', '', value)
except TypeError:
raise TypeError('Input {} must be a text string. Got {}: {}.'.format(
input_name, type(value), value))
final_len = truncate_len - uuid_len
if len(val) > final_len:
val = val[:final_len]
if uuid_len > 0:
return val + str(uuid.uuid4())[:uuid_len]
return val
def fixed_string_length(value, target_len=32):
"""Truncate a string or add trailing spaces to hit a target character length.
This is useful when trying to construct human-readable tables of text.
"""
if len(value) > target_len:
return value[:target_len]
elif len(value) < target_len:
return value + ' ' * (target_len - len(value))
else:
return value
def invalid_dict_error(invalid_dict, error):
"""Raise a ValueError for an invalid dictionary that failed to serialize.
This error message will include the identifier (and display_name) if they are
present within the invalid_dict, making it easier for ens users to find the
invalid object within large objects like Models.
Args:
invalid_dict: A dictionary of an invalid honeybee object that failed
to serialize.
error:
"""
obj_type = invalid_dict['type'].replace('Abridged', '') \
if 'type' in invalid_dict else 'Honeybee Object'
obj_id = invalid_dict['identifier'] if 'identifier' in invalid_dict else ''
full_id = '{}[{}]'.format(invalid_dict['display_name'], obj_id) \
if 'display_name' in invalid_dict else obj_id
raise ValueError('{} "{}" is invalid:\n{}'.format(obj_type, full_id, error))
wrapper = '"' if os.name == 'nt' else '\''
"""String wrapper."""
def normpath(value):
"""Normalize path eliminating double slashes, etc and put it in quotes if needed."""
value = os.path.normpath(value)
if ' ' in value:
value = '{0}{1}{0}'.format(wrapper, value)
return value