openstudio-extension 0.7.1 → 0.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.gitignore +1 -0
- data/CHANGELOG.md +14 -0
- data/LICENSE.md +1 -1
- data/README.md +2 -0
- data/lib/openstudio/extension/runner.rb +12 -8
- data/lib/openstudio/extension/runner_config.rb +33 -6
- data/lib/openstudio/extension/version.rb +1 -1
- data/openstudio-extension.gemspec +6 -6
- metadata +15 -66
- data/lib/openstudio/extension/core/CreateResults.rb +0 -1033
- data/lib/openstudio/extension/core/check_air_sys_temps.rb +0 -160
- data/lib/openstudio/extension/core/check_calibration.rb +0 -125
- data/lib/openstudio/extension/core/check_cond_zns.rb +0 -54
- data/lib/openstudio/extension/core/check_domestic_hot_water.rb +0 -304
- data/lib/openstudio/extension/core/check_envelope_conductance.rb +0 -423
- data/lib/openstudio/extension/core/check_eui_by_end_use.rb +0 -132
- data/lib/openstudio/extension/core/check_eui_reasonableness.rb +0 -105
- data/lib/openstudio/extension/core/check_fan_pwr.rb +0 -68
- data/lib/openstudio/extension/core/check_internal_loads.rb +0 -363
- data/lib/openstudio/extension/core/check_mech_sys_capacity.rb +0 -196
- data/lib/openstudio/extension/core/check_mech_sys_efficiency.rb +0 -296
- data/lib/openstudio/extension/core/check_mech_sys_part_load_eff.rb +0 -434
- data/lib/openstudio/extension/core/check_mech_sys_type.rb +0 -109
- data/lib/openstudio/extension/core/check_part_loads.rb +0 -421
- data/lib/openstudio/extension/core/check_placeholder.rb +0 -45
- data/lib/openstudio/extension/core/check_plant_cap.rb +0 -93
- data/lib/openstudio/extension/core/check_plant_temps.rb +0 -129
- data/lib/openstudio/extension/core/check_plenum_loads.rb +0 -57
- data/lib/openstudio/extension/core/check_pump_pwr.rb +0 -78
- data/lib/openstudio/extension/core/check_sch_coord.rb +0 -211
- data/lib/openstudio/extension/core/check_schedules.rb +0 -281
- data/lib/openstudio/extension/core/check_simultaneous_heating_and_cooling.rb +0 -128
- data/lib/openstudio/extension/core/check_supply_air_and_thermostat_temp_difference.rb +0 -118
- data/lib/openstudio/extension/core/check_weather_files.rb +0 -102
- data/lib/openstudio/extension/core/deer_vintages.rb +0 -281
- data/lib/openstudio/extension/core/os_lib_aedg_measures.rb +0 -461
- data/lib/openstudio/extension/core/os_lib_constructions.rb +0 -353
- data/lib/openstudio/extension/core/os_lib_geometry.rb +0 -1169
- data/lib/openstudio/extension/core/os_lib_helper_methods.rb +0 -383
- data/lib/openstudio/extension/core/os_lib_hvac.rb +0 -2163
- data/lib/openstudio/extension/core/os_lib_lighting_and_equipment.rb +0 -184
- data/lib/openstudio/extension/core/os_lib_model_generation.rb +0 -3584
- data/lib/openstudio/extension/core/os_lib_model_simplification.rb +0 -1019
- data/lib/openstudio/extension/core/os_lib_outdoorair_and_infiltration.rb +0 -135
- data/lib/openstudio/extension/core/os_lib_reporting_qaqc.rb +0 -170
- data/lib/openstudio/extension/core/os_lib_schedules.rb +0 -933
@@ -1,434 +0,0 @@
|
|
1
|
-
# *******************************************************************************
|
2
|
-
# OpenStudio(R), Copyright (c) Alliance for Sustainable Energy, LLC.
|
3
|
-
# See also https://openstudio.net/license
|
4
|
-
# *******************************************************************************
|
5
|
-
|
6
|
-
module OsLib_QAQC
|
7
|
-
# include any general notes about QAQC method here
|
8
|
-
# 'standard performance curves' based on what is in OpenStudio standards for prototype building
|
9
|
-
# initially the tollerance will be hard coded vs. passed in as a method argument
|
10
|
-
|
11
|
-
# checks the number of unmet hours in the model
|
12
|
-
def check_mech_sys_part_load_eff(category, target_standard, min_pass, max_pass, name_only = false)
|
13
|
-
if target_standard.include?('90.1')
|
14
|
-
display_standard = "ASHRAE #{target_standard}"
|
15
|
-
else
|
16
|
-
display_standard = target_standard
|
17
|
-
end
|
18
|
-
|
19
|
-
component_type_array = ['ChillerElectricEIR', 'CoilCoolingDXSingleSpeed', 'CoilCoolingDXTwoSpeed', 'CoilHeatingDXSingleSpeed']
|
20
|
-
|
21
|
-
# summary of the check
|
22
|
-
check_elems = OpenStudio::AttributeVector.new
|
23
|
-
check_elems << OpenStudio::Attribute.new('name', 'Mechanical System Part Load Efficiency')
|
24
|
-
check_elems << OpenStudio::Attribute.new('category', category)
|
25
|
-
check_elems << OpenStudio::Attribute.new('description', "Check 40% and 80% part load efficency against #{display_standard} for the following compenent types: #{component_type_array.join(', ')}. Checking EIR Function of Part Load Ratio curve for chiller and EIR Function of Flow Fraction for DX coils.")
|
26
|
-
# TODO: - add in check for VAV fan
|
27
|
-
|
28
|
-
# stop here if only name is requested this is used to populate display name for arguments
|
29
|
-
if name_only == true
|
30
|
-
results = []
|
31
|
-
check_elems.each do |elem|
|
32
|
-
results << elem.valueAsString
|
33
|
-
end
|
34
|
-
return results
|
35
|
-
end
|
36
|
-
|
37
|
-
# Versions of OpenStudio greater than 2.4.0 use a modified version of
|
38
|
-
# openstudio-standards with different method calls. These methods
|
39
|
-
# require a "Standard" object instead of the standard being passed into method calls.
|
40
|
-
# This Standard object is used throughout the QAQC check.
|
41
|
-
if OpenStudio::VersionString.new(OpenStudio.openStudioVersion) < OpenStudio::VersionString.new('2.4.3')
|
42
|
-
use_old_gem_code = true
|
43
|
-
else
|
44
|
-
use_old_gem_code = false
|
45
|
-
std = Standard.build(target_standard)
|
46
|
-
end
|
47
|
-
|
48
|
-
begin
|
49
|
-
# TODO: - in future would be nice to dynamically genrate list of possible options from standards json
|
50
|
-
chiller_air_cooled_condenser_types = ['WithCondenser', 'WithoutCondenser']
|
51
|
-
chiller_water_cooled_compressor_types = ['Reciprocating', 'Scroll', 'Rotary Screw', 'Centrifugal']
|
52
|
-
absorption_types = ['Single Effect', 'Double Effect Indirect Fired', 'Double Effect Direct Fired']
|
53
|
-
|
54
|
-
# check getChillerElectricEIRs objects (will also have curve check in different script)
|
55
|
-
@model.getChillerElectricEIRs.each do |component|
|
56
|
-
# get curve and evaluate
|
57
|
-
electric_input_to_cooling_output_ratio_function_of_PLR = component.electricInputToCoolingOutputRatioFunctionOfPLR
|
58
|
-
curve_40_pct = electric_input_to_cooling_output_ratio_function_of_PLR.evaluate(0.4)
|
59
|
-
curve_80_pct = electric_input_to_cooling_output_ratio_function_of_PLR.evaluate(0.8)
|
60
|
-
|
61
|
-
# find ac properties
|
62
|
-
if use_old_gem_code
|
63
|
-
search_criteria = component.find_search_criteria(target_standard)
|
64
|
-
else
|
65
|
-
search_criteria = std.chiller_electric_eir_find_search_criteria(component)
|
66
|
-
end
|
67
|
-
|
68
|
-
# extend search_criteria for absorption_type
|
69
|
-
absorption_types.each do |absorption_type|
|
70
|
-
if component.name.to_s.include?(absorption_type)
|
71
|
-
search_criteria['absorption_type'] = absorption_type
|
72
|
-
next
|
73
|
-
end
|
74
|
-
end
|
75
|
-
# extend search_criteria for condenser type or compressor type
|
76
|
-
if search_criteria['cooling_type'] == 'AirCooled'
|
77
|
-
chiller_air_cooled_condenser_types.each do |condenser_type|
|
78
|
-
if component.name.to_s.include?(condenser_type)
|
79
|
-
search_criteria['condenser_type'] = condenser_type
|
80
|
-
next
|
81
|
-
end
|
82
|
-
end
|
83
|
-
# if no match and also no absorption_type then issue warning
|
84
|
-
if !search_criteria.key?('condenser_type') || search_criteria['condenser_type'].nil?
|
85
|
-
if !search_criteria.key?('absorption_type') || search_criteria['absorption_type'].nil?
|
86
|
-
check_elems << OpenStudio::Attribute.new('flag', "Can't find unique search criteria for #{component.name}. #{search_criteria}")
|
87
|
-
next # don't go past here
|
88
|
-
end
|
89
|
-
end
|
90
|
-
elsif search_criteria['cooling_type'] == 'WaterCooled'
|
91
|
-
chiller_air_cooled_condenser_types.each do |compressor_type|
|
92
|
-
if component.name.to_s.include?(compressor_type)
|
93
|
-
search_criteria['compressor_type'] = compressor_type
|
94
|
-
next
|
95
|
-
end
|
96
|
-
end
|
97
|
-
# if no match and also no absorption_type then issue warning
|
98
|
-
if !search_criteria.key?('compressor_type') || search_criteria['compressor_type'].nil?
|
99
|
-
if !search_criteria.key?('absorption_type') || search_criteria['absorption_type'].nil?
|
100
|
-
check_elems << OpenStudio::Attribute.new('flag', "Can't find unique search criteria for #{component.name}. #{search_criteria}")
|
101
|
-
next # don't go past here
|
102
|
-
end
|
103
|
-
end
|
104
|
-
end
|
105
|
-
|
106
|
-
# lookup chiller
|
107
|
-
if use_old_gem_code
|
108
|
-
capacity_w = component.find_capacity
|
109
|
-
else
|
110
|
-
capacity_w = std.chiller_electric_eir_find_capacity(component)
|
111
|
-
end
|
112
|
-
capacity_tons = OpenStudio.convert(capacity_w, 'W', 'ton').get
|
113
|
-
|
114
|
-
if use_old_gem_code
|
115
|
-
chlr_props = component.model.find_object($os_standards['chillers'], search_criteria, capacity_tons, Date.today)
|
116
|
-
chlr_props
|
117
|
-
chlr_props = std.model_find_object(std.standards_data['chillers'], search_criteria, capacity_tons, Date.today)
|
118
|
-
end
|
119
|
-
if chlr_props.nil?
|
120
|
-
check_elems << OpenStudio::Attribute.new('flag', "Didn't find chiller for #{component.name}. #{search_criteria}")
|
121
|
-
next # don't go past here in loop if can't find curve
|
122
|
-
end
|
123
|
-
|
124
|
-
# temp model to hold temp curve
|
125
|
-
model_temp = OpenStudio::Model::Model.new
|
126
|
-
|
127
|
-
# create temp curve
|
128
|
-
target_curve_name = chlr_props['eirfplr']
|
129
|
-
if target_curve_name.nil?
|
130
|
-
check_elems << OpenStudio::Attribute.new('flag', "Can't find target eirfplr curve for #{component.name}")
|
131
|
-
next # don't go past here in loop if can't find curve
|
132
|
-
end
|
133
|
-
if use_old_gem_code
|
134
|
-
temp_curve = model_temp.add_curve(target_curve_name)
|
135
|
-
else
|
136
|
-
temp_curve = std.model_add_curve(model_temp, target_curve_name)
|
137
|
-
end
|
138
|
-
|
139
|
-
target_curve_40_pct = temp_curve.evaluate(0.4)
|
140
|
-
target_curve_80_pct = temp_curve.evaluate(0.8)
|
141
|
-
|
142
|
-
# check curve at two points
|
143
|
-
if curve_40_pct < target_curve_40_pct * (1.0 - min_pass)
|
144
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
145
|
-
elsif curve_40_pct > target_curve_40_pct * (1.0 + max_pass)
|
146
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
147
|
-
end
|
148
|
-
if curve_80_pct < target_curve_80_pct * (1.0 - min_pass)
|
149
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
150
|
-
elsif curve_80_pct > target_curve_80_pct * (1.0 + max_pass)
|
151
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
152
|
-
end
|
153
|
-
end
|
154
|
-
|
155
|
-
# check getCoilCoolingDXSingleSpeeds objects (will also have curve check in different script)
|
156
|
-
@model.getCoilCoolingDXSingleSpeeds.each do |component|
|
157
|
-
# get curve and evaluate
|
158
|
-
eir_function_of_flow_fraction_curve = component.energyInputRatioFunctionOfFlowFractionCurve
|
159
|
-
curve_40_pct = eir_function_of_flow_fraction_curve.evaluate(0.4)
|
160
|
-
curve_80_pct = eir_function_of_flow_fraction_curve.evaluate(0.8)
|
161
|
-
|
162
|
-
# find ac properties
|
163
|
-
if use_old_gem_code
|
164
|
-
search_criteria = component.find_search_criteria(target_standard)
|
165
|
-
else
|
166
|
-
search_criteria = std.coil_dx_find_search_criteria(component)
|
167
|
-
end
|
168
|
-
|
169
|
-
if use_old_gem_code
|
170
|
-
capacity_w = component.find_capacity
|
171
|
-
else
|
172
|
-
capacity_w = std.coil_cooling_dx_single_speed_find_capacity(component)
|
173
|
-
end
|
174
|
-
capacity_btu_per_hr = OpenStudio.convert(capacity_w, 'W', 'Btu/hr').get
|
175
|
-
|
176
|
-
if use_old_gem_code
|
177
|
-
if component.heat_pump?
|
178
|
-
ac_props = component.model.find_object($os_standards['heat_pumps'], search_criteria, capacity_btu_per_hr, Date.today)
|
179
|
-
else
|
180
|
-
ac_props = component.model.find_object($os_standards['unitary_acs'], search_criteria, capacity_btu_per_hr, Date.today)
|
181
|
-
end
|
182
|
-
else
|
183
|
-
if std.coil_dx_heat_pump?(component)
|
184
|
-
ac_props = std.model_find_object(std.standards_data['heat_pumps'], search_criteria, capacity_btu_per_hr, Date.today)
|
185
|
-
else
|
186
|
-
ac_props = std.model_find_object(std.standards_data['unitary_acs'], search_criteria, capacity_btu_per_hr, Date.today)
|
187
|
-
end
|
188
|
-
end
|
189
|
-
|
190
|
-
# temp model to hold temp curve
|
191
|
-
model_temp = OpenStudio::Model::Model.new
|
192
|
-
|
193
|
-
# create temp curve
|
194
|
-
target_curve_name = ac_props['cool_eir_fflow']
|
195
|
-
if target_curve_name.nil?
|
196
|
-
check_elems << OpenStudio::Attribute.new('flag', "Can't find target cool_eir_fflow curve for #{component.name}")
|
197
|
-
next # don't go past here in loop if can't find curve
|
198
|
-
end
|
199
|
-
if use_old_gem_code
|
200
|
-
temp_curve = model_temp.add_curve(target_curve_name)
|
201
|
-
else
|
202
|
-
temp_curve = std.model_add_curve(model_temp, target_curve_name)
|
203
|
-
end
|
204
|
-
target_curve_40_pct = temp_curve.evaluate(0.4)
|
205
|
-
target_curve_80_pct = temp_curve.evaluate(0.8)
|
206
|
-
|
207
|
-
# check curve at two points
|
208
|
-
if curve_40_pct < target_curve_40_pct * (1.0 - min_pass)
|
209
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
210
|
-
elsif curve_40_pct > target_curve_40_pct * (1.0 + max_pass)
|
211
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
212
|
-
end
|
213
|
-
if curve_80_pct < target_curve_80_pct * (1.0 - min_pass)
|
214
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
215
|
-
elsif curve_80_pct > target_curve_80_pct * (1.0 + max_pass)
|
216
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
217
|
-
end
|
218
|
-
end
|
219
|
-
|
220
|
-
# check CoilCoolingDXTwoSpeed objects (will also have curve check in different script)
|
221
|
-
@model.getCoilCoolingDXTwoSpeeds.each do |component|
|
222
|
-
# get curve and evaluate
|
223
|
-
eir_function_of_flow_fraction_curve = component.energyInputRatioFunctionOfFlowFractionCurve
|
224
|
-
curve_40_pct = eir_function_of_flow_fraction_curve.evaluate(0.4)
|
225
|
-
curve_80_pct = eir_function_of_flow_fraction_curve.evaluate(0.8)
|
226
|
-
|
227
|
-
# find ac properties
|
228
|
-
if use_old_gem_code
|
229
|
-
search_criteria = component.find_search_criteria(target_standard)
|
230
|
-
else
|
231
|
-
search_criteria = std.coil_dx_find_search_criteria(component)
|
232
|
-
end
|
233
|
-
|
234
|
-
if use_old_gem_code
|
235
|
-
capacity_w = component.find_capacity
|
236
|
-
else
|
237
|
-
capacity_w = std.coil_cooling_dx_two_speed_find_capacity(component)
|
238
|
-
end
|
239
|
-
capacity_btu_per_hr = OpenStudio.convert(capacity_w, 'W', 'Btu/hr').get
|
240
|
-
|
241
|
-
if use_old_gem_code
|
242
|
-
ac_props = component.model.find_object($os_standards['unitary_acs'], search_criteria, capacity_btu_per_hr, Date.today)
|
243
|
-
else
|
244
|
-
ac_props = std.model_find_object(std.standards_data['unitary_acs'], search_criteria, capacity_btu_per_hr, Date.today)
|
245
|
-
end
|
246
|
-
|
247
|
-
# temp model to hold temp curve
|
248
|
-
model_temp = OpenStudio::Model::Model.new
|
249
|
-
|
250
|
-
# create temp curve
|
251
|
-
target_curve_name = ac_props['cool_eir_fflow']
|
252
|
-
if target_curve_name.nil?
|
253
|
-
check_elems << OpenStudio::Attribute.new('flag', "Can't find target cool_eir_flow curve for #{component.name}")
|
254
|
-
next # don't go past here in loop if can't find curve
|
255
|
-
end
|
256
|
-
if use_old_gem_code
|
257
|
-
temp_curve = model_temp.add_curve(target_curve_name)
|
258
|
-
else
|
259
|
-
temp_curve = std.model_add_curve(model_temp, target_curve_name)
|
260
|
-
end
|
261
|
-
target_curve_40_pct = temp_curve.evaluate(0.4)
|
262
|
-
target_curve_80_pct = temp_curve.evaluate(0.8)
|
263
|
-
|
264
|
-
# check curve at two points
|
265
|
-
if curve_40_pct < target_curve_40_pct * (1.0 - min_pass)
|
266
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
267
|
-
elsif curve_40_pct > target_curve_40_pct * (1.0 + max_pass)
|
268
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
269
|
-
end
|
270
|
-
if curve_80_pct < target_curve_80_pct * (1.0 - min_pass)
|
271
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
272
|
-
elsif curve_80_pct > target_curve_80_pct * (1.0 + max_pass)
|
273
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
274
|
-
end
|
275
|
-
end
|
276
|
-
|
277
|
-
# check CoilCoolingDXTwoSpeed objects (will also have curve check in different script)
|
278
|
-
@model.getCoilHeatingDXSingleSpeeds.each do |component|
|
279
|
-
# get curve and evaluate
|
280
|
-
eir_function_of_flow_fraction_curve = component.energyInputRatioFunctionofFlowFractionCurve # why lowercase of here but not in CoilCoolingDX objects
|
281
|
-
curve_40_pct = eir_function_of_flow_fraction_curve.evaluate(0.4)
|
282
|
-
curve_80_pct = eir_function_of_flow_fraction_curve.evaluate(0.8)
|
283
|
-
|
284
|
-
# find ac properties
|
285
|
-
if use_old_gem_code
|
286
|
-
search_criteria = component.find_search_criteria(target_standard)
|
287
|
-
else
|
288
|
-
search_criteria = std.coil_dx_find_search_criteria(component)
|
289
|
-
end
|
290
|
-
|
291
|
-
if use_old_gem_code
|
292
|
-
capacity_w = component.find_capacity
|
293
|
-
else
|
294
|
-
capacity_w = std.coil_heating_dx_single_speed_find_capacity(component)
|
295
|
-
end
|
296
|
-
capacity_btu_per_hr = OpenStudio.convert(capacity_w, 'W', 'Btu/hr').get
|
297
|
-
|
298
|
-
if use_old_gem_code
|
299
|
-
ac_props = component.model.find_object($os_standards['heat_pumps_heating'], search_criteria, capacity_btu_per_hr, Date.today)
|
300
|
-
else
|
301
|
-
ac_props = std.model_find_object(std.standards_data['heat_pumps_heating'], search_criteria, capacity_btu_per_hr, Date.today)
|
302
|
-
end
|
303
|
-
if ac_props.nil?
|
304
|
-
target_curve_name = nil
|
305
|
-
else
|
306
|
-
target_curve_name = ac_props['heat_eir_fflow']
|
307
|
-
end
|
308
|
-
|
309
|
-
# temp model to hold temp curve
|
310
|
-
model_temp = OpenStudio::Model::Model.new
|
311
|
-
|
312
|
-
# create temp curve
|
313
|
-
if target_curve_name.nil?
|
314
|
-
check_elems << OpenStudio::Attribute.new('flag', "Can't find target curve for #{component.name}")
|
315
|
-
next # don't go past here in loop if can't find curve
|
316
|
-
end
|
317
|
-
if use_old_gem_code
|
318
|
-
temp_curve = model_temp.add_curve(target_curve_name)
|
319
|
-
else
|
320
|
-
temp_curve = std.model_add_curve(model_temp, target_curve_name)
|
321
|
-
end
|
322
|
-
|
323
|
-
# Ensure that the curve was found in standards before attempting to evaluate
|
324
|
-
if temp_curve.nil?
|
325
|
-
check_elems << OpenStudio::Attribute.new('flag', "Can't find coefficients of curve called #{target_curve_name} for #{component.name}, cannot check part-load performance.")
|
326
|
-
next
|
327
|
-
end
|
328
|
-
|
329
|
-
target_curve_40_pct = temp_curve.evaluate(0.4)
|
330
|
-
target_curve_80_pct = temp_curve.evaluate(0.8)
|
331
|
-
|
332
|
-
# check curve at two points
|
333
|
-
if curve_40_pct < target_curve_40_pct * (1.0 - min_pass)
|
334
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
335
|
-
elsif curve_40_pct > target_curve_40_pct * (1.0 + max_pass)
|
336
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
337
|
-
end
|
338
|
-
if curve_80_pct < target_curve_80_pct * (1.0 - min_pass)
|
339
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
340
|
-
elsif curve_80_pct > target_curve_80_pct * (1.0 + max_pass)
|
341
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
342
|
-
end
|
343
|
-
end
|
344
|
-
|
345
|
-
# check
|
346
|
-
@model.getFanVariableVolumes.each do |component|
|
347
|
-
# skip if not on multi-zone system.
|
348
|
-
if component.airLoopHVAC.is_initialized
|
349
|
-
airloop = component.airLoopHVAC.get
|
350
|
-
|
351
|
-
next unless airloop.thermalZones.size > 1.0
|
352
|
-
end
|
353
|
-
|
354
|
-
# skip of brake horsepower is 0
|
355
|
-
if use_old_gem_code
|
356
|
-
next if component.brake_horsepower == 0.0
|
357
|
-
else
|
358
|
-
next if std.fan_brake_horsepower(component) == 0.0
|
359
|
-
end
|
360
|
-
|
361
|
-
# temp model for use by temp model and target curve
|
362
|
-
model_temp = OpenStudio::Model::Model.new
|
363
|
-
|
364
|
-
# get coeficents for fan
|
365
|
-
model_fan_coefs = []
|
366
|
-
model_fan_coefs << component.fanPowerCoefficient1.get
|
367
|
-
model_fan_coefs << component.fanPowerCoefficient2.get
|
368
|
-
model_fan_coefs << component.fanPowerCoefficient3.get
|
369
|
-
model_fan_coefs << component.fanPowerCoefficient4.get
|
370
|
-
model_fan_coefs << component.fanPowerCoefficient5.get
|
371
|
-
|
372
|
-
# make model curve
|
373
|
-
model_curve = OpenStudio::Model::CurveQuartic.new(model_temp)
|
374
|
-
model_curve.setCoefficient1Constant(model_fan_coefs[0])
|
375
|
-
model_curve.setCoefficient2x(model_fan_coefs[1])
|
376
|
-
model_curve.setCoefficient3xPOW2(model_fan_coefs[2])
|
377
|
-
model_curve.setCoefficient4xPOW3(model_fan_coefs[3])
|
378
|
-
model_curve.setCoefficient5xPOW4(model_fan_coefs[4])
|
379
|
-
curve_40_pct = model_curve.evaluate(0.4)
|
380
|
-
curve_80_pct = model_curve.evaluate(0.8)
|
381
|
-
|
382
|
-
# get target coefs
|
383
|
-
target_fan = OpenStudio::Model::FanVariableVolume.new(model_temp)
|
384
|
-
if use_old_gem_code
|
385
|
-
target_fan.set_control_type('Multi Zone VAV with Static Pressure Reset')
|
386
|
-
else
|
387
|
-
std.fan_variable_volume_set_control_type(target_fan, 'Multi Zone VAV with VSD and Static Pressure Reset')
|
388
|
-
end
|
389
|
-
|
390
|
-
# get coeficents for fan
|
391
|
-
target_fan_coefs = []
|
392
|
-
target_fan_coefs << target_fan.fanPowerCoefficient1.get
|
393
|
-
target_fan_coefs << target_fan.fanPowerCoefficient2.get
|
394
|
-
target_fan_coefs << target_fan.fanPowerCoefficient3.get
|
395
|
-
target_fan_coefs << target_fan.fanPowerCoefficient4.get
|
396
|
-
target_fan_coefs << target_fan.fanPowerCoefficient5.get
|
397
|
-
|
398
|
-
# make model curve
|
399
|
-
target_curve = OpenStudio::Model::CurveQuartic.new(model_temp)
|
400
|
-
target_curve.setCoefficient1Constant(target_fan_coefs[0])
|
401
|
-
target_curve.setCoefficient2x(target_fan_coefs[1])
|
402
|
-
target_curve.setCoefficient3xPOW2(target_fan_coefs[2])
|
403
|
-
target_curve.setCoefficient4xPOW3(target_fan_coefs[3])
|
404
|
-
target_curve.setCoefficient5xPOW4(target_fan_coefs[4])
|
405
|
-
target_curve_40_pct = target_curve.evaluate(0.4)
|
406
|
-
target_curve_80_pct = target_curve.evaluate(0.8)
|
407
|
-
|
408
|
-
# check curve at two points
|
409
|
-
if curve_40_pct < target_curve_40_pct * (1.0 - min_pass)
|
410
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
411
|
-
elsif curve_40_pct > target_curve_40_pct * (1.0 + max_pass)
|
412
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 40% of #{curve_40_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_40_pct.round(2)} for #{display_standard}.")
|
413
|
-
end
|
414
|
-
if curve_80_pct < target_curve_80_pct * (1.0 - min_pass)
|
415
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{min_pass * 100} % below the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
416
|
-
elsif curve_80_pct > target_curve_80_pct * (1.0 + max_pass)
|
417
|
-
check_elems << OpenStudio::Attribute.new('flag', "The curve value at 80% of #{curve_80_pct.round(2)} for #{component.name} is more than #{max_pass * 100} % above the typical value of #{target_curve_80_pct.round(2)} for #{display_standard}.")
|
418
|
-
end
|
419
|
-
end
|
420
|
-
rescue StandardError => e
|
421
|
-
# brief description of ruby error
|
422
|
-
check_elems << OpenStudio::Attribute.new('flag', "Error prevented QAQC check from running (#{e}).")
|
423
|
-
|
424
|
-
# backtrace of ruby error for diagnostic use
|
425
|
-
if @error_backtrace then check_elems << OpenStudio::Attribute.new('flag', e.backtrace.join("\n").to_s) end
|
426
|
-
end
|
427
|
-
|
428
|
-
# add check_elms to new attribute
|
429
|
-
check_elem = OpenStudio::Attribute.new('check', check_elems)
|
430
|
-
|
431
|
-
return check_elem
|
432
|
-
# note: registerWarning and registerValue will be added for checks downstream using os_lib_reporting_qaqc.rb
|
433
|
-
end
|
434
|
-
end
|
@@ -1,109 +0,0 @@
|
|
1
|
-
# *******************************************************************************
|
2
|
-
# OpenStudio(R), Copyright (c) Alliance for Sustainable Energy, LLC.
|
3
|
-
# See also https://openstudio.net/license
|
4
|
-
# *******************************************************************************
|
5
|
-
|
6
|
-
module OsLib_QAQC
|
7
|
-
# include any general notes about QAQC method here
|
8
|
-
|
9
|
-
# checks the number of unmet hours in the model
|
10
|
-
def check_mech_sys_type(category, target_standard, name_only = false)
|
11
|
-
# summary of the check
|
12
|
-
check_elems = OpenStudio::AttributeVector.new
|
13
|
-
check_elems << OpenStudio::Attribute.new('name', 'Mechanical System Type')
|
14
|
-
check_elems << OpenStudio::Attribute.new('category', category)
|
15
|
-
|
16
|
-
# add ASHRAE to display of target standard if includes with 90.1
|
17
|
-
if target_standard.include?('90.1 2013')
|
18
|
-
check_elems << OpenStudio::Attribute.new('description', 'Check against ASHRAE 90.1 2013 Tables G3.1.1 A-B. Infers the baseline system type based on the equipment serving the zone and their heating/cooling fuels. Only does a high-level inference; does not look for the presence/absence of required controls, etc.')
|
19
|
-
else
|
20
|
-
check_elems << OpenStudio::Attribute.new('description', 'Check against ASHRAE 90.1. Infers the baseline system type based on the equipment serving the zone and their heating/cooling fuels. Only does a high-level inference; does not look for the presence/absence of required controls, etc.')
|
21
|
-
end
|
22
|
-
|
23
|
-
# stop here if only name is requested this is used to populate display name for arguments
|
24
|
-
if name_only == true
|
25
|
-
results = []
|
26
|
-
check_elems.each do |elem|
|
27
|
-
results << elem.valueAsString
|
28
|
-
end
|
29
|
-
return results
|
30
|
-
end
|
31
|
-
|
32
|
-
# Versions of OpenStudio greater than 2.4.0 use a modified version of
|
33
|
-
# openstudio-standards with different method calls. These methods
|
34
|
-
# require a "Standard" object instead of the standard being passed into method calls.
|
35
|
-
# This Standard object is used throughout the QAQC check.
|
36
|
-
if OpenStudio::VersionString.new(OpenStudio.openStudioVersion) < OpenStudio::VersionString.new('2.4.3')
|
37
|
-
use_old_gem_code = true
|
38
|
-
else
|
39
|
-
use_old_gem_code = false
|
40
|
-
std = Standard.build(target_standard)
|
41
|
-
end
|
42
|
-
|
43
|
-
begin
|
44
|
-
# Get the actual system type for all zones in the model
|
45
|
-
act_zone_to_sys_type = {}
|
46
|
-
@model.getThermalZones.each do |zone|
|
47
|
-
if use_old_gem_code
|
48
|
-
act_zone_to_sys_type[zone] = zone.infer_system_type
|
49
|
-
else
|
50
|
-
act_zone_to_sys_type[zone] = std.thermal_zone_infer_system_type(zone)
|
51
|
-
end
|
52
|
-
end
|
53
|
-
|
54
|
-
# Get the baseline system type for all zones in the model
|
55
|
-
if use_old_gem_code
|
56
|
-
climate_zone = @model.get_building_climate_zone_and_building_type['climate_zone']
|
57
|
-
else
|
58
|
-
climate_zone = std.model_get_building_properties(@model)['climate_zone']
|
59
|
-
end
|
60
|
-
|
61
|
-
if use_old_gem_code
|
62
|
-
req_zone_to_sys_type = @model.get_baseline_system_type_by_zone(target_standard, climate_zone)
|
63
|
-
else
|
64
|
-
req_zone_to_sys_type = std.model_get_baseline_system_type_by_zone(@model, climate_zone)
|
65
|
-
end
|
66
|
-
|
67
|
-
# Compare the actual to the correct
|
68
|
-
@model.getThermalZones.each do |zone|
|
69
|
-
# TODO: - skip if plenum
|
70
|
-
is_plenum = false
|
71
|
-
zone.spaces.each do |space|
|
72
|
-
if use_old_gem_code
|
73
|
-
if space.plenum?
|
74
|
-
is_plenum = true
|
75
|
-
end
|
76
|
-
else
|
77
|
-
if std.space_plenum?(space)
|
78
|
-
is_plenum = true
|
79
|
-
end
|
80
|
-
end
|
81
|
-
end
|
82
|
-
next if is_plenum
|
83
|
-
|
84
|
-
req_sys_type = req_zone_to_sys_type[zone]
|
85
|
-
act_sys_type = act_zone_to_sys_type[zone]
|
86
|
-
|
87
|
-
if act_sys_type == req_sys_type
|
88
|
-
puts "#{zone.name} system type = #{act_sys_type}"
|
89
|
-
else
|
90
|
-
if req_sys_type == '' then req_sys_type = 'Unknown' end
|
91
|
-
puts "#{zone.name} baseline system type is incorrect. Supposed to be #{req_sys_type}, but was #{act_sys_type} instead."
|
92
|
-
check_elems << OpenStudio::Attribute.new('flag', "#{zone.name} baseline system type is incorrect. Supposed to be #{req_sys_type}, but was #{act_sys_type} instead.")
|
93
|
-
end
|
94
|
-
end
|
95
|
-
rescue StandardError => e
|
96
|
-
# brief description of ruby error
|
97
|
-
check_elems << OpenStudio::Attribute.new('flag', "Error prevented QAQC check from running (#{e}).")
|
98
|
-
|
99
|
-
# backtrace of ruby error for diagnostic use
|
100
|
-
if @error_backtrace then check_elems << OpenStudio::Attribute.new('flag', e.backtrace.join("\n").to_s) end
|
101
|
-
end
|
102
|
-
|
103
|
-
# add check_elms to new attribute
|
104
|
-
check_elem = OpenStudio::Attribute.new('check', check_elems)
|
105
|
-
|
106
|
-
return check_elem
|
107
|
-
# note: registerWarning and registerValue will be added for checks downstream using os_lib_reporting_qaqc.rb
|
108
|
-
end
|
109
|
-
end
|