honeybee-openstudio 2.15.0 → 2.16.4
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/honeybee-openstudio.gemspec +1 -1
- data/lib/from_openstudio/geometry/aperture.rb +1 -1
- data/lib/from_openstudio/geometry/door.rb +1 -1
- data/lib/from_openstudio/geometry/face.rb +1 -1
- data/lib/from_openstudio/geometry/room.rb +1 -1
- data/lib/from_openstudio/geometry/shade.rb +2 -2
- data/lib/honeybee/_defaults/model.json +189 -294
- data/lib/honeybee/hvac/template.rb +1 -1
- data/lib/honeybee/model_object.rb +1 -1
- data/lib/measures/from_gbxml_model/LICENSE.md +23 -0
- data/lib/measures/from_gbxml_model/README.md +32 -0
- data/lib/measures/from_gbxml_model/measure.rb +114 -0
- data/lib/measures/from_gbxml_model/measure.xml +131 -0
- data/lib/measures/from_gbxml_model/tests/from_gbxml_model_test.rb +107 -0
- data/lib/measures/from_honeybee_model/README.md +1 -1
- data/lib/measures/from_honeybee_model_to_gbxml/LICENSE.md +23 -0
- data/lib/measures/from_honeybee_model_to_gbxml/README.md +32 -0
- data/lib/measures/from_honeybee_model_to_gbxml/measure.rb +113 -0
- data/lib/measures/from_honeybee_model_to_gbxml/measure.xml +88 -0
- data/lib/measures/from_honeybee_model_to_gbxml/tests/from_honeybee_model_to_gbxml_test.rb +105 -0
- data/lib/measures/from_idf_model/LICENSE.md +23 -0
- data/lib/measures/from_idf_model/README.md +32 -0
- data/lib/measures/from_idf_model/measure.rb +114 -0
- data/lib/measures/from_idf_model/measure.xml +110 -0
- data/lib/measures/from_idf_model/tests/from_idf_model_test.rb +107 -0
- data/lib/measures/from_openstudio_model/LICENSE.md +23 -0
- data/lib/measures/from_openstudio_model/README.md +32 -0
- data/lib/measures/from_openstudio_model/measure.rb +114 -0
- data/lib/measures/from_openstudio_model/measure.xml +95 -0
- data/lib/measures/from_openstudio_model/tests/from_openstudio_model_test.rb +107 -0
- data/lib/to_openstudio/hvac/template.rb +103 -11
- metadata +22 -2
@@ -0,0 +1,107 @@
|
|
1
|
+
# *******************************************************************************
|
2
|
+
# Honeybee OpenStudio Gem, Copyright (c) 2020, Alliance for Sustainable
|
3
|
+
# Energy, LLC, Ladybug Tools LLC and other contributors. All rights reserved.
|
4
|
+
#
|
5
|
+
# Redistribution and use in source and binary forms, with or without
|
6
|
+
# modification, are permitted provided that the following conditions are met:
|
7
|
+
#
|
8
|
+
# (1) Redistributions of source code must retain the above copyright notice,
|
9
|
+
# this list of conditions and the following disclaimer.
|
10
|
+
#
|
11
|
+
# (2) Redistributions in binary form must reproduce the above copyright notice,
|
12
|
+
# this list of conditions and the following disclaimer in the documentation
|
13
|
+
# and/or other materials provided with the distribution.
|
14
|
+
#
|
15
|
+
# (3) Neither the name of the copyright holder nor the names of any contributors
|
16
|
+
# may be used to endorse or promote products derived from this software without
|
17
|
+
# specific prior written permission from the respective party.
|
18
|
+
#
|
19
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS
|
20
|
+
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
21
|
+
# THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
22
|
+
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE
|
23
|
+
# UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF
|
24
|
+
# THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
25
|
+
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
|
26
|
+
# OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
27
|
+
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
28
|
+
# STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
29
|
+
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
30
|
+
# *******************************************************************************
|
31
|
+
|
32
|
+
require 'openstudio'
|
33
|
+
require 'openstudio/ruleset/ShowRunnerOutput'
|
34
|
+
require 'minitest/autorun'
|
35
|
+
require_relative '../measure.rb'
|
36
|
+
require 'fileutils'
|
37
|
+
|
38
|
+
class FromOpenstudioModel_Test < Minitest::Test
|
39
|
+
# method to apply arguments, run measure, and assert results (only populate args hash with non-default argument values)
|
40
|
+
def apply_measure_to_model(test_name, args, model_name = nil, result_value = 'Success', warnings_count = 0, info_count = nil)
|
41
|
+
# create an instance of the measure
|
42
|
+
measure = FromOpenstudioModel.new
|
43
|
+
|
44
|
+
# create an instance of a runner
|
45
|
+
runner = OpenStudio::Measure::OSRunner.new(OpenStudio::WorkflowJSON.new)
|
46
|
+
|
47
|
+
if model_name.nil?
|
48
|
+
# make an empty model
|
49
|
+
model = OpenStudio::Model::Model.new
|
50
|
+
else
|
51
|
+
# load the test model
|
52
|
+
translator = OpenStudio::OSVersion::VersionTranslator.new
|
53
|
+
path = OpenStudio::Path.new(File.dirname(__FILE__) + '/' + model_name)
|
54
|
+
model = translator.loadModel(path)
|
55
|
+
assert(!model.empty?)
|
56
|
+
model = model.get
|
57
|
+
end
|
58
|
+
|
59
|
+
# get arguments
|
60
|
+
arguments = measure.arguments(model)
|
61
|
+
argument_map = OpenStudio::Measure.convertOSArgumentVectorToMap(arguments)
|
62
|
+
|
63
|
+
# populate argument with specified hash value if specified
|
64
|
+
arguments.each do |arg|
|
65
|
+
temp_arg_var = arg.clone
|
66
|
+
if args.key?(arg.name)
|
67
|
+
assert(temp_arg_var.setValue(args[arg.name]), "could not set #{arg.name} to #{args[arg.name]}.")
|
68
|
+
end
|
69
|
+
argument_map[arg.name] = temp_arg_var
|
70
|
+
end
|
71
|
+
|
72
|
+
# run the measure
|
73
|
+
measure.run(model, runner, argument_map)
|
74
|
+
result = runner.result
|
75
|
+
|
76
|
+
puts "this is measure #{measure.run(model, runner, argument_map)}"
|
77
|
+
# show the output
|
78
|
+
puts "measure results for #{test_name}"
|
79
|
+
show_output(result)
|
80
|
+
|
81
|
+
# assert that it ran correctly
|
82
|
+
if result_value.nil? then result_value = 'Success' end
|
83
|
+
assert_equal(result_value, result.value.valueName)
|
84
|
+
|
85
|
+
# check count of warning and info messages
|
86
|
+
unless info_count.nil? then assert(result.info.size == info_count) end
|
87
|
+
unless warnings_count.nil? then assert(result.warnings.size == warnings_count, "warning count (#{result.warnings.size}) did not match expectation (#{warnings_count})") end
|
88
|
+
|
89
|
+
# if 'Fail' passed in make sure at least one error message (while not typical there may be more than one message)
|
90
|
+
if result_value == 'Fail' then assert(result.errors.size >= 1) end
|
91
|
+
|
92
|
+
end
|
93
|
+
|
94
|
+
def test_example_model_with_output_path
|
95
|
+
args = {}
|
96
|
+
args['openstudio_model'] = File.join(File.dirname(__FILE__) + '/../../../../spec/samples/osm/exampleModel.osm')
|
97
|
+
args['output_file_path'] = File.join(File.dirname(__FILE__) + '/output/exampleModel.hbjson')
|
98
|
+
apply_measure_to_model(__method__.to_s.gsub('test_',''),args, nil)
|
99
|
+
end
|
100
|
+
|
101
|
+
def test_example_model_without_output_path
|
102
|
+
args = {}
|
103
|
+
args['openstudio_model'] = File.join(File.dirname(__FILE__) + '/../../../../spec/samples/osm/exampleModel.osm')
|
104
|
+
apply_measure_to_model(__method__.to_s.gsub('test_',''),args, nil)
|
105
|
+
end
|
106
|
+
|
107
|
+
end
|
@@ -83,27 +83,36 @@ module Honeybee
|
|
83
83
|
air_loops = openstudio_model.getAirLoopHVACs
|
84
84
|
unless air_loops.length == $air_loop_count # check if any new loops were added
|
85
85
|
$air_loop_count = air_loops.length
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
86
|
+
os_air_terminal = zones[0].airLoopHVACTerminal
|
87
|
+
unless os_air_terminal.empty?
|
88
|
+
os_air_terminal = os_air_terminal.get
|
89
|
+
os_air_loop_opt = os_air_terminal.airLoopHVAC
|
90
|
+
unless os_air_loop_opt.empty?
|
91
|
+
os_air_loop = os_air_loop_opt.get
|
92
|
+
loop_name = os_air_loop.name
|
93
|
+
unless loop_name.empty?
|
94
|
+
# set the name of the air loop to align with the HVAC name
|
95
|
+
if @hash[:display_name]
|
96
|
+
clean_name = @hash[:display_name].to_s.gsub(/[^.A-Za-z0-9_-] /, " ")
|
97
|
+
os_air_loop.setName(clean_name + ' - ' + loop_name.get)
|
98
|
+
end
|
99
|
+
end
|
91
100
|
end
|
92
101
|
end
|
93
102
|
end
|
94
103
|
|
95
104
|
# assign the economizer type if there's an air loop and the economizer is specified
|
96
|
-
if @hash[:economizer_type] &&
|
105
|
+
if @hash[:economizer_type] && os_air_loop
|
97
106
|
oasys = os_air_loop.airLoopHVACOutdoorAirSystem
|
98
107
|
unless oasys.empty?
|
99
|
-
|
100
|
-
|
101
|
-
|
108
|
+
os_oasys = oasys.get
|
109
|
+
oactrl = os_oasys.getControllerOutdoorAir
|
110
|
+
oactrl.setEconomizerControlType(@hash[:economizer_type])
|
102
111
|
end
|
103
112
|
end
|
104
113
|
|
105
114
|
# set the sensible heat recovery if there's an air loop and the heat recovery is specified
|
106
|
-
if @hash[:sensible_heat_recovery] && @hash[:sensible_heat_recovery] !=
|
115
|
+
if @hash[:sensible_heat_recovery] && @hash[:sensible_heat_recovery] != 0 && os_air_loop
|
107
116
|
erv = get_existing_erv(os_air_loop)
|
108
117
|
unless erv
|
109
118
|
erv = create_erv(openstudio_model, os_air_loop)
|
@@ -116,7 +125,7 @@ module Honeybee
|
|
116
125
|
end
|
117
126
|
|
118
127
|
# set the latent heat recovery if there's an air loop and the heat recovery is specified
|
119
|
-
if @hash[:latent_heat_recovery] && @hash[:latent_heat_recovery] !=
|
128
|
+
if @hash[:latent_heat_recovery] && @hash[:latent_heat_recovery] != 0 && os_air_loop
|
120
129
|
erv = get_existing_erv(os_air_loop)
|
121
130
|
unless erv
|
122
131
|
erv = create_erv(openstudio_model, os_air_loop)
|
@@ -128,6 +137,60 @@ module Honeybee
|
|
128
137
|
erv.setLatentEffectivenessat75HeatingAirFlow(@hash[:latent_heat_recovery])
|
129
138
|
end
|
130
139
|
|
140
|
+
# assign demand controlled ventilation if there's an air loop
|
141
|
+
if @hash[:demand_controlled_ventilation] && os_air_loop
|
142
|
+
oasys = os_air_loop.airLoopHVACOutdoorAirSystem
|
143
|
+
unless oasys.empty?
|
144
|
+
os_oasys = oasys.get
|
145
|
+
oactrl = os_oasys.getControllerOutdoorAir
|
146
|
+
vent_ctrl = oactrl.controllerMechanicalVentilation
|
147
|
+
vent_ctrl.setDemandControlledVentilationNoFail(true)
|
148
|
+
oactrl.resetMinimumFractionofOutdoorAirSchedule
|
149
|
+
end
|
150
|
+
end
|
151
|
+
|
152
|
+
# assign the DOAS availability schedule if there's an air loop and it is specified
|
153
|
+
if @hash[:doas_availability_schedule] && os_air_loop
|
154
|
+
schedule = openstudio_model.getScheduleByName(@hash[:doas_availability_schedule])
|
155
|
+
unless schedule.empty?
|
156
|
+
avail_sch = schedule.get
|
157
|
+
os_air_loop.setAvailabilitySchedule(avail_sch)
|
158
|
+
end
|
159
|
+
end
|
160
|
+
|
161
|
+
# set the outdoor air controller to respect room-level ventilation schedules if they exist
|
162
|
+
if os_air_loop
|
163
|
+
oasys = os_air_loop.airLoopHVACOutdoorAirSystem
|
164
|
+
unless oasys.empty?
|
165
|
+
os_oasys = oasys.get
|
166
|
+
oactrl = os_oasys.getControllerOutdoorAir
|
167
|
+
oa_sch, oa_sch_name = nil, nil
|
168
|
+
zones.each do |zone|
|
169
|
+
oa_spec = zone.spaces[0].designSpecificationOutdoorAir
|
170
|
+
unless oa_spec.empty?
|
171
|
+
oa_spec = oa_spec.get
|
172
|
+
space_oa_sch = oa_spec.outdoorAirFlowRateFractionSchedule
|
173
|
+
unless space_oa_sch.empty?
|
174
|
+
space_oa_sch = space_oa_sch.get
|
175
|
+
space_oa_sch_name = space_oa_sch.name
|
176
|
+
unless space_oa_sch_name.empty?
|
177
|
+
space_oa_sch_name = space_oa_sch_name.get
|
178
|
+
if oa_sch_name.nil? || space_oa_sch_name == oa_sch_name
|
179
|
+
oa_sch, oa_sch_name = space_oa_sch, space_oa_sch_name
|
180
|
+
else
|
181
|
+
oa_sch = nil
|
182
|
+
end
|
183
|
+
end
|
184
|
+
end
|
185
|
+
end
|
186
|
+
end
|
187
|
+
if oa_sch
|
188
|
+
oactrl.resetMinimumFractionofOutdoorAirSchedule
|
189
|
+
oactrl.setMinimumOutdoorAirSchedule(oa_sch)
|
190
|
+
end
|
191
|
+
end
|
192
|
+
end
|
193
|
+
|
131
194
|
# if the systems are PTAC and there is ventilation, ensure the system includes it
|
132
195
|
if equipment_type.include?('PTAC') || equipment_type.include?('PTHP')
|
133
196
|
always_on = openstudio_model.getScheduleByName('Always On').get
|
@@ -164,6 +227,20 @@ module Honeybee
|
|
164
227
|
end
|
165
228
|
end
|
166
229
|
|
230
|
+
# assign an electric humidifier if there's an air loop and the zones have a humidistat
|
231
|
+
if os_air_loop
|
232
|
+
humidistat_exists = false
|
233
|
+
zones.each do |zone|
|
234
|
+
h_stat = zone.zoneControlHumidistat
|
235
|
+
unless h_stat.empty?
|
236
|
+
humidistat_exists = true
|
237
|
+
end
|
238
|
+
end
|
239
|
+
if humidistat_exists
|
240
|
+
humidifier = create_humidifier(openstudio_model, os_air_loop)
|
241
|
+
end
|
242
|
+
end
|
243
|
+
|
167
244
|
# set all plants to non-coincident sizing to avoid simualtion control issues on design days
|
168
245
|
openstudio_model.getPlantLoops.each do |loop|
|
169
246
|
sizing = loop.sizingPlant
|
@@ -209,5 +286,20 @@ module Honeybee
|
|
209
286
|
heat_ex
|
210
287
|
end
|
211
288
|
|
289
|
+
def create_humidifier(model, os_air_loop)
|
290
|
+
# create an electric humidifier
|
291
|
+
humidifier = OpenStudio::Model::HumidifierSteamElectric.new(model)
|
292
|
+
humidifier.setName(@hash[:identifier] + '_Humidifier Unit')
|
293
|
+
humid_controller = OpenStudio::Model::SetpointManagerMultiZoneHumidityMinimum.new(model)
|
294
|
+
humid_controller.setName(@hash[:identifier] + '_Humidifier Controller')
|
295
|
+
|
296
|
+
# add the humidifier to the air loop
|
297
|
+
supply_node = os_air_loop.supplyOutletNode
|
298
|
+
humidifier.addToNode(supply_node)
|
299
|
+
humid_controller.addToNode(supply_node)
|
300
|
+
|
301
|
+
humidifier
|
302
|
+
end
|
303
|
+
|
212
304
|
end #TemplateHVAC
|
213
305
|
end #Honeybee
|
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: honeybee-openstudio
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 2.
|
4
|
+
version: 2.16.4
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Tanushree Charan
|
@@ -11,7 +11,7 @@ authors:
|
|
11
11
|
autorequire:
|
12
12
|
bindir: exe
|
13
13
|
cert_chain: []
|
14
|
-
date: 2021-
|
14
|
+
date: 2021-05-05 00:00:00.000000000 Z
|
15
15
|
dependencies:
|
16
16
|
- !ruby/object:Gem::Dependency
|
17
17
|
name: json_pure
|
@@ -152,16 +152,36 @@ files:
|
|
152
152
|
- lib/honeybee/ventcool/opening.rb
|
153
153
|
- lib/honeybee/ventcool/simulation.rb
|
154
154
|
- lib/measures/.gitkeep
|
155
|
+
- lib/measures/from_gbxml_model/LICENSE.md
|
156
|
+
- lib/measures/from_gbxml_model/README.md
|
157
|
+
- lib/measures/from_gbxml_model/measure.rb
|
158
|
+
- lib/measures/from_gbxml_model/measure.xml
|
159
|
+
- lib/measures/from_gbxml_model/tests/from_gbxml_model_test.rb
|
155
160
|
- lib/measures/from_honeybee_model/LICENSE.md
|
156
161
|
- lib/measures/from_honeybee_model/README.md
|
157
162
|
- lib/measures/from_honeybee_model/measure.rb
|
158
163
|
- lib/measures/from_honeybee_model/measure.xml
|
159
164
|
- lib/measures/from_honeybee_model/tests/from_honeybee_model_test.rb
|
165
|
+
- lib/measures/from_honeybee_model_to_gbxml/LICENSE.md
|
166
|
+
- lib/measures/from_honeybee_model_to_gbxml/README.md
|
167
|
+
- lib/measures/from_honeybee_model_to_gbxml/measure.rb
|
168
|
+
- lib/measures/from_honeybee_model_to_gbxml/measure.xml
|
169
|
+
- lib/measures/from_honeybee_model_to_gbxml/tests/from_honeybee_model_to_gbxml_test.rb
|
160
170
|
- lib/measures/from_honeybee_simulation_parameter/LICENSE.md
|
161
171
|
- lib/measures/from_honeybee_simulation_parameter/README.md
|
162
172
|
- lib/measures/from_honeybee_simulation_parameter/measure.rb
|
163
173
|
- lib/measures/from_honeybee_simulation_parameter/measure.xml
|
164
174
|
- lib/measures/from_honeybee_simulation_parameter/tests/from_honeybee_simulation_parameter_test.rb
|
175
|
+
- lib/measures/from_idf_model/LICENSE.md
|
176
|
+
- lib/measures/from_idf_model/README.md
|
177
|
+
- lib/measures/from_idf_model/measure.rb
|
178
|
+
- lib/measures/from_idf_model/measure.xml
|
179
|
+
- lib/measures/from_idf_model/tests/from_idf_model_test.rb
|
180
|
+
- lib/measures/from_openstudio_model/LICENSE.md
|
181
|
+
- lib/measures/from_openstudio_model/README.md
|
182
|
+
- lib/measures/from_openstudio_model/measure.rb
|
183
|
+
- lib/measures/from_openstudio_model/measure.xml
|
184
|
+
- lib/measures/from_openstudio_model/tests/from_openstudio_model_test.rb
|
165
185
|
- lib/to_openstudio.rb
|
166
186
|
- lib/to_openstudio/construction/air.rb
|
167
187
|
- lib/to_openstudio/construction/opaque.rb
|