# ******************************************************************************* # Honeybee OpenStudio Gem, Copyright (c) 2020, Alliance for Sustainable # Energy, LLC, Ladybug Tools LLC and other contributors. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # (1) Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # (2) Redistributions in binary form must reproduce the above copyright notice, # this list of conditions and the following disclaimer in the documentation # and/or other materials provided with the distribution. # # (3) Neither the name of the copyright holder nor the names of any contributors # may be used to endorse or promote products derived from this software without # specific prior written permission from the respective party. # # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE # UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF # THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, # EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT # OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, # STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # ******************************************************************************* require 'honeybee/model' require 'openstudio' module Honeybee def self.write_schedule_csv(schedule_csv_dir, schedule_csv) filename = schedule_csv[:filename] columns = schedule_csv[:columns] if !columns.empty? n = columns[0].size path = File.join(schedule_csv_dir, filename) File.open(path, 'w') do |file| (0...n).each do |i| row = [] columns.each do |column| row << column[i] end file.puts row.join(',') end # make sure data is written to the disk one way or the other begin file.fsync rescue file.flush end end end end class Model attr_reader :openstudio_model attr_reader :schedule_csv_dir, :include_datetimes, :schedule_csvs # if a schedule csv dir is specified then ScheduleFixedIntervalAbridged objects # will be translated to ScheduleFile objects instead of ScheduleFixedInterval # the optional schedule_csv_include_datetimes argument controls whether schedule csv # files include a first column of date times for verification def set_schedule_csv_dir(schedule_csv_dir, include_datetimes = false) @schedule_csv_dir = schedule_csv_dir @include_datetimes = include_datetimes end # convert to openstudio model, clears errors and warnings def to_openstudio_model(openstudio_model=nil, log_report=true) @errors = [] @warnings = [] if log_report puts 'Starting Model translation from Honeybee to OpenStudio' end @openstudio_model = if openstudio_model openstudio_model else OpenStudio::Model::Model.new end # create all openstudio objects in the model create_openstudio_objects(log_report) @openstudio_model end private # create OpenStudio objects in the OpenStudio model def create_openstudio_objects(log_report=true) # assign the building name and a default standards building type so measures can run building = @openstudio_model.getBuilding if @hash[:display_name] clean_name = @hash[:display_name].to_s.gsub(/[^.A-Za-z0-9_-] /, " ") building.setName(clean_name) else building.setName(@hash[:identifier]) end if building.standardsBuildingType.empty? building.setStandardsBuildingType('MediumOffice') end # initialize a global variable for whether the AFN is used instead of simple ventilation $use_simple_vent = true if @hash[:properties].key?(:energy) && @hash[:properties][:energy][:ventilation_simulation_control] vent_sim_control = @hash[:properties][:energy][:ventilation_simulation_control] if vent_sim_control[:vent_control_type] && vent_sim_control[:vent_control_type] != 'SingleZone' $use_simple_vent = false vsim_cntrl = VentilationSimulationControl.new(vent_sim_control) $afn_reference_crack = vsim_cntrl.to_openstudio(@openstudio_model) end end # initialize global hashes for various model properties $gas_gap_hash = Hash.new # hash to track gas gaps in case they are split by shades $air_boundary_hash = Hash.new # hash to track any air boundary constructions $window_shade_hash = Hash.new # hash to track any window constructions with shade $window_dynamic_hash = Hash.new # hash to track any dynamic window constructions $programtype_shw_hash = Hash.new # hash to track ServiceHotWater objects $programtype_setpoint_hash = Hash.new # hash to track Setpoint objects $interior_afn_srf_hash = Hash.new # track whether an adjacent surface is already in the AFN $shw_for_plant = nil # track whether a hot water plant is needed $detailed_hvac_hash = Hash.new # track the detailed HVAC systems that have been assigned $ironbug_exe = nil # ironbug executable; will be overwritten by any detailed HVACs $generators = [] # track the generator objects in the model # create all of the non-geometric model elements if @hash[:properties].key?(:energy) if log_report # schedules are used by all other objects and come first puts 'Translating Schedules' end if @hash[:properties][:energy][:schedule_type_limits] create_schedule_type_limits(@hash[:properties][:energy][:schedule_type_limits]) end if @hash[:properties][:energy][:schedules] create_schedules(@hash[:properties][:energy][:schedules], false, true) end if log_report puts 'Translating Materials' end if @hash[:properties][:energy][:materials] create_materials(@hash[:properties][:energy][:materials]) end if log_report puts 'Translating Constructions' end if @hash[:properties][:energy][:constructions] create_constructions(@hash[:properties][:energy][:constructions]) end if log_report puts 'Translating ConstructionSets' end if @hash[:properties][:energy][:construction_sets] create_construction_sets(@hash[:properties][:energy][:construction_sets]) end if log_report puts 'Translating ProgramTypes' end if @hash[:properties][:energy][:program_types] create_program_types(@hash[:properties][:energy][:program_types]) end end # create the default construction set to catch any cases of unassigned constructions if log_report puts 'Translating Default ConstructionSet' end create_default_construction_set # create the geometry and add any extra properties to it if log_report puts 'Translating Room Geometry' end create_rooms unless $window_shade_hash.empty? if log_report puts 'Translating Window Shading Control' end create_shading_control end unless $window_dynamic_hash.empty? if log_report puts 'Translating Dynamic Windows' end create_dynamic_windows end if log_report puts 'Translating HVAC Systems' end if @hash[:properties].key?(:energy) create_hvacs create_hot_water_plant end if log_report puts 'Translating Context Shade Geometry' end create_orphaned_shades create_orphaned_faces create_orphaned_apertures create_orphaned_doors create_shade_meshes unless $generators.empty? if log_report puts 'Translating Electricity Generators' end if @hash[:properties][:energy][:electric_load_center] load_center_object = ElectricLoadCenter.new( @hash[:properties][:energy][:electric_load_center]) else empty_hash = Hash.new load_center_object = ElectricLoadCenter.new(empty_hash) end load_center_object.to_openstudio(@openstudio_model, $generators) end end def create_materials(material_dicts, check_existing=false) material_dicts.each do |material| # check if there's already a material in the model with the identifier add_obj = true if check_existing object = @openstudio_model.getMaterialByName(material[:identifier]) if object.is_initialized add_obj = false end end # add the material object to the Model if add_obj material_type = material[:type] case material_type when 'EnergyMaterial' material_object = EnergyMaterial.new(material) when 'EnergyMaterialNoMass' material_object = EnergyMaterialNoMass.new(material) when 'EnergyMaterialVegetation' material_object = EnergyMaterialVegetation.new(material) when 'EnergyWindowMaterialSimpleGlazSys' material_object = EnergyWindowMaterialSimpleGlazSys.new(material) when 'EnergyWindowMaterialGlazing' material_object = EnergyWindowMaterialGlazing.new(material) when 'EnergyWindowFrame' material_object = EnergyWindowFrame.new(material) when 'EnergyWindowMaterialGas' material_object = EnergyWindowMaterialGas.new(material) $gas_gap_hash[material[:identifier]] = material_object when 'EnergyWindowMaterialGasMixture' material_object = EnergyWindowMaterialGasMixture.new(material) $gas_gap_hash[material[:identifier]] = material_object when 'EnergyWindowMaterialGasCustom' material_object = EnergyWindowMaterialGasCustom.new(material) $gas_gap_hash[material[:identifier]] = material_object when 'EnergyWindowMaterialBlind' material_object = EnergyWindowMaterialBlind.new(material) when 'EnergyWindowMaterialShade' material_object = EnergyWindowMaterialShade.new(material) else raise "Unknown material type #{material_type}" end material_object.to_openstudio(@openstudio_model) end end end def create_constructions(construction_dicts, check_existing=false) construction_dicts.each do |construction| # check if there's already a construction in the model with the identifier add_obj = true if check_existing object = @openstudio_model.getConstructionByName(construction[:identifier]) if object.is_initialized add_obj = false end end # add the construction object to the Model if add_obj construction_type = construction[:type] case construction_type when 'OpaqueConstructionAbridged' construction_object = OpaqueConstructionAbridged.new(construction) when 'WindowConstructionAbridged' construction_object = WindowConstructionAbridged.new(construction) when 'WindowConstructionShadeAbridged' construction_object = WindowConstructionShadeAbridged.new(construction) $window_shade_hash[construction[:identifier]] = construction_object when 'WindowConstructionDynamicAbridged' construction_object = WindowConstructionDynamicAbridged.new(construction) $window_dynamic_hash[construction[:identifier]] = construction_object when 'ShadeConstruction' construction_object = ShadeConstruction.new(construction) when 'AirBoundaryConstructionAbridged' construction_object = AirBoundaryConstructionAbridged.new(construction) $air_boundary_hash[construction[:identifier]] = construction else raise "Unknown construction type #{construction_type}." end construction_object.to_openstudio(@openstudio_model) end end end def create_construction_sets(construction_set_dicts, check_existing=false) construction_set_dicts.each do |construction_set| # check if there's already a construction set in the model with the identifier add_obj = true if check_existing object = @openstudio_model.getDefaultConstructionSetByName( construction_set[:identifier]) if object.is_initialized add_obj = false end end # add the construction set object to the Model if add_obj construction_set_object = ConstructionSetAbridged.new(construction_set) construction_set_object.to_openstudio(@openstudio_model) end end end def create_schedule_type_limits(stl_dicts, check_existing=false) stl_dicts.each do |schedule_type_limit| # check if there's already a schedule type limit in the model with the identifier add_obj = true if check_existing object = @openstudio_model.getScheduleTypeLimitsByName( schedule_type_limit[:identifier]) if object.is_initialized add_obj = false end end # add the schedule type limit object to the Model if add_obj schedule_type_limit_object = ScheduleTypeLimit.new(schedule_type_limit) schedule_type_limit_object.to_openstudio(@openstudio_model) end end end def create_schedules(schedule_dicts, check_existing=false, check_leap_year=true) # clear out schedule_csvs @schedule_csvs = {} if check_leap_year is_leap_year = :unknown schedule_dicts.each do |schedule| # set is leap year = true in case start date has 3 integers this_leap_year = false if schedule[:start_date] && schedule[:start_date][2] this_leap_year = true end if is_leap_year == :unknown is_leap_year = this_leap_year elsif is_leap_year != this_leap_year raise("Mixed leap year information.") end end if is_leap_year != :unknown year_description = openstudio_model.getYearDescription year_description.setIsLeapYear(is_leap_year) end end schedule_dicts.each do |schedule| # check if there's already a schedule in the model with the identifier add_obj = true if check_existing object = @openstudio_model.getScheduleByName(schedule[:identifier]) if object.is_initialized add_obj = false end end # add the schedule object to the Model if add_obj schedule_type = schedule[:type] case schedule_type when 'ScheduleRulesetAbridged' schedule_object = ScheduleRulesetAbridged.new(schedule) when 'ScheduleFixedIntervalAbridged' schedule_object = ScheduleFixedIntervalAbridged.new(schedule) else raise("Unknown schedule type #{schedule_type}.") end schedule_object.to_openstudio(@openstudio_model, @schedule_csv_dir, @include_datetimes, @schedule_csvs) end end # write schedule csvs @schedule_csvs.each_value do |schedule_csv| Honeybee.write_schedule_csv(@schedule_csv_dir, schedule_csv) end end def create_program_types(program_dicts, check_existing=false) program_dicts.each do |space_type| # check if there's already a space type in the model with the identifier add_obj = true if check_existing object = @openstudio_model.getSpaceTypeByName(space_type[:identifier]) if object.is_initialized add_obj = false end end # add the space type object to the Model if add_obj space_type_object = ProgramTypeAbridged.new(space_type) space_type_object.to_openstudio(@openstudio_model) end end end def create_default_construction_set # create the materials, constructions and construction set create_materials(@@standards[:materials], true) create_constructions(@@standards[:constructions], true) create_construction_sets(@@standards[:construction_sets], true) # write the fractional schedule type and always on schedule if they are not there @@standards[:schedule_type_limits].each do |sch_type_limit| if sch_type_limit[:identifier] == 'Fractional' create_schedule_type_limits([sch_type_limit], true) end end @@standards[:schedules].each do |schedule| if schedule[:identifier] == 'Always On' create_schedules([schedule], true, false) end end # set the default construction set to the building level of the Model construction_id = 'Default Generic Construction Set' construction = @openstudio_model.getDefaultConstructionSetByName(construction_id) unless construction.empty? os_constructionset = construction.get @openstudio_model.getBuilding.setDefaultConstructionSet(os_constructionset) end end def create_rooms if @hash[:rooms] $air_mxing_array = [] # list to track any air mixing between Rooms @hash[:rooms].each do |room| room_object = Room.new(room) openstudio_room = room_object.to_openstudio(@openstudio_model) # for rooms with hot water objects definied in the ProgramType, make a new WaterUse:Equipment if room[:properties].key?(:energy) if room[:properties][:energy][:program_type] && !room[:properties][:energy][:service_hot_water] program_type_id = room[:properties][:energy][:program_type] shw_hash = $programtype_shw_hash[program_type_id] unless shw_hash.nil? if shw_hash[:flow_per_area].to_f != 0 shw_object = ServiceHotWaterAbridged.new(shw_hash) openstudio_shw = shw_object.to_openstudio( @openstudio_model, openstudio_room, room[:properties][:energy][:shw]) $shw_for_plant = shw_object end end end # for rooms with setpoint objects defined in the ProgramType, make a new thermostat if room[:properties][:energy][:program_type] && !room[:properties][:energy][:setpoint] thermal_zone = openstudio_room.thermalZone() unless thermal_zone.empty? thermal_zone_object = thermal_zone.get program_type_id = room[:properties][:energy][:program_type] setpoint_hash = $programtype_setpoint_hash[program_type_id] unless setpoint_hash.nil? # program type has no setpoint thermostat_object = SetpointThermostat.new(setpoint_hash) openstudio_thermostat = thermostat_object.to_openstudio(@openstudio_model) thermal_zone_object.setThermostatSetpointDualSetpoint(openstudio_thermostat) if setpoint_hash[:humidifying_schedule] or setpoint_hash[:dehumidifying_schedule] humidistat_object = SetpointHumidistat.new(setpoint_hash) openstudio_humidistat = humidistat_object.to_openstudio(@openstudio_model) thermal_zone_object.setZoneControlHumidistat(openstudio_humidistat) end end end end end end # create mixing objects between Rooms $air_mxing_array.each do |air_mix_props| zone_mixing = OpenStudio::Model::ZoneMixing.new(air_mix_props[0]) zone_mixing.setDesignFlowRate(air_mix_props[1]) flow_sch_ref = @openstudio_model.getScheduleByName(air_mix_props[2]) unless flow_sch_ref.empty? flow_sched = flow_sch_ref.get zone_mixing.setSchedule(flow_sched) end source_zone_ref = @openstudio_model.getThermalZoneByName(air_mix_props[3]) unless source_zone_ref.empty? source_zone = source_zone_ref.get zone_mixing.setSourceZone(source_zone) end end end end def create_shading_control # assign any shading control objects to windows with shades # this is run as a separate step once all logic about construction sets is in place sub_faces = @openstudio_model.getSubSurfaces() sub_faces.each do |sub_face| constr_ref = sub_face.construction unless constr_ref.empty? constr = constr_ref.get constr_name_ref = constr.name unless constr_name_ref.empty? constr_name = constr_name_ref.get unless $window_shade_hash[constr_name].nil? window_shd_constr = $window_shade_hash[constr_name] os_shd_control = window_shd_constr.to_openstudio_shading_control(@openstudio_model) sub_face.setShadingControl(os_shd_control) end end end end end def create_dynamic_windows # create the actuators and EMS program for any dynamic windows WindowConstructionDynamicAbridged.add_sub_faces_to_window_dynamic_hash(@openstudio_model) $window_dynamic_hash.each do |constr_id, constr_obj| constr_obj.ems_program_to_openstudio(@openstudio_model) end end def create_hvacs if @hash[:properties][:energy][:hvacs] $air_loop_count = 0 # track the total number of air loops in the model # gather all of the hashes of the HVACs hvac_hashes = Hash.new @hash[:properties][:energy][:hvacs].each do |hvac| hvac_hashes[hvac[:identifier]] = hvac hvac_hashes[hvac[:identifier]]['rooms'] = [] end # loop through the rooms and track which are assigned to each HVAC if @hash[:rooms] @hash[:rooms].each do |room| if room[:properties][:energy][:hvac] hvac_hashes[room[:properties][:energy][:hvac]]['rooms'] << room[:identifier] end end end hvac_hashes.each_value do |hvac| system_type = hvac[:type] if system_type == 'IdealAirSystemAbridged' ideal_air_system = IdealAirSystemAbridged.new(hvac) hvac['rooms'].each do |room_id| os_ideal_air = ideal_air_system.to_openstudio(@openstudio_model) # enforce a strict naming system for each zone so results can be matched os_ideal_air.setName(room_id + ' Ideal Loads Air System') zone_get = @openstudio_model.getThermalZoneByName(room_id) unless zone_get.empty? os_thermal_zone = zone_get.get os_ideal_air.addToThermalZone(os_thermal_zone) # set the humidistat if the zone has one humid_get = os_thermal_zone.zoneControlHumidistat unless humid_get.empty? os_ideal_air.setDehumidificationControlType('Humidistat') os_ideal_air.setHumidificationControlType('Humidistat') end end end elsif system_type == 'DetailedHVAC' $detailed_hvac_hash[hvac[:identifier]] = hvac[:specification] unless hvac[:ironbug_exe].nil? $ironbug_exe = hvac[:ironbug_exe] end elsif TemplateHVAC.types.include?(system_type) template_system = TemplateHVAC.new(hvac) template_system.to_openstudio(@openstudio_model, hvac['rooms']) end end end end def create_hot_water_plant # create a hot water plant if there's any service hot water in the model unless $shw_for_plant.nil? if @hash[:properties][:energy][:shws].nil? shw_list = [] else shw_list = @hash[:properties][:energy][:shws] end unless $shw_for_plant.shw_connections['default_district_shw'].nil? def_hash = {} def_hash[:equipment_type] = 'Default_District_SHW' def_hash[:identifier] = 'default_district_shw' shw_list << def_hash end $shw_for_plant.add_hot_water_plants(@openstudio_model, shw_list) end end def create_orphaned_shades if @hash[:orphaned_shades] if $orphan_groups shading_surface_group = OpenStudio::Model::ShadingSurfaceGroup.new(@openstudio_model) shading_surface_group.setShadingSurfaceType('Building') end @hash[:orphaned_shades].each do |shade| shade_object = Shade.new(shade) openstudio_shade = shade_object.to_openstudio(@openstudio_model) if $orphan_groups openstudio_shade.setShadingSurfaceGroup(shading_surface_group) end end end end def create_orphaned_faces if @hash[:orphaned_faces] if $orphan_groups shading_surface_group = OpenStudio::Model::ShadingSurfaceGroup.new(@openstudio_model) shading_surface_group.setShadingSurfaceType('Building') end @hash[:orphaned_faces].each do |face| face_object = Face.new(face) openstudio_shade = face_object.to_openstudio_shade(@openstudio_model, shading_surface_group) if $orphan_groups openstudio_shade.setShadingSurfaceGroup(shading_surface_group) end end end end def create_orphaned_apertures if @hash[:orphaned_apertures] if $orphan_groups shading_surface_group = OpenStudio::Model::ShadingSurfaceGroup.new(@openstudio_model) shading_surface_group.setShadingSurfaceType('Building') end @hash[:orphaned_apertures].each do |ap| ap_object = Aperture.new(ap) openstudio_shade = ap_object.to_openstudio_shade(@openstudio_model, shading_surface_group) if $orphan_groups openstudio_shade.setShadingSurfaceGroup(shading_surface_group) end end end end def create_orphaned_doors if @hash[:orphaned_doors] if $orphan_groups shading_surface_group = OpenStudio::Model::ShadingSurfaceGroup.new(@openstudio_model) shading_surface_group.setShadingSurfaceType('Building') end @hash[:orphaned_doors].each do |dr| dr_object = Door.new(dr) openstudio_shade = dr_object.to_openstudio_shade(@openstudio_model, shading_surface_group) if $orphan_groups openstudio_shade.setShadingSurfaceGroup(shading_surface_group) end end end end def create_shade_meshes if @hash[:shade_meshes] @hash[:shade_meshes].each do |shade_mesh| shade_mesh_object = ShadeMesh.new(shade_mesh) openstudio_shades = shade_mesh_object.to_openstudio(@openstudio_model) if $orphan_groups shading_surface_group = OpenStudio::Model::ShadingSurfaceGroup.new(@openstudio_model) shading_surface_group.setShadingSurfaceType('Building') shading_surface_group.setName(shade_mesh[:identifier]) unless shade_mesh[:display_name].nil? shading_surface_group.setDisplayName(shade_mesh[:display_name]) end openstudio_shades.each do |openstudio_shade| openstudio_shade.setShadingSurfaceGroup(shading_surface_group) end end end end end #TODO: create runlog for errors. end # Model end # Honeybee