# *******************************************************************************
# OpenStudio(R), Copyright (c) 2008-2021, Alliance for Sustainable Energy, LLC.
# 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.
#
# (4) Other than as required in clauses (1) and (2), distributions in any form
# of modifications or other derivative works may not use the "OpenStudio"
# trademark, "OS", "os", or any other confusingly similar designation without
# specific prior written permission from Alliance for Sustainable Energy, LLC.
#
# 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 'json'
require 'openstudio-standards'
module OsLib_Reporting
# setup - get model, sql, and setup web assets path
def self.setup(runner)
results = {}
# get the last model
model = runner.lastOpenStudioModel
if model.empty?
runner.registerError('Cannot find last model.')
return false
end
model = model.get
# get the last idf
workspace = runner.lastEnergyPlusWorkspace
if workspace.empty?
runner.registerError('Cannot find last idf file.')
return false
end
workspace = workspace.get
# get the last sql file
sqlFile = runner.lastEnergyPlusSqlFile
if sqlFile.empty?
runner.registerError('Cannot find last sql file.')
return false
end
sqlFile = sqlFile.get
model.setSqlFile(sqlFile)
# populate hash to pass to measure
results[:model] = model
# results[:workspace] = workspace
results[:sqlFile] = sqlFile
return results
end
def self.ann_env_pd(sqlFile)
# get the weather file run period (as opposed to design day run period)
ann_env_pd = nil
sqlFile.availableEnvPeriods.each do |env_pd|
env_type = sqlFile.environmentType(env_pd)
if env_type.is_initialized
if env_type.get == OpenStudio::EnvironmentType.new('WeatherRunPeriod')
ann_env_pd = env_pd
end
end
end
return ann_env_pd
end
def self.create_xls
require 'rubyXL'
book = ::RubyXL::Workbook.new
# delete initial worksheet
return book
end
def self.save_xls(book)
file = book.write 'excel-file.xlsx'
return file
end
# write an Excel file from table data
# the Excel Functions are not currently being used, left in as example
# Requires ruby Gem which isn't currently supported in OpenStudio GUIs.
# Current setup is simple, creates new workbook for each table
# Could be updated to have one section per workbook
def self.write_xls(table_data, book)
worksheet = book.add_worksheet table_data[:title]
row_cnt = 0
# write the header row
header = table_data[:header]
header.each_with_index do |h, i|
worksheet.add_cell(row_cnt, i, h)
end
worksheet.change_row_fill(row_cnt, '0ba53d')
# loop over data rows
data = table_data[:data]
data.each do |d|
row_cnt += 1
d.each_with_index do |c, i|
worksheet.add_cell(row_cnt, i, c)
end
end
return book
end
# cleanup - prep html and close sql
def self.cleanup(html_in_path)
# TODO: - would like to move code here, but couldn't get it working. May look at it again later on.
return html_out_path
end
# clean up unkown strings used for runner.registerValue names
def self.reg_val_string_prep(string)
# replace non alpha-numberic characters with an underscore
string = string.gsub(/[^0-9a-z]/i, '_')
# snake case string
string = OpenStudio.toUnderscoreCase(string)
return string
end
# hard code fuel types (for E+ 9.4 shouldn't have it twice, should eventually come form OS)
def self.fuel_type_names(extended = false)
# get short or extended list (not using now)
fuel_types = []
OpenStudio::EndUseFuelType.getValues.each do |fuel_type|
# convert integer to string
fuel_name = OpenStudio::EndUseFuelType.new(fuel_type).valueDescription
next if fuel_name == 'Water'
fuel_types << fuel_name
end
return fuel_types
end
# create template section
def self.template_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
template_tables = []
# gather data for section
@template_section = {}
@template_section[:title] = 'Tasty Treats'
@template_section[:tables] = template_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @template_section
end
# notes:
# The data below would typically come from the model or simulation results
# You can loop through objects to make a table for each item of that type, such as air loops
# If a section will only have one table you can leave the table title blank and just rely on the section title
# these will be updated later to support graphs
# create table
template_table_01 = {}
template_table_01[:title] = 'Fruit'
template_table_01[:header] = ['Definition', 'Value']
template_table_01[:units] = ['', '$/pound']
template_table_01[:data] = []
# add rows to table
template_table_01[:data] << ['Banana', 0.23]
template_table_01[:data] << ['Apple', 0.75]
template_table_01[:data] << ['Orange', 0.50]
# add table to array of tables
template_tables << template_table_01
# using helper method that generates table for second example
template_tables << OsLib_Reporting.template_table(model, sqlFile, runner, is_ip_units = true)
return @template_section
end
# create template section
def self.template_table(model, sqlFile, runner, is_ip_units = true)
# create a second table
template_table = {}
template_table[:title] = 'Ice Cream'
template_table[:header] = ['Definition', 'Base Flavor', 'Toppings', 'Value']
template_table[:units] = ['', '', '', 'scoop']
template_table[:data] = []
# add rows to table
template_table[:data] << ['Vanilla', 'Vanilla', 'NA', 1.5]
template_table[:data] << ['Rocky Road', 'Chocolate', 'Nuts', 1.5]
template_table[:data] << ['Mint Chip', 'Mint', 'Chocolate Chips', 1.5]
return template_table
end
# building_summary section
def self.building_summary_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
general_tables = []
# gather data for section
@building_summary_section = {}
@building_summary_section[:title] = 'Model Summary'
@building_summary_section[:tables] = general_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @building_summary_section
end
# add in general information from method
general_tables << OsLib_Reporting.general_building_information_table(model, sqlFile, runner, is_ip_units)
general_tables << OsLib_Reporting.weather_summary_table(model, sqlFile, runner, is_ip_units)
general_tables << OsLib_Reporting.design_day_table(model, sqlFile, runner, is_ip_units)
general_tables << OsLib_Reporting.setpoint_not_met_summary_table(model, sqlFile, runner, is_ip_units)
general_tables << OsLib_Reporting.setpoint_not_met_criteria_table(model, sqlFile, runner, is_ip_units)
# general_tables << OsLib_Reporting.site_performance_table(model,sqlFile,runner)
site_power_generation_table = OsLib_Reporting.site_power_generation_table(model, sqlFile, runner, is_ip_units)
if site_power_generation_table
general_tables << OsLib_Reporting.site_power_generation_table(model, sqlFile, runner, is_ip_units)
end
return @building_summary_section
end
# annual_overview section
def self.annual_overview_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
annual_tables = []
# gather data for section
@annual_overview_section = {}
@annual_overview_section[:title] = 'Annual Overview'
@annual_overview_section[:tables] = annual_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @annual_overview_section
end
# add in annual overview from method
annual_tables << OsLib_Reporting.output_data_end_use_table(model, sqlFile, runner, is_ip_units)
annual_tables << OsLib_Reporting.output_data_energy_use_table(model, sqlFile, runner, is_ip_units)
annual_tables << OsLib_Reporting.output_data_end_use_electricity_table(model, sqlFile, runner, is_ip_units)
annual_tables << OsLib_Reporting.output_data_end_use_gas_table(model, sqlFile, runner, is_ip_units)
return @annual_overview_section
end
# create table with general building information
# this just makes a table, and not a full section. It feeds into another method that makes a full section
def self.general_building_information_table(model, sqlFile, runner, is_ip_units = true)
# general building information type data output
general_building_information = {}
general_building_information[:title] = 'Building Summary' # name will be with section
general_building_information[:header] = ['Data', 'Value']
general_building_information[:units] = [] # won't populate for this table since each row has different units
general_building_information[:data] = []
# structure ID / building name
display = 'Building Name'
target_units = 'n/a'
value = model.getBuilding.name.to_s
general_building_information[:data] << [display, value]
runner.registerValue(OsLib_Reporting.reg_val_string_prep(display), value, target_units)
# total site energy
display = 'Total Site Energy'
source_units = 'GJ'
if is_ip_units
target_units = 'kBtu'
else
target_units = 'kWh'
end
value = OpenStudio.convert(sqlFile.totalSiteEnergy.get, source_units, target_units).get
value_neat = "#{OpenStudio.toNeatString(value, 0, true)} #{target_units}"
runner.registerValue(OsLib_Reporting.reg_val_string_prep(display), value, target_units)
general_building_information[:data] << [display, value_neat]
# net site energy
display = 'Net Site Energy'
source_units = 'GJ'
if is_ip_units
target_units = 'kBtu'
else
target_units = 'kWh'
end
value = OpenStudio.convert(sqlFile.netSiteEnergy.get, source_units, target_units).get
value_neat = "#{OpenStudio.toNeatString(value, 0, true)} #{target_units}"
runner.registerValue(OsLib_Reporting.reg_val_string_prep(display), value, target_units)
# always register value, but only add to table if net is different than total
if sqlFile.totalSiteEnergy.get != sqlFile.netSiteEnergy.get
general_building_information[:data] << [display, value_neat]
end
# total building area
query = 'SELECT Value FROM tabulardatawithstrings WHERE '
query << "ReportName='AnnualBuildingUtilityPerformanceSummary' and "
query << "ReportForString='Entire Facility' and "
query << "TableName='Building Area' and "
query << "RowName='Total Building Area' and "
query << "ColumnName='Area' and "
query << "Units='m2';"
query_results = sqlFile.execAndReturnFirstDouble(query)
if query_results.empty?
runner.registerWarning('Did not find value for total building area.')
return false
else
display = 'Total Building Area'
source_units = 'm^2'
if is_ip_units
target_units = 'ft^2'
else
target_units = source_units
end
value = OpenStudio.convert(query_results.get, source_units, target_units).get
value_neat = "#{OpenStudio.toNeatString(value, 0, true)} #{target_units}"
general_building_information[:data] << [display, value_neat]
runner.registerValue(OsLib_Reporting.reg_val_string_prep(display), value, target_units)
end
# code to check OS vs. E+ area
energy_plus_area = query_results.get
open_studio_area = model.getBuilding.floorArea
if (energy_plus_area - open_studio_area).abs >= 1.0
runner.registerWarning("EnergyPlus reported area is #{query_results.get.round} (m^2). OpenStudio reported area is #{model.getBuilding.floorArea.round} (m^2).")
end
# total EUI
eui = sqlFile.totalSiteEnergy.get / energy_plus_area
display = 'Total Site EUI'
source_units = 'GJ/m^2'
if is_ip_units
target_units = 'kBtu/ft^2'
else
target_units = 'kWh/m^2'
end
if query_results.get > 0.0 # don't calculate EUI if building doesn't have any area
value = OpenStudio.convert(eui, source_units, target_units).get
value_neat = "#{OpenStudio.toNeatString(value, 2, true)} #{target_units}"
runner.registerValue(OsLib_Reporting.reg_val_string_prep(display), value, target_units) # is it ok not to calc EUI if no area in model
else
value_neat = "can't calculate Total EUI."
end
general_building_information[:data] << [display.to_s, value_neat]
# net EUI
eui = sqlFile.netSiteEnergy.get / energy_plus_area
display = 'EUI'
if query_results.get > 0.0 # don't calculate EUI if building doesn't have any area
value = OpenStudio.convert(eui, source_units, target_units).get
value_neat = "#{OpenStudio.toNeatString(value, 2, true)} #{target_units}"
runner.registerValue(OsLib_Reporting.reg_val_string_prep(display), value, target_units) # is it ok not to calc EUI if no area in model
else
value_neat = 'Net EUI could not be calculated.'
end
# always register value, but only add to table if net is different than total
if sqlFile.totalSiteEnergy.get != sqlFile.netSiteEnergy.get
general_building_information[:data] << ['Net Site EUI', value_neat]
end
# TODO: - add total EUI for conditioned floor area if not the same as total. Doesn't seem necessary to also calculate net conditioned EUI if it exists as a unique value.
# conditioned building area
query = 'SELECT Value FROM tabulardatawithstrings WHERE '
query << "ReportName='AnnualBuildingUtilityPerformanceSummary' and "
query << "ReportForString='Entire Facility' and "
query << "TableName='Building Area' and "
query << "RowName='Net Conditioned Building Area' and "
query << "ColumnName='Area' and "
query << "Units='m2';"
query_results = sqlFile.execAndReturnFirstDouble(query)
if query_results.empty?
runner.registerWarning('Did not find value for net conditioned building area.')
return false
elsif query_results.get == 0.0
display_a = 'Conditioned Building Area'
source_units_a = 'm^2'
if is_ip_units
target_units_a = 'ft^2'
else
target_units_a = source_units_a
end
value_a = OpenStudio.convert(query_results.get, source_units_a, target_units_a).get
value_neat_a = "#{OpenStudio.toNeatString(value_a, 0, true)} #{target_units_a}"
runner.registerValue(OsLib_Reporting.reg_val_string_prep(display_a), value_a, target_units_a)
# conditioned EUI
eui = sqlFile.totalSiteEnergy.get / energy_plus_area
display_e = 'Conditioned Site EUI'
source_units_e = 'GJ/m^2'
if is_ip_units
target_units_e = 'kBtu/ft^2'
else
target_units_e = 'kWh/m^2'
end
value_e = OpenStudio.convert(eui, source_units_e, target_units_e).get
value_neat_e = "#{OpenStudio.toNeatString(value_e, 2, true)} #{target_units_e}"
runner.registerValue(OsLib_Reporting.reg_val_string_prep(display_e), value_e, target_units_e)
# always register value, but only add to table if net is different than total
if energy_plus_area - query_results.get >= 1.0
general_building_information[:data] << [display_a, value_neat_a]
general_building_information[:data] << [display_e, value_neat_e]
end
end
# get standards building type
building_type = 'n/a'
if model.getBuilding.standardsBuildingType.is_initialized
building_type = model.getBuilding.standardsBuildingType.get
end
general_building_information[:data] << ['OpenStudio Standards Building Type', building_type]
return general_building_information
end
# create table of space type breakdown
def self.space_type_breakdown_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# space type data output
output_data_space_type_breakdown = {}
output_data_space_type_breakdown[:title] = ''
output_data_space_type_breakdown[:header] = ['Space Type Name', 'Floor Area', 'Standards Building Type', 'Standards Space Type']
if is_ip_units
target_units = 'ft^2'
else
target_units = 'm^2'
end
output_data_space_type_breakdown[:units] = ['', target_units, '', '']
output_data_space_type_breakdown[:data] = []
output_data_space_type_breakdown[:chart_type] = 'simple_pie'
output_data_space_type_breakdown[:chart] = []
# gather data for section
@output_data_space_type_breakdown_section = {}
@output_data_space_type_breakdown_section[:title] = 'Space Type Breakdown'
@output_data_space_type_breakdown_section[:tables] = [output_data_space_type_breakdown] # only one table for this section
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @output_data_space_type_breakdown_section
end
space_types = model.getSpaceTypes
space_types.sort.each do |spaceType|
next if spaceType.floorArea == 0
# get color
color = spaceType.renderingColor
if !color.empty?
color = color.get
red = color.renderingRedValue
green = color.renderingGreenValue
blue = color.renderingBlueValue
color = "rgb(#{red},#{green},#{blue})"
else
# TODO: - this should set red green and blue as separate values
color = 'rgb(20,20,20)' # maybe do random or let d3 pick color instead of this?
end
# data for space type breakdown
display = spaceType.name.get
floor_area_si = spaceType.floorArea
floor_area_si = 0
# loop through spaces so I can skip if not included in floor area
spaceType.spaces.each do |space|
next if !space.partofTotalFloorArea
floor_area_si += space.floorArea * space.multiplier
end
# julien
source_units = 'm^2'
if is_ip_units
target_units = 'ft^2'
else
target_units = source_units
end
value = OpenStudio.convert(floor_area_si, 'm^2', target_units).get
num_people = nil
value_neat = OpenStudio.toNeatString(value, 0, true)
# get standards information
if spaceType.standardsBuildingType.is_initialized
standards_building_type = spaceType.standardsBuildingType.get
else
standards_building_type = ''
end
if spaceType.standardsSpaceType.is_initialized
standards_space_type = spaceType.standardsSpaceType.get
else
standards_space_type = ''
end
output_data_space_type_breakdown[:data] << [display, value_neat, standards_building_type, standards_space_type]
runner.registerValue("space_type_#{OsLib_Reporting.reg_val_string_prep(display)}", value, target_units)
# data for graph
output_data_space_type_breakdown[:chart] << JSON.generate(label: display, value: value, color: color)
end
spaces = model.getSpaces
# count area of spaces that have no space type
no_space_type_area_counter = 0
spaces.each do |space|
if space.spaceType.empty?
next if !space.partofTotalFloorArea
no_space_type_area_counter += space.floorArea * space.multiplier
end
end
if no_space_type_area_counter > 0
display = 'No Space Type'
# julien
source_units = 'm^2'
if is_ip_units
target_units = 'ft^2'
else
target_units = source_units
end
value = OpenStudio.convert(no_space_type_area_counter, 'm^2', target_units).get
value_neat = OpenStudio.toNeatString(value, 0, true)
output_data_space_type_breakdown[:data] << [display, value_neat]
runner.registerValue("space_type_#{OsLib_Reporting.reg_val_string_prep(display)}", value, target_units)
# data for graph
color = 'rgb(20,20,20)' # maybe do random or let d3 pick color instead of this?
output_data_space_type_breakdown[:chart] << JSON.generate(label: 'No SpaceType Assigned',
value: OpenStudio.convert(no_space_type_area_counter, 'm^2', target_units),
color: color)
end
return @output_data_space_type_breakdown_section
end
# create table with general building information
# this just makes a table, and not a full section. It feeds into another method that makes a full section
def self.output_data_end_use_table(model, sqlFile, runner, is_ip_units = true)
# end use data output
output_data_end_use = {}
output_data_end_use[:title] = 'End Use'
output_data_end_use[:header] = ['End Use', 'Consumption']
if is_ip_units
target_units = 'kBtu'
else
target_units = 'kWh'
end
output_data_end_use[:units] = ['', target_units]
output_data_end_use[:data] = []
output_data_end_use[:chart_type] = 'simple_pie'
output_data_end_use[:chart] = []
end_use_colors = ['#EF1C21', '#0071BD', '#F7DF10', '#DEC310', '#4A4D4A', '#B5B2B5', '#FF79AD', '#632C94', '#F75921', '#293094', '#CE5921', '#FFB239', '#29AAE7', '#8CC739']
# loop through fuels for consumption tables
counter = 0
OpenStudio::EndUseCategoryType.getValues.each do |end_use|
# get end uses
end_use = OpenStudio::EndUseCategoryType.new(end_use).valueDescription
# loop through fuels
total_end_use = 0.0
fuel_type_names.each do |fuel_type|
query_fuel = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='AnnualBuildingUtilityPerformanceSummary' and TableName='End Uses' and RowName= '#{end_use}' and ColumnName= '#{fuel_type}'"
results_fuel = sqlFile.execAndReturnFirstDouble(query_fuel).get
total_end_use += results_fuel
end
# convert value and populate table
value = OpenStudio.convert(total_end_use, 'GJ', target_units).get
value_neat = OpenStudio.toNeatString(value, 0, true)
output_data_end_use[:data] << [end_use, value_neat]
runner.registerValue("end_use_#{end_use.downcase.tr(' ', '_')}", value, target_units)
if value > 0
output_data_end_use[:chart] << JSON.generate(label: end_use, value: value, color: end_use_colors[counter])
end
counter += 1
end
return output_data_end_use
end
# create table with general building information
# this just makes a table, and not a full section. It feeds into another method that makes a full section
def self.output_data_end_use_electricity_table(model, sqlFile, runner, is_ip_units = true)
# end use data output
output_data_end_use_electricity = {}
output_data_end_use_electricity[:title] = 'EUI - Electricity'
output_data_end_use_electricity[:header] = ['End Use', 'Consumption']
target_units = 'kWh'
output_data_end_use_electricity[:units] = ['', target_units]
output_data_end_use_electricity[:data] = []
output_data_end_use_electricity[:chart_type] = 'simple_pie'
output_data_end_use_electricity[:chart] = []
end_use_colors = ['#EF1C21', '#0071BD', '#F7DF10', '#DEC310', '#4A4D4A', '#B5B2B5', '#FF79AD', '#632C94', '#F75921', '#293094', '#CE5921', '#FFB239', '#29AAE7', '#8CC739']
# loop through fuels for consumption tables
counter = 0
OpenStudio::EndUseCategoryType.getValues.each do |end_use|
# get end uses
end_use = OpenStudio::EndUseCategoryType.new(end_use).valueDescription
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='AnnualBuildingUtilityPerformanceSummary' and TableName='End Uses' and RowName= '#{end_use}' and ColumnName= 'Electricity'"
results = sqlFile.execAndReturnFirstDouble(query)
value = OpenStudio.convert(results.get, 'GJ', target_units).get
value_neat = OpenStudio.toNeatString(value, 0, true)
output_data_end_use_electricity[:data] << [end_use, value_neat]
runner.registerValue("end_use_electricity_#{end_use.downcase.tr(' ', '_')}", value, target_units)
if value > 0
output_data_end_use_electricity[:chart] << JSON.generate(label: end_use, value: value, color: end_use_colors[counter])
end
counter += 1
end
return output_data_end_use_electricity
end
# create table with general building information
# this just makes a table, and not a full section. It feeds into another method that makes a full section
def self.output_data_end_use_gas_table(model, sqlFile, runner, is_ip_units = true)
# end use data output
output_data_end_use_gas = {}
output_data_end_use_gas[:title] = 'EUI - Gas'
output_data_end_use_gas[:header] = ['End Use', 'Consumption']
if is_ip_units
target_units = 'therm'
target_display_unit = 'therms'
else
target_units = 'kWh'
target_display_unit = 'kWh'
end
output_data_end_use_gas[:units] = ['', target_display_unit]
output_data_end_use_gas[:data] = []
output_data_end_use_gas[:chart_type] = 'simple_pie'
output_data_end_use_gas[:chart] = []
output_data_end_use_gas[:chart_type] = 'simple_pie'
output_data_end_use_gas[:chart] = []
end_use_colors = ['#EF1C21', '#0071BD', '#F7DF10', '#DEC310', '#4A4D4A', '#B5B2B5', '#FF79AD', '#632C94', '#F75921', '#293094', '#CE5921', '#FFB239', '#29AAE7', '#8CC739']
# loop through fuels for consumption tables
counter = 0
OpenStudio::EndUseCategoryType.getValues.each do |end_use|
# get end uses
end_use = OpenStudio::EndUseCategoryType.new(end_use).valueDescription
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='AnnualBuildingUtilityPerformanceSummary' and TableName='End Uses' and RowName= '#{end_use}' and ColumnName= 'Natural Gas'"
results = sqlFile.execAndReturnFirstDouble(query)
value = OpenStudio.convert(results.get, 'GJ', target_units).get
value_neat = OpenStudio.toNeatString(value, 0, true)
output_data_end_use_gas[:data] << [end_use, value_neat]
runner.registerValue("end_use_natural_gas_#{end_use.downcase.tr(' ', '_')}", value, target_units)
if value > 0
output_data_end_use_gas[:chart] << JSON.generate(label: end_use, value: value, color: end_use_colors[counter])
end
counter += 1
end
return output_data_end_use_gas
end
# create table with general building information
# this just makes a table, and not a full section. It feeds into another method that makes a full section
def self.output_data_energy_use_table(model, sqlFile, runner, is_ip_units = true)
# energy use data output
output_data_energy_use = {}
output_data_energy_use[:title] = 'Energy Use'
output_data_energy_use[:header] = ['Fuel', 'Consumption']
if is_ip_units
target_units = 'kBtu'
else
target_units = 'kWh'
end
output_data_energy_use[:units] = ['', target_units]
output_data_energy_use[:data] = []
output_data_energy_use[:chart_type] = 'simple_pie'
output_data_energy_use[:chart] = []
# list of colors for fuel. Also used for cash flow chart
color = []
color << '#DDCC77' # Electricity
color << '#999933' # Natural Gas
# not used for 9.4
# color << '#AA4499' # Additional Fuel
# TODO: - new colors to add for 9.4 (for nwo using color of Additional Fuel)
color << '#AA4499'
color << '#AA4499'
color << '#AA4499'
color << '#AA4499'
color << '#AA4499'
color << '#AA4499'
color << '#AA4499'
color << '#88CCEE' # District Cooling
color << '#CC6677' # District Heating
# color << "#332288" # Water (not used here but is in cash flow chart)
# color << "#117733" # Capital and O&M (not used here but is in cash flow chart)
# loop through fuels for consumption tables
counter = 0
fuel_type_names.each do |fuel_type| # OpenStudio::EndUseFuelType.getValues
# get fuel type and units
# fuel_type = OpenStudio::EndUseFuelType.new(fuel_type).valueDescription
next if fuel_type == 'Water'
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='AnnualBuildingUtilityPerformanceSummary' and TableName='End Uses' and RowName= 'Total End Uses' and ColumnName= '#{fuel_type}'"
results = sqlFile.execAndReturnFirstDouble(query)
# julien
source_units = 'kWh'
if is_ip_units
target_units = 'kBtu'
else
target_units = source_units
end
target_units = 'kBtu'
value = OpenStudio.convert(results.get, 'GJ', target_units).get
value_neat = OpenStudio.toNeatString(value, 0, true)
output_data_energy_use[:data] << [fuel_type, value_neat]
runner.registerValue("fuel_#{fuel_type.downcase.tr(' ', '_')}", value, target_units)
if value > 0
output_data_energy_use[:chart] << JSON.generate(label: fuel_type, value: value, color: color[counter])
end
counter += 1
end
return output_data_energy_use
end
# create table for advisory messages
def self.setpoint_not_met_summary_table(model, sqlFile, runner, is_ip_units = true)
# unmet hours data output
setpoint_not_met_summary = {}
setpoint_not_met_summary[:title] = 'Unmet Hours Summary'
setpoint_not_met_summary[:header] = ['Time Setpoint Not Met', 'Time']
target_units = 'hr'
setpoint_not_met_summary[:units] = ['', target_units]
setpoint_not_met_summary[:data] = []
# create string for rows (transposing from what is in tabular data)
setpoint_not_met_cat = []
setpoint_not_met_cat << 'During Heating'
setpoint_not_met_cat << 'During Cooling'
setpoint_not_met_cat << 'During Occupied Heating'
setpoint_not_met_cat << 'During Occupied Cooling'
# loop through messages
setpoint_not_met_cat.each do |cat|
# Retrieve end use percentages from table
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='SystemSummary' and TableName = 'Time Setpoint Not Met' and RowName= 'Facility' and ColumnName='#{cat}';"
setpoint_not_met_cat_value = sqlFile.execAndReturnFirstDouble(query)
if setpoint_not_met_cat_value.empty?
runner.registerWarning("Did not find value for #{cat}.")
return false
else
# net site energy
display = cat
source_units = 'hr'
value = setpoint_not_met_cat_value.get
value_neat = value # OpenStudio::toNeatString(value,0,true)
setpoint_not_met_summary[:data] << [display, value_neat]
runner.registerValue("unmet_hours_#{OsLib_Reporting.reg_val_string_prep(display)}", value, target_units)
end
end # setpoint_not_met_cat.each do
return setpoint_not_met_summary
end
# create table for setpoint_not_met_criteria
def self.setpoint_not_met_criteria_table(model, sqlFile, runner, is_ip_units = true)
# unmet hours data output
tolerance_summary = {}
tolerance_summary[:title] = 'Unmet Hours Tolerance'
tolerance_summary[:header] = ['Tolerance for Time Setpoint Not Met', 'Temperature']
# julien
source_units = 'C'
if is_ip_units
target_units = 'F'
else
target_units = source_units
end
tolerance_summary[:units] = ['', target_units]
tolerance_summary[:data] = []
# create string for rows (transposing from what is in tabular data)
setpoint_not_met_cat = []
setpoint_not_met_cat << 'Heating'
setpoint_not_met_cat << 'Cooling'
# loop through messages
setpoint_not_met_cat.each do |cat|
# Retrieve end use percentages from table
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='AnnualBuildingUtilityPerformanceSummary' and TableName = 'Setpoint Not Met Criteria' and RowName= 'Tolerance for Zone #{cat} Setpoint Not Met Time' and ColumnName='Degrees';"
setpoint_not_met_cat_value = sqlFile.execAndReturnFirstDouble(query)
if setpoint_not_met_cat_value.empty?
runner.registerWarning("Did not find value for #{cat}.")
return false
else
# net site energy
display = cat
# julien
source_units = 'K'
if is_ip_units
target_units = 'R'
else
target_units = source_units
end
value = OpenStudio.convert(setpoint_not_met_cat_value.get.to_f, source_units, target_units).get
value_neat = value.round(2)
tolerance_summary[:data] << [display, value_neat]
runner.registerValue("unmet_hours_tolerance_#{cat.downcase}", value, target_units)
end
end # setpoint_not_met_cat.each do
return tolerance_summary
end
# Generalized method for getting component performance characteristics
def self.general_component_summary_logic(component, is_ip_units)
data_arrays = []
# Convert to HVAC Component
component = component.to_HVACComponent
return data_arrays if component.empty?
component = component.get
# Skip object types that are not informative
types_to_skip = ['SetpointManager:MixedAir', 'Node']
idd_obj_type = component.iddObject.name.gsub('OS:', '')
return data_arrays if types_to_skip.include?(idd_obj_type)
# Only report the component type once
comp_name_used = false
# Airflow, heating and cooling capacity, and water flow
summary_types = []
summary_types << ['Air Flow Rate', 'maxAirFlowRate', 'm^3/s', 4, 'cfm', 0]
summary_types << ['Heating Capacity', 'maxHeatingCapacity', 'W', 1, 'Btu/hr', 1]
summary_types << ['Cooling Capacity', 'maxCoolingCapacity', 'W', 1, 'ton', 1]
summary_types << ['Water Flow Rate', 'maxWaterFlowRate', 'm^3/s', 4, 'gal/min', 2]
summary_types << ['Rated Power', 'ratedPower', 'W', 1, 'W', 1]
summary_types.each do |s|
val_name = s[0]
val_method = s[1]
units_si = s[2]
decimal_places_si = s[3]
units_ip = s[4]
decimal_places_ip = s[5]
# Get the value and skip if not available
val_si = component.public_send(val_method)
next if val_si.empty?
# Determine if the value was autosized or hard sized
siz = 'Hard Sized'
if component.public_send("#{val_method}Autosized").is_initialized
if component.public_send("#{val_method}Autosized").get
siz = 'Autosized'
end
end
# Convert and report the value
source_units = units_si
if is_ip_units
target_units = units_ip
decimal_places = decimal_places_ip
else
target_units = source_units
decimal_places = decimal_places_si
end
val_ip = OpenStudio.convert(val_si.get, source_units, target_units).get
val_ip_neat = OpenStudio.toNeatString(val_ip, decimal_places, true)
if is_ip_units
data_arrays << ['', val_name, "#{val_ip_neat} #{units_ip}", siz, '']
else
data_arrays << ['', val_name, "#{val_ip_neat} #{units_si}", siz, '']
end
end
# Performance characteristics (specific to each component type)
perf_chars = component.performanceCharacteristics.each do |char|
perf_val = char[0].to_s.to_f
perf_name = char[1]
display_units = '' # For Display
# Unit conversion for pressure rise and pump head
if perf_name.downcase.include?('pressure rise')
source_units = 'Pa'
if is_ip_units
target_units = 'inH_{2}O'
display_units = 'in w.g.'
else
target_units = source_units
display_units = source_units
end
perf_val = OpenStudio.convert(perf_val, source_units, target_units).get
perf_val = OpenStudio.toNeatString(perf_val, 2, true)
elsif perf_name.downcase.include?('pump head')
source_units = 'Pa'
if is_ip_units
target_units = 'ftH_{2}O'
display_units = 'ft H2O'
n_decimals = 1
else
target_units = source_units
display_units = source_units
n_decimals = 0
end
perf_val = OpenStudio.convert(perf_val, source_units, target_units).get
perf_val = OpenStudio.toNeatString(perf_val, n_decimals, true)
elsif perf_name.downcase.include?('efficiency') || perf_name.downcase.include?('effectiveness')
display_units = '%'
perf_val *= 100
perf_val = OpenStudio.toNeatString(perf_val, 1, true)
end
# Report the value
data_arrays << ['', perf_name, "#{perf_val} #{display_units}", '', '']
end
return data_arrays
end
# Gives the Plant Loop connection information for an HVAC Component
def self.water_component_logic(component, is_ip_units)
data_arrays = []
component = component.to_HVACComponent
return data_arrays if component.empty?
component = component.get
# Only deal with plant-connected components
return data_arrays unless component.respond_to?('plantLoop')
# Report the plant loop name
if component.plantLoop.is_initialized
data_arrays << ['', 'Plant Loop', component.plantLoop.get.name, '', '']
end
return data_arrays
end
# Shows the calculated brake horsepower for fans and pumps
def self.motor_component_logic(component, is_ip_units)
data_arrays = []
# Skip exhaust fans for now because of bug in openstudio-standards
return data_arrays if component.to_FanZoneExhaust.is_initialized
concrete_comp = component.cast_to_concrete_type
component = concrete_comp unless component.nil?
# Only deal with plant-connected components
return data_arrays unless component.respond_to?('brake_horsepower')
# Report the plant loop name
bhp = component.brake_horsepower
bhp_neat = OpenStudio.toNeatString(bhp, 2, true)
data_arrays << ['', 'Brake Horsepower', "#{bhp_neat} HP", '', '']
return data_arrays
end
# Shows the setpoint manager details depending on type
def self.spm_logic(component, is_ip_units)
data_arrays = []
case component.iddObject.name
when 'OS:SetpointManager:Scheduled'
# Constrol type and temperature range
setpoint = component.to_SetpointManagerScheduled.get
supply_air_temp_schedule = setpoint.schedule
schedule_values = OsLib_Schedules.getMinMaxAnnualProfileValue(component.model, supply_air_temp_schedule)
if schedule_values.nil?
schedule_values_pretty = "can't inspect schedule"
target_units = ''
else
if setpoint.controlVariable.to_s == 'Temperature'
# julien
source_units = 'C'
if is_ip_units
target_units = 'F'
else
target_units = source_units
end
schedule_values_pretty = "#{OpenStudio.convert(schedule_values['min'], source_units, target_units).get.round(1)} to #{OpenStudio.convert(schedule_values['max'], source_units, target_units).get.round(1)}"
else # TODO: - add support for other control variables
schedule_values_pretty = "#{schedule_values['min']} to #{schedule_values['max']}"
target_units = 'raw si values'
end
end
data_arrays << ['', "Control Variable - #{setpoint.controlVariable}", "#{schedule_values_pretty} #{target_units}", '', '']
when 'OS:SetpointManager:SingleZone:Reheat'
# Control Zone
setpoint = component.to_SetpointManagerSingleZoneReheat.get
control_zone = setpoint.controlZone
if control_zone.is_initialized
control_zone_name = control_zone.get.name
else
control_zone_name = ''
end
data_arrays << ['', 'Control Zone', control_zone_name, '', '']
when 'OS:SetpointManager:FollowOutdoorAirTemperature'
setpoint = component.to_SetpointManagerFollowOutdoorAirTemperature.get
ref_temp_type = setpoint.referenceTemperatureType
data_arrays << [setpoint.iddObject.name, 'Reference Temperature Type', ref_temp_type, '', '']
when 'OS:SetpointManager:OutdoorAirReset'
setpoint = component.to_SetpointManagerOutdoorAirReset.get
source_units = 'C'
if is_ip_units
target_units = 'F'
else
target_units = source_units
end
wt_at_hi_oat_f = OpenStudio.convert(setpoint.setpointatOutdoorHighTemperature, source_units, target_units).get.round(1)
wt_at_lo_oat_f = OpenStudio.convert(setpoint.setpointatOutdoorLowTemperature, source_units, target_units).get.round(1)
hi_oat_f = OpenStudio.convert(setpoint.outdoorHighTemperature, source_units, target_units).get.round(1)
lo_oat_f = OpenStudio.convert(setpoint.outdoorLowTemperature, source_units, target_units).get.round(1)
# julien
if is_ip_units
desc = "#{wt_at_lo_oat_f} F to #{wt_at_hi_oat_f.round} F btwn OAT #{lo_oat_f} F to #{hi_oat_f} F."
else
desc = "#{wt_at_lo_oat_f} C to #{wt_at_hi_oat_f.round} C btwn OAT #{lo_oat_f} C to #{hi_oat_f} C."
end
data_arrays << [setpoint.iddObject.name, 'Reset', desc, '', '']
when 'OS:SetpointManager:Warmest'
setpoint = component.to_SetpointManagerWarmest.get
source_units = 'C'
if is_ip_units
target_units = 'F'
else
target_units = source_units
end
min_sat_f = OpenStudio.convert(setpoint.minimumSetpointTemperature, source_units, target_units).get.round(1)
max_sat_f = OpenStudio.convert(setpoint.maximumSetpointTemperature, source_units, target_units).get.round(1)
desc = "#{min_sat_f} #{target_units} to #{max_sat_f.round} #{target_units}"
data_arrays << [setpoint.iddObject.name, 'Reset SAT per Worst Zone', desc, '', '']
when 'OS:SetpointManager:WarmestTemperatureFlow'
setpoint = component.to_SetpointManagerWarmestTemperatureFlow.get
source_units = 'C'
if is_ip_units
target_units = 'F'
else
target_units = source_units
end
min_sat_f = OpenStudio.convert(setpoint.minimumSetpointTemperature, source_units, target_units).get.round(1)
max_sat_f = OpenStudio.convert(setpoint.maximumSetpointTemperature, source_units, target_units).get.round(1)
desc = "#{min_sat_f} #{target_units} to #{max_sat_f.round} #{target_units}, #{setpoint.strategy}"
data_arrays << [setpoint.iddObject.name, 'Reset SAT & Flow per Worst Zone', desc, '', '']
end
return data_arrays
end
# summary of what to show for each type of air loop component
def self.air_loop_component_summary_logic(component, model, is_ip_units)
# Generic component logic first
data_arrays = general_component_summary_logic(component, is_ip_units)
# Water component logic
data_arrays += water_component_logic(component, is_ip_units)
# Motor component logic
data_arrays += motor_component_logic(component, is_ip_units)
# Setpoint manager logic
data_arrays += spm_logic(component, is_ip_units)
# Unique logic for subset of components
case component.iddObject.name
when 'OS:AirLoopHVAC:OutdoorAirSystem'
component = component.to_AirLoopHVACOutdoorAirSystem.get
controller_oa = component.getControllerOutdoorAir
# Min OA
# julien
source_units = 'm^3/s'
if is_ip_units
target_units = 'cfm'
n_decimals = 0
else
target_units = 'm^3/h'
n_decimals = 0
end
if controller_oa.minimumOutdoorAirFlowRate.is_initialized
value = OpenStudio.convert(controller_oa.minimumOutdoorAirFlowRate.get, source_units, target_units).get
value_neat = OpenStudio.toNeatString(value, n_decimals, true)
siz = 'Hard Sized'
elsif controller_oa.autosizedMinimumOutdoorAirFlowRate.is_initialized
value = OpenStudio.convert(controller_oa.autosizedMinimumOutdoorAirFlowRate.get, source_units, target_units).get
value_neat = OpenStudio.toNeatString(value, n_decimals, true)
siz = 'Autosized'
else
value_neat = 'Autosized'
end
data_arrays << ['', 'Minimum Outdoor Air Flow Rate', "#{value_neat} #{target_units}", siz, '']
# Max OA
if controller_oa.maximumOutdoorAirFlowRate.is_initialized
value_ip = OpenStudio.convert(controller_oa.maximumOutdoorAirFlowRate.get, source_units, target_units).get
value_ip_neat = OpenStudio.toNeatString(value_ip, n_decimals, true)
siz = 'Hard Sized'
elsif controller_oa.autosizedMaximumOutdoorAirFlowRate.is_initialized
value_ip = OpenStudio.convert(controller_oa.autosizedMaximumOutdoorAirFlowRate.get, source_units, target_units).get
value_ip_neat = OpenStudio.toNeatString(value_ip, n_decimals, true)
siz = 'Autosized'
else
value_ip_neat = 'Autosized'
end
data_arrays << ['', 'Maximum Outdoor Air Flow Rate', "#{value_ip_neat} #{target_units}", siz, '']
end
# Make the component type the first element of the first row
if !data_arrays.empty?
data_arrays[0][0] = component.iddObject.name.gsub('OS:', '')
end
return data_arrays
end
# create table air loop summary
def self.air_loops_detail_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
output_data_air_loop_tables = []
# gather data for section
@output_data_air_loop_section = {}
@output_data_air_loop_section[:title] = 'Air Loops Detail'
@output_data_air_loop_section[:tables] = output_data_air_loop_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @output_data_air_loop_section
end
# loop through air loops
model.getAirLoopHVACs.sort.each do |air_loop|
# air loop data output
output_data_air_loops = {}
output_data_air_loops[:title] = air_loop.name.get # TODO: - confirm first that it has name
output_data_air_loops[:header] = ['Object', 'Description', 'Value', 'Sizing', 'Count']
output_data_air_loops[:units] = [] # not using units for these tables
output_data_air_loops[:data] = []
output_data_air_loops[:data] << [{ sub_header: 'supply' }, '', '', '', '']
# hold values for later use
dcv_setting = 'na' # should hit this if there isn't an outdoor air object on the loop
economizer_setting = 'na' # should hit this if there isn't an outdoor air object on the loop
# loop through components
air_loop.supplyComponents.each do |component|
# skip some object types, but look for node with setpoint manager
if component.to_Node.is_initialized
setpoint_managers = component.to_Node.get.setpointManagers
if !setpoint_managers.empty?
# setpoint type
setpoint = setpoint_managers[0] # TODO: - could have more than one in some situations
data_arrays = OsLib_Reporting.air_loop_component_summary_logic(setpoint, model, is_ip_units)
data_arrays.each do |data_array|
output_data_air_loops[:data] << data_array
end
end
else
# populate table for everything but setpoint managers, which are added above.
data_arrays = OsLib_Reporting.air_loop_component_summary_logic(component, model, is_ip_units)
data_arrays.each do |data_array|
output_data_air_loops[:data] << data_array
end
end
# gather controls information to use later
if component.to_AirLoopHVACOutdoorAirSystem.is_initialized
hVACComponent = component.to_AirLoopHVACOutdoorAirSystem.get
# get ControllerOutdoorAir
controller_oa = hVACComponent.getControllerOutdoorAir
# get ControllerMechanicalVentilation
controller_mv = controller_oa.controllerMechanicalVentilation
# get dcv value
dcv_setting = controller_mv.demandControlledVentilation
if dcv_setting
dcv_setting = 'On'
else
dcv_setting = 'Off'
end
# get economizer setting
economizer_setting = controller_oa.getEconomizerControlType
end
end
output_data_air_loops[:data] << [{ sub_header: 'demand' }, '', '', '', '']
# demand side summary, list of terminal types used, and number of zones
thermal_zones = []
terminals = []
cooling_temp_ranges = []
heating_temps_ranges = []
air_loop.demandComponents.each do |component|
# gather array of thermal zones and terminals
if component.to_ThermalZone.is_initialized
thermal_zone = component.to_ThermalZone.get
thermal_zones << thermal_zone
thermal_zone.equipment.each do |zone_equip|
next if zone_equip.to_ZoneHVACComponent.is_initialized # should only find terminals
terminals << zone_equip.iddObject.name.gsub('OS:', '')
end
# populate thermostat ranges
if thermal_zone.thermostatSetpointDualSetpoint.is_initialized
thermostat = thermal_zone.thermostatSetpointDualSetpoint.get
if thermostat.coolingSetpointTemperatureSchedule.is_initialized
schedule_values = OsLib_Schedules.getMinMaxAnnualProfileValue(model, thermostat.coolingSetpointTemperatureSchedule.get)
unless schedule_values.nil?
cooling_temp_ranges << schedule_values['min']
cooling_temp_ranges << schedule_values['max']
end
end
if thermostat.heatingSetpointTemperatureSchedule.is_initialized
schedule_values = OsLib_Schedules.getMinMaxAnnualProfileValue(model, thermostat.heatingSetpointTemperatureSchedule.get)
unless schedule_values.nil?
heating_temps_ranges << schedule_values['min']
heating_temps_ranges << schedule_values['max']
end
end
end
end
end
# get floor area of thermal zones
total_loop_floor_area = 0
thermal_zones.each do |zone|
total_loop_floor_area += zone.floorArea
end
source_units = 'm^2'
if is_ip_units
target_units = 'ft^2'
else
target_units = source_units
end
total_loop_floor_area = OpenStudio.convert(total_loop_floor_area, source_units, target_units).get
total_loop_floor_area_neat = OpenStudio.toNeatString(total_loop_floor_area, 0, true)
# output zone and terminal data
if is_ip_units
output_data_air_loops[:data] << ['Thermal Zones', 'Total Floor Area', "#{total_loop_floor_area_neat} ft^2", '', thermal_zones.size]
else
output_data_air_loops[:data] << ['Thermal Zones', 'Total Floor Area', "#{total_loop_floor_area_neat} m^2", '', thermal_zones.size]
end
# heating and cooling temperature range data
source_units = 'C'
if is_ip_units
target_units = 'F'
target_units_display = 'F'
else
target_units = source_units
target_units_display = 'C'
end
if cooling_temp_ranges.empty?
cooling_temp_ranges_pretty = "can't inspect schedules"
else
cooling_temp_ranges_pretty = "#{OpenStudio.convert(cooling_temp_ranges.min, source_units, target_units).get.round(1)} to #{OpenStudio.convert(cooling_temp_ranges.max, source_units, target_units).get.round(1)}"
end
if heating_temps_ranges.empty?
heating_temps_ranges_pretty = "can't inspect schedules"
else
heating_temps_ranges_pretty = "#{OpenStudio.convert(heating_temps_ranges.min, source_units, target_units).get.round(1)} to #{OpenStudio.convert(heating_temps_ranges.max, source_units, target_units).get.round(1)}"
end
output_data_air_loops[:data] << ['Thermal Zones', 'Cooling Setpoint Range', "#{cooling_temp_ranges_pretty} #{target_units_display}", '', '']
output_data_air_loops[:data] << ['Thermal Zones', 'Heating Setpoint Range', "#{heating_temps_ranges_pretty} #{target_units_display}", '', '']
output_data_air_loops[:data] << ['Terminal Types Used', terminals.uniq.sort.join(', '), '', '', terminals.size]
# controls summary
# julien
source_units = 'C'
if is_ip_units
target_units = 'F'
target_units_display = 'F'
else
target_units = source_units
target_units_display = 'C'
end
output_data_air_loops[:data] << [{ sub_header: 'controls' }, '', '', '', '']
output_data_air_loops[:data] << ['HVAC Operation Schedule', '', air_loop.availabilitySchedule.name, '', ''] # I think this is a bool
output_data_air_loops[:data] << ['Night Cycle Setting', '', air_loop.nightCycleControlType, '', '']
output_data_air_loops[:data] << ['Economizer Setting', '', economizer_setting, '', '']
output_data_air_loops[:data] << ['Demand Controlled Ventilation Status', '', dcv_setting, '', '']
htg_sat_si = air_loop.sizingSystem.centralHeatingDesignSupplyAirTemperature
htg_sat_ip = OpenStudio.toNeatString(OpenStudio.convert(htg_sat_si, source_units, target_units).get, 1, true)
output_data_air_loops[:data] << ['Central Heating Design Supply Air Temperature', '', "#{htg_sat_ip} #{target_units_display}", '', '']
clg_sat_si = air_loop.sizingSystem.centralCoolingDesignSupplyAirTemperature
clg_sat_ip = OpenStudio.toNeatString(OpenStudio.convert(clg_sat_si, source_units, target_units).get, 1, true)
output_data_air_loops[:data] << ['Central Cooling Design Supply Air Temperature', '', "#{clg_sat_ip} #{target_units_display}", '', '']
output_data_air_loops[:data] << ['Load to Size On', '', air_loop.sizingSystem.typeofLoadtoSizeOn, '', '']
# populate tables for section
output_data_air_loop_tables << output_data_air_loops
end
return @output_data_air_loop_section
end
# summary of what to show for each type of plant loop component
def self.plant_loop_component_summary_logic(component, model, is_ip_units)
# Generic component logic first
data_arrays = general_component_summary_logic(component, is_ip_units)
# Motor component logic
data_arrays += motor_component_logic(component, is_ip_units)
# Setpoint manager logic
data_arrays += spm_logic(component, is_ip_units)
# Make the component type the first element of the first row
if !data_arrays.empty?
data_arrays[0][0] = component.iddObject.name.gsub('OS:', '')
end
return data_arrays
end
# create table plant loop summary
def self.plant_loops_detail_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
output_data_plant_loop_tables = []
# gather data for section
@output_data_plant_loop_section = {}
@output_data_plant_loop_section[:title] = 'Plant Loops Detail'
@output_data_plant_loop_section[:tables] = output_data_plant_loop_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @output_data_plant_loop_section
end
# loop through plant loops
model.getPlantLoops.sort.each do |plant_loop|
# plant loop data output
output_data_plant_loops = {}
output_data_plant_loops[:title] = plant_loop.name.get # TODO: - confirm first that it has name
output_data_plant_loops[:header] = ['Object', 'Description', 'Value', 'Sizing', 'Count']
output_data_plant_loops[:units] = [] # not using units for these tables
output_data_plant_loops[:data] = []
output_data_plant_loops[:data] << [{ sub_header: 'supply' }, '', '', '', '']
plant_loop.supplyComponents.each do |component|
if component.to_ThermalZone.is_initialized
end
# skip some object types
next if component.to_PipeAdiabatic.is_initialized
next if component.to_Splitter.is_initialized
next if component.to_Mixer.is_initialized
if component.to_Node.is_initialized
setpoint_managers = component.to_Node.get.setpointManagers
if !setpoint_managers.empty?
# setpoint type
setpoint = setpoint_managers[0] # TODO: - could have more than one in some situations
data_arrays = OsLib_Reporting.plant_loop_component_summary_logic(setpoint, model, is_ip_units)
data_arrays.each do |data_array| # typically just one, but in some cases there are a few
output_data_plant_loops[:data] << data_array
end
end
else
# populate table for everything but setpoint managers, which are added above.
data_arrays = OsLib_Reporting.plant_loop_component_summary_logic(component, model, is_ip_units)
data_arrays.each do |data_array| # typically just one, but in some cases there are a few
output_data_plant_loops[:data] << data_array
end
end
end
# loop through demand components
output_data_plant_loops[:data] << [{ sub_header: 'demand' }, '', '', '', '']
# keep track of terminal count to report later
terminal_connections = [] # Not sure how I want to list in display
# loop through plant demand components
plant_loop.demandComponents.each do |component|
# flag for terminal connecxtions
terminal_connection = false
# skip some object types
next if component.to_PipeAdiabatic.is_initialized
next if component.to_Splitter.is_initialized
next if component.to_Mixer.is_initialized
next if component.to_Node.is_initialized
# determine if water to air
if component.to_WaterToAirComponent.is_initialized
component = component.to_WaterToAirComponent.get
if component.airLoopHVAC.is_initialized
description = 'Air Loop'
value = component.airLoopHVAC.get.name
else
# this is a terminal connection
terminal_connection = true
terminal_connections << component
end
elsif component.to_WaterToWaterComponent.is_initialized
description = 'Plant Loop'
component = component.to_WaterToWaterComponent.get
ww_loop = component.plantLoop
if ww_loop.is_initialized
value = ww_loop.get.name
else
value = ''
end
else # water use connections would go here
description = component.name
value = ''
end
# don't report here if this component is connected to a terminal
next if terminal_connection == true
output_data_plant_loops[:data] << [component.iddObject.name.gsub('OS:', ''), description, value, '', '']
end
# report terminal connections
if !terminal_connections.empty?
output_data_plant_loops[:data] << ['Air Terminal Connections', '', '', '', terminal_connections.size]
end
output_data_plant_loops[:data] << [{ sub_header: 'controls' }, '', '', '', '']
# Min loop flow rate
source_units = 'm^3/s'
if is_ip_units
target_units = 'gal/min'
n_decimals = 2
else
target_units = 'm^3/h'
n_decimals = 0
end
if plant_loop.minimumLoopFlowRate.is_initialized
value = OpenStudio.convert(plant_loop.minimumLoopFlowRate.get, source_units, target_units).get
value_neat = OpenStudio.toNeatString(value, n_decimals, true)
siz = 'Hard Sized'
elsif plant_loop.autosizedMinimumLoopFlowRate.is_initialized
value = OpenStudio.convert(plant_loop.autosizedMinimumLoopFlowRate.get, source_units, target_units).get
value_neat = OpenStudio.toNeatString(value, n_decimals, true)
siz = 'Autosized'
else
value_neat = 'Autosized'
end
output_data_plant_loops[:data] << ['Loop Flow Rate Range', 'Minimum Loop Flow Rate', "#{value_neat} #{target_units}", siz, '']
# Max loop flow rate
if plant_loop.maximumLoopFlowRate.is_initialized
value = OpenStudio.convert(plant_loop.maximumLoopFlowRate.get, source_units, target_units).get
value_neat = OpenStudio.toNeatString(value, n_decimals, true)
siz = 'Hard Sized'
elsif
plant_loop.autosizedMaximumLoopFlowRate.is_initialized
value = OpenStudio.convert(plant_loop.autosizedMaximumLoopFlowRate.get, source_units, target_units).get
value_neat = OpenStudio.toNeatString(value, n_decimals, true)
siz = 'Autosized'
else
value_neat = 'Autosized'
end
output_data_plant_loops[:data] << ['Loop Flow Rate Range', 'Maximum Loop Flow Rate', "#{value_neat} #{target_units}", siz, '']
# loop temperatures
source_units = 'C'
if is_ip_units
target_units = 'F'
else
target_units = 'C'
end
min_temp = plant_loop.minimumLoopTemperature
max_temp = plant_loop.maximumLoopTemperature
value_neat = "#{OpenStudio.convert(min_temp, source_units, target_units).get.round(1)} to #{OpenStudio.convert(max_temp, source_units, target_units).get.round(1)}"
output_data_plant_loops[:data] << ['Loop Temperature Range', '', "#{value_neat} #{target_units}", '', '']
# get values out of sizing plant
sizing_plant = plant_loop.sizingPlant
source_units = 'C'
if is_ip_units
target_units = 'F'
else
target_units = 'C'
end
loop_exit_temp = sizing_plant.designLoopExitTemperature
value_neat = OpenStudio.toNeatString(OpenStudio.convert(loop_exit_temp, source_units, target_units).get, 1, true)
output_data_plant_loops[:data] << ['Loop Design Exit Temperature', '', "#{value_neat} #{target_units}", '', '']
source_units = 'K'
if is_ip_units
target_units = 'R'
else
target_units = 'K'
end
loop_design_temp_diff = sizing_plant.loopDesignTemperatureDifference
value_neat = OpenStudio.toNeatString(OpenStudio.convert(loop_design_temp_diff, source_units, target_units).get, 1, true)
output_data_plant_loops[:data] << ['Loop Design Temperature Difference', '', "#{value_neat} #{target_units}", '', '']
# Equipment staging
output_data_plant_loops[:data] << ['Equipment Loading/Staging', '', plant_loop.loadDistributionScheme, '', '']
# push tables
output_data_plant_loop_tables << output_data_plant_loops
end
return @output_data_plant_loop_section
end
# summary of what to show for each type of zone equipment component
def self.zone_equipment_component_summary_logic(component, model, is_ip_units)
# Generic component logic first
data_arrays = general_component_summary_logic(component, is_ip_units)
# Motor component logic
data_arrays += motor_component_logic(component, is_ip_units)
return data_arrays
end
# create table plant loop summary
def self.zone_equipment_detail_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
output_data_zone_equipment = []
# gather data for section
@output_data_zone_equipment_section = {}
@output_data_zone_equipment_section[:title] = 'Zone Equipment Detail'
@output_data_zone_equipment_section[:tables] = output_data_zone_equipment
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @output_data_zone_equipment_section
end
# loop through thermal zones
model.getThermalZones.sort.each do |zone|
# plant loop data output
output_data_zone_equipment = {}
output_data_zone_equipment[:title] = zone.name.get # TODO: - confirm that zone has a name
output_data_zone_equipment[:header] = ['Object', 'Description', 'Value', 'Sizing', 'Count']
output_data_zone_equipment[:units] = [] # not using units for these tables
output_data_zone_equipment[:data] = []
zone.equipment.sort.each do |zone_equip|
next unless zone_equip.to_ZoneHVACComponent.is_initialized # skip any terminals
data_arrays = OsLib_Reporting.zone_equipment_component_summary_logic(zone_equip, model, is_ip_units)
data_arrays.each do |data_array| # typically just one, but in some cases there are a few
output_data_zone_equipment[:data] << data_array
end
# Make the component type the first element of the first row
if !data_arrays.empty?
data_arrays[0][0] = zone_equip.iddObject.name.gsub('OS:', '')
end
end
# push table to array
if !output_data_zone_equipment[:data].empty?
@output_data_zone_equipment_section[:tables] << output_data_zone_equipment
end
end
return @output_data_zone_equipment_section
end
# create table for constructions
def self.envelope_section_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# Versions of OpenStudio greater than 2.4.0 use a modified version of
# openstudio-standards with different method calls. These methods
# require a "Standard" object instead of the standard being passed into method calls.
# This Standard object is used throughout the QAQC check.
if OpenStudio::VersionString.new(OpenStudio.openStudioVersion) < OpenStudio::VersionString.new('2.4.2')
use_old_gem_code = true
else
use_old_gem_code = false
std = Standard.build('90.1-2013') # actual standard doesn't matter in this case
end
# array to hold tables
envelope_tables = []
# gather data for section
@envelope_section = {}
@envelope_section[:title] = 'Envelope Summary'
@envelope_section[:tables] = envelope_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @envelope_section
end
# summary of exterior constructions used in the model for base surfaces
surface_data = {}
surface_data[:title] = 'Base Surface Constructions'
surface_data[:header] = ['Construction', 'Net Area', 'Surface Count', 'R Value']
source_area_units = 'm^2'
source_rvalue_units = 'm^2*K/W'
source_ufactor_units = 'W/m^2*K'
if is_ip_units
target_area_units = 'ft^2'
target_rvalue_units = 'ft^2*h*R/Btu'
target_ufactor_units = 'Btu/ft^2*h*R'
n_decimals_area = 0
n_decimals_rvalue = 2
else
target_area_units = source_area_units
target_rvalue_units = source_rvalue_units
target_ufactor_units = source_ufactor_units
n_decimals_area = 1
n_decimals_rvalue = 2
end
surface_data[:units] = ['', target_area_units, '', target_rvalue_units]
surface_data[:data] = []
# construction details
# TODO
# add table with subheads for each construction, and rows for each material layer
# construction_details = {}
# construction_details[:title] = 'Construction Details'
# construction_details[:header] = ['Material']
# construction_details[:data] = []
# loop through surfaces to get constructions
ext_const_base = {}
model.getSurfaces.each do |surface|
next if surface.outsideBoundaryCondition != 'Outdoors'
if ext_const_base.include? surface.construction.get
ext_const_base[surface.construction.get] += 1
else
ext_const_base[surface.construction.get] = 1
end
end
ext_const_base.sort.each do |construction, count|
net_area = construction.getNetArea
net_area_conv = OpenStudio.convert(net_area, source_area_units, target_area_units).get
net_area_neat = OpenStudio.toNeatString(net_area_conv, n_decimals_area, true)
surface_count = count
if construction.thermalConductance.is_initialized
thermal_conductance = construction.thermalConductance.get
r_value = OpenStudio.convert(1 / thermal_conductance, source_rvalue_units, target_rvalue_units).get
r_value_neat = OpenStudio.toNeatString(r_value, n_decimals_rvalue, true)
else
r_value_neat = ''
end
surface_data[:data] << [construction.name, net_area_neat, surface_count, r_value_neat]
runner.registerValue(OsLib_Reporting.reg_val_string_prep(construction.name.to_s), net_area_conv, target_area_units)
# TODO: Get this working like subsurfaces
# construction_details[:data] << [{ sub_header: "Material Layers in Construction '#{construction.name}':"}]
# construction.layers.each do |l|
# construction_details[:data] << [l.name]
# end
end
envelope_tables << surface_data
# summary of exterior constructions used in the model for sub surfaces
sub_surface_data = {}
sub_surface_data[:title] = 'Sub Surface Constructions'
sub_surface_data[:header] = ['Construction', 'Net Area', 'Surface Count', 'U-factor', 'SHGC', 'VLT']
sub_surface_data[:units] = ['', target_area_units, '', target_ufactor_units]
sub_surface_data[:data] = []
construction_details = {}
construction_details[:title] = 'Sub Surface Construction Details (Material Layers)'
construction_details[:header] = ['Material Name']
construction_details[:data] = []
# loop through sub_surfaces to get constructions
ext_const_sub = {}
model.getSubSurfaces.each do |sub_surface|
next if sub_surface.outsideBoundaryCondition != 'Outdoors'
if ext_const_sub.include? sub_surface.construction.get
ext_const_sub[sub_surface.construction.get] += 1
else
ext_const_sub[sub_surface.construction.get] = 1
end
end
ext_const_sub.sort.each do |construction, count|
net_area = construction.getNetArea
net_area_conv = OpenStudio.convert(net_area, source_area_units, target_area_units).get
net_area_neat = OpenStudio.toNeatString(net_area_conv, n_decimals_area, true)
surface_count = count
vlt_neat = 'n/a'
shgc_neat = 'n/a'
u_factor_neat = ''
if construction.to_Construction.is_initialized
construction_root = construction.to_Construction.get
if construction_root.isFenestration
if use_old_gem_code
shgc = construction_root.calculated_solar_heat_gain_coefficient
else
shgc = std.construction_calculated_solar_heat_gain_coefficient(construction_root)
end
shgc_neat = OpenStudio.toNeatString(shgc, n_decimals_rvalue, false)
if use_old_gem_code
vlt = construction_root.calculated_visible_transmittance
else
vlt = std.construction_calculated_visible_transmittance(construction_root)
end
vlt_neat = OpenStudio.toNeatString(vlt, n_decimals_rvalue, false)
if use_old_gem_code
u_factor = construction_root.calculated_u_factor
else
u_factor = std.construction_calculated_u_factor(construction_root)
end
ufactor_conv = OpenStudio.convert(u_factor, source_ufactor_units, target_ufactor_units).get
ufactor_neat = OpenStudio.toNeatString(ufactor_conv, n_decimals_rvalue, false)
else
u_factor = construction.thermalConductance.get
u_factor_conv = OpenStudio.convert(u_factor, source_ufactor_units, target_ufactor_units).get
u_factor_neat = OpenStudio.toNeatString(u_factor_conv, n_decimals_rvalue, false)
end
# add layer details for each construction
construction_details[:data] << [{ sub_header: "Material Layers in Construction '#{construction.name}':" }]
construction_root.layers.each do |l|
construction_details[:data] << [l.name]
end
end
sub_surface_data[:data] << [construction.name, net_area_neat, surface_count, ufactor_neat, shgc_neat, vlt_neat]
runner.registerValue(OsLib_Reporting.reg_val_string_prep(construction.name.to_s), net_area_conv, target_area_units)
end
envelope_tables << sub_surface_data
envelope_tables << construction_details
# Conditioned Window-Wall Ratio and Skylight-Roof Ratio
fenestration_data = {}
fenestration_data[:title] = 'Window-to-Wall and Skylight-to-Roof area Ratios'
fenestration_data[:header] = ['Description', 'Total', 'North', 'East', 'South', 'West']
target_units = '%' # ratios reported as percentages
fenestration_data[:units] = ['', target_units, target_units, target_units, target_units, target_units]
fenestration_data[:data] = []
# create string for rows
fenestrations = []
fenestrations << 'Gross Window-Wall Ratio' # [%]
# loop rows
fenestrations.each do |fenestration|
query0 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='Total'"
query1 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='North (315 to 45 deg)'"
query2 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='East (45 to 135 deg)'"
query3 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='South (135 to 225 deg)'"
query4 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='West (225 to 315 deg)'"
query5 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Skylight-Roof Ratio' and RowName='Skylight-Roof Ratio'"
query6 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Conditioned Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='Total'"
query7 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Conditioned Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='North (315 to 45 deg)'"
query8 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Conditioned Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='East (45 to 135 deg)'"
query9 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Conditioned Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='South (135 to 225 deg)'"
query10 = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='InputVerificationandResultsSummary' and TableName='Conditioned Window-Wall Ratio' and RowName='#{fenestration}' and ColumnName='West (225 to 315 deg)'"
total = sqlFile.execAndReturnFirstDouble(query0)
north = sqlFile.execAndReturnFirstDouble(query1)
east = sqlFile.execAndReturnFirstDouble(query2)
south = sqlFile.execAndReturnFirstDouble(query3)
west = sqlFile.execAndReturnFirstDouble(query4)
skylight = sqlFile.execAndReturnFirstDouble(query5)
total_cond = sqlFile.execAndReturnFirstDouble(query6)
north_cond = sqlFile.execAndReturnFirstDouble(query7)
east_cond = sqlFile.execAndReturnFirstDouble(query8)
south_cond = sqlFile.execAndReturnFirstDouble(query9)
west_cond = sqlFile.execAndReturnFirstDouble(query10)
if total.empty? || north.empty? || east.empty? || south.empty? || west.empty? || total_cond.empty? || north_cond.empty? || east.empty? || south_cond.empty? || west_cond.empty? || skylight.empty?
runner.registerWarning('Did not find value for Window or Skylight Ratio')
return false
else
# add data
display = fenestration
fenestration_data[:data] << [display, total.get, north.get, east.get, south.get, west.get]
fenestration_data[:data] << ["#{display} (Conditioned)", total_cond.get, north_cond.get, east_cond.get, south_cond.get, west_cond.get]
runner.registerValue(OsLib_Reporting.reg_val_string_prep(display).to_s, total.get, target_units)
runner.registerValue("#{OsLib_Reporting.reg_val_string_prep(display)}_conditioned", total_cond.get, target_units)
# skylight
# skylight seems to provide back percentage vs. fraction. Changing to fraction to match vertical fenestration.
fenestration_data[:data] << ['Skylight-Roof Ratio', skylight.get, '', '', '', '']
runner.registerValue('skylight_roof_ratio', skylight.get, target_units)
end
end
envelope_tables << fenestration_data
return @envelope_section
end
# create table for service water heating
def self.water_use_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# water use equipment from model
water_use_data = {}
water_use_data[:title] = ''
water_use_data[:header] = ['Instance', 'Plant Loop', 'Definition', 'Space', 'Peak Flow Rate', 'Flow Rate Schedule', 'Target Temp Range']
source_units = 'm^3/s'
source_units_temp = 'C'
if is_ip_units
target_units = 'gal/min'
n_decimals = 2
target_units_temp = 'F'
else
target_units = 'm^3/h'
n_decimals = 2
target_units_temp = 'C'
end
water_use_data[:units] = ['', '', '', '', target_units, '', target_units_temp]
water_use_data[:data] = []
# gather data for section
@water_use_data_section = {}
@water_use_data_section[:title] = 'Water Use Equipment'
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @water_use_data_section
end
# loop through water use equipment
water_use_equipment = model.getWaterUseEquipments
water_use_equipment.sort.each do |instance|
water_use_equipment_def = instance.waterUseEquipmentDefinition
if instance.waterUseConnections.is_initialized && instance.waterUseConnections.get.plantLoop.is_initialized
plant_loop = instance.waterUseConnections.get.plantLoop.get.name
else
plant_loop = ''
end
if instance.flowRateFractionSchedule.is_initialized
water_use_equipment_flow_rate_sch = instance.flowRateFractionSchedule.get.name
else
water_use_equipment_flow_rate_sch = ''
end
if instance.space.is_initialized
space = instance.space.get.name
else
space = ''
end
peak_flow_rate = water_use_equipment_def.peakFlowRate
peak_flow_rate_conv = OpenStudio.convert(peak_flow_rate, source_units, target_units).get
peak_flow_rate_neat = OpenStudio.toNeatString(peak_flow_rate_conv, n_decimals, true)
if water_use_equipment_def.targetTemperatureSchedule.is_initialized
target_temp_sch = water_use_equipment_def.targetTemperatureSchedule.get
schedule_values = OsLib_Schedules.getMinMaxAnnualProfileValue(model, target_temp_sch)
if !schedule_values.nil?
min_conv = OpenStudio.convert(schedule_values['min'], source_units_temp, target_units_temp).get
max_conv = OpenStudio.convert(schedule_values['max'], source_units_temp, target_units_temp).get
target_temp_range = "#{min_conv.round(1)} to #{max_conv.round(1)}"
else
target_temp_range = "can't inspect schedule."
end
else
target_temp_range = ''
end
water_use_data[:data] << [instance.name, plant_loop, water_use_equipment_def.name, space, peak_flow_rate_neat, water_use_equipment_flow_rate_sch, target_temp_range]
runner.registerValue(OsLib_Reporting.reg_val_string_prep(instance.name.to_s), peak_flow_rate_conv, target_units)
end
# don't create empty table
if !water_use_data[:data].empty?
@water_use_data_section[:tables] = [water_use_data] # only one table for this section
else
@water_use_data_section[:tables] = []
end
return @water_use_data_section
end
# create table for exterior lights
def self.exterior_light_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# Exterior Lighting from output
# gather data for section
@ext_light_data_section = {}
@ext_light_data_section[:title] = 'Exterior Lighting'
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @ext_light_data_section
end
# data for query
report_name = 'LightingSummary'
table_name = 'Exterior Lighting'
columns = ['Description', 'Total Power', 'Astronomical', 'Schedule Name', 'Annual Consumption']
columns_query = ['', 'Total Watts', 'Astronomical Clock/Schedule', 'Schedule Name', 'Consumption']
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
rows << row_name
end
# Zone-level Lighting Summary
table = {}
table[:title] = 'Exterior Lighting'
table[:header] = columns
table[:source_units] = ['', 'W', '', '', 'GJ'] # used for conversion, not needed for rendering.
table[:units] = ['', 'W', '', '', 'kWh']
table[:data] = []
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
columns_query.each do |header|
column_counter += 1
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
if table[:source_units][column_counter] != ''
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, table[:source_units][column_counter], table[:units][column_counter]).get
row_data << row_data_ip.round(2)
else
results = sqlFile.execAndReturnFirstString(query)
row_data << results
end
end
table[:data] << row_data
end
# don't create empty table
if table[:data].size > 1 # did > 1 instead of > 0 because I thought empty one might still have one row
@ext_light_data_section[:tables] = [table] # only one table for this section
else
@ext_light_data_section[:tables] = []
end
return @ext_light_data_section
end
# create table for elevators
# TODO: - update this to be custom load with user supplied string (or strings)
def self.elevator_data_table(model, sqlFile, runner, name_only = false, is_ip_units = true)
# elevators from model
@elevator_data = {}
@elevator_data[:title] = 'Elevators'
@elevator_data[:header] = ['Instance', 'Definition', 'Thermal Zone', 'Power Per Elevator', 'Units', 'Count']
@elevator_data[:data] = []
elec_equip_instances = model.getElectricEquipments
elec_equip_instances.sort.each do |instance|
elec_equip_def = instance.electricEquipmentDefinition
# see if it is expected and valid object
next if elec_equip_def.name.to_s != 'ElevatorElecEquipDef'
unless instance.space.is_initialized
runner.registerWarning("#{instance.name} doesn't have a space.")
next
end
# get other data
elev_space = instance.space.get
elev_zone = elev_space.thermalZone.get # should check this
elev_power = elec_equip_def.designLevel.get # should check this
elev_power_neat = OpenStudio.toNeatString(elev_power, 0, true)
units = 'W'
count = instance.multiplier
@elevator_data[:data] << [instance.name.to_s, elec_equip_def.name, elev_zone.name.get, elev_power_neat, units, OpenStudio.toNeatString(count, 2, true)]
runner.registerValue(OsLib_Reporting.reg_val_string_prep(instance.name.to_s), elev_power, units)
end
return @elevator_data
end
# create table of space type details
def self.space_type_details_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
output_data_space_type_detail_tables = []
# gather data for section
@output_data_space_type_section = {}
@output_data_space_type_section[:title] = 'Space Type Summary'
@output_data_space_type_section[:tables] = output_data_space_type_detail_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @output_data_space_type_section
end
source_units_surf = 'm^2'
source_units_temp = 'C'
source_units_FlowRate = 'm^3/s'
source_units_flowperSpaceFloorArea = 'm/s'
if is_ip_units
target_units_surf = 'ft^2'
target_units_temp = 'F'
target_units_FlowRate = 'ft^3/min'
def_units_powerPerFloorArea = 'W/ft^2'
inst_units_FlowRate = 'cfm'
target_units_flowperSpaceFloorArea = 'ft/min'
inst_units_outdoorAirFlowperPerson = 'cfm/person'
inst_units_flowperSpaceFloorArea = 'cfm/ floor area ft^2'
inst_units_flowperExteriorSurfaceArea = 'cfm/ext surf area ft^2'
inst_units_flowperExteriorWallArea = 'cfm/ext wall area ft^2'
def_units_surfaceAreaPerFloorArea = 'ft^2/floor area ft^2'
def_units_surfaceAreaPerPerson = 'ft^2/person'
def_units_peoplePerFloorArea = 'people/ft^2'
else
target_units_surf = source_units_surf
target_units_temp = source_units_temp
target_units_FlowRate = 'm^3/h'
def_units_powerPerFloorArea = 'W/m^2'
inst_units_FlowRate = 'm^3/h'
target_units_flowperSpaceFloorArea = 'm/s'
inst_units_outdoorAirFlowperPerson = 'm^3/h/person'
inst_units_flowperSpaceFloorArea = 'm^3/h/ floor area m^2'
inst_units_flowperExteriorSurfaceArea = 'm^3/h/ext surf area m^2'
inst_units_flowperExteriorWallArea = 'm^3/h/ext wall area m^2'
def_units_surfaceAreaPerFloorArea = 'm^2/floor area m^2'
def_units_surfaceAreaPerPerson = 'm^2/person'
def_units_peoplePerFloorArea = 'people/m^2'
end
# loop through space types
model.getSpaceTypes.sort.each do |spaceType|
next if spaceType.floorArea == 0
# get floor area
floor_area_si = spaceType.floorArea
# create variable for number of people
num_people = nil
# gather list of spaces and zones in space type
zone_name_list = []
space_name_list = []
spaceType.spaces.each do |space|
# grabspace and zone names
space_name_list << space.name.to_s
if space.thermalZone.is_initialized
zone_name_list << space.thermalZone.get.name.to_s
end
end
# output_data_space_type_details[:data] << [space_name_list.uniq.join(","),space_name_list.uniq.size,"Spaces",""]
# output_data_space_type_details[:data] << [zone_name_list.uniq.join(","),zone_name_list.uniq.size,"Thermal Zones",""]
# space type details data output
output_data_space_type_details = {}
output_data_space_type_details[:title] = "#{spaceType.name}
(#{space_name_list.uniq.size} spaces and #{zone_name_list.uniq.size} thermal zones)"
output_data_space_type_details[:header] = ['Definition', 'Value', 'Unit', 'Inst. Multiplier']
output_data_space_type_details[:units] = [] # won't use this for these tables since units change
output_data_space_type_details[:data] = []
# data for space type details
instances = spaceType.internalMass
instances.each do |instance|
def_display = instance.definition.name
if instance.surfaceArea.is_initialized && instance.surfaceArea.get > 0
def_value = OpenStudio.convert(instance.surfaceArea.get, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = target_units_surf
elsif instance.surfaceAreaPerFloorArea.is_initialized && instance.surfaceAreaPerFloorArea.get > 0
def_value = instance.surfaceAreaPerFloorArea.get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = def_units_surfaceAreaPerFloorArea
elsif instance.surfaceAreaPerPerson.is_initialized && instance.surfaceAreaPerPerson.get > 0
def_value = OpenStudio.convert(instance.surfaceAreaPerPerson.get, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = def_units_surfaceAreaPerPerson
end
count = instance.multiplier
output_data_space_type_details[:data] << [def_display, def_value_neat, def_units, count]
end
instances = spaceType.people
instances.each do |instance|
def_display = instance.definition.name
if instance.numberOfPeople.is_initialized && instance.numberOfPeople.get > 0
def_value = instance.numberOfPeople.get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = 'people'
elsif instance.peoplePerFloorArea.is_initialized && instance.peoplePerFloorArea.get > 0
def_value = instance.peoplePerFloorArea.get / OpenStudio.convert(1, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 4, true)
def_units = def_units_peoplePerFloorArea
elsif instance.spaceFloorAreaPerPerson.is_initialized && instance.spaceFloorAreaPerPerson.get > 0
def_value = OpenStudio.convert(instance.spaceFloorAreaPerPerson.get, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = def_units_surfaceAreaPerPerson
end
count = instance.multiplier
output_data_space_type_details[:data] << [def_display, def_value_neat, def_units, count]
end
instances = spaceType.electricEquipment
instances.each do |instance|
def_display = instance.definition.name
if instance.designLevel.is_initialized && instance.designLevel.get > 0
def_value = instance.designLevel.get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = 'W'
elsif instance.powerPerFloorArea.is_initialized && instance.powerPerFloorArea.get > 0
def_value = instance.powerPerFloorArea.get / OpenStudio.convert(1, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 4, true)
def_units = def_units_powerPerFloorArea
elsif instance.powerPerPerson .is_initialized && instance.powerPerPerson .get > 0
def_value = OpenStudio.convert(instance.powerPerPerson .get, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = 'W/person'
end
count = instance.multiplier
output_data_space_type_details[:data] << [def_display, def_value_neat, def_units, count]
end
instances = spaceType.gasEquipment
instances.each do |instance|
def_display = instance.definition.name
if instance.designLevel.is_initialized && instance.designLevel.get > 0
def_value = instance.designLevel.get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = 'W'
elsif instance.powerPerFloorArea.is_initialized && instance.powerPerFloorArea.get > 0
def_value = instance.powerPerFloorArea.get / OpenStudio.convert(1, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 4, true)
def_units = def_units_powerPerFloorArea
elsif instance.powerPerPerson .is_initialized && instance.powerPerPerson .get > 0
def_value = OpenStudio.convert(instance.powerPerPerson .get, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = 'W/person'
end
count = instance.multiplier
output_data_space_type_details[:data] << [def_display, def_value_neat, def_units, count]
end
instances = spaceType.lights
instances.each do |instance|
def_display = instance.definition.name
if instance.lightingLevel.is_initialized && instance.lightingLevel.get > 0
def_value = instance.lightingLevel.get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = 'W'
elsif instance.powerPerFloorArea.is_initialized && instance.powerPerFloorArea.get > 0
def_value = instance.powerPerFloorArea.get / OpenStudio.convert(1, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 4, true)
def_units = def_units_powerPerFloorArea
elsif instance.powerPerPerson .is_initialized && instance.powerPerPerson .get > 0
def_value = OpenStudio.convert(instance.powerPerPerson .get, source_units_surf, target_units_surf).get
def_value_neat = OpenStudio.toNeatString(def_value, 0, true)
def_units = 'W/person'
end
count = instance.multiplier
output_data_space_type_details[:data] << [def_display, def_value_neat, def_units, count]
end
instances = spaceType.spaceInfiltrationDesignFlowRates
instances.each do |instance|
instance_display = instance.name
if instance.designFlowRate.is_initialized
inst_value = OpenStudio.convert(instance.designFlowRate.get, source_units_FlowRate, target_units_FlowRate).get
inst_value_neat = OpenStudio.toNeatString(inst_value, 4, true)
inst_units = inst_units_FlowRate
count = ''
output_data_space_type_details[:data] << [instance_display, inst_value_neat, inst_units, count]
end
if instance.flowperSpaceFloorArea.is_initialized
inst_value = OpenStudio.convert(instance.flowperSpaceFloorArea.get, source_units_flowperSpaceFloorArea, target_units_flowperSpaceFloorArea).get
inst_value_neat = OpenStudio.toNeatString(inst_value, 4, true)
inst_units = inst_units_flowperSpaceFloorArea
count = ''
output_data_space_type_details[:data] << [instance_display, inst_value_neat, inst_units, count]
end
if instance.flowperExteriorSurfaceArea.is_initialized
inst_value = OpenStudio.convert(instance.flowperExteriorSurfaceArea.get, source_units_flowperSpaceFloorArea, target_units_flowperSpaceFloorArea).get
inst_value_neat = OpenStudio.toNeatString(inst_value, 4, true)
inst_units = inst_units_flowperExteriorSurfaceArea
count = ''
output_data_space_type_details[:data] << [instance_display, inst_value_neat, inst_units, count]
end
if instance.flowperExteriorWallArea.is_initialized # uses same input as exterior surface area but different calc method
inst_value = OpenStudio.convert(instance.flowperExteriorWallArea.get, source_units_flowperSpaceFloorArea, target_units_flowperSpaceFloorArea).get
inst_value_neat = OpenStudio.toNeatString(inst_value, 4, true)
inst_units = inst_units_flowperExteriorWallArea
count = ''
output_data_space_type_details[:data] << [instance_display, inst_value_neat, inst_units, count]
end
if instance.airChangesperHour.is_initialized
inst_value = instance.airChangesperHour.get
inst_value_neat = OpenStudio.toNeatString(inst_value, 4, true)
inst_units = 'ach'
count = ''
output_data_space_type_details[:data] << [instance_display, inst_value_neat, inst_units, count]
end
end
if spaceType.designSpecificationOutdoorAir.is_initialized
instance = spaceType.designSpecificationOutdoorAir.get
instance_display = instance.name
if instance.to_DesignSpecificationOutdoorAir.is_initialized
instance = instance.to_DesignSpecificationOutdoorAir.get
outdoor_air_method = instance.outdoorAirMethod
count = ''
# calculate and report various methods
if instance.outdoorAirFlowperPerson > 0
inst_value = OpenStudio.convert(instance.outdoorAirFlowperPerson, source_units_FlowRate, target_units_FlowRate).get
inst_value_neat = OpenStudio.toNeatString(inst_value, 4, true)
inst_units = inst_units_outdoorAirFlowperPerson
output_data_space_type_details[:data] << ["#{instance_display} (outdoor air method #{outdoor_air_method})", inst_value_neat, inst_units, count]
end
if instance.outdoorAirFlowperFloorArea > 0
inst_value = OpenStudio.convert(instance.outdoorAirFlowperFloorArea, source_units_flowperSpaceFloorArea, target_units_flowperSpaceFloorArea).get
inst_value_neat = OpenStudio.toNeatString(inst_value, 4, true)
inst_units = inst_units_flowperSpaceFloorArea
output_data_space_type_details[:data] << ["#{instance_display} (outdoor air method #{outdoor_air_method})", inst_value_neat, inst_units, count]
end
if instance.outdoorAirFlowRate > 0
inst_value = OpenStudio.convert(instance.outdoorAirFlowRate, source_units_FlowRate, target_units_FlowRate).get
inst_value_neat = OpenStudio.toNeatString(inst_value, 4, true)
inst_units = inst_units_FlowRate
output_data_space_type_details[:data] << ["#{instance_display} (outdoor air method #{outdoor_air_method})", inst_value_neat, inst_units, count]
end
if instance.outdoorAirFlowAirChangesperHour > 0
inst_value = instance.outdoorAirFlowAirChangesperHour
inst_value_neat = OpenStudio.toNeatString(inst_value, 4, true)
inst_units = 'ach'
output_data_space_type_details[:data] << ["#{instance_display} (outdoor air method #{outdoor_air_method})", inst_value_neat, inst_units, count]
end
end
end
# add table to array of tables
output_data_space_type_detail_tables << output_data_space_type_details
end
return @output_data_space_type_section
end
# create template section
def self.weather_summary_table(model, sqlFile, runner, is_ip_units)
# data for query
report_name = 'InputVerificationandResultsSummary'
table_name = 'General'
columns = ['', 'Value']
rows = ['Weather File', 'Latitude', 'Longitude', 'Elevation', 'Time Zone', 'North Axis Angle']
# create table
table = {}
table[:title] = 'Weather Summary'
table[:header] = columns
table[:units] = []
table[:data] = []
source_units = 'm'
if is_ip_units
target_units = 'ft'
else
target_units = 'm'
end
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
table[:header].each do |header|
column_counter += 1
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
results = sqlFile.execAndReturnFirstString(query) # this is first time I needed string vs. double for weather file
# TODO: - would be nice to get units from first column
if row == 'Elevation' then results = "#{OpenStudio.convert(results.get.to_f, source_units, target_units).get.round} #{target_units}" end
# if row == "Elevation" then results = "#{results.class} (f)" end
row_data << results
end
table[:data] << row_data
end
# add in climate zone from OpenStudio model
# get ashrae climate zone from model
climate_zone = ''
climateZones = model.getClimateZones
climateZones.climateZones.each do |climateZone|
if climateZone.institution == 'ASHRAE'
climate_zone = climateZone.value
next
end
end
table[:data] << ['ASHRAE Climate Zone', climate_zone]
return table
end
# create design_day_table
def self.design_day_table(model, sqlFile, runner, is_ip_units)
# data for query
report_name = 'ClimaticDataSummary'
table_name = 'SizingPeriod:DesignDay'
columns = ['', 'Maximum Dry Bulb', 'Daily Temperature Range', 'Humidity Value', 'Humidity Type', 'Wind Speed', 'Wind Direction']
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
rows << row_name
end
# create table
table = {}
table[:title] = 'Sizing Period Design Days'
table[:header] = columns
table[:source_units] = ['', 'C', 'K', '', '', 'm/s', '']
if is_ip_units
table[:units] = ['', 'F', 'R', '', '', 'mph', '']
else
table[:units] = table[:source_units]
end
table[:data] = []
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
table[:header].each do |header|
column_counter += 1
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row.gsub("'", "''")}' and ColumnName= '#{header}'"
if header == 'Humidity Type'
results = sqlFile.execAndReturnFirstString(query).get
if is_ip_units
# TODO: <11-12-18, jmarrec> # That doesn't seem right to me
results = results.gsub('[C]', '[F]')
results = results.gsub('[J/kg]', '[Btu/lb]')
end
# any other types?
elsif header == 'Humidity Value'
results = sqlFile.execAndReturnFirstDouble(query).get
# get humidity units for conversion
query_humidity_type = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row.gsub("'", "''")}' and ColumnName= 'Humidity Type'"
results_units = sqlFile.execAndReturnFirstString(query_humidity_type).get
if is_ip_units
if results_units.include?('[C]')
results = OpenStudio.convert(results, 'C', 'F').get.round(2)
elsif results_units.include?('[J/kg]')
results = OpenStudio.convert(results, 'J/kg', 'Btu/lb').get.round(2)
# any other types?
end
end
else
results_si = sqlFile.execAndReturnFirstString(query).get.to_s.delete(' ').to_f
if is_ip_units
results = OpenStudio.convert(results_si, table[:source_units][column_counter], table[:units][column_counter]).get.round(2)
else
results = results_si
end
end
row_data << results
end
table[:data] << row_data
end
return table
end
# create template section
def self.building_performance_table(model, sqlFile, runner, is_ip_units = true)
# create a second table
building_performance_table = {}
building_performance_table[:title] = 'Building Performance'
building_performance_table[:header] = ['Description', 'Value']
building_performance_table[:units] = []
building_performance_table[:data] = []
# add rows to table
# building_performance_table[:data] << ["Vanilla",1.5]
return building_performance_table
end
# create template section
def self.site_performance_table(model, sqlFile, runner, is_ip_units = true)
# create a second table
site_performance_table = {}
site_performance_table[:title] = 'Site Performance'
site_performance_table[:header] = ['Description', 'Value']
site_performance_table[:units] = []
site_performance_table[:data] = []
# add rows to table
# site_performance_table[:data] << ["Vanilla",1.5]
return site_performance_table
end
# create template section
def self.site_power_generation_table(model, sqlFile, runner, is_ip_units = true)
# create a second table
site_power_generation_table = {}
site_power_generation_table[:title] = 'Renewable Energy Source Summary'
site_power_generation_table[:header] = ['', 'Rated Capacity', 'Annual Energy Generated']
site_power_generation_table[:source_units] = ['', 'kW', 'GJ']
site_power_generation_table[:units] = ['', 'kW', 'kWh']
site_power_generation_table[:data] = []
# create string for LEED advisories
rows = []
rows << 'Photovoltaic'
rows << 'Wind'
# loop through advisory messages
value_found = false
rows.each do |row|
row_data = [row]
column_counter = -1
site_power_generation_table[:header].each do |header|
column_counter += 1
next if column_counter == 0
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='LEEDsummary' and RowName= '#{row}' and ColumnName='#{header}';"
data = sqlFile.execAndReturnFirstDouble(query).get
data_ip = OpenStudio.convert(data, site_power_generation_table[:source_units][column_counter], site_power_generation_table[:units][column_counter]).get
if data > 0 then value_found = true end
row_data << data_ip.round(2)
end
site_power_generation_table[:data] << row_data
end
if value_found
return site_power_generation_table
else
return false
end
end
# create template section
def self.monthly_overview_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
monthly_tables = []
# gather data for section
@monthly_overview_section = {}
@monthly_overview_section[:title] = 'Monthly Overview'
@monthly_overview_section[:tables] = monthly_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @monthly_overview_section
end
# check value of reg_monthly_details in measures
report_detailed = false
runner.workflow.workflowSteps.each do |step|
next if !step.to_MeasureStep.is_initialized
measure_step = step.to_MeasureStep.get
next if measure_step.result.is_initialized # don't look at upstream measures that have already run
arg_test = measure_step.getArgument('reg_monthly_details')
if arg_test.is_initialized
report_detailed = arg_test.get.valueAsBoolean
end
break # only want to check the first measure that doesn't have results yet
end
# alert user if report_detailed is requested
if report_detailed then runner.registerInfo('Monthly End Use by Fuel registerValues have been requested.') end
# end use colors by index
end_use_colors = ['#EF1C21', '#0071BD', '#F7DF10', '#DEC310', '#4A4D4A', '#B5B2B5', '#FF79AD', '#632C94', '#F75921', '#293094', '#CE5921', '#FFB239', '#29AAE7', '#8CC739']
# sorted end use array to pass in for stacked bar chart order
end_use_order = []
month_order = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']
# loop through fuels for consumption tables
fuel_type_names.each do |fuel_type|
# get fuel type and units
if fuel_type == 'Electricity'
units = '"kWh"'
unit_str = 'kWh'
else
if is_ip_units
units = '"Million Btu"'
unit_str = 'MBtu'
else
units = '"kWh"'
unit_str = 'kWh'
end
end
# create table
monthly_fuel = {}
monthly_fuel[:title] = "#{fuel_type} Consumption (#{unit_str})"
monthly_fuel[:header] = ['', 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec', 'Total']
monthly_fuel[:units] = []
monthly_fuel[:data] = []
monthly_fuel[:chart_type] = 'vertical_stacked_bar'
monthly_fuel[:chart_attributes] = { value: monthly_fuel[:title], label_x: 'Month', sort_yaxis: end_use_order, sort_xaxis: month_order }
monthly_fuel[:chart] = []
# has to hold monthly totals for fuel
monthly_total = {}
# rest counter for each fuel type
site_energy_use = 0.0
fuel_type_aggregation = 0.0
# loop through end uses
OpenStudio::EndUseCategoryType.getValues.each do |category_type|
category_str = OpenStudio::EndUseCategoryType.new(category_type).valueDescription
end_use_order << category_str
row_data = [category_str]
fuel_and_category_aggregation = 0.0
OpenStudio::MonthOfYear.getValues.each do |month|
if month >= 1 && month <= 12
valInJ = nil
if fuel_type_names.include?(fuel_type)
if !sqlFile.energyConsumptionByMonth(OpenStudio::EndUseFuelType.new(fuel_type),
OpenStudio::EndUseCategoryType.new(category_type),
OpenStudio::MonthOfYear.new(month)).empty?
valInJ = sqlFile.energyConsumptionByMonth(OpenStudio::EndUseFuelType.new(fuel_type),
OpenStudio::EndUseCategoryType.new(category_type),
OpenStudio::MonthOfYear.new(month)).get
end
else
# manual sql query for additional fuel types
report_name = "BUILDING ENERGY PERFORMANCE - #{fuel_type.upcase}"
row_name = OpenStudio::MonthOfYear.new(month).valueDescription
column_name = "#{category_str.upcase}:#{fuel_type.upcase}"
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and ReportForString='Meter' and RowName= '#{row_name}' and ColumnName='#{column_name}' and Units='J';"
if !sqlFile.execAndReturnFirstDouble(query).empty?
valInJ = sqlFile.execAndReturnFirstDouble(query).get
end
end
if !valInJ.nil?
fuel_and_category_aggregation += valInJ
valInUnits = OpenStudio.convert(valInJ, 'J', unit_str).get
# report values for monthly tables
if report_detailed
# return just first three characters of month
month_str = OpenStudio::MonthOfYear.new(month).valueDescription[0..2]
# this specific string chosen to match design case for a specific project
prefix_str = OpenStudio.toUnderscoreCase("end_use_#{fuel_type}_#{category_str}_#{month_str}")
runner.registerValue(prefix_str.downcase.gsub(' ', '_'), valInUnits, unit_str)
end
# populate hash for monthly totals
month = monthly_fuel[:header][month]
if monthly_total[month]
monthly_total[month] += valInJ
else
monthly_total[month] = valInJ
end
monthly_fuel[:chart] << JSON.generate(label: category_str, label_x: month, value: valInUnits, color: end_use_colors[category_type])
# for some reason sometimes 0 comes through here, show as blank of 0
if valInUnits > 0
row_data << valInUnits.round(2)
else
row_data << ''
end
else
row_data << ''
# populate hash for monthly totals
month = monthly_fuel[:header][month]
if monthly_total[month]
# do nothing
else
monthly_total[month] = 0.0
end
end
end
end
prefix_str = OpenStudio.toUnderscoreCase("#{fuel_type}_#{category_str}")
runner.registerValue("#{prefix_str}_ip", OpenStudio.convert(fuel_and_category_aggregation, 'J', unit_str).get, unit_str)
fuel_type_aggregation += fuel_and_category_aggregation
row_total = OpenStudio.convert(fuel_and_category_aggregation, 'J', unit_str).get
if row_total == 0
row_data << ''
else
row_data << row_total.round(2)
end
monthly_fuel[:data] << row_data
end
runner.registerValue(OpenStudio.toUnderscoreCase("#{fuel_type}_ip"),
OpenStudio.convert(fuel_type_aggregation, 'J', unit_str).get,
unit_str)
site_energy_use += fuel_type_aggregation
# add row for totals
row_data = ['Total']
monthly_total.each do |k, v|
if OpenStudio.convert(v, 'J', unit_str).get == 0
row_data << ''
else
row_data << OpenStudio.convert(v, 'J', unit_str).get.round(2)
end
# add monthly consumption by fuel from table to runner.registerValues
if report_detailed
# return jsut first three characters of month
month_str = k[0..2]
prefix_str = OpenStudio.toUnderscoreCase("#{fuel_type}_ip_#{month_str}")
runner.registerValue(prefix_str.downcase.gsub(' ', '_'), OpenStudio.convert(v, 'J', unit_str).get, unit_str)
end
end
table_total = OpenStudio.convert(site_energy_use, 'J', unit_str).get
row_data << table_total.round(2)
monthly_fuel[:data] << row_data
# add table to array of tables if table total is > 0
if table_total > 0
monthly_tables << monthly_fuel
end
end
# loop through fuels for peak demand tables
fuel_type_names.each do |fuel_type|
# get fuel type and units
if fuel_type == 'Electricity'
unit_str = 'kW'
else
if is_ip_units
unit_str = 'kBtu/hr' # TODO: - update units ?
else
unit_str = 'W'
end
end
# create table
monthly_fuel = {}
monthly_fuel[:title] = "#{fuel_type} Peak Demand (#{unit_str})"
monthly_fuel[:header] = ['', 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']
monthly_fuel[:units] = []
monthly_fuel[:data] = []
monthly_fuel[:chart_type] = 'vertical_stacked_bar'
monthly_fuel[:chart_attributes] = { value: monthly_fuel[:title], label_x: 'Month', sort_yaxis: end_use_order, sort_xaxis: month_order }
monthly_fuel[:chart] = []
# has to hold monthly totals for fuel
monthly_total = {}
# test for non 0 value in table
value_found = false
# loop through end uses
OpenStudio::EndUseCategoryType.getValues.each do |category_type|
category_str = OpenStudio::EndUseCategoryType.new(category_type).valueDescription
row_data = [category_str]
OpenStudio::MonthOfYear.getValues.each do |month|
if month >= 1 && month <= 12
if !sqlFile.peakEnergyDemandByMonth(OpenStudio::EndUseFuelType.new(fuel_type),
OpenStudio::EndUseCategoryType.new(category_type),
OpenStudio::MonthOfYear.new(month)).empty?
valInJ = sqlFile.peakEnergyDemandByMonth(OpenStudio::EndUseFuelType.new(fuel_type),
OpenStudio::EndUseCategoryType.new(category_type),
OpenStudio::MonthOfYear.new(month)).get
valInUnits = OpenStudio.convert(valInJ, 'W', unit_str).get
# populate hash for monthly totals
month = monthly_fuel[:header][month]
if monthly_total[month]
monthly_total[month] += valInJ
else
monthly_total[month] = valInJ
end
monthly_fuel[:chart] << JSON.generate(label: category_str, label_x: month, value: valInUnits, color: end_use_colors[category_type])
# for some reason sometimes 0 comes through here, show as blank of 0
if valInUnits > 0
row_data << valInUnits.round(4)
value_found = true
else
row_data << ''
end
else
row_data << ''
month = monthly_fuel[:header][month]
if monthly_total[month]
# do nothing
else
monthly_total[month] = 0.0
end
end
end
end
monthly_fuel[:data] << row_data
end
# add row for totals
row_data = ['Total']
monthly_total.each do |k, v|
if OpenStudio.convert(v, 'W', unit_str).get == 0
row_data << ''
else
row_data << OpenStudio.convert(v, 'W', unit_str).get.round(2)
end
end
monthly_fuel[:data] << row_data
# add table if value found
if value_found
monthly_tables << monthly_fuel
end
end
return @monthly_overview_section
end
# create utility section
def self.utility_bills_rates_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
utility_bills_rates_tables = []
# gather data for section
@utility_bills_rates_section = {}
@utility_bills_rates_section[:title] = 'Utility Bills/Rates'
@utility_bills_rates_section[:tables] = utility_bills_rates_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @utility_bills_rates_section
end
# create table
utility_table = {}
utility_table[:title] = 'Energy Type Summary'
utility_table[:header] = ['', 'Utility Rate', 'Average Rate', 'Units of Energy', 'Units of Demand']
utility_table[:query_column] = ['', 'Utility Rate', 'Virtual Rate', 'Units of Energy', 'Units of Demand']
utility_table[:units] = ['', '', '$/unit energy', '', '']
utility_table[:data] = []
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='LEEDsummary' and TableName='EAp2-3. Energy Type Summary'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
next if row_name == 'Other' # other currently mixes water and dist htg/clg. Don't want to show that
rows << row_name
end
# loop through rows
value_found = false
rows.each do |row|
data = [row]
utility_table[:query_column].each do |header|
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='LEEDsummary' and TableName='EAp2-3. Energy Type Summary' and RowName= '#{row}' and ColumnName='#{header}';"
data << sqlFile.execAndReturnFirstString(query).get
if sqlFile.execAndReturnFirstString(query).get.to_f > 0
value_found = true
end
end
utility_table[:data] << data
end
# add table to array of tables
if value_found
utility_bills_rates_tables << utility_table
end
# create table
energy_cost_table = {}
energy_cost_table[:title] = 'Energy Cost Summary'
energy_cost_table[:header] = ['', 'Total Energy Cost'] # skipping Process Subtotal
energy_cost_table[:units] = ['', '$'] # skipping Process Subtotal
energy_cost_table[:data] = []
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='LEEDsummary' and TableName='EAp2-7. Energy Cost Summary'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
rows << row_name
end
# loop through rows
value_found = false
rows.each do |row|
data = [row]
energy_cost_table[:header].each do |header|
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='LEEDsummary' and tableName = 'EAp2-7. Energy Cost Summary' and RowName= '#{row}' and ColumnName='#{header}';"
data << sqlFile.execAndReturnFirstDouble(query).get
if sqlFile.execAndReturnFirstDouble(query).get > 0
value_found = true
end
end
energy_cost_table[:data] << data
end
# add table to array of tables
if value_found
utility_bills_rates_tables << energy_cost_table
end
return @utility_bills_rates_section
end
# create unmet hours
def self.zone_condition_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
zone_condition_tables = []
# gather data for section
@zone_condition_section = {}
@zone_condition_section[:title] = 'Zone Conditions'
@zone_condition_section[:tables] = zone_condition_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @zone_condition_section
end
temperature_bins_temps_ip = [56, 61, 66, 68, 70, 72, 74, 76, 78, 83, 88]
temperature_bins_temps_si = [13, 16, 18, 20, 21, 22, 23, 24, 26, 28, 30]
# temperature_bins_temps_ip.each do |i|
# temperature_bins_temps_si << OpenStudio.convert(i, 'F', 'C').get.round(2)
# end
# hash to store hours
temperature_bins = {}
if is_ip_units
for i in 0..(temperature_bins_temps_ip.size - 1)
if i == 0
temperature_bins["< #{temperature_bins_temps_ip[i]}"] = 0
else
temperature_bins["#{temperature_bins_temps_ip[i - 1]}-#{temperature_bins_temps_ip[i]}"] = 0
end
end
# catchall bin for values over the top
temperature_bins[">= #{temperature_bins_temps_ip.last}"] = 0
else
for i in 0..(temperature_bins_temps_si.size - 1)
if i == 0
temperature_bins["< #{temperature_bins_temps_si[i]}"] = 0
else
temperature_bins["#{temperature_bins_temps_si[i - 1]}-#{temperature_bins_temps_si[i]}"] = 0
end
end
# catchall bin for values over the top
temperature_bins[">= #{temperature_bins_temps_si.last}"] = 0
end
# create table
temperature_table = {}
temperature_table[:title] = 'Temperature (Table values represent hours spent in each temperature range)'
temperature_table[:header] = ['Zone', 'Unmet Htg', 'Unmet Htg - Occ']
temperature_bins.each do |k, v|
temperature_table[:header] << k
end
temperature_table[:header] += ['Unmet Clg', 'Unmet Clg - Occ', 'Mean Temp']
temperature_table[:units] = ['', 'hr', 'hr']
temperature_bins.each do |k, v|
if is_ip_units
temperature_table[:units] << 'F'
else
temperature_table[:units] << 'C'
end
end
if is_ip_units
temperature_table[:units] += ['hr', 'hr', 'F']
else
temperature_table[:units] += ['hr', 'hr', 'C']
end
temperature_table[:data] = []
temperature_table[:data_color] = []
# get time series data for each zone
ann_env_pd = OsLib_Reporting.ann_env_pd(sqlFile)
if ann_env_pd
# get keys
keys = sqlFile.availableKeyValues(ann_env_pd, 'Hourly', 'Zone Air Temperature')
keys.each do |key|
# reset bin values
temperature_bins.each do |k, v|
temperature_bins[k] = 0
end
# get desired variable
output_timeseries = sqlFile.timeSeries(ann_env_pd, 'Hourly', 'Zone Air Temperature', key)
# loop through timeseries and move the data from an OpenStudio timeseries to a normal Ruby array (vector)
if output_timeseries.is_initialized # checks to see if time_series exists
output_timeseries = output_timeseries.get.values
temp_counter = 0
temp_sum = 0
for i in 0..(output_timeseries.size - 1)
# add to counter and sum
temp_counter += 1
temp_sum += output_timeseries[i]
found_bin = false
for j in 0..(temperature_bins_temps_si.size - 1)
if found_bin == false && output_timeseries[i] < temperature_bins_temps_si[j]
temperature_bins[temperature_bins.keys[j]] += 1
found_bin = true
end
end
# add to top if larger than all other hash values
if !found_bin
temperature_bins[temperature_bins.keys.last] += 1
end
end # end of for i in 0..(output_timeseries.size - 1)
else
runner.registerWarning("Didn't find data for Zone Air Temperature") # not getting triggered when variable missing
end # end of if output_timeseries.is_initialized
# get unmet hours for each zone from tabular data
query_htg = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='SystemSummary' and TableName = 'Time Setpoint Not Met' and RowName= '#{key}' and ColumnName='During Heating';"
unmet_htg = sqlFile.execAndReturnFirstDouble(query_htg).get
query_clg = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='SystemSummary' and TableName = 'Time Setpoint Not Met' and RowName= '#{key}' and ColumnName='During Cooling';"
unmet_clg = sqlFile.execAndReturnFirstDouble(query_clg).get
query_htg_occ = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='SystemSummary' and TableName = 'Time Setpoint Not Met' and RowName= '#{key}' and ColumnName='During Occupied Heating';"
unmet_htg_occ = sqlFile.execAndReturnFirstDouble(query_htg_occ).get
query_clg_occ = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='SystemSummary' and TableName = 'Time Setpoint Not Met' and RowName= '#{key}' and ColumnName='During Occupied Cooling';"
unmet_clg_occ = sqlFile.execAndReturnFirstDouble(query_clg_occ).get
# get mean temp
if is_ip_units
mean = OpenStudio.convert(temp_sum / temp_counter.to_f, 'C', 'F').get
else
mean = temp_sum / temp_counter.to_f
end
# add rows to table
row_data = [key, unmet_htg.round, unmet_htg_occ.round]
row_color = ['', '', '']
temperature_bins.each do |k, v|
row_data << v
if v > 2000
row_color << 'indianred'
elsif v > 1000
row_color << 'orange'
elsif v > 500
row_color << 'yellow'
else
row_color << ''
end
end
if is_ip_units
row_data += [unmet_clg.round, unmet_clg_occ.round, "#{mean.round(1)} (F)"]
else
row_data += [unmet_clg.round, unmet_clg_occ.round, "#{mean.round(1)} (C)"]
end
row_color += ['', '', '']
temperature_table[:data] << row_data
temperature_table[:data_color] << row_color
end
else
runner.registerWarning('An annual simulation was not run. Cannot get annual timeseries data')
return false
end
# add table to array of tables
zone_condition_tables << temperature_table
humidity_bins_ip = [30, 35, 40, 45, 50, 55, 60, 65, 70, 75, 80]
humidity_bins_si = humidity_bins_ip
# hash to store hours
humidity_bins = {}
for i in 0..(humidity_bins_ip.size - 1)
if i == 0
humidity_bins["< #{humidity_bins_ip[i]}"] = 0
else
humidity_bins["#{humidity_bins_ip[i - 1]}-#{humidity_bins_ip[i]}"] = 0
end
end
# add catch all bin at top
humidity_bins[">= #{humidity_bins_ip.last}"] = 0
# create table
humidity_table = {}
humidity_table[:title] = 'Humidity (Table values represent hours spent in each Humidity range)'
humidity_table[:header] = ['Zone']
humidity_bins.each do |k, v|
humidity_table[:header] << k
end
humidity_table[:header] += ['Mean Relative Humidity']
humidity_table[:units] = ['']
humidity_bins.each do |k, v|
humidity_table[:units] << '%'
end
humidity_table[:units] += ['%']
humidity_table[:data] = []
humidity_table[:data_color] = []
# get time series data for each zone
ann_env_pd = OsLib_Reporting.ann_env_pd(sqlFile)
if ann_env_pd
# store values about humidity fir reguster values
zone_max_hours_over_70_rh = 0
zone_max_hours_over_55_rh = 0
rh_hours_threshold = 10 # hr
num_zones_x_hours_over_70 = 0
num_zones_x_hours_over_55 = 0
# get keys
keys = sqlFile.availableKeyValues(ann_env_pd, 'Hourly', 'Zone Air Relative Humidity')
keys.each do |key|
# reset bin values
humidity_bins.each do |k, v|
humidity_bins[k] = 0
end
# reset humidity zone flag
zone_rh_count_hr_55 = 0.0
zone_rh_count_hr_70 = 0.0
# get desired variable
output_timeseries = sqlFile.timeSeries(ann_env_pd, 'Hourly', 'Zone Air Relative Humidity', key)
# loop through timeseries and move the data from an OpenStudio timeseries to a normal Ruby array (vector)
if output_timeseries.is_initialized # checks to see if time_series exists
output_timeseries = output_timeseries.get.values
humidity_counter = 0
humidity_sum = 0
for i in 0..(output_timeseries.size - 1)
# add to counter and sum
humidity_counter += 1
humidity_sum += output_timeseries[i]
found_bin = false
for j in 0..(humidity_bins_si.size - 1)
if found_bin == false && output_timeseries[i] < humidity_bins_si[j]
humidity_bins[humidity_bins.keys[j]] += 1
found_bin = true
end
end
# add to top if larger than all other hash values
if !found_bin
humidity_bins[humidity_bins.keys.last] += 1
end
end # end of for i in 0..(output_timeseries.size - 1)
else
runner.registerWarning("Didn't find data for Zone Air Relative Humidity") # not getting triggered when variable missing
end # end of if output_timeseries.is_initialized
# get mean humidity
mean = humidity_sum / humidity_counter.to_f
# add rows to table
row_data = [key]
row_color = ['']
humidity_bins.each do |k, v|
row_data << v
if v > 2000
row_color << 'indianred'
elsif v > 1000
row_color << 'orange'
elsif v > 500
row_color << 'yellow'
else
row_color << ''
end
# populate rh data for register_values
# catch greater than 70 and 80 for runner.registerValue
if ['55-60', '60-65', '65-70', '70-75', '75-80', '>= 80'].include?(k)
zone_rh_count_hr_55 += v
end
if ['70-75', '75-80', '>= 80'].include?(k)
zone_rh_count_hr_70 += v
end
end
row_data += ["#{mean.round(1)} (%)"]
row_color += ['']
humidity_table[:data] << row_data
humidity_table[:data_color] << row_color
# apply rh zones and max hours
if zone_rh_count_hr_55 >= rh_hours_threshold then num_zones_x_hours_over_55 += 1 end
if zone_rh_count_hr_70 >= rh_hours_threshold then num_zones_x_hours_over_70 += 1 end
if zone_max_hours_over_55_rh < zone_rh_count_hr_55 then zone_max_hours_over_55_rh = zone_rh_count_hr_55 end
if zone_max_hours_over_70_rh < zone_rh_count_hr_70 then zone_max_hours_over_70_rh = zone_rh_count_hr_70 end
# add rh runner.registerValues to be used as output in analyses
runner.registerValue('zone_max_hours_over_70_rh', zone_max_hours_over_70_rh, 'hr')
runner.registerValue('zone_max_hours_over_55_rh', zone_max_hours_over_55_rh, 'hr')
runner.registerValue('num_zones_x_hours_over_70', num_zones_x_hours_over_70, 'zones')
runner.registerValue('num_zones_x_hours_over_55', num_zones_x_hours_over_55, 'zones')
end
else
runner.registerWarning('An annual simulation was not run. Cannot get annual timeseries data')
return false
end
# add table to array of tables
zone_condition_tables << humidity_table
return @zone_condition_section
end
# create interior_lighting section
def self.interior_lighting_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
interior_lighting_tables = []
# gather data for section
@interior_lighting_section = {}
@interior_lighting_section[:title] = 'Interior Lighting Summary'
@interior_lighting_section[:tables] = interior_lighting_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @interior_lighting_section
end
# data for query
report_name = 'LightingSummary'
table_name = 'Interior Lighting'
columns = ['Lights ', 'Zone', 'Lighting Power Density', 'Total Power', 'Schedule Name', 'Scheduled Hours/Week', 'Actual Load Hours/Week', 'Return Air Fraction', 'Annual Consumption']
columns_query = ['', 'Zone', 'Lighting Power Density', 'Total Power', 'Schedule Name', 'Scheduled Hours/Week', 'Full Load Hours/Week', 'Return Air Fraction', 'Consumption']
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
next if row_name == 'Interior Lighting Total' # skipping this on purpose, may give odd results in some instances
rows << row_name
end
# Zone-level Lighting Summary
table = {}
table[:title] = 'Zone Lighting'
table[:header] = columns
source_units_area = 'm^2'
source_units_lpd = 'W/m^2'
source_units_energy = 'GJ'
if is_ip_units
target_units_area = 'ft^2'
target_units_lpd = 'W/ft^2'
target_units_energy = 'kWh'
else
target_units_area = 'm^2'
target_units_lpd = 'W/m^2'
target_units_energy = 'kWh'
end
table[:source_units] = ['', '', source_units_lpd, 'W', '', 'hr', 'hr', '', source_units_energy] # used for conversion, not needed for rendering.
table[:units] = ['', '', target_units_lpd, 'W', '', 'hr', 'hr', '', target_units_energy]
table[:data] = []
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
columns_query.each do |header|
column_counter += 1
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
if table[:source_units][column_counter] != ''
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, table[:source_units][column_counter], table[:units][column_counter]).get
row_data << row_data_ip.round(2)
else
results = sqlFile.execAndReturnFirstString(query)
row_data << results
end
end
table[:data] << row_data
end
# Space-level lighting loads table(s)
space_lighting_table = {}
space_lighting_table[:title] = 'Space Lighting Details'
space_lighting_table[:header] = ['Load Name', 'Definition Name', 'Load Type', 'Load (units)', 'Multiplier', 'Total Load (W)']
space_lighting_table[:data] = []
spaces = model.getSpaces
spaces.each do |space|
table_row = []
area = OpenStudio.convert(space.floorArea, source_units_area, target_units_area).get
lp = OpenStudio.convert(space.lightingPowerPerFloorArea, source_units_lpd, target_units_lpd).get
space_lighting_table[:data] << [{ sub_header: "Space Name: '#{space.name}', Area: #{area.round(0)} #{target_units_area},
Total LPD: #{lp.round(2)} #{target_units_lpd}" }, '', '', '', '', '']
lights_found = 0
if space.spaceType.is_initialized
space.spaceType.get.lights.each do |lights_object|
tlp = ''
def_name = lights_object.lightsDefinition.name
lights_found += 1
if lights_object.lightsDefinition.designLevelCalculationMethod == 'LightingLevel'
val = "#{lights_object.lightsDefinition.lightingLevel.to_f.round(0)} (W)"
tlp = lights_object.lightsDefinition.lightingLevel.to_f * lights_object.multiplier
end
if lights_object.lightsDefinition.designLevelCalculationMethod == 'Watts/Area'
val_conv = OpenStudio.convert(lights_object.lightsDefinition.wattsperSpaceFloorArea.to_f, source_units_lpd, target_units_lpd).get
val = "#{val_conv.to_f.round(2)} (#{target_units_lpd})"
tlp = (lights_object.lightsDefinition.wattsperSpaceFloorArea.to_f * space.floorArea) * lights_object.multiplier
end
if lights_object.lightsDefinition.designLevelCalculationMethod == 'Watts/Person'
val = "#{lights_object.lightsDefinition.wattsperPerson.to_f.round(2)} (W/Person)"
tlp = (lights_object.lightsDefinition.wattsperPerson.to_f * space.numberOfPeople) * lights_object.multiplier
end
space_lighting_table[:data] << [lights_object.name.to_s, def_name, 'Spacetype', val, lights_object.multiplier.round(0), tlp.round(0)]
end
end
space.lights.each do |sl|
tlp = ''
def_name = sl.lightsDefinition.name
lights_found += 1
if sl.lightsDefinition.designLevelCalculationMethod == 'LightingLevel'
val = "#{sl.lightsDefinition.lightingLevel.to_f.round(0)} (W)"
tlp = sl.lightsDefinition.lightingLevel.to_f * sl.multiplier
end
if sl.lightsDefinition.designLevelCalculationMethod == 'Watts/Area'
val_conv = OpenStudio.convert(sl.lightsDefinition.wattsperSpaceFloorArea.to_f, source_units_lpd, target_units_lpd).get
val = "#{val_conv.to_f.round(2)} (#{target_units_lpd})"
tlp = (sl.lightsDefinition.wattsperSpaceFloorArea.to_f * space.floorArea) * sl.multiplier
end
if sl.lightsDefinition.designLevelCalculationMethod == 'Watts/Person'
val = "#{sl.lightsDefinition.wattsperPerson.to_f.round(2)} (W/Person)"
tlp = (sl.lightsDefinition.wattsperPerson.to_f * space.numberOfPeople) * sl.multiplier
end
space_lighting_table[:data] << [sl.name.to_s, def_name, 'Space', val, sl.multiplier.round(0), tlp.round(0)]
end
space_lighting_table[:data] << ['-', '-', '-', '-', '-', '-'] if lights_found == 0
end
# Lighting Controls
source_units_illuminance = 'lux'
if is_ip_units
target_units_illuminance = 'fc'
else
target_units_illuminance = source_units_illuminance
end
lighting_controls_table = {}
lighting_controls_table[:title] = 'Lighting Controls Details'
lighting_controls_table[:header] = ['Space Name', 'Control Name', 'Zone Controlled (type, fraction)', "Illuminance Setpoint (#{target_units_illuminance})"]
lighting_controls_table[:data] = []
model.getSpaces.sort.each do |space|
thermal_zone = space.thermalZone.get
zone_control = 'n/a'
space.daylightingControls.each do |dc|
if thermal_zone.primaryDaylightingControl.is_initialized && dc.isPrimaryDaylightingControl
zone_control = "#{thermal_zone.name} (primary, #{thermal_zone.fractionofZoneControlledbyPrimaryDaylightingControl.round(1)})"
end
if thermal_zone.secondaryDaylightingControl.is_initialized && dc.isSecondaryDaylightingControl
zone_control = "#{thermal_zone.name} (secondary, #{thermal_zone.fractionofZoneControlledbySecondaryDaylightingControl.round(1)})"
end
illuminance_conv = OpenStudio.convert(dc.illuminanceSetpoint, source_units_illuminance, target_units_illuminance).get
lighting_controls_table[:data] << [space.name, dc.name, zone_control, illuminance_conv.round(0)]
end
end
# add tables to report
interior_lighting_tables << table
interior_lighting_tables << space_lighting_table
interior_lighting_tables << lighting_controls_table
return @interior_lighting_section
end
# create plug_loads section
def self.plug_loads_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
plug_loads_tables = []
# gather data for section
@plug_loads_section = {}
@plug_loads_section[:title] = 'Plug Loads Summary'
@plug_loads_section[:tables] = plug_loads_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @plug_loads_section
end
# data for query
report_name = 'EnergyMeters'
table_name = 'Annual and Peak Values - Electricity'
columns = ['', 'Electricity Annual Value'] # TODO: - would be nice to make this more like lighting summary
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
next unless row_name.include?('InteriorEquipment:Electricity:Zone:')
rows << row_name
end
# create table
table = {}
table[:title] = 'Electric Plug Load Consumption'
table[:header] = columns
source_units_energy = 'GJ'
target_units_energy = 'kWh'
table[:source_units] = ['', source_units_energy] # used for conversation, not needed for rendering.
table[:units] = ['', target_units_energy]
table[:data] = []
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
table[:header].each do |header|
column_counter += 1
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, table[:source_units][column_counter], table[:units][column_counter]).get
row_data << row_data_ip.round(2)
end
table[:data] << row_data
end
# add table to array of tables
if !table[:data].empty? then plug_loads_tables << table end
# space-level electric plug loads inputs table
table = {}
table[:title] = 'Space-level Electric Plug Loads'
table[:header] = ['Equipment Name', 'Definition', 'Load (units)', 'Inheritance Level', 'Multiplier', 'Total Load (W)']
table[:data] = []
model.getSpaces.sort.each do |space|
space_elec_equip = {}
# check for equipment from both inheritance paths
space_elec_equip['spacetype'] = space.spaceType.is_initialized ? space.spaceType.get.electricEquipment : []
space_elec_equip['space'] = space.electricEquipment
# space subheading if any equipment found
if is_ip_units
area = space.floorArea * 10.7639
table[:data] << [{ sub_header: "Space Name: #{space.name}, Area: #{area.round(0)} ft^2" }, '', '', '', '', ''] if !space_elec_equip['spacetype'].empty? || !space_elec_equip['space'].empty?
else
area = space.floorArea
table[:data] << [{ sub_header: "Space Name: #{space.name}, Area: #{area.round(0)} m^2" }, '', '', '', '', ''] if !space_elec_equip['spacetype'].empty? || !space_elec_equip['space'].empty?
end
# spacetype equipment
space_elec_equip.each do |inheritance_level, elec_equip_array|
elec_equip_array.each do |elec_equip|
if elec_equip.electricEquipmentDefinition.designLevelCalculationMethod == 'Watts/Area'
if is_ip_units
ee_power = elec_equip.electricEquipmentDefinition.wattsperSpaceFloorArea.to_f * 0.092903 # IP
ee_power = "#{ee_power.round(2)} (W/ft^2)"
ee_total_power = ((elec_equip.powerPerFloorArea.to_f * space.floorArea))
else
ee_power = elec_equip.electricEquipmentDefinition.wattsperSpaceFloorArea.to_f
ee_power = "#{ee_power.round(2)} (W/m^2)"
ee_total_power = ((elec_equip.powerPerFloorArea.to_f * space.floorArea))
end
end
if elec_equip.electricEquipmentDefinition.designLevelCalculationMethod == 'Watts/Person'
ee_power = "#{elec_equip.electricEquipmentDefinition.wattsperPerson .to_f.round(2)} (W/person)"
ee_total_power = (elec_equip.powerPerPerson.to_f * space.numberOfPeople)
end
if elec_equip.electricEquipmentDefinition.designLevelCalculationMethod == 'EquipmentLevel'
ee_power = "#{elec_equip.electricEquipmentDefinition.designLevel.to_f.round(0)} (W)"
ee_total_power = elec_equip.designLevel.to_f.round(0)
end
table[:data] << [elec_equip.name, elec_equip.electricEquipmentDefinition.name, ee_power, inheritance_level, elec_equip.multiplier.round(1), ee_total_power.round(0)]
end
end
end # space-level elec plug loads table
# if data, add table to report
plug_loads_tables << table if !table[:data].empty?
# space-level gas plug loads inputs table
table = {}
table[:title] = 'Space-level Gas Plug Loads'
if is_ip_units
table[:header] = ['Equipment Name', 'Definition', 'Load (units)', 'Inheritance Level', 'Multiplier', 'Total Load (BTU/hr)']
else
table[:header] = ['Equipment Name', 'Definition', 'Load (units)', 'Inheritance Level', 'Multiplier', 'Total Load (W)']
end
table[:data] = []
model.getSpaces.sort.each do |space|
space_gas_equip = {}
# check for equipment from both inheritance paths
space_gas_equip['spacetype'] = space.spaceType.is_initialized ? space.spaceType.get.gasEquipment : []
space_gas_equip['space'] = space.gasEquipment
# space subheading if any equipment found
if is_ip_units
area = space.floorArea.to_f * 10.7639 # --> IP
else
area = space.floorArea.to_f
end
if is_ip_units
table[:data] << [{ sub_header: "Space Name: #{space.name}, Area: #{area.round(0)} ft^2" }, '', '', '', '', ''] if !space_gas_equip['spacetype'].empty? || !space_gas_equip['space'].empty?
else
table[:data] << [{ sub_header: "Space Name: #{space.name}, Area: #{area.round(0)} m^2" }, '', '', '', '', ''] if !space_gas_equip['spacetype'].empty? || !space_gas_equip['space'].empty?
end
# spacetype equipment
space_gas_equip.each do |inheritance_level, gas_equip_array|
gas_equip_array.each do |gas_equip|
if gas_equip.gasEquipmentDefinition.designLevelCalculationMethod == 'Watts/Area'
if is_ip_units
ge_power = gas_equip.powerPerFloorArea.to_f * 0.316998331 # W/m^2 --> BTU/hr/ft^2 (OpenStudio.convert() !work (!))
ge_total_power = ((ge_power * area) * gas_equip.multiplier).to_f
ge_power = "#{ge_power.to_f.round(2)} (BTU/hr/ft^2)"
else
ge_power = gas_equip.powerPerFloorArea.to_f
ge_total_power = ((ge_power * area) * gas_equip.multiplier).to_f
ge_power = "#{ge_power.to_f.round(2)} (W/m^2)"
end
end
if gas_equip.gasEquipmentDefinition.designLevelCalculationMethod == 'Watts/Person'
if is_ip_units
ge_power = gas_equip.powerPerPerson.to_f * 3.412142 # W --> BTU/hr
ge_total_power = ((ge_power * space.numberOfPeople) * gas_equip.multiplier).to_f
ge_power = "#{ge_power.to_f.round(2)} (BTU/hr/person)"
else
ge_power = gas_equip.powerPerPerson.to_f
ge_total_power = ((ge_power * space.numberOfPeople) * gas_equip.multiplier).to_f
ge_power = "#{ge_power.to_f.round(2)} (W/person)"
end
end
if gas_equip.gasEquipmentDefinition.designLevelCalculationMethod == 'EquipmentLevel'
if is_ip_units
ge_power = gas_equip.designLevel.to_f * 3.412142 # W --> BTU/hr
ge_power = "#{ge_power.to_f.round(0)} (BTU/hr)"
ge_total_power = gas_equip.designLevel.to_f * gas_equip.multiplier
else
ge_power = gas_equip.designLevel.to_f
ge_power = "#{ge_power.to_f.round(0)} (W)"
ge_total_power = gas_equip.designLevel.to_f * gas_equip.multiplier
end
end
table[:data] << [gas_equip.name, gas_equip.gasEquipmentDefinition.name, ge_power, inheritance_level, gas_equip.multiplier.round(1), ge_total_power.round(0)]
end
end
end # space-level gas plug loads table
# if data, add table to report
plug_loads_tables << table if !table[:data].empty?
# data for query
# TODO: - need to test this in model with gas equipment
report_name = 'EnergyMeters'
table_name = 'Annual and Peak Values - Gas'
columns = ['', 'Gas Annual Value'] # TODO: - would be nice to make this more like lighting summary
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
next unless row_name.include?('InteriorEquipment:Gas:Zone:')
rows << row_name
end
# create table
table = {}
table[:title] = 'Gas Plug Load Consumption'
table[:header] = columns
source_units_energy = 'GJ'
if is_ip_units
target_units_energy = 'kBtu'
else
target_units_energy = 'kWh'
end
table[:source_units] = ['', source_units_energy] # used for conversation, not needed for rendering.
table[:units] = ['', target_units_energy]
table[:data] = []
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
table[:header].each do |header|
column_counter += 1
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, table[:source_units][column_counter], table[:units][column_counter]).get
row_data << row_data_ip.round(2)
end
table[:data] << row_data
end
# add table to array of tables
if !table[:data].empty? then plug_loads_tables << table end
return @plug_loads_section
end
# create unmet hours
def self.hvac_load_profile(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
hvac_load_profile_tables = []
# gather data for section
@hvac_load_profile_section = {}
@hvac_load_profile_section[:title] = 'HVAC Load Profiles'
@hvac_load_profile_section[:tables] = hvac_load_profile_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @hvac_load_profile_section
end
month_order = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']
# create table
hvac_load_profile_monthly_table = {}
hvac_load_profile_monthly_table[:title] = 'Monthly Load Profiles'
hvac_load_profile_monthly_table[:header] = ['', 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']
hvac_load_profile_monthly_table[:units] = []
hvac_load_profile_monthly_table[:data] = []
hvac_load_profile_monthly_table[:chart_type] = 'vertical_grouped_bar_with_comp_line'
if is_ip_units
hvac_load_profile_monthly_table[:chart_attributes] = { value_left: 'Cooling/Heating Load (MBtu)', label_x: 'Month', value_right: 'Average Outdoor Air Dry Bulb (F)', sort_xaxis: month_order }
else
hvac_load_profile_monthly_table[:chart_attributes] = { value_left: 'Cooling/Heating Load (kWh)', label_x: 'Month', value_right: 'Average Outdoor Air Dry Bulb (C)', sort_xaxis: month_order }
end
hvac_load_profile_monthly_table[:chart] = []
# hash to store monthly values
cooling_monthly = { 'Jan' => 0, 'Feb' => 0, 'Mar' => 0, 'Apr' => 0, 'May' => 0, 'Jun' => 0, 'Jul' => 0, 'Aug' => 0, 'Sep' => 0, 'Oct' => 0, 'Nov' => 0, 'Dec' => 0 }
heating_monthly = { 'Jan' => 0, 'Feb' => 0, 'Mar' => 0, 'Apr' => 0, 'May' => 0, 'Jun' => 0, 'Jul' => 0, 'Aug' => 0, 'Sep' => 0, 'Oct' => 0, 'Nov' => 0, 'Dec' => 0 }
# units for conversion
source_units = 'J'
if is_ip_units
target_units = 'MBtu'
else
target_units = 'kWh'
end
# loop through fuel types
fuel_type_names.each do |fuel_type|
OpenStudio::MonthOfYear.getValues.each do |month|
if month >= 1 && month <= 12
if fuel_type == 'Natural Gas' then fuel_type = 'Gas' end
# get cooling value for this fuel and month
if !sqlFile.energyConsumptionByMonth(OpenStudio::EndUseFuelType.new(fuel_type),
OpenStudio::EndUseCategoryType.new('Cooling'),
OpenStudio::MonthOfYear.new(month)).empty?
cooling_valInJ = sqlFile.energyConsumptionByMonth(OpenStudio::EndUseFuelType.new(fuel_type),
OpenStudio::EndUseCategoryType.new('Cooling'),
OpenStudio::MonthOfYear.new(month)).get
cooling_valInUnits = OpenStudio.convert(cooling_valInJ, source_units, target_units).get
else
cooling_valInUnits = 0
end
# get heating value for this fuel and month
if !sqlFile.energyConsumptionByMonth(OpenStudio::EndUseFuelType.new(fuel_type),
OpenStudio::EndUseCategoryType.new('Heating'),
OpenStudio::MonthOfYear.new(month)).empty?
heating_valInJ = sqlFile.energyConsumptionByMonth(OpenStudio::EndUseFuelType.new(fuel_type),
OpenStudio::EndUseCategoryType.new('Heating'),
OpenStudio::MonthOfYear.new(month)).get
heating_valInUnits = OpenStudio.convert(heating_valInJ, source_units, target_units).get
else
heating_valInUnits = 0
end
# create or add to hash to sum across fuel types
month = hvac_load_profile_monthly_table[:header][month]
if cooling_monthly.key?(month)
cooling_monthly[month] = cooling_monthly[month].to_f + cooling_valInUnits
else
cooling_monthly[month] = cooling_valInUnits
end
if heating_monthly.key?(month)
heating_monthly[month] = heating_monthly[month].to_f + heating_valInUnits
else
heating_monthly[month] = heating_monthly
end
end
end
end
# populate dry bulb data
if is_ip_units
dry_bulb_monthly = ['Average Outdoor Air Dry Bulb (F)']
else
dry_bulb_monthly = ['Average Outdoor Air Dry Bulb (C)']
end
ann_env_pd = OsLib_Reporting.ann_env_pd(sqlFile)
if ann_env_pd
# get desired variable
output_timeseries = sqlFile.timeSeries(ann_env_pd, 'Monthly', 'Site Outdoor Air Drybulb Temperature', 'Environment')
# loop through timeseries and move the data from an OpenStudio timeseries to a normal Ruby array (vector)
if output_timeseries.is_initialized # checks to see if time_series exists
# see if filler needed at start or end of table/chart
num_blanks_start = output_timeseries.get.dateTimes[0].date.monthOfYear.value - 2
num_blanks_end = 12 - output_timeseries.get.values.size - num_blanks_start
# fill in blank data for partial year simulations
for i in 0..(num_blanks_start - 1)
month = hvac_load_profile_monthly_table[:header][i + 1]
dry_bulb_monthly << ''
end
output_timeseries = output_timeseries.get.values
for i in 0..(output_timeseries.size - 1)
month = hvac_load_profile_monthly_table[:header][i + 1 + num_blanks_start]
if is_ip_units
value = OpenStudio.convert(output_timeseries[i], 'C', 'F').get
else
value = OpenStudio.convert(output_timeseries[i], 'C', 'C').get
end
dry_bulb_monthly << value.round(1)
hvac_load_profile_monthly_table[:chart] << JSON.generate(label: 'Outdoor Temp', label_x: month, value2: value, color: 'green')
end # end of for i in 0..(output_timeseries.size - 1)
# fill in blank data for partial year simulations
for i in 0..(num_blanks_end - 1)
month = hvac_load_profile_monthly_table[:header][i]
dry_bulb_monthly << ''
end
else
# TODO: - see why this is getting thrown on some models
runner.registerWarning("Didn't find data for Site Outdoor Air Drybulb Temperature")
end # end of if output_timeseries.is_initialized
else
runner.registerWarning('An annual simulation was not run. Cannot get annual timeseries data')
return false
end
# populate tables
hvac_load_profile_monthly_table[:data] << dry_bulb_monthly
if is_ip_units
cooling_array = ['Cooling Load (MBtu)']
else
cooling_array = ['Cooling Load (kWh)']
end
cooling_monthly.each do |k, v|
cooling_array << v.round(2)
# populate chart
hvac_load_profile_monthly_table[:chart] << JSON.generate(label: 'Cooling Load', label_x: k, value: v, color: '#0071BD')
end
hvac_load_profile_monthly_table[:data] << cooling_array
if is_ip_units
heating_array = ['Heating Load (MBtu)']
else
heating_array = ['Heating Load (kWh)']
end
heating_monthly.each do |k, v|
heating_array << v.round(2)
# populate chart
hvac_load_profile_monthly_table[:chart] << JSON.generate(label: 'Heating Load', label_x: k, value: v, color: '#EF1C21')
end
hvac_load_profile_monthly_table[:data] << heating_array
# add table to array of tables
hvac_load_profile_tables << hvac_load_profile_monthly_table
# create table
hvac_part_load_profile_table = {}
hvac_part_load_profile_table[:title] = 'Part Load Profiles'
hvac_part_load_profile_table[:header] = ['Load', 'Clg: Cutoff', 'Clg: Hours', 'Clg: Hours', 'Htg: Cutoff', 'Htg: Hours', 'Htg: Hours']
if is_ip_units
hvac_part_load_profile_table[:units] = ['%', 'MBtu', '%', 'hr', 'MBtu', '%', 'hr']
else
hvac_part_load_profile_table[:units] = ['%', 'kWh', '%', 'hr', 'kWh', '%', 'hr']
end
hvac_part_load_profile_table[:data] = []
# add rows to table
hvac_part_load_profile_table[:data] << ['0-5', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['5-10', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['10-25', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['15-20', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['20-25', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['25-30', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['30-35', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['35-40', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['40-45', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['45-50', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['50-55', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['55-60', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['60-65', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['65-70', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['70-75', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['75-80', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['80-85', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['85-90', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['90-95', '', '', '', '', '', '']
hvac_part_load_profile_table[:data] << ['95-100', '', '', '', '', '', '']
# TODO: - add table to array of tables
# hvac_load_profile_tables << hvac_part_load_profile_table
return @hvac_load_profile_section
end
def self.zone_summary_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
template_tables = []
# gather data for section
@zone_summary_section = {}
@zone_summary_section[:title] = 'Zone Overview'
@zone_summary_section[:tables] = template_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @zone_summary_section
end
# data for query
report_name = 'InputVerificationandResultsSummary'
table_name = 'Zone Summary'
columns = ['', 'Area', 'Conditioned (Y/N)', 'Part of Total Floor Area (Y/N)', 'Volume', 'Multiplier', 'Above Ground Gross Wall Area', 'Underground Gross Wall Area', 'Window Glass Area', 'Lighting', 'People', 'Plug and Process']
# test looking at getting entire table to get rows
# query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}'"
# results = sqlFile.execAndReturnVectorOfString(query).get
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
rows << row_name
end
# rows = ['Total','Conditioned Total','Unconditioned Total','Not Part of Total']
# create zone_summary_table
zone_summary_table = {}
zone_summary_table[:title] = table_name
zone_summary_table[:header] = columns
source_units_area = 'm^2'
source_units_area_per_person = 'm^2/person'
source_units_volume = 'm^3'
source_units_pd = 'W/m^2'
if is_ip_units
target_units_area = 'ft^2'
target_units_area_per_person = 'ft^2/person'
target_units_volume = 'ft^3'
target_units_pd = 'W/ft^2'
else
target_units_area = 'm^2'
target_units_area_per_person = 'm^2/person'
target_units_volume = 'm^3'
target_units_pd = 'W/m^2'
end
zone_summary_table[:units] = ['', target_units_area, '', '', target_units_volume, '', target_units_area, target_units_area, target_units_area, target_units_pd, target_units_area_per_person, target_units_pd]
zone_summary_table[:source_units] = ['', source_units_area, '', '', source_units_volume, '', source_units_area, source_units_area, source_units_area, source_units_pd, source_units_area_per_person, source_units_pd] # used for conversation, not needed for rendering.
zone_summary_table[:data] = []
# run query and populate zone_summary_table
rows.each do |row|
row_data = [row]
column_counter = -1
zone_summary_table[:header].each do |header|
column_counter += 1
next if header == ''
if header == 'Multiplier' then header = 'Multipliers' end # what we want to show is different than what is in E+ table
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
if zone_summary_table[:source_units][column_counter] != ''
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, zone_summary_table[:source_units][column_counter], zone_summary_table[:units][column_counter]).get
row_data << row_data_ip.round(2)
else
results = sqlFile.execAndReturnFirstString(query)
row_data << results
end
end
zone_summary_table[:data] << row_data
end
# add zone_summary_table to array of tables
template_tables << zone_summary_table
# data for query
report_name = 'HVACSizingSummary'
table_01_name = 'Zone Sensible Cooling'
table_02_name = 'Zone Sensible Heating'
columns = ['', 'Heating/Cooling', 'Calculated Design Load', 'Design Load With Sizing Factor', 'Calculated Design Air Flow', 'Design Air Flow With Sizing Factor', 'Date/Time Of Peak', 'Outdoor Temperature at Peak Load', 'Outdoor Humidity Ratio at Peak Load']
columns_query = ['', 'Heating/Cooling', 'Calculated Design Load', 'User Design Load', 'Calculated Design Air Flow', 'User Design Air Flow', 'Date/Time Of Peak {TIMESTAMP}', 'Outdoor Temperature at Peak Load', 'Outdoor Humidity Ratio at Peak Load']
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_01_name}'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
rows << row_name
end
# create zone_dd_table
zone_dd_table = {}
zone_dd_table[:title] = 'Zone Sensible Cooling and Heating Sensible Sizing'
zone_dd_table[:header] = columns
source_units_power = 'W'
source_units_air_flow = 'm^3/s'
source_units_temp = 'C'
if is_ip_units
target_units_power_clg = 'ton'
target_units_power_htg = 'kBtu/h'
target_units_air_flow = 'ft^3/min'
target_units_temp = 'F'
# This one's a ratio... so it doesn't matter really, we don't convert
# but we want to display something pretty
target_units_humrat_display = 'lbWater/lbAir'
else
target_units_power_clg = 'W'
target_units_power_htg = 'W'
target_units_air_flow = 'm^3/h'
target_units_temp = 'C'
target_units_humrat_display = 'kgWater/kgAir'
end
zone_dd_table[:units] = ['', '', '', '', target_units_air_flow, target_units_air_flow, '', target_units_temp, target_units_humrat_display]
# used for convertion, not needed for rendering.
zone_dd_table[:source_units] = ['', '', '', '', source_units_air_flow, source_units_air_flow, '', source_units_temp, target_units_humrat_display]
zone_dd_table[:data] = []
# run query and populate zone_dd_table
rows.each do |row|
# populate cooling row
row_data = [row, 'Cooling']
column_counter = -1
columns_query.each do |header|
column_counter += 1
next if header == '' || header == 'Heating/Cooling'
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_01_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
if zone_dd_table[:source_units][column_counter] != ''
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, zone_dd_table[:source_units][column_counter], zone_dd_table[:units][column_counter]).get
row_data << row_data_ip.round(2)
elsif header == 'Calculated Design Load' || header == 'User Design Load'
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, source_units_power, target_units_power_clg).get
row_data << "#{row_data_ip.round(2)} (#{target_units_power_clg})"
else
results = sqlFile.execAndReturnFirstString(query)
row_data << results
end
end
zone_dd_table[:data] << row_data
# populate heating row
row_data = [row, 'Heating']
column_counter = -1
columns_query.each do |header|
column_counter += 1
next if header == '' || header == 'Heating/Cooling'
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_02_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
if zone_dd_table[:source_units][column_counter] != ''
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, zone_dd_table[:source_units][column_counter], zone_dd_table[:units][column_counter]).get
row_data << row_data_ip.round(2)
elsif header == 'Calculated Design Load' || header == 'User Design Load'
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, source_units_power, target_units_power_htg).get
row_data << "#{row_data_ip.round(2)} (#{target_units_power_htg})"
else
results = sqlFile.execAndReturnFirstString(query)
row_data << results
end
end
zone_dd_table[:data] << row_data
end
# add zone_dd_table to array of tables
template_tables << zone_dd_table
return @zone_summary_section
end
# create air_loop_summary section
def self.air_loop_summary_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
air_loop_summary_tables = []
# gather data for section
@air_loop_summary_section = {}
@air_loop_summary_section[:title] = 'Air Loops Summary'
@air_loop_summary_section[:tables] = air_loop_summary_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @air_loop_summary_section
end
# create table
air_loop_summary_table = {}
air_loop_summary_table[:title] = 'Part load histograms for fans; graphical for annual, tabular for annual and monthly'
air_loop_summary_table[:header] = []
air_loop_summary_table[:units] = []
air_loop_summary_table[:data] = []
# add table to array of tables
air_loop_summary_tables << air_loop_summary_table
return @air_loop_summary_section
end
# create plant_loop_summary section
def self.plant_loop_summary_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
plant_loop_summary_tables = []
# gather data for section
@outdoor_air_section = {}
@outdoor_air_section[:title] = 'Plant Loops Summary'
@outdoor_air_section[:tables] = plant_loop_summary_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @outdoor_air_section
end
# create table
plant_loop_summary_table = {}
plant_loop_summary_table[:title] = 'Part load histograms for chillers, boilers, pumps'
plant_loop_summary_table[:header] = []
plant_loop_summary_table[:units] = []
plant_loop_summary_table[:data] = []
# add table to array of tables
plant_loop_summary_tables << plant_loop_summary_table
return @outdoor_air_section
end
# create outdoor_air_section
def self.outdoor_air_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
outdoor_air_section_tables = []
# gather data for section
@outdoor_air_section = {}
@outdoor_air_section[:title] = 'Outdoor Air'
@outdoor_air_section[:tables] = outdoor_air_section_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @outdoor_air_section
end
# data for query
report_name = 'OutdoorAirSummary'
table_name = 'Average Outdoor Air During Occupied Hours'
min_table_name = 'Minimum Outdoor Air During Occupied Hours'
columns = ['', 'Average Number of Occupants', 'Nominal Number of Occupants', 'Zone Volume', 'Avg. Mechanical Ventilation', 'Min. Mechanical Ventilation', 'Avg. Infiltration', 'Min. Infiltration', 'Avg. Simple Ventilation', 'Min. Simple Ventilation']
# populate dynamic rows
rows_name_query = "SELECT DISTINCT RowName FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}'"
row_names = sqlFile.execAndReturnVectorOfString(rows_name_query).get
rows = []
row_names.each do |row_name|
rows << row_name
end
# create table
table = {}
table[:title] = 'Average and Minimum Outdoor Air During Occupied Hours'
table[:header] = columns
source_units_volume = 'm^3'
if is_ip_units
target_units_volume = 'ft^3'
else
target_units_volume = 'm^3'
end
table[:units] = ['', '', '', target_units_volume, 'ach', 'ach', 'ach', 'ach', 'ach', 'ach']
table[:source_units] = ['', '', '', source_units_volume, 'ach', 'ach', 'ach', 'ach', 'ach', 'ach']
table[:data] = []
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
table[:header].each do |header|
column_counter += 1
next if header == ''
if header.include? 'Avg. '
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header.gsub('Avg. ', '')}'"
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, table[:source_units][column_counter], table[:units][column_counter]).get
row_data << row_data_ip.round(4)
elsif header.include? 'Min. '
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{min_table_name}' and RowName= '#{row}' and ColumnName= '#{header.gsub('Min. ', '')}'"
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, table[:source_units][column_counter], table[:units][column_counter]).get
row_data << row_data_ip.round(4)
elsif header == 'Zone Volume'
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, table[:source_units][column_counter], table[:units][column_counter]).get
row_data << row_data_ip.round(0)
else
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, table[:source_units][column_counter], table[:units][column_counter]).get
row_data << row_data_ip.round(4)
end
end
table[:data] << row_data
end
# add table to array of tables
outdoor_air_section_tables << table
return @outdoor_air_section
end
# create cost_summary_section
def self.cost_summary_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
cost_summary_section_tables = []
# gather data for section
@cost_summary_section = {}
@cost_summary_section[:title] = 'Cash Flow'
@cost_summary_section[:tables] = cost_summary_section_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @cost_summary_section
end
# order for fuel types
yaxis_order = ['Electricity Purchased', 'Natural Gas', 'District Heating', 'District Cooling', 'Additional', 'Water', 'Capital and O&M']
# create table
cost_summary_table = {}
cost_summary_table[:title] = 'Annual Cash Flow
(Not adjusted for inflation or utility escalation)'
cost_summary_table[:header] = ['Year', 'Capital and O&M', 'Electricity', 'Natural Gas', 'District Heating', 'District Cooling', 'Additional', 'Water']
cost_summary_table[:units] = ['', '$', '$', '$', '$', '$', '$', '$']
cost_summary_table[:data] = []
cost_summary_table[:chart_type] = 'vertical_stacked_bar'
cost_summary_table[:chart_attributes] = { value: 'Annual Cash Flow ($)', label_x: 'Date', sort_yaxis: yaxis_order }
cost_summary_table[:chart] = []
# inflation approach
inf_appr_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Life-Cycle Cost Parameters' AND RowName='Inflation Approach' AND ColumnName='Value'"
inf_appr = sqlFile.execAndReturnFirstString(inf_appr_query)
if inf_appr.is_initialized
if inf_appr.get == 'ConstantDollar'
inf_appr = 'Constant Dollar'
elsif inf_appr.get == 'CurrentDollar'
inf_appr = 'Current Dollar'
else
runner.registerWarning("Inflation approach: #{inf_appr.get} not recognized")
return false
end
runner.registerValue('inflation_approach', inf_appr)
else
runner.registerWarning('Could not determine inflation approach used')
return false
end
# base year
base_yr_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Life-Cycle Cost Parameters' AND RowName='Base Date' AND ColumnName='Value'"
base_yr = sqlFile.execAndReturnFirstString(base_yr_query)
if base_yr.is_initialized
if base_yr.get =~ /\d\d\d\d/
base_yr = base_yr.get.match(/\d\d\d\d/)[0].to_f
else
runner.registerWarning("Could not determine the analysis start year from #{base_yr.get}")
return false
end
else
runner.registerWarning('Could not determine analysis start year')
return false
end
# analysis length
length_yrs_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Life-Cycle Cost Parameters' AND RowName='Length of Study Period in Years' AND ColumnName='Value'"
length_yrs = sqlFile.execAndReturnFirstInt(length_yrs_query)
if length_yrs.is_initialized
length_yrs = length_yrs.get
runner.registerValue('analysis_length', length_yrs, 'yrs')
else
runner.registerWarning('Could not determine analysis length')
return false
end
# record the cash flow in these hashes
cap_cash_flow = {}
om_cash_flow = {}
energy_cash_flow = {}
electricity_cash_flow = {}
gas_cash_flow = {}
water_cash_flow = {}
tot_cash_flow = {}
additional_cash_flow = {}
data_cashFlow = []
data_running_total = []
running_total = 0
color = []
color << '#DDCC77' # Electricity
color << '#999933' # Natural Gas
color << '#AA4499' # Additional Fuel
color << '#88CCEE' # District Cooling
color << '#CC6677' # District Heating
color << '#332288' # Water (not used here but is in cash flow chart)
color << '#117733' # Capital and O&M (not used here but is in cash flow chart)
# loop through each year and record the cash flow
for i in 0..(length_yrs - 1) do
new_yr = base_yr + i
yr = "January #{new_yr.round}" # note: two spaces removed from earlier version of sql file
ann_cap_cash = 0.0
ann_om_cash = 0.0
ann_energy_cash = 0.0
ann_electricity_cash = 0.0
ann_gas_cash = 0.0
ann_dist_htg_cash = 0.0
ann_dist_clg_cash = 0.0
ann_water_cash = 0.0
ann_tot_cash = 0.0
# capital cash flow
cap_cash_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Capital Cash Flow by Category (Without Escalation)' AND RowName='#{yr}' AND ColumnName='Total'"
cap_cash = sqlFile.execAndReturnFirstDouble(cap_cash_query)
if cap_cash.is_initialized
ann_cap_cash += cap_cash.get
ann_tot_cash += cap_cash.get
end
# o&m cash flow (excluding utility costs)
om_types = ['Maintenance', 'Repair', 'Operation', 'Replacement', 'MinorOverhaul', 'MajorOverhaul', 'OtherOperational']
om_types.each do |om_type|
om_cash_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Operating Cash Flow by Category (Without Escalation)' AND RowName='#{yr}' AND ColumnName='#{om_type}'"
om_cash = sqlFile.execAndReturnFirstDouble(om_cash_query)
if om_cash.is_initialized
ann_om_cash += om_cash.get
ann_tot_cash += om_cash.get
end
end
# energy cost cash flows (by fuel)
electricity_purchased_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Energy and Water Cost Cash Flows (Without Escalation)' AND RowName='#{yr}' AND ColumnName='ElectricityPurchased'"
electricity_purchased = sqlFile.execAndReturnFirstDouble(electricity_purchased_query)
if electricity_purchased.is_initialized
ann_electricity_cash += electricity_purchased.get
end
gas_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Energy and Water Cost Cash Flows (Without Escalation)' AND RowName='#{yr}' AND ColumnName='Gas'"
gas = sqlFile.execAndReturnFirstDouble(gas_query)
if gas.is_initialized
ann_gas_cash += gas.get
end
dist_htg_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Energy and Water Cost Cash Flows (Without Escalation)' AND RowName='#{yr}' AND ColumnName='DistrictHeating'"
dist_htg = sqlFile.execAndReturnFirstDouble(dist_htg_query)
if dist_htg.is_initialized
ann_dist_htg_cash += dist_htg.get
end
dist_clg_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Energy and Water Cost Cash Flows (Without Escalation)' AND RowName='#{yr}' AND ColumnName='DistrictCooling'"
dist_clg = sqlFile.execAndReturnFirstDouble(dist_clg_query)
if dist_clg.is_initialized
ann_dist_clg_cash += dist_clg.get
end
# energy cash flow
energy_cash_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Operating Cash Flow by Category (Without Escalation)' AND RowName='#{yr}' AND ColumnName='Energy'"
energy_cash = sqlFile.execAndReturnFirstDouble(energy_cash_query)
if energy_cash.is_initialized
ann_energy_cash += energy_cash.get
ann_tot_cash += energy_cash.get
end
# water cash flow
water_cash_query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='Life-Cycle Cost Report' AND ReportForString='Entire Facility' AND TableName='Operating Cash Flow by Category (Without Escalation)' AND RowName='#{yr}' AND ColumnName='Water'"
water_cash = sqlFile.execAndReturnFirstDouble(water_cash_query)
if water_cash.is_initialized
ann_water_cash += water_cash.get
ann_tot_cash += water_cash.get
end
# log the values for this year
cap_cash_flow[yr] = ann_cap_cash
om_cash_flow[yr] = ann_om_cash
electricity_cash_flow[yr] = ann_electricity_cash
gas_cash_flow[yr] = ann_gas_cash
energy_cash_flow[yr] = ann_energy_cash
water_cash_flow[yr] = ann_water_cash
tot_cash_flow[yr] = ann_tot_cash
ann_additional_cash = ann_energy_cash - ann_electricity_cash - ann_gas_cash - ann_dist_htg_cash - ann_dist_clg_cash
additional_cash_flow[yr] = ann_additional_cash
# populate table row
cost_summary_table[:data] << [yr, ann_cap_cash + ann_om_cash, ann_electricity_cash, ann_gas_cash, ann_dist_htg_cash, ann_dist_clg_cash, ann_additional_cash.round(2), ann_water_cash]
# gather graph data
if ann_electricity_cash > 0
cost_summary_table[:chart] << JSON.generate(label: 'Electricity Purchased', label_x: yr, value: ann_electricity_cash, color: color[0])
end
if ann_gas_cash > 0
cost_summary_table[:chart] << JSON.generate(label: 'Natural Gas', label_x: yr, value: ann_gas_cash, color: color[1])
end
if ann_additional_cash > 0
cost_summary_table[:chart] << JSON.generate(label: 'Additional', label_x: yr, value: ann_additional_cash, color: color[2])
end
if ann_dist_clg_cash > 0
cost_summary_table[:chart] << JSON.generate(label: 'District Cooling', label_x: yr, value: ann_dist_clg_cash, color: color[3])
end
if ann_dist_htg_cash > 0
cost_summary_table[:chart] << JSON.generate(label: 'District Heating', label_x: yr, value: ann_dist_htg_cash, color: color[4])
end
if ann_water_cash > 0
cost_summary_table[:chart] << JSON.generate(label: 'Water', label_x: yr, value: ann_water_cash, color: color[5])
end
if ann_cap_cash + ann_om_cash > 0
cost_summary_table[:chart] << JSON.generate(label: 'Capital and O&M', label_x: yr, value: ann_cap_cash + ann_om_cash, color: color[6])
end
# gather running total data for line plot
running_total += ann_tot_cash
end # next year
# add table to array of tables
if running_total > 0
cost_summary_section_tables << cost_summary_table
else
# don't make chart of no data to add to it.
cost_summary_table[:chart] = []
end
return @cost_summary_section
end
# create source_energy_section
def self.source_energy_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
source_energy_section_tables = []
# gather data for section
@source_energy_section = {}
@source_energy_section[:title] = 'Site and Source Summary'
@source_energy_section[:tables] = source_energy_section_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @source_energy_section
end
# data for query
report_name = 'AnnualBuildingUtilityPerformanceSummary'
table_name = 'Site and Source Energy'
columns = ['', 'Total Energy', 'Energy Per Total Building Area', 'Energy Per Conditioned Building Area']
rows = ['Total Site Energy', 'Net Site Energy', 'Total Source Energy', 'Net Source Energy']
# create table
source_energy_table = {}
source_energy_table[:title] = table_name
source_energy_table[:header] = columns
source_units_total = 'GJ'
source_units_area = 'MJ/m^2'
if is_ip_units
target_units_total = 'kBtu'
target_units_area = 'kBtu/ft^2'
else
target_units_total = 'kWh'
target_units_area = 'kWh/m^2'
end
source_energy_table[:units] = ['', target_units_total, target_units_area, target_units_area]
source_energy_table[:source_units] = ['', source_units_total, source_units_area, source_units_area] # used for conversation, not needed for rendering.
source_energy_table[:data] = []
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
source_energy_table[:header].each do |header|
column_counter += 1
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, source_energy_table[:source_units][column_counter], source_energy_table[:units][column_counter]).get
row_data << row_data_ip.round(1)
end
source_energy_table[:data] << row_data
end
# add table to array of tables
source_energy_section_tables << source_energy_table
# data for query
report_name = 'AnnualBuildingUtilityPerformanceSummary'
table_name = 'Site to Source Energy Conversion Factors'
columns = ['', 'Site=>Source Conversion Factor']
rows = ['Electricity', 'Natural Gas', 'District Cooling', 'District Heating'] # TODO: - complete this and add logic to skip row if not used in model
# create table
source_energy_table = {}
source_energy_table[:title] = table_name
source_energy_table[:header] = columns
source_energy_table[:units] = []
source_energy_table[:data] = []
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
source_energy_table[:header].each do |header|
column_counter += 1
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
results = sqlFile.execAndReturnFirstDouble(query).to_f
row_data << results.round(3)
end
source_energy_table[:data] << row_data
end
# add table to array of tables
source_energy_section_tables << source_energy_table
return @source_energy_section
end
# create co2_and_other_emissions_section
def self.co2_and_other_emissions_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
co2_and_other_emissions_section_tables = []
# gather data for section
@co2_and_other_emissions_section = {}
@co2_and_other_emissions_section[:title] = 'CO2 and Other Emissions'
@co2_and_other_emissions_section[:tables] = co2_and_other_emissions_section_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @co2_and_other_emissions_section
end
# data for query
report_name = 'EnergyMeters'
table_name = 'Annual and Peak Values - Other by Weight/Mass'
columns = ['', 'Annual Value', 'Minimum Value', 'Timestamp of Minimum', 'Maximum Value', 'Timestamp of Maximum']
rows = ['Carbon Equivalent:Facility', 'CarbonEquivalentEmissions:Carbon Equivalent']
# create table
table = {}
table[:title] = table_name
table[:header] = columns
source_units_total = 'kg'
source_units_rate = 'kg/s'
if is_ip_units
target_units_total = 'lb'
target_units_rate = 'lb/s'
else
target_units_total = 'kg'
target_units_rate = 'kg/s'
end
table[:units] = ['', target_units_total, target_units_rate, '', target_units_rate, '']
table[:source_units] = ['', source_units_total, source_units_rate, '', source_units_rate, ''] # used for conversation, not needed for rendering.
table[:data] = []
# run query and populate table
rows.each do |row|
row_data = [row]
column_counter = -1
table[:header].each do |header|
column_counter += 1
next if header == ''
query = "SELECT Value FROM tabulardatawithstrings WHERE ReportName='#{report_name}' and TableName='#{table_name}' and RowName= '#{row}' and ColumnName= '#{header}'"
if table[:source_units][column_counter] != ''
results = sqlFile.execAndReturnFirstDouble(query)
row_data_ip = OpenStudio.convert(results.to_f, table[:source_units][column_counter], table[:units][column_counter]).get
row_data << row_data_ip.round(2)
else
results = sqlFile.execAndReturnFirstString(query)
row_data << results
end
end
table[:data] << row_data
end
# add table to array of tables
co2_and_other_emissions_section_tables << table
return @co2_and_other_emissions_section
end
# create typical_load_profiles_section
def self.typical_load_profiles_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
typical_load_profiles_section_tables = []
# gather data for section
@typical_load_profiles_section = {}
@typical_load_profiles_section[:title] = 'Typical Load Profiles'
@typical_load_profiles_section[:tables] = typical_load_profiles_section_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @typical_load_profiles_section
end
# create table
typical_load_profiles_table = {}
typical_load_profiles_table[:title] = 'Content To Be Determined'
typical_load_profiles_table[:header] = []
typical_load_profiles_table[:units] = []
typical_load_profiles_table[:data] = []
# add table to array of tables
typical_load_profiles_section_tables << typical_load_profiles_table
return @typical_load_profiles_section
end
# create schedules_overview_section
def self.schedules_overview_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
schedules_overview_section_tables = []
# gather data for section
@schedules_overview_section = {}
@schedules_overview_section[:title] = 'Schedule Overview'
@schedules_overview_section[:tables] = schedules_overview_section_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @schedules_overview_section
end
# create table
schedules_overview_table = {}
schedules_overview_table[:title] = ''
schedules_overview_table[:header] = ['Schedule', 'Type Limits', 'Rules', 'Use Count']
schedules_overview_table[:units] = []
schedules_overview_table[:data] = []
schedules_overview_table[:chart_type] = 'multi_step_line_grid'
schedules_overview_table[:chart] = [] # for this chart type, this contains an array of charts
# add table to array of tables
schedules_overview_section_tables << schedules_overview_table
model.getSchedules.sort.each do |schedule|
next unless schedule.to_ScheduleRuleset.is_initialized
schedule = schedule.to_ScheduleRuleset.get
next if schedule.nonResourceObjectUseCount(true) == 0 # true excludes children
# hash to hold chart
chart = { chart_data: [], chart_attributes: {} }
# get schedule and type limits
type_limits = nil
if schedule.scheduleTypeLimits.is_initialized
type_limits = schedule.scheduleTypeLimits.get.unitType
end
# populate table with high level schedule information
schedules_overview_table[:data] << [schedule.name, type_limits, schedule.scheduleRules.size, schedule.nonResourceObjectUseCount(true)]
# array to hold profiles
profiles = []
# get default profile
profiles << [schedule.defaultDaySchedule, 'default profile']
# get design days
summer_design = schedule.summerDesignDaySchedule
profiles << [summer_design, 'summer design day']
winter_design = schedule.winterDesignDaySchedule
profiles << [winter_design, 'winter design day']
# get rules
schedule.scheduleRules.each do |rule|
# add days of week to text
rule.applySunday ? (sun = 'Sun') : (sun = '')
rule.applyMonday ? (mon = 'Mon') : (mon = '')
rule.applyTuesday ? (tue = 'Tue') : (tue = '')
rule.applyWednesday ? (wed = 'Wed') : (wed = '')
rule.applyThursday ? (thu = 'Thu') : (thu = '')
rule.applyFriday ? (fri = 'Fri') : (fri = '')
rule.applySaturday ? (sat = 'Sat') : (sat = '')
# add dates to text
if rule.startDate.is_initialized
date = rule.startDate.get
start = date
else
start = ''
end
if rule.endDate.is_initialized
date = rule.endDate.get
finish = date
else
finish = ''
end
text = "(#{sun}#{mon}#{tue}#{wed}#{thu}#{fri}#{sat}) #{start}-#{finish}"
profiles << [rule.daySchedule, text]
end
# rule colors by index (starting with default, then summer, winter, then p1,p2, etc)
rule_colors = ['#88CCEE', 'red', 'blue', '#AA4499', '#332286', '#117733', '#99995B', '#DDCC77', '#CC6677', '#882255', '#6699CC', '#661100', '#AA4466', '#505050']
# temp test of profiles
profile_counter = -2
profiles.each do |array|
profile = array[0]
text = array[1]
if profile_counter == -2
name = text.to_s
elsif profile_counter < 1
name = " #{text}"
else
name = "Priority #{profile_counter} - #{text}"
end
# update counter
profile_counter += 1
times = profile.times
values = profile.values
(1..times.size).each do |index|
# add for this index value
time_double = times[index - 1].hours + times[index - 1].minutes / 60.0
value = values[index - 1]
# populate chart with datapoint
# chart[:chart_data] << JSON.generate({:label => name, :label_x => time_double, :value => value, :color => rule_colors[profile_counter+1]})
end
# add datapoint for 24
time = OpenStudio::Time.new(0, 24, 0, 0)
val = profile.getValue(time)
# chart[:chart_data] << JSON.generate({:label => name, :label_x => 24.0, :value => val, :color => rule_colors[profile_counter+1]})
# add datapoint for 0
time = OpenStudio::Time.new(0, 0, 0, 0)
val = profile.getValue(time)
# chart[:chart_data] << JSON.generate({:label => name, :label_x => 0.0, :value => val, :color => rule_colors[profile_counter+1]})
# get datapoint every 15min (doing this because I could get step to work just with input profile values)
(1..24).each do |i|
fractional_hours = i / 1.0
hr = fractional_hours.truncate
min = ((fractional_hours - fractional_hours.truncate) * 60.0).truncate
time = OpenStudio::Time.new(0, hr, min, 0)
val = profile.getValue(time)
# add unit conversion depending on type limits
if type_limits == 'Temperature'
if is_ip_units
val = OpenStudio.convert(val, 'C', 'F').get
else
val = OpenStudio.convert(val, 'C', 'C').get
end
end
# populate chart with datapoint
chart[:chart_data] << JSON.generate(label: name, label_x: fractional_hours, value: val, color: rule_colors[profile_counter + 1])
end
end
# populate chart attributes
if type_limits == 'Temperature'
if is_ip_units
chart[:chart_attributes][:value] = "#{type_limits} (F)"
else
chart[:chart_attributes][:value] = "#{type_limits} (C)"
end
elsif type_limits == 'ActivityLevel'
chart[:chart_attributes][:value] = "#{type_limits} (W)"
else
chart[:chart_attributes][:value] = type_limits
end
chart[:chart_attributes][:label_x] = 'Hours'
chart[:chart_attributes][:title] = schedule.name
# push chart to array of charts
schedules_overview_table[:chart] << chart
end
return @schedules_overview_section
end
# create measure_warning_section (creates tables and runner.registerValues)
def self.measure_warning_section(model, sqlFile, runner, name_only = false, is_ip_units = true)
# array to hold tables
measure_tables = []
# gather data for section
@measure_warnings_section = {}
@measure_warnings_section[:title] = 'Measure Warnings'
@measure_warnings_section[:tables] = measure_tables
# stop here if only name is requested this is used to populate display name for arguments
if name_only == true
return @measure_warnings_section
end
# will be used for registerValues
num_measures_with_warnings = 0
num_warnings = 0
num_measures = 0
# loop through workflow steps
runner.workflow.workflowSteps.each do |step|
if step.to_MeasureStep.is_initialized
measure_step = step.to_MeasureStep.get
measure_name = measure_step.measureDirName
num_measures += 1
if measure_step.name.is_initialized
measure_name = measure_step.name.get # this is instance name in PAT
end
if measure_step.result.is_initialized
result = measure_step.result.get
# create and populate table if warnings exist
if !result.warnings.empty?
measure_table_01 = {}
measure_table_01[:title] = measure_name
measure_table_01[:header] = ['Warning']
measure_table_01[:data] = []
num_measures_with_warnings += 1
# step through warnings
start_counter = num_warnings
result.warnings.each do |step|
# add rows to table and register value
num_warnings += 1
if num_warnings < start_counter + 25
measure_table_01[:data] << [step.logMessage]
elsif num_warnings == start_counter + 25
measure_table_01[:data] << ["* See OSW file for full list of warnings. This measure has #{result.warnings.size} warnings."]
end
end
# add table to section
measure_tables << measure_table_01
end
else
# puts "No result for #{measure_name}"
end
else
# puts "This step is not a measure"
end
end
# check for warnings in openstudio_results and add table for it if there are warnings.
if !runner.result.stepWarnings.empty?
measure_table_os_results = {}
measure_table_os_results[:title] = 'OpenStudio Results'
measure_table_os_results[:header] = ['Warning']
measure_table_os_results[:data] = []
num_measures_with_warnings += 1
runner.result.stepWarnings.each do |warning|
measure_table_os_results[:data] << [warning]
num_warnings += 1
end
measure_tables << measure_table_os_results
end
# add summary table (even when there are no warnings)
measure_table_summary = {}
measure_table_summary[:title] = 'Measure Warning Summary'
measure_table_summary[:header] = ['Description', 'Count']
measure_table_summary[:data] = []
# add summary rows
measure_table_summary[:data] << ['Number of measures in workflow', num_measures]
measure_table_summary[:data] << ['Number of measures with warnings', num_measures_with_warnings]
measure_table_summary[:data] << ['Total number of warnings', num_warnings]
# add table to section
measure_tables.insert(0, measure_table_summary)
runner.registerValue('number_of_measures_with_warnings', num_measures_with_warnings)
runner.registerValue('number_warnings', num_warnings)
return @measure_warnings_section
end
end