openstudio-extension 0.1.0 → 0.1.1
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 +9 -0
- data/.rubocop.yml +9 -0
- data/Gemfile +3 -1
- data/Jenkinsfile +10 -0
- data/README.md +230 -12
- data/Rakefile +88 -3
- data/bin/console +3 -3
- data/doc_templates/LICENSE.md +27 -0
- data/doc_templates/README.md.erb +42 -0
- data/doc_templates/copyright_erb.txt +36 -0
- data/doc_templates/copyright_js.txt +4 -0
- data/doc_templates/copyright_ruby.txt +34 -0
- data/init_templates/README.md +37 -0
- data/init_templates/gemspec.txt +32 -0
- data/init_templates/openstudio_module.rb +50 -0
- data/init_templates/spec.rb +47 -0
- data/init_templates/spec_helper.rb +49 -0
- data/init_templates/template_gemfile.txt +17 -0
- data/init_templates/template_rakefile.txt +15 -0
- data/init_templates/version.rb +40 -0
- data/lib/files/openstudio-extension-gem-test.ddy +536 -0
- data/lib/files/openstudio-extension-gem-test.epw +8768 -0
- data/lib/files/openstudio-extension-gem-test.stat +554 -0
- data/lib/measures/openstudio_extension_test_measure/LICENSE.md +27 -0
- data/lib/measures/openstudio_extension_test_measure/README.md +26 -0
- data/lib/measures/openstudio_extension_test_measure/README.md.erb +42 -0
- data/lib/measures/openstudio_extension_test_measure/measure.rb +72 -0
- data/lib/measures/openstudio_extension_test_measure/measure.xml +83 -0
- data/lib/measures/openstudio_extension_test_measure/resources/os_lib_helper_methods.rb +399 -0
- data/lib/measures/openstudio_extension_test_measure/tests/OpenStudioExtensionTestMeasure_Test.rb +75 -0
- data/lib/openstudio/extension.rb +220 -0
- data/lib/openstudio/extension/core/CreateResults.rb +879 -0
- data/lib/openstudio/extension/core/check_air_sys_temps.rb +190 -0
- data/lib/openstudio/extension/core/check_calibration.rb +155 -0
- data/lib/openstudio/extension/core/check_cond_zns.rb +84 -0
- data/lib/openstudio/extension/core/check_domestic_hot_water.rb +334 -0
- data/lib/openstudio/extension/core/check_envelope_conductance.rb +453 -0
- data/lib/openstudio/extension/core/check_eui_by_end_use.rb +162 -0
- data/lib/openstudio/extension/core/check_eui_reasonableness.rb +135 -0
- data/lib/openstudio/extension/core/check_fan_pwr.rb +98 -0
- data/lib/openstudio/extension/core/check_internal_loads.rb +393 -0
- data/lib/openstudio/extension/core/check_mech_sys_capacity.rb +226 -0
- data/lib/openstudio/extension/core/check_mech_sys_efficiency.rb +326 -0
- data/lib/openstudio/extension/core/check_mech_sys_part_load_eff.rb +464 -0
- data/lib/openstudio/extension/core/check_mech_sys_type.rb +139 -0
- data/lib/openstudio/extension/core/check_part_loads.rb +451 -0
- data/lib/openstudio/extension/core/check_placeholder.rb +75 -0
- data/lib/openstudio/extension/core/check_plant_cap.rb +123 -0
- data/lib/openstudio/extension/core/check_plant_temps.rb +159 -0
- data/lib/openstudio/extension/core/check_plenum_loads.rb +87 -0
- data/lib/openstudio/extension/core/check_pump_pwr.rb +108 -0
- data/lib/openstudio/extension/core/check_sch_coord.rb +241 -0
- data/lib/openstudio/extension/core/check_schedules.rb +311 -0
- data/lib/openstudio/extension/core/check_simultaneous_heating_and_cooling.rb +158 -0
- data/lib/openstudio/extension/core/check_supply_air_and_thermostat_temp_difference.rb +148 -0
- data/lib/openstudio/extension/core/check_weather_files.rb +132 -0
- data/lib/openstudio/extension/core/deer_vintages.rb +311 -0
- data/lib/openstudio/extension/core/os_lib_aedg_measures.rb +491 -0
- data/lib/openstudio/extension/core/os_lib_cofee.rb +259 -0
- data/lib/openstudio/extension/core/os_lib_constructions.rb +378 -0
- data/lib/openstudio/extension/core/os_lib_geometry.rb +1022 -0
- data/lib/openstudio/extension/core/os_lib_helper_methods.rb +399 -0
- data/lib/openstudio/extension/core/os_lib_hvac.rb +2171 -0
- data/lib/openstudio/extension/core/os_lib_lighting_and_equipment.rb +214 -0
- data/lib/openstudio/extension/core/os_lib_model_generation.rb +817 -0
- data/lib/openstudio/extension/core/os_lib_model_simplification.rb +1049 -0
- data/lib/openstudio/extension/core/os_lib_outdoorair_and_infiltration.rb +165 -0
- data/lib/openstudio/extension/core/os_lib_reporting.rb +4652 -0
- data/lib/openstudio/extension/core/os_lib_reporting_qaqc.rb +200 -0
- data/lib/openstudio/extension/core/os_lib_schedules.rb +963 -0
- data/lib/openstudio/extension/rake_task.rb +149 -0
- data/lib/openstudio/extension/runner.rb +644 -0
- data/lib/openstudio/extension/version.rb +40 -0
- data/openstudio-extension.gemspec +20 -15
- metadata +150 -14
- data/.travis.yml +0 -7
- data/lib/OpenStudio/Extension/rake_task.rb +0 -84
- data/lib/OpenStudio/Extension/version.rb +0 -33
- data/lib/OpenStudio/extension.rb +0 -65
@@ -0,0 +1,139 @@
|
|
1
|
+
# *******************************************************************************
|
2
|
+
# OpenStudio(R), Copyright (c) 2008-2019, Alliance for Sustainable Energy, LLC.
|
3
|
+
# All rights reserved.
|
4
|
+
# Redistribution and use in source and binary forms, with or without
|
5
|
+
# modification, are permitted provided that the following conditions are met:
|
6
|
+
#
|
7
|
+
# (1) Redistributions of source code must retain the above copyright notice,
|
8
|
+
# this list of conditions and the following disclaimer.
|
9
|
+
#
|
10
|
+
# (2) Redistributions in binary form must reproduce the above copyright notice,
|
11
|
+
# this list of conditions and the following disclaimer in the documentation
|
12
|
+
# and/or other materials provided with the distribution.
|
13
|
+
#
|
14
|
+
# (3) Neither the name of the copyright holder nor the names of any contributors
|
15
|
+
# may be used to endorse or promote products derived from this software without
|
16
|
+
# specific prior written permission from the respective party.
|
17
|
+
#
|
18
|
+
# (4) Other than as required in clauses (1) and (2), distributions in any form
|
19
|
+
# of modifications or other derivative works may not use the "OpenStudio"
|
20
|
+
# trademark, "OS", "os", or any other confusingly similar designation without
|
21
|
+
# specific prior written permission from Alliance for Sustainable Energy, LLC.
|
22
|
+
#
|
23
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS
|
24
|
+
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
25
|
+
# THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
26
|
+
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE
|
27
|
+
# UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF
|
28
|
+
# THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
29
|
+
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
|
30
|
+
# OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
31
|
+
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
32
|
+
# STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
33
|
+
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
34
|
+
# *******************************************************************************
|
35
|
+
|
36
|
+
module OsLib_QAQC
|
37
|
+
# include any general notes about QAQC method here
|
38
|
+
|
39
|
+
# checks the number of unmet hours in the model
|
40
|
+
def check_mech_sys_type(category, target_standard, name_only = false)
|
41
|
+
# summary of the check
|
42
|
+
check_elems = OpenStudio::AttributeVector.new
|
43
|
+
check_elems << OpenStudio::Attribute.new('name', 'Mechanical System Type')
|
44
|
+
check_elems << OpenStudio::Attribute.new('category', category)
|
45
|
+
|
46
|
+
# add ASHRAE to display of target standard if includes with 90.1
|
47
|
+
if target_standard.include?('90.1 2013')
|
48
|
+
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.')
|
49
|
+
else
|
50
|
+
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.')
|
51
|
+
end
|
52
|
+
|
53
|
+
# stop here if only name is requested this is used to populate display name for arguments
|
54
|
+
if name_only == true
|
55
|
+
results = []
|
56
|
+
check_elems.each do |elem|
|
57
|
+
results << elem.valueAsString
|
58
|
+
end
|
59
|
+
return results
|
60
|
+
end
|
61
|
+
|
62
|
+
# Versions of OpenStudio greater than 2.4.0 use a modified version of
|
63
|
+
# openstudio-standards with different method calls. These methods
|
64
|
+
# require a "Standard" object instead of the standard being passed into method calls.
|
65
|
+
# This Standard object is used throughout the QAQC check.
|
66
|
+
if OpenStudio::VersionString.new(OpenStudio.openStudioVersion) < OpenStudio::VersionString.new('2.4.3')
|
67
|
+
use_old_gem_code = true
|
68
|
+
else
|
69
|
+
use_old_gem_code = false
|
70
|
+
std = Standard.build(target_standard)
|
71
|
+
end
|
72
|
+
|
73
|
+
begin
|
74
|
+
# Get the actual system type for all zones in the model
|
75
|
+
act_zone_to_sys_type = {}
|
76
|
+
@model.getThermalZones.each do |zone|
|
77
|
+
if use_old_gem_code
|
78
|
+
act_zone_to_sys_type[zone] = zone.infer_system_type
|
79
|
+
else
|
80
|
+
act_zone_to_sys_type[zone] = std.thermal_zone_infer_system_type(zone)
|
81
|
+
end
|
82
|
+
end
|
83
|
+
|
84
|
+
# Get the baseline system type for all zones in the model
|
85
|
+
if use_old_gem_code
|
86
|
+
climate_zone = @model.get_building_climate_zone_and_building_type['climate_zone']
|
87
|
+
else
|
88
|
+
climate_zone = std.model_get_building_climate_zone_and_building_type(@model)['climate_zone']
|
89
|
+
end
|
90
|
+
|
91
|
+
if use_old_gem_code
|
92
|
+
req_zone_to_sys_type = @model.get_baseline_system_type_by_zone(target_standard, climate_zone)
|
93
|
+
else
|
94
|
+
req_zone_to_sys_type = std.model_get_baseline_system_type_by_zone(@model, climate_zone)
|
95
|
+
end
|
96
|
+
|
97
|
+
# Compare the actual to the correct
|
98
|
+
@model.getThermalZones.each do |zone|
|
99
|
+
# TODO: - skip if plenum
|
100
|
+
is_plenum = false
|
101
|
+
zone.spaces.each do |space|
|
102
|
+
if use_old_gem_code
|
103
|
+
if space.plenum?
|
104
|
+
is_plenum = true
|
105
|
+
end
|
106
|
+
else
|
107
|
+
if std.space_plenum?(space)
|
108
|
+
is_plenum = true
|
109
|
+
end
|
110
|
+
end
|
111
|
+
end
|
112
|
+
next if is_plenum
|
113
|
+
|
114
|
+
req_sys_type = req_zone_to_sys_type[zone]
|
115
|
+
act_sys_type = act_zone_to_sys_type[zone]
|
116
|
+
|
117
|
+
if act_sys_type == req_sys_type
|
118
|
+
puts "#{zone.name} system type = #{act_sys_type}"
|
119
|
+
else
|
120
|
+
if req_sys_type == '' then req_sys_type = 'Unknown' end
|
121
|
+
puts "#{zone.name} baseline system type is incorrect. Supposed to be #{req_sys_type}, but was #{act_sys_type} instead."
|
122
|
+
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.")
|
123
|
+
end
|
124
|
+
end
|
125
|
+
rescue StandardError => e
|
126
|
+
# brief description of ruby error
|
127
|
+
check_elems << OpenStudio::Attribute.new('flag', "Error prevented QAQC check from running (#{e}).")
|
128
|
+
|
129
|
+
# backtrace of ruby error for diagnostic use
|
130
|
+
if @error_backtrace then check_elems << OpenStudio::Attribute.new('flag', e.backtrace.join("\n").to_s) end
|
131
|
+
end
|
132
|
+
|
133
|
+
# add check_elms to new attribute
|
134
|
+
check_elem = OpenStudio::Attribute.new('check', check_elems)
|
135
|
+
|
136
|
+
return check_elem
|
137
|
+
# note: registerWarning and registerValue will be added for checks downstream using os_lib_reporting_qaqc.rb
|
138
|
+
end
|
139
|
+
end
|
@@ -0,0 +1,451 @@
|
|
1
|
+
# *******************************************************************************
|
2
|
+
# OpenStudio(R), Copyright (c) 2008-2019, Alliance for Sustainable Energy, LLC.
|
3
|
+
# All rights reserved.
|
4
|
+
# Redistribution and use in source and binary forms, with or without
|
5
|
+
# modification, are permitted provided that the following conditions are met:
|
6
|
+
#
|
7
|
+
# (1) Redistributions of source code must retain the above copyright notice,
|
8
|
+
# this list of conditions and the following disclaimer.
|
9
|
+
#
|
10
|
+
# (2) Redistributions in binary form must reproduce the above copyright notice,
|
11
|
+
# this list of conditions and the following disclaimer in the documentation
|
12
|
+
# and/or other materials provided with the distribution.
|
13
|
+
#
|
14
|
+
# (3) Neither the name of the copyright holder nor the names of any contributors
|
15
|
+
# may be used to endorse or promote products derived from this software without
|
16
|
+
# specific prior written permission from the respective party.
|
17
|
+
#
|
18
|
+
# (4) Other than as required in clauses (1) and (2), distributions in any form
|
19
|
+
# of modifications or other derivative works may not use the "OpenStudio"
|
20
|
+
# trademark, "OS", "os", or any other confusingly similar designation without
|
21
|
+
# specific prior written permission from Alliance for Sustainable Energy, LLC.
|
22
|
+
#
|
23
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS
|
24
|
+
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
25
|
+
# THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
26
|
+
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE
|
27
|
+
# UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF
|
28
|
+
# THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
29
|
+
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
|
30
|
+
# OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
31
|
+
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
32
|
+
# STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
33
|
+
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
34
|
+
# *******************************************************************************
|
35
|
+
|
36
|
+
module OsLib_QAQC
|
37
|
+
# Bin the hourly part load ratios into 10% bins
|
38
|
+
def bin_part_loads_by_ten_pcts(hrly_plrs)
|
39
|
+
bins = Array.new(10, 0)
|
40
|
+
op_hrs = 0.0
|
41
|
+
hrly_plrs.each do |plr|
|
42
|
+
op_hrs += 1.0 if plr > 0
|
43
|
+
if plr <= 0.1 # add below-zero % PLRs to final bin
|
44
|
+
bins[0] += 1
|
45
|
+
elsif plr > 0.1 && plr <= 0.2
|
46
|
+
bins[1] += 1
|
47
|
+
elsif plr > 0.2 && plr <= 0.3
|
48
|
+
bins[2] += 1
|
49
|
+
elsif plr > 0.3 && plr <= 0.4
|
50
|
+
bins[3] += 1
|
51
|
+
elsif plr > 0.4 && plr <= 0.5
|
52
|
+
bins[4] += 1
|
53
|
+
elsif plr > 0.5 && plr <= 0.6
|
54
|
+
bins[5] += 1
|
55
|
+
elsif plr > 0.6 && plr <= 0.7
|
56
|
+
bins[6] += 1
|
57
|
+
elsif plr > 0.7 && plr <= 0.8
|
58
|
+
bins[7] += 1
|
59
|
+
elsif plr > 0.8 && plr <= 0.9
|
60
|
+
bins[8] += 1
|
61
|
+
elsif plr > 0.9 # add over-100% PLRs to final bin
|
62
|
+
bins[9] += 1
|
63
|
+
end
|
64
|
+
end
|
65
|
+
|
66
|
+
# Convert bins from hour counts to % of operating hours.
|
67
|
+
bins.each_with_index do |bin, i|
|
68
|
+
bins[i] = bins[i] / op_hrs
|
69
|
+
end
|
70
|
+
|
71
|
+
return bins
|
72
|
+
end
|
73
|
+
|
74
|
+
# Check primary heating and cooling equipment part load ratios
|
75
|
+
# to find equipment that is significantly oversized or undersized.
|
76
|
+
def check_part_loads(category, target_standard, max_pct_delta = 0.1, name_only = false)
|
77
|
+
# summary of the check
|
78
|
+
check_elems = OpenStudio::AttributeVector.new
|
79
|
+
check_elems << OpenStudio::Attribute.new('name', 'Part Load')
|
80
|
+
check_elems << OpenStudio::Attribute.new('category', category)
|
81
|
+
check_elems << OpenStudio::Attribute.new('description', 'Check that equipment operates at reasonable part load ranges.')
|
82
|
+
|
83
|
+
# stop here if only name is requested this is used to populate display name for arguments
|
84
|
+
if name_only == true
|
85
|
+
results = []
|
86
|
+
check_elems.each do |elem|
|
87
|
+
results << elem.valueAsString
|
88
|
+
end
|
89
|
+
return results
|
90
|
+
end
|
91
|
+
|
92
|
+
std = Standard.build(target_standard)
|
93
|
+
|
94
|
+
begin
|
95
|
+
# Establish limits for % of operating hrs expected above 90% part load
|
96
|
+
expected_pct_hrs_above_90 = 0.1
|
97
|
+
|
98
|
+
# get the weather file run period (as opposed to design day run period)
|
99
|
+
ann_env_pd = nil
|
100
|
+
@sql.availableEnvPeriods.each do |env_pd|
|
101
|
+
env_type = @sql.environmentType(env_pd)
|
102
|
+
if env_type.is_initialized
|
103
|
+
if env_type.get == OpenStudio::EnvironmentType.new('WeatherRunPeriod')
|
104
|
+
ann_env_pd = env_pd
|
105
|
+
break
|
106
|
+
end
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
# only try to get the annual timeseries if an annual simulation was run
|
111
|
+
if ann_env_pd.nil?
|
112
|
+
check_elems << OpenStudio::Attribute.new('flag', 'Cannot find the annual simulation run period, cannot check equipment part load ratios.')
|
113
|
+
return check_elem
|
114
|
+
end
|
115
|
+
|
116
|
+
# Boilers
|
117
|
+
@model.getBoilerHotWaters.each do |equip|
|
118
|
+
# Get the timeseries part load ratio data
|
119
|
+
key_value = equip.name.get.to_s.upcase # must be in all caps.
|
120
|
+
time_step = 'Hourly'
|
121
|
+
variable_name = 'Boiler Part Load Ratio'
|
122
|
+
ts = @sql.timeSeries(ann_env_pd, time_step, variable_name, key_value)
|
123
|
+
if ts.empty?
|
124
|
+
check_elems << OpenStudio::Attribute.new('flag', "#{variable_name} Timeseries not found for #{key_value}.")
|
125
|
+
next
|
126
|
+
end
|
127
|
+
|
128
|
+
# Convert to array
|
129
|
+
ts = ts.get.values
|
130
|
+
plrs = []
|
131
|
+
for i in 0..(ts.size - 1)
|
132
|
+
plrs << ts[i]
|
133
|
+
end
|
134
|
+
|
135
|
+
# Bin part load ratios
|
136
|
+
pct_hrs_above_90 = bin_part_loads_by_ten_pcts(plrs)[9]
|
137
|
+
|
138
|
+
# Check top-end part load ratio bins
|
139
|
+
if ((pct_hrs_above_90 - expected_pct_hrs_above_90) / pct_hrs_above_90).abs > max_pct_delta
|
140
|
+
check_elems << OpenStudio::Attribute.new('flag', "For #{equip.name}, the actual hrs above 90% part load of #{(pct_hrs_above_90 * 100).round(2)}% is more than #{(max_pct_delta * 100.0).round(2)}% different from the expected #{(expected_pct_hrs_above_90 * 100).round(2)}% of hrs above 90% part load. This could indicate significantly oversized or undersized equipment.")
|
141
|
+
end
|
142
|
+
end
|
143
|
+
|
144
|
+
# Chillers
|
145
|
+
@model.getChillerElectricEIRs.each do |equip|
|
146
|
+
# Get the timeseries part load ratio data
|
147
|
+
key_value = equip.name.get.to_s.upcase # must be in all caps.
|
148
|
+
time_step = 'Hourly'
|
149
|
+
variable_name = 'Chiller Part Load Ratio'
|
150
|
+
ts = @sql.timeSeries(ann_env_pd, time_step, variable_name, key_value)
|
151
|
+
if ts.empty?
|
152
|
+
check_elems << OpenStudio::Attribute.new('flag', "#{variable_name} Timeseries not found for #{key_value}.")
|
153
|
+
next
|
154
|
+
end
|
155
|
+
|
156
|
+
# Convert to array
|
157
|
+
ts = ts.get.values
|
158
|
+
plrs = []
|
159
|
+
for i in 0..(ts.size - 1)
|
160
|
+
plrs << ts[i]
|
161
|
+
end
|
162
|
+
|
163
|
+
# Bin part load ratios
|
164
|
+
pct_hrs_above_90 = bin_part_loads_by_ten_pcts(plrs)[9]
|
165
|
+
|
166
|
+
# Check top-end part load ratio bins
|
167
|
+
if ((pct_hrs_above_90 - expected_pct_hrs_above_90) / pct_hrs_above_90).abs > max_pct_delta
|
168
|
+
check_elems << OpenStudio::Attribute.new('flag', "For #{equip.name}, the actual hrs above 90% part load of #{(pct_hrs_above_90 * 100).round(2)}% is more than #{(max_pct_delta * 100.0).round(2)}% different from the expected #{(expected_pct_hrs_above_90 * 100).round(2)}% of hrs above 90% part load. This could indicate significantly oversized or undersized equipment.")
|
169
|
+
end
|
170
|
+
end
|
171
|
+
|
172
|
+
# Cooling Towers (Single Speed)
|
173
|
+
@model.getCoolingTowerSingleSpeeds.each do |equip|
|
174
|
+
# Get the design fan power
|
175
|
+
if equip.fanPoweratDesignAirFlowRate.is_initialized
|
176
|
+
dsn_pwr = equip.fanPoweratDesignAirFlowRate.get
|
177
|
+
elsif equip.autosizedFanPoweratDesignAirFlowRate.is_initialized
|
178
|
+
dsn_pwr = equip.autosizedFanPoweratDesignAirFlowRate.get
|
179
|
+
else
|
180
|
+
check_elems << OpenStudio::Attribute.new('flag', "Could not determine peak power for #{equip.name}, cannot check part load ratios.")
|
181
|
+
next
|
182
|
+
end
|
183
|
+
|
184
|
+
# Get the timeseries fan power
|
185
|
+
key_value = equip.name.get.to_s.upcase # must be in all caps.
|
186
|
+
time_step = 'Hourly'
|
187
|
+
variable_name = 'Cooling Tower Fan Electric Power'
|
188
|
+
ts = @sql.timeSeries(ann_env_pd, time_step, variable_name, key_value)
|
189
|
+
if ts.empty?
|
190
|
+
check_elems << OpenStudio::Attribute.new('flag', "#{variable_name} Timeseries not found for #{key_value}.")
|
191
|
+
next
|
192
|
+
end
|
193
|
+
|
194
|
+
# Convert to array
|
195
|
+
ts = ts.get.values
|
196
|
+
plrs = []
|
197
|
+
for i in 0..(ts.size - 1)
|
198
|
+
plrs << ts[i] / dsn_pwr.to_f
|
199
|
+
end
|
200
|
+
|
201
|
+
# Bin part load ratios
|
202
|
+
pct_hrs_above_90 = bin_part_loads_by_ten_pcts(plrs)[9]
|
203
|
+
|
204
|
+
# Check top-end part load ratio bins
|
205
|
+
if ((pct_hrs_above_90 - expected_pct_hrs_above_90) / pct_hrs_above_90).abs > max_pct_delta
|
206
|
+
check_elems << OpenStudio::Attribute.new('flag', "For #{equip.name}, the actual hrs above 90% part load of #{(pct_hrs_above_90 * 100).round(2)}% is more than #{(max_pct_delta * 100.0).round(2)}% different from the expected #{(expected_pct_hrs_above_90 * 100).round(2)}% of hrs above 90% part load. This could indicate significantly oversized or undersized equipment.")
|
207
|
+
end
|
208
|
+
end
|
209
|
+
|
210
|
+
# Cooling Towers (Two Speed)
|
211
|
+
@model.getCoolingTowerTwoSpeeds.each do |equip|
|
212
|
+
# Get the design fan power
|
213
|
+
if equip.highFanSpeedFanPower.is_initialized
|
214
|
+
dsn_pwr = equip.highFanSpeedFanPower.get
|
215
|
+
elsif equip.autosizedHighFanSpeedFanPower.is_initialized
|
216
|
+
dsn_pwr = equip.autosizedHighFanSpeedFanPower.get
|
217
|
+
else
|
218
|
+
check_elems << OpenStudio::Attribute.new('flag', "Could not determine peak power for #{equip.name}, cannot check part load ratios.")
|
219
|
+
next
|
220
|
+
end
|
221
|
+
|
222
|
+
# Get the timeseries fan power
|
223
|
+
key_value = equip.name.get.to_s.upcase # must be in all caps.
|
224
|
+
time_step = 'Hourly'
|
225
|
+
variable_name = 'Cooling Tower Fan Electric Power'
|
226
|
+
ts = @sql.timeSeries(ann_env_pd, time_step, variable_name, key_value)
|
227
|
+
if ts.empty?
|
228
|
+
check_elems << OpenStudio::Attribute.new('flag', "#{variable_name} Timeseries not found for #{key_value}.")
|
229
|
+
next
|
230
|
+
end
|
231
|
+
|
232
|
+
# Convert to array
|
233
|
+
ts = ts.get.values
|
234
|
+
plrs = []
|
235
|
+
for i in 0..(ts.size - 1)
|
236
|
+
plrs << ts[i] / dsn_pwr.to_f
|
237
|
+
end
|
238
|
+
|
239
|
+
# Bin part load ratios
|
240
|
+
pct_hrs_above_90 = bin_part_loads_by_ten_pcts(plrs)[9]
|
241
|
+
|
242
|
+
# Check top-end part load ratio bins
|
243
|
+
if ((pct_hrs_above_90 - expected_pct_hrs_above_90) / pct_hrs_above_90).abs > max_pct_delta
|
244
|
+
check_elems << OpenStudio::Attribute.new('flag', "For #{equip.name}, the actual hrs above 90% part load of #{(pct_hrs_above_90 * 100).round(2)}% is more than #{(max_pct_delta * 100.0).round(2)}% different from the expected #{(expected_pct_hrs_above_90 * 100).round(2)}% of hrs above 90% part load. This could indicate significantly oversized or undersized equipment.")
|
245
|
+
end
|
246
|
+
end
|
247
|
+
|
248
|
+
# Cooling Towers (Variable Speed)
|
249
|
+
@model.getCoolingTowerVariableSpeeds.each do |equip|
|
250
|
+
# Get the design fan power
|
251
|
+
if equip.designFanPower.is_initialized
|
252
|
+
dsn_pwr = equip.designFanPower.get
|
253
|
+
elsif equip.autosizedDesignFanPower.is_initialized
|
254
|
+
dsn_pwr = equip.autosizedDesignFanPower.get
|
255
|
+
else
|
256
|
+
check_elems << OpenStudio::Attribute.new('flag', "Could not determine peak power for #{equip.name}, cannot check part load ratios.")
|
257
|
+
next
|
258
|
+
end
|
259
|
+
|
260
|
+
# Get the timeseries fan power
|
261
|
+
key_value = equip.name.get.to_s.upcase # must be in all caps.
|
262
|
+
time_step = 'Hourly'
|
263
|
+
variable_name = 'Cooling Tower Fan Electric Power'
|
264
|
+
ts = @sql.timeSeries(ann_env_pd, time_step, variable_name, key_value)
|
265
|
+
if ts.empty?
|
266
|
+
check_elems << OpenStudio::Attribute.new('flag', "#{variable_name} Timeseries not found for #{key_value}.")
|
267
|
+
next
|
268
|
+
end
|
269
|
+
|
270
|
+
# Convert to array
|
271
|
+
ts = ts.get.values
|
272
|
+
plrs = []
|
273
|
+
for i in 0..(ts.size - 1)
|
274
|
+
plrs << ts[i] / dsn_pwr.to_f
|
275
|
+
end
|
276
|
+
|
277
|
+
# Bin part load ratios
|
278
|
+
pct_hrs_above_90 = bin_part_loads_by_ten_pcts(plrs)[9]
|
279
|
+
|
280
|
+
# Check top-end part load ratio bins
|
281
|
+
if ((pct_hrs_above_90 - expected_pct_hrs_above_90) / pct_hrs_above_90).abs > max_pct_delta
|
282
|
+
check_elems << OpenStudio::Attribute.new('flag', "For #{equip.name}, the actual hrs above 90% part load of #{(pct_hrs_above_90 * 100).round(2)}% is more than #{(max_pct_delta * 100.0).round(2)}% different from the expected #{(expected_pct_hrs_above_90 * 100).round(2)}% of hrs above 90% part load. This could indicate significantly oversized or undersized equipment.")
|
283
|
+
end
|
284
|
+
end
|
285
|
+
|
286
|
+
# DX Cooling Coils (Single Speed)
|
287
|
+
@model.getCoilCoolingDXSingleSpeeds.each do |equip|
|
288
|
+
# Get the design coil capacity
|
289
|
+
if equip.ratedTotalCoolingCapacity.is_initialized
|
290
|
+
dsn_pwr = equip.ratedTotalCoolingCapacity.get
|
291
|
+
elsif equip.autosizedRatedTotalCoolingCapacity.is_initialized
|
292
|
+
dsn_pwr = equip.autosizedRatedTotalCoolingCapacity.get
|
293
|
+
else
|
294
|
+
check_elems << OpenStudio::Attribute.new('flag', "Could not determine capacity for #{equip.name}, cannot check part load ratios.")
|
295
|
+
next
|
296
|
+
end
|
297
|
+
|
298
|
+
# Get the timeseries coil capacity
|
299
|
+
key_value = equip.name.get.to_s.upcase # must be in all caps.
|
300
|
+
time_step = 'Hourly'
|
301
|
+
variable_name = 'Cooling Coil Total Cooling Rate'
|
302
|
+
ts = @sql.timeSeries(ann_env_pd, time_step, variable_name, key_value)
|
303
|
+
if ts.empty?
|
304
|
+
check_elems << OpenStudio::Attribute.new('flag', "#{variable_name} Timeseries not found for #{key_value}.")
|
305
|
+
next
|
306
|
+
end
|
307
|
+
|
308
|
+
# Convert to array
|
309
|
+
ts = ts.get.values
|
310
|
+
plrs = []
|
311
|
+
for i in 0..(ts.size - 1)
|
312
|
+
plrs << ts[i] / dsn_pwr.to_f
|
313
|
+
end
|
314
|
+
|
315
|
+
# Bin part load ratios
|
316
|
+
pct_hrs_above_90 = bin_part_loads_by_ten_pcts(plrs)[9]
|
317
|
+
|
318
|
+
# Check top-end part load ratio bins
|
319
|
+
if ((pct_hrs_above_90 - expected_pct_hrs_above_90) / pct_hrs_above_90).abs > max_pct_delta
|
320
|
+
check_elems << OpenStudio::Attribute.new('flag', "For #{equip.name}, the actual hrs above 90% part load of #{(pct_hrs_above_90 * 100).round(2)}% is more than #{(max_pct_delta * 100.0).round(2)}% different from the expected #{(expected_pct_hrs_above_90 * 100).round(2)}% of hrs above 90% part load. This could indicate significantly oversized or undersized equipment.")
|
321
|
+
end
|
322
|
+
end
|
323
|
+
|
324
|
+
# DX Cooling Coils (Two Speed)
|
325
|
+
@model.getCoilCoolingDXTwoSpeeds.each do |equip|
|
326
|
+
# Get the design coil capacity
|
327
|
+
if equip.ratedHighSpeedTotalCoolingCapacity.is_initialized
|
328
|
+
dsn_pwr = equip.ratedHighSpeedTotalCoolingCapacity.get
|
329
|
+
elsif equip.autosizedRatedHighSpeedTotalCoolingCapacity.is_initialized
|
330
|
+
dsn_pwr = equip.autosizedRatedHighSpeedTotalCoolingCapacity.get
|
331
|
+
else
|
332
|
+
check_elems << OpenStudio::Attribute.new('flag', "Could not determine capacity for #{equip.name}, cannot check part load ratios.")
|
333
|
+
next
|
334
|
+
end
|
335
|
+
|
336
|
+
# Get the timeseries coil capacity
|
337
|
+
key_value = equip.name.get.to_s.upcase # must be in all caps.
|
338
|
+
time_step = 'Hourly'
|
339
|
+
variable_name = 'Cooling Coil Total Cooling Rate'
|
340
|
+
ts = @sql.timeSeries(ann_env_pd, time_step, variable_name, key_value)
|
341
|
+
if ts.empty?
|
342
|
+
check_elems << OpenStudio::Attribute.new('flag', "#{variable_name} Timeseries not found for #{key_value}.")
|
343
|
+
next
|
344
|
+
end
|
345
|
+
|
346
|
+
# Convert to array
|
347
|
+
ts = ts.get.values
|
348
|
+
plrs = []
|
349
|
+
for i in 0..(ts.size - 1)
|
350
|
+
plrs << ts[i] / dsn_pwr.to_f
|
351
|
+
end
|
352
|
+
|
353
|
+
# Bin part load ratios
|
354
|
+
pct_hrs_above_90 = bin_part_loads_by_ten_pcts(plrs)[9]
|
355
|
+
|
356
|
+
# Check top-end part load ratio bins
|
357
|
+
if ((pct_hrs_above_90 - expected_pct_hrs_above_90) / pct_hrs_above_90).abs > max_pct_delta
|
358
|
+
check_elems << OpenStudio::Attribute.new('flag', "For #{equip.name}, the actual hrs above 90% part load of #{(pct_hrs_above_90 * 100).round(2)}% is more than #{(max_pct_delta * 100.0).round(2)}% different from the expected #{(expected_pct_hrs_above_90 * 100).round(2)}% of hrs above 90% part load. This could indicate significantly oversized or undersized equipment.")
|
359
|
+
end
|
360
|
+
end
|
361
|
+
|
362
|
+
# DX Cooling Coils (Variable Speed)
|
363
|
+
@model.getCoilCoolingDXVariableSpeeds.each do |equip|
|
364
|
+
# Get the design coil capacity
|
365
|
+
if equip.grossRatedTotalCoolingCapacityAtSelectedNominalSpeedLevel.is_initialized
|
366
|
+
dsn_pwr = equip.grossRatedTotalCoolingCapacityAtSelectedNominalSpeedLevel.get
|
367
|
+
elsif equip.autosizedGrossRatedTotalCoolingCapacityAtSelectedNominalSpeedLevel.is_initialized
|
368
|
+
dsn_pwr = equip.autosizedGrossRatedTotalCoolingCapacityAtSelectedNominalSpeedLevel.get
|
369
|
+
else
|
370
|
+
check_elems << OpenStudio::Attribute.new('flag', "Could not determine capacity for #{equip.name}, cannot check part load ratios.")
|
371
|
+
next
|
372
|
+
end
|
373
|
+
|
374
|
+
# Get the timeseries coil capacity
|
375
|
+
key_value = equip.name.get.to_s.upcase # must be in all caps.
|
376
|
+
time_step = 'Hourly'
|
377
|
+
variable_name = 'Cooling Coil Total Cooling Rate'
|
378
|
+
ts = @sql.timeSeries(ann_env_pd, time_step, variable_name, key_value)
|
379
|
+
if ts.empty?
|
380
|
+
check_elems << OpenStudio::Attribute.new('flag', "#{variable_name} Timeseries not found for #{key_value}.")
|
381
|
+
next
|
382
|
+
end
|
383
|
+
|
384
|
+
# Convert to array
|
385
|
+
ts = ts.get.values
|
386
|
+
plrs = []
|
387
|
+
for i in 0..(ts.size - 1)
|
388
|
+
plrs << ts[i] / dsn_pwr.to_f
|
389
|
+
end
|
390
|
+
|
391
|
+
# Bin part load ratios
|
392
|
+
pct_hrs_above_90 = bin_part_loads_by_ten_pcts(plrs)[9]
|
393
|
+
|
394
|
+
# Check top-end part load ratio bins
|
395
|
+
if ((pct_hrs_above_90 - expected_pct_hrs_above_90) / pct_hrs_above_90).abs > max_pct_delta
|
396
|
+
check_elems << OpenStudio::Attribute.new('flag', "For #{equip.name}, the actual hrs above 90% part load of #{(pct_hrs_above_90 * 100).round(2)}% is more than #{(max_pct_delta * 100.0).round(2)}% different from the expected #{(expected_pct_hrs_above_90 * 100).round(2)}% of hrs above 90% part load. This could indicate significantly oversized or undersized equipment.")
|
397
|
+
end
|
398
|
+
end
|
399
|
+
|
400
|
+
# Gas Heating Coils
|
401
|
+
@model.getCoilHeatingGass.each do |equip|
|
402
|
+
# Get the design coil capacity
|
403
|
+
if equip.nominalCapacity.is_initialized
|
404
|
+
dsn_pwr = equip.nominalCapacity.get
|
405
|
+
elsif equip.autosizedNominalCapacity.is_initialized
|
406
|
+
dsn_pwr = equip.autosizedNominalCapacity.get
|
407
|
+
else
|
408
|
+
check_elems << OpenStudio::Attribute.new('flag', "Could not determine capacity for #{equip.name}, cannot check part load ratios.")
|
409
|
+
next
|
410
|
+
end
|
411
|
+
|
412
|
+
# Get the timeseries coil capacity
|
413
|
+
key_value = equip.name.get.to_s.upcase # must be in all caps.
|
414
|
+
time_step = 'Hourly'
|
415
|
+
variable_name = 'Heating Coil Air Heating Rate'
|
416
|
+
ts = @sql.timeSeries(ann_env_pd, time_step, variable_name, key_value)
|
417
|
+
if ts.empty?
|
418
|
+
check_elems << OpenStudio::Attribute.new('flag', "#{variable_name} Timeseries not found for #{key_value}.")
|
419
|
+
next
|
420
|
+
end
|
421
|
+
|
422
|
+
# Convert to array
|
423
|
+
ts = ts.get.values
|
424
|
+
plrs = []
|
425
|
+
for i in 0..(ts.size - 1)
|
426
|
+
plrs << ts[i] / dsn_pwr.to_f
|
427
|
+
end
|
428
|
+
|
429
|
+
# Bin part load ratios
|
430
|
+
pct_hrs_above_90 = bin_part_loads_by_ten_pcts(plrs)[9]
|
431
|
+
|
432
|
+
# Check top-end part load ratio bins
|
433
|
+
if ((pct_hrs_above_90 - expected_pct_hrs_above_90) / pct_hrs_above_90).abs > max_pct_delta
|
434
|
+
check_elems << OpenStudio::Attribute.new('flag', "For #{equip.name}, the actual hrs above 90% part load of #{(pct_hrs_above_90 * 100).round(2)}% is more than #{(max_pct_delta * 100.0).round(2)}% different from the expected #{(expected_pct_hrs_above_90 * 100).round(2)}% of hrs above 90% part load. This could indicate significantly oversized or undersized equipment.")
|
435
|
+
end
|
436
|
+
end
|
437
|
+
rescue StandardError => e
|
438
|
+
# brief description of ruby error
|
439
|
+
check_elems << OpenStudio::Attribute.new('flag', "Error prevented QAQC check from running (#{e}).")
|
440
|
+
|
441
|
+
# backtrace of ruby error for diagnostic use
|
442
|
+
if @error_backtrace then check_elems << OpenStudio::Attribute.new('flag', e.backtrace.join("\n").to_s) end
|
443
|
+
end
|
444
|
+
|
445
|
+
# add check_elms to new attribute
|
446
|
+
check_elem = OpenStudio::Attribute.new('check', check_elems)
|
447
|
+
|
448
|
+
return check_elem
|
449
|
+
# note: registerWarning and registerValue will be added for checks downstream using os_lib_reporting_qaqc.rb
|
450
|
+
end
|
451
|
+
end
|