openstudio-calibration 0.1.2
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +29 -0
- data/CHANGELOG.md +14 -0
- data/Gemfile +20 -0
- data/Jenkinsfile +6 -0
- data/LICENSE.md +27 -0
- data/README.md +33 -0
- data/Rakefile +50 -0
- data/doc_templates/LICENSE.md +27 -0
- data/doc_templates/README.md.erb +42 -0
- data/doc_templates/copyright_erb.txt +36 -0
- data/doc_templates/copyright_js.txt +4 -0
- data/doc_templates/copyright_ruby.txt +34 -0
- data/lib/files/.gitkeep +0 -0
- data/lib/measures/AddMonthlyJSONUtilityData/LICENSE.md +27 -0
- data/lib/measures/AddMonthlyJSONUtilityData/README.md +96 -0
- data/lib/measures/AddMonthlyJSONUtilityData/README.md.erb +42 -0
- data/lib/measures/AddMonthlyJSONUtilityData/measure.rb +280 -0
- data/lib/measures/AddMonthlyJSONUtilityData/measure.xml +163 -0
- data/lib/measures/AddMonthlyUtilityData/LICENSE.md +27 -0
- data/lib/measures/AddMonthlyUtilityData/README.md +64 -0
- data/lib/measures/AddMonthlyUtilityData/README.md.erb +42 -0
- data/lib/measures/AddMonthlyUtilityData/measure.rb +312 -0
- data/lib/measures/AddMonthlyUtilityData/measure.xml +115 -0
- data/lib/measures/AddMonthlyUtilityData/tests/AddMonthlyUtilityData_Test.rb +88 -0
- data/lib/measures/AddMonthlyUtilityData/tests/electric_billed_usages.json +198 -0
- data/lib/measures/AddMonthlyUtilityData/tests/gas_billed_usages.json +198 -0
- data/lib/measures/CalibrationReports/LICENSE.md +27 -0
- data/lib/measures/CalibrationReports/README.md +26 -0
- data/lib/measures/CalibrationReports/README.md.erb +42 -0
- data/lib/measures/CalibrationReports/measure.rb +511 -0
- data/lib/measures/CalibrationReports/measure.xml +98 -0
- data/lib/measures/CalibrationReports/resources/report.html.in +379 -0
- data/lib/measures/CalibrationReports/tests/CalibrationReports_Test.rb +523 -0
- data/lib/measures/CalibrationReports/tests/ExampleModel.osm +10497 -0
- data/lib/measures/CalibrationReports/tests/ExampleModelNoDemandInput.osm +10560 -0
- data/lib/measures/CalibrationReports/tests/ExampleModelNoGasInput.osm +10456 -0
- data/lib/measures/CalibrationReports/tests/USA_CO_Golden-NREL.724666_TMY3.epw +8768 -0
- data/lib/measures/CalibrationReportsEnhanced/LICENSE.md +27 -0
- data/lib/measures/CalibrationReportsEnhanced/README.md +143 -0
- data/lib/measures/CalibrationReportsEnhanced/README.md.erb +42 -0
- data/lib/measures/CalibrationReportsEnhanced/measure.rb +798 -0
- data/lib/measures/CalibrationReportsEnhanced/measure.xml +372 -0
- data/lib/measures/CalibrationReportsEnhanced/resources/report.html.in +380 -0
- data/lib/measures/CalibrationReportsEnhanced/tests/CalibrationReportsEnhanced_Test.rb +734 -0
- data/lib/measures/CalibrationReportsEnhanced/tests/ExampleModel.osm +10497 -0
- data/lib/measures/CalibrationReportsEnhanced/tests/ExampleModel_FuelOil.osm +12740 -0
- data/lib/measures/CalibrationReportsEnhanced/tests/USA_CO_Golden-NREL.724666_TMY3.epw +8768 -0
- data/lib/measures/CalibrationReportsEnhanced21/LICENSE.md +27 -0
- data/lib/measures/CalibrationReportsEnhanced21/README.md +109 -0
- data/lib/measures/CalibrationReportsEnhanced21/README.md.erb +42 -0
- data/lib/measures/CalibrationReportsEnhanced21/measure.rb +690 -0
- data/lib/measures/CalibrationReportsEnhanced21/measure.xml +269 -0
- data/lib/measures/CalibrationReportsEnhanced21/resources/report.html.in +380 -0
- data/lib/measures/CoilCoolingDXSingleSpeedMultiplier/LICENSE.md +27 -0
- data/lib/measures/CoilCoolingDXSingleSpeedMultiplier/README.md +48 -0
- data/lib/measures/CoilCoolingDXSingleSpeedMultiplier/README.md.erb +42 -0
- data/lib/measures/CoilCoolingDXSingleSpeedMultiplier/measure.rb +237 -0
- data/lib/measures/CoilCoolingDXSingleSpeedMultiplier/measure.xml +93 -0
- data/lib/measures/CoilCoolingDXSingleSpeedPercentChange/LICENSE.md +27 -0
- data/lib/measures/CoilCoolingDXSingleSpeedPercentChange/README.md +48 -0
- data/lib/measures/CoilCoolingDXSingleSpeedPercentChange/README.md.erb +42 -0
- data/lib/measures/CoilCoolingDXSingleSpeedPercentChange/measure.rb +228 -0
- data/lib/measures/CoilCoolingDXSingleSpeedPercentChange/measure.xml +93 -0
- data/lib/measures/CoilCoolingDXTwoSpeedMultiplier/LICENSE.md +27 -0
- data/lib/measures/CoilCoolingDXTwoSpeedMultiplier/README.md +64 -0
- data/lib/measures/CoilCoolingDXTwoSpeedMultiplier/README.md.erb +42 -0
- data/lib/measures/CoilCoolingDXTwoSpeedMultiplier/measure.rb +280 -0
- data/lib/measures/CoilCoolingDXTwoSpeedMultiplier/measure.xml +111 -0
- data/lib/measures/CoilCoolingDXTwoSpeedPercentChange/LICENSE.md +27 -0
- data/lib/measures/CoilCoolingDXTwoSpeedPercentChange/README.md +64 -0
- data/lib/measures/CoilCoolingDXTwoSpeedPercentChange/README.md.erb +42 -0
- data/lib/measures/CoilCoolingDXTwoSpeedPercentChange/measure.rb +270 -0
- data/lib/measures/CoilCoolingDXTwoSpeedPercentChange/measure.xml +111 -0
- data/lib/measures/CoilCoolingWaterMultiplier/LICENSE.md +27 -0
- data/lib/measures/CoilCoolingWaterMultiplier/README.md +80 -0
- data/lib/measures/CoilCoolingWaterMultiplier/README.md.erb +42 -0
- data/lib/measures/CoilCoolingWaterMultiplier/measure.rb +324 -0
- data/lib/measures/CoilCoolingWaterMultiplier/measure.xml +129 -0
- data/lib/measures/CoilCoolingWaterPercentChange/LICENSE.md +27 -0
- data/lib/measures/CoilCoolingWaterPercentChange/README.md +80 -0
- data/lib/measures/CoilCoolingWaterPercentChange/README.md.erb +42 -0
- data/lib/measures/CoilCoolingWaterPercentChange/measure.rb +311 -0
- data/lib/measures/CoilCoolingWaterPercentChange/measure.xml +129 -0
- data/lib/measures/CoilHeatingElectricMultiplier/LICENSE.md +27 -0
- data/lib/measures/CoilHeatingElectricMultiplier/README.md +48 -0
- data/lib/measures/CoilHeatingElectricMultiplier/README.md.erb +42 -0
- data/lib/measures/CoilHeatingElectricMultiplier/measure.rb +240 -0
- data/lib/measures/CoilHeatingElectricMultiplier/measure.xml +93 -0
- data/lib/measures/CoilHeatingElectricPercentChange/LICENSE.md +27 -0
- data/lib/measures/CoilHeatingElectricPercentChange/README.md +48 -0
- data/lib/measures/CoilHeatingElectricPercentChange/README.md.erb +42 -0
- data/lib/measures/CoilHeatingElectricPercentChange/measure.rb +231 -0
- data/lib/measures/CoilHeatingElectricPercentChange/measure.xml +93 -0
- data/lib/measures/CoilHeatingGasMultiplier/LICENSE.md +27 -0
- data/lib/measures/CoilHeatingGasMultiplier/README.md +64 -0
- data/lib/measures/CoilHeatingGasMultiplier/README.md.erb +42 -0
- data/lib/measures/CoilHeatingGasMultiplier/measure.rb +282 -0
- data/lib/measures/CoilHeatingGasMultiplier/measure.xml +111 -0
- data/lib/measures/CoilHeatingGasPercentChange/LICENSE.md +27 -0
- data/lib/measures/CoilHeatingGasPercentChange/README.md +64 -0
- data/lib/measures/CoilHeatingGasPercentChange/README.md.erb +42 -0
- data/lib/measures/CoilHeatingGasPercentChange/measure.rb +271 -0
- data/lib/measures/CoilHeatingGasPercentChange/measure.xml +111 -0
- data/lib/measures/CoilHeatingWaterMultiplier/LICENSE.md +27 -0
- data/lib/measures/CoilHeatingWaterMultiplier/README.md +48 -0
- data/lib/measures/CoilHeatingWaterMultiplier/README.md.erb +42 -0
- data/lib/measures/CoilHeatingWaterMultiplier/measure.rb +235 -0
- data/lib/measures/CoilHeatingWaterMultiplier/measure.xml +93 -0
- data/lib/measures/CoilHeatingWaterPercentChange/LICENSE.md +27 -0
- data/lib/measures/CoilHeatingWaterPercentChange/README.md +48 -0
- data/lib/measures/CoilHeatingWaterPercentChange/README.md.erb +42 -0
- data/lib/measures/CoilHeatingWaterPercentChange/measure.rb +226 -0
- data/lib/measures/CoilHeatingWaterPercentChange/measure.xml +93 -0
- data/lib/measures/ConstructionLayerZeroMaterialProperties/LICENSE.md +27 -0
- data/lib/measures/ConstructionLayerZeroMaterialProperties/README.md +88 -0
- data/lib/measures/ConstructionLayerZeroMaterialProperties/README.md.erb +42 -0
- data/lib/measures/ConstructionLayerZeroMaterialProperties/measure.rb +221 -0
- data/lib/measures/ConstructionLayerZeroMaterialProperties/measure.xml +194 -0
- data/lib/measures/ElectricBaseboardEfficiencyAndCapacity/LICENSE.md +27 -0
- data/lib/measures/ElectricBaseboardEfficiencyAndCapacity/README.md +40 -0
- data/lib/measures/ElectricBaseboardEfficiencyAndCapacity/README.md.erb +42 -0
- data/lib/measures/ElectricBaseboardEfficiencyAndCapacity/measure.rb +102 -0
- data/lib/measures/ElectricBaseboardEfficiencyAndCapacity/measure.xml +78 -0
- data/lib/measures/ExteriorWallThermalPropertiesMultiplier/LICENSE.md +27 -0
- data/lib/measures/ExteriorWallThermalPropertiesMultiplier/README.md +48 -0
- data/lib/measures/ExteriorWallThermalPropertiesMultiplier/README.md.erb +42 -0
- data/lib/measures/ExteriorWallThermalPropertiesMultiplier/measure.rb +263 -0
- data/lib/measures/ExteriorWallThermalPropertiesMultiplier/measure.xml +97 -0
- data/lib/measures/ExteriorWallThermalPropertiesPercentChange/LICENSE.md +27 -0
- data/lib/measures/ExteriorWallThermalPropertiesPercentChange/README.md +48 -0
- data/lib/measures/ExteriorWallThermalPropertiesPercentChange/README.md.erb +42 -0
- data/lib/measures/ExteriorWallThermalPropertiesPercentChange/measure.rb +253 -0
- data/lib/measures/ExteriorWallThermalPropertiesPercentChange/measure.xml +97 -0
- data/lib/measures/FansMultiplier/LICENSE.md +27 -0
- data/lib/measures/FansMultiplier/README.md +64 -0
- data/lib/measures/FansMultiplier/README.md.erb +42 -0
- data/lib/measures/FansMultiplier/measure.rb +297 -0
- data/lib/measures/FansMultiplier/measure.xml +166 -0
- data/lib/measures/FansPercentChange/LICENSE.md +27 -0
- data/lib/measures/FansPercentChange/README.md +64 -0
- data/lib/measures/FansPercentChange/README.md.erb +42 -0
- data/lib/measures/FansPercentChange/measure.rb +286 -0
- data/lib/measures/FansPercentChange/measure.xml +166 -0
- data/lib/measures/GeneralCalibrationMeasureMultiplier/LICENSE.md +27 -0
- data/lib/measures/GeneralCalibrationMeasureMultiplier/README.md +112 -0
- data/lib/measures/GeneralCalibrationMeasureMultiplier/README.md.erb +42 -0
- data/lib/measures/GeneralCalibrationMeasureMultiplier/measure.rb +639 -0
- data/lib/measures/GeneralCalibrationMeasureMultiplier/measure.xml +229 -0
- data/lib/measures/GeneralCalibrationMeasurePercentChange/LICENSE.md +27 -0
- data/lib/measures/GeneralCalibrationMeasurePercentChange/README.md +112 -0
- data/lib/measures/GeneralCalibrationMeasurePercentChange/README.md.erb +42 -0
- data/lib/measures/GeneralCalibrationMeasurePercentChange/measure.rb +756 -0
- data/lib/measures/GeneralCalibrationMeasurePercentChange/measure.xml +229 -0
- data/lib/measures/HardSizeHvac/LICENSE.md +27 -0
- data/lib/measures/HardSizeHvac/README.md +26 -0
- data/lib/measures/HardSizeHvac/README.md.erb +42 -0
- data/lib/measures/HardSizeHvac/measure.rb +88 -0
- data/lib/measures/HardSizeHvac/measure.xml +81 -0
- data/lib/measures/MaalkaMonthlyJSONUtilityData/LICENSE.md +27 -0
- data/lib/measures/MaalkaMonthlyJSONUtilityData/README.md +104 -0
- data/lib/measures/MaalkaMonthlyJSONUtilityData/README.md.erb +42 -0
- data/lib/measures/MaalkaMonthlyJSONUtilityData/measure.rb +304 -0
- data/lib/measures/MaalkaMonthlyJSONUtilityData/measure.xml +216 -0
- data/lib/measures/RValueOfInsulationForConstructionMultiplier/LICENSE.md +27 -0
- data/lib/measures/RValueOfInsulationForConstructionMultiplier/README.md +40 -0
- data/lib/measures/RValueOfInsulationForConstructionMultiplier/README.md.erb +42 -0
- data/lib/measures/RValueOfInsulationForConstructionMultiplier/measure.rb +196 -0
- data/lib/measures/RValueOfInsulationForConstructionMultiplier/measure.xml +132 -0
- data/lib/measures/RValueOfInsulationForConstructionPercentageChange/LICENSE.md +27 -0
- data/lib/measures/RValueOfInsulationForConstructionPercentageChange/README.md +40 -0
- data/lib/measures/RValueOfInsulationForConstructionPercentageChange/README.md.erb +42 -0
- data/lib/measures/RValueOfInsulationForConstructionPercentageChange/measure.rb +189 -0
- data/lib/measures/RValueOfInsulationForConstructionPercentageChange/measure.xml +132 -0
- data/lib/measures/RoofThermalPropertiesMultiplier/LICENSE.md +27 -0
- data/lib/measures/RoofThermalPropertiesMultiplier/README.md +48 -0
- data/lib/measures/RoofThermalPropertiesMultiplier/README.md.erb +42 -0
- data/lib/measures/RoofThermalPropertiesMultiplier/measure.rb +263 -0
- data/lib/measures/RoofThermalPropertiesMultiplier/measure.xml +97 -0
- data/lib/measures/RoofThermalPropertiesPercentChange/LICENSE.md +27 -0
- data/lib/measures/RoofThermalPropertiesPercentChange/README.md +48 -0
- data/lib/measures/RoofThermalPropertiesPercentChange/README.md.erb +42 -0
- data/lib/measures/RoofThermalPropertiesPercentChange/measure.rb +253 -0
- data/lib/measures/RoofThermalPropertiesPercentChange/measure.xml +97 -0
- data/lib/measures/TimeseriesObjectiveFunction/LICENSE.md +27 -0
- data/lib/measures/TimeseriesObjectiveFunction/README.md +207 -0
- data/lib/measures/TimeseriesObjectiveFunction/README.md.erb +42 -0
- data/lib/measures/TimeseriesObjectiveFunction/measure.rb +748 -0
- data/lib/measures/TimeseriesObjectiveFunction/measure.xml +435 -0
- data/lib/measures/TimeseriesObjectiveFunction/resources/report.html.erb +288 -0
- data/lib/measures/TimeseriesPlot/LICENSE.md +27 -0
- data/lib/measures/TimeseriesPlot/README.md +56 -0
- data/lib/measures/TimeseriesPlot/README.md.erb +42 -0
- data/lib/measures/TimeseriesPlot/measure.rb +302 -0
- data/lib/measures/TimeseriesPlot/measure.xml +137 -0
- data/lib/measures/TimeseriesPlot/resources/report.html.erb +270 -0
- data/lib/measures/WaterHeaterMixedMultiplier/LICENSE.md +27 -0
- data/lib/measures/WaterHeaterMixedMultiplier/README.md +64 -0
- data/lib/measures/WaterHeaterMixedMultiplier/README.md.erb +42 -0
- data/lib/measures/WaterHeaterMixedMultiplier/measure.rb +255 -0
- data/lib/measures/WaterHeaterMixedMultiplier/measure.xml +180 -0
- data/lib/measures/WaterHeaterMixedPercentChange/LICENSE.md +27 -0
- data/lib/measures/WaterHeaterMixedPercentChange/README.md +64 -0
- data/lib/measures/WaterHeaterMixedPercentChange/README.md.erb +42 -0
- data/lib/measures/WaterHeaterMixedPercentChange/measure.rb +245 -0
- data/lib/measures/WaterHeaterMixedPercentChange/measure.xml +180 -0
- data/lib/measures/zone_report/LICENSE.md +27 -0
- data/lib/measures/zone_report/README.md +26 -0
- data/lib/measures/zone_report/README.md.erb +42 -0
- data/lib/measures/zone_report/measure.rb +706 -0
- data/lib/measures/zone_report/measure.xml +67 -0
- data/lib/measures/zone_report/resources/report.html.in +342 -0
- data/lib/openstudio-calibration-measures.rb +36 -0
- data/lib/openstudio/calibration_measures.rb +37 -0
- data/lib/openstudio/calibration_measures/extension.rb +49 -0
- data/lib/openstudio/calibration_measures/version.rb +40 -0
- data/openstudio-calibration.gemspec +30 -0
- metadata +331 -0
@@ -0,0 +1,180 @@
|
|
1
|
+
<measure>
|
2
|
+
<schema_version>3.0</schema_version>
|
3
|
+
<name>water_heater_mixed_percent_change</name>
|
4
|
+
<uid>c42bf4e7-2bbb-44c6-ab91-e6ddb53e050d</uid>
|
5
|
+
<version_id>414fa696-b1c4-41b2-a1e2-1904af51cf16</version_id>
|
6
|
+
<version_modified>20200325T030949Z</version_modified>
|
7
|
+
<xml_checksum>D28BCE9C</xml_checksum>
|
8
|
+
<class_name>WaterHeaterMixedPercentChange</class_name>
|
9
|
+
<display_name>Water Heater Mixed Percent Change</display_name>
|
10
|
+
<description>This is a general purpose measure to calibrate WaterHeaterMixed with a Percent Change.</description>
|
11
|
+
<modeler_description>It will be used for calibration of WaterHeaterMixed. User can choose between a SINGLE WaterHeaterMixed or ALL the WaterHeaterMixed objects.</modeler_description>
|
12
|
+
<arguments>
|
13
|
+
<argument>
|
14
|
+
<name>water_heater</name>
|
15
|
+
<display_name>Apply the Measure to a SINGLE WaterHeaterMixed, ALL the WaterHeaterMixeds or NONE.</display_name>
|
16
|
+
<type>Choice</type>
|
17
|
+
<required>true</required>
|
18
|
+
<model_dependent>false</model_dependent>
|
19
|
+
<default_value>*All WaterHeaterMixeds*</default_value>
|
20
|
+
<choices>
|
21
|
+
<choice>
|
22
|
+
<value>{ae86c587-54d9-4c1a-b9f4-b9c4b80a0059}</value>
|
23
|
+
<display_name>*All WaterHeaterMixeds*</display_name>
|
24
|
+
</choice>
|
25
|
+
<choice>
|
26
|
+
<value>0</value>
|
27
|
+
<display_name>*None*</display_name>
|
28
|
+
</choice>
|
29
|
+
</choices>
|
30
|
+
</argument>
|
31
|
+
<argument>
|
32
|
+
<name>maximum_capacity_multiplier</name>
|
33
|
+
<display_name>Percent Change for Heater Maximum Capacity.</display_name>
|
34
|
+
<description>Percent Change for Heater Maximum Capacity.</description>
|
35
|
+
<type>Double</type>
|
36
|
+
<required>true</required>
|
37
|
+
<model_dependent>false</model_dependent>
|
38
|
+
<default_value>0</default_value>
|
39
|
+
</argument>
|
40
|
+
<argument>
|
41
|
+
<name>minimum_capacity_multiplier</name>
|
42
|
+
<display_name>Percent Change for Heater Minimum Capacity.</display_name>
|
43
|
+
<description>Percent Change for Heater Minimum Capacity.</description>
|
44
|
+
<type>Double</type>
|
45
|
+
<required>true</required>
|
46
|
+
<model_dependent>false</model_dependent>
|
47
|
+
<default_value>0</default_value>
|
48
|
+
</argument>
|
49
|
+
<argument>
|
50
|
+
<name>thermal_efficiency_multiplier</name>
|
51
|
+
<display_name>Percent Change for Thermal Efficiency.</display_name>
|
52
|
+
<description>Percent Change for Thermal Efficiency.</description>
|
53
|
+
<type>Double</type>
|
54
|
+
<required>true</required>
|
55
|
+
<model_dependent>false</model_dependent>
|
56
|
+
<default_value>0</default_value>
|
57
|
+
</argument>
|
58
|
+
<argument>
|
59
|
+
<name>fuel_type</name>
|
60
|
+
<display_name>Heater Fuel Type.</display_name>
|
61
|
+
<description>Heater Fuel Type.</description>
|
62
|
+
<type>Choice</type>
|
63
|
+
<required>true</required>
|
64
|
+
<model_dependent>false</model_dependent>
|
65
|
+
<default_value>NaturalGas</default_value>
|
66
|
+
<choices>
|
67
|
+
<choice>
|
68
|
+
<value>NaturalGas</value>
|
69
|
+
<display_name>NaturalGas</display_name>
|
70
|
+
</choice>
|
71
|
+
<choice>
|
72
|
+
<value>Electricity</value>
|
73
|
+
<display_name>Electricity</display_name>
|
74
|
+
</choice>
|
75
|
+
<choice>
|
76
|
+
<value>PropaneGas</value>
|
77
|
+
<display_name>PropaneGas</display_name>
|
78
|
+
</choice>
|
79
|
+
</choices>
|
80
|
+
</argument>
|
81
|
+
</arguments>
|
82
|
+
<outputs/>
|
83
|
+
<provenances/>
|
84
|
+
<tags>
|
85
|
+
<tag>Service Water Heating.Water Heating</tag>
|
86
|
+
</tags>
|
87
|
+
<attributes>
|
88
|
+
<attribute>
|
89
|
+
<name>Intended Software Tool</name>
|
90
|
+
<value>Analysis Spreadsheet</value>
|
91
|
+
<datatype>string</datatype>
|
92
|
+
</attribute>
|
93
|
+
<attribute>
|
94
|
+
<name>Intended Use Case</name>
|
95
|
+
<value>Model Articulation</value>
|
96
|
+
<datatype>string</datatype>
|
97
|
+
</attribute>
|
98
|
+
<attribute>
|
99
|
+
<name>Intended Use Case</name>
|
100
|
+
<value>Calibration</value>
|
101
|
+
<datatype>string</datatype>
|
102
|
+
</attribute>
|
103
|
+
<attribute>
|
104
|
+
<name>Intended Use Case</name>
|
105
|
+
<value>Sensitivity Analysis</value>
|
106
|
+
<datatype>string</datatype>
|
107
|
+
</attribute>
|
108
|
+
<attribute>
|
109
|
+
<name>Measure Type</name>
|
110
|
+
<value>ModelMeasure</value>
|
111
|
+
<datatype>string</datatype>
|
112
|
+
</attribute>
|
113
|
+
<attribute>
|
114
|
+
<name>Intended Software Tool</name>
|
115
|
+
<value>Apply Measure Now</value>
|
116
|
+
<datatype>string</datatype>
|
117
|
+
</attribute>
|
118
|
+
<attribute>
|
119
|
+
<name>Intended Software Tool</name>
|
120
|
+
<value>OpenStudio Application</value>
|
121
|
+
<datatype>string</datatype>
|
122
|
+
</attribute>
|
123
|
+
<attribute>
|
124
|
+
<name>Intended Software Tool</name>
|
125
|
+
<value>Parametric Analysis Tool</value>
|
126
|
+
<datatype>string</datatype>
|
127
|
+
</attribute>
|
128
|
+
<attribute>
|
129
|
+
<name>Intended Software Tool</name>
|
130
|
+
<value>Analysis Spreadsheet</value>
|
131
|
+
<datatype>string</datatype>
|
132
|
+
</attribute>
|
133
|
+
<attribute>
|
134
|
+
<name>Intended Use Case</name>
|
135
|
+
<value>Model Articulation</value>
|
136
|
+
<datatype>string</datatype>
|
137
|
+
</attribute>
|
138
|
+
<attribute>
|
139
|
+
<name>Intended Use Case</name>
|
140
|
+
<value>Calibration</value>
|
141
|
+
<datatype>string</datatype>
|
142
|
+
</attribute>
|
143
|
+
<attribute>
|
144
|
+
<name>Intended Use Case</name>
|
145
|
+
<value>Sensitivity Analysis</value>
|
146
|
+
<datatype>string</datatype>
|
147
|
+
</attribute>
|
148
|
+
</attributes>
|
149
|
+
<files>
|
150
|
+
<file>
|
151
|
+
<filename>LICENSE.md</filename>
|
152
|
+
<filetype>md</filetype>
|
153
|
+
<usage_type>license</usage_type>
|
154
|
+
<checksum>E0468DD6</checksum>
|
155
|
+
</file>
|
156
|
+
<file>
|
157
|
+
<filename>README.md.erb</filename>
|
158
|
+
<filetype>erb</filetype>
|
159
|
+
<usage_type>readmeerb</usage_type>
|
160
|
+
<checksum>703C9964</checksum>
|
161
|
+
</file>
|
162
|
+
<file>
|
163
|
+
<filename>README.md</filename>
|
164
|
+
<filetype>md</filetype>
|
165
|
+
<usage_type>readme</usage_type>
|
166
|
+
<checksum>04987893</checksum>
|
167
|
+
</file>
|
168
|
+
<file>
|
169
|
+
<version>
|
170
|
+
<software_program>OpenStudio</software_program>
|
171
|
+
<identifier>2.1.0</identifier>
|
172
|
+
<min_compatible>2.1.0</min_compatible>
|
173
|
+
</version>
|
174
|
+
<filename>measure.rb</filename>
|
175
|
+
<filetype>rb</filetype>
|
176
|
+
<usage_type>script</usage_type>
|
177
|
+
<checksum>019A4EE1</checksum>
|
178
|
+
</file>
|
179
|
+
</files>
|
180
|
+
</measure>
|
@@ -0,0 +1,27 @@
|
|
1
|
+
OpenStudio(R), Copyright (c) 2008-2020, Alliance for Sustainable Energy, LLC. All rights reserved.
|
2
|
+
|
3
|
+
Redistribution and use in source and binary forms, with or without modification, are permitted
|
4
|
+
provided that the following conditions are met:
|
5
|
+
|
6
|
+
(1) Redistributions of source code must retain the above copyright notice, this list of conditions
|
7
|
+
and the following disclaimer.
|
8
|
+
|
9
|
+
(2) Redistributions in binary form must reproduce the above copyright notice, this list of conditions
|
10
|
+
and the following disclaimer in the documentation and/or other materials provided with the distribution.
|
11
|
+
|
12
|
+
(3) Neither the name of the copyright holder nor the names of any contributors may be used to endorse
|
13
|
+
or promote products derived from this software without specific prior written permission from the
|
14
|
+
respective party.
|
15
|
+
|
16
|
+
(4) Other than as required in clauses (1) and (2), distributions in any form of modifications or other
|
17
|
+
derivative works may not use the "OpenStudio" trademark, "OS", "os", or any other confusingly similar
|
18
|
+
designation without specific prior written permission from Alliance for Sustainable Energy, LLC.
|
19
|
+
|
20
|
+
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED
|
21
|
+
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
22
|
+
PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER, THE UNITED STATES GOVERNMENT,
|
23
|
+
OR ANY CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
24
|
+
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
25
|
+
OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
26
|
+
STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
27
|
+
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
@@ -0,0 +1,26 @@
|
|
1
|
+
|
2
|
+
|
3
|
+
###### (Automatically generated documentation)
|
4
|
+
|
5
|
+
# Zone Report
|
6
|
+
|
7
|
+
## Description
|
8
|
+
|
9
|
+
|
10
|
+
## Modeler Description
|
11
|
+
|
12
|
+
|
13
|
+
## Measure Type
|
14
|
+
ReportingMeasure
|
15
|
+
|
16
|
+
## Taxonomy
|
17
|
+
|
18
|
+
|
19
|
+
## Arguments
|
20
|
+
|
21
|
+
|
22
|
+
|
23
|
+
|
24
|
+
This measure does not have any user arguments
|
25
|
+
|
26
|
+
|
@@ -0,0 +1,42 @@
|
|
1
|
+
<%#= README.md.erb is used to auto-generate README.md. %>
|
2
|
+
<%#= To manually maintain README.md throw away README.md.erb and manually edit README.md %>
|
3
|
+
###### (Automatically generated documentation)
|
4
|
+
|
5
|
+
# <%= name %>
|
6
|
+
|
7
|
+
## Description
|
8
|
+
<%= description %>
|
9
|
+
|
10
|
+
## Modeler Description
|
11
|
+
<%= modelerDescription %>
|
12
|
+
|
13
|
+
## Measure Type
|
14
|
+
<%= measureType %>
|
15
|
+
|
16
|
+
## Taxonomy
|
17
|
+
<%= taxonomy %>
|
18
|
+
|
19
|
+
## Arguments
|
20
|
+
|
21
|
+
<% arguments.each do |argument| %>
|
22
|
+
### <%= argument[:display_name] %>
|
23
|
+
<%= argument[:description] %>
|
24
|
+
**Name:** <%= argument[:name] %>,
|
25
|
+
**Type:** <%= argument[:type] %>,
|
26
|
+
**Units:** <%= argument[:units] %>,
|
27
|
+
**Required:** <%= argument[:required] %>,
|
28
|
+
**Model Dependent:** <%= argument[:model_dependent] %>
|
29
|
+
<% end %>
|
30
|
+
|
31
|
+
<% if arguments.size == 0 %>
|
32
|
+
<%= "This measure does not have any user arguments" %>
|
33
|
+
<% end %>
|
34
|
+
|
35
|
+
<% if outputs.size > 0 %>
|
36
|
+
## Outputs
|
37
|
+
<% output_names = [] %>
|
38
|
+
<% outputs.each do |output| %>
|
39
|
+
<% output_names << output[:display_name] %>
|
40
|
+
<% end %>
|
41
|
+
<%= output_names.join(", ") %>
|
42
|
+
<% end %>
|
@@ -0,0 +1,706 @@
|
|
1
|
+
# *******************************************************************************
|
2
|
+
# OpenStudio(R), Copyright (c) 2008-2020, Alliance for Sustainable Energy, LLC.
|
3
|
+
# All rights reserved.
|
4
|
+
# Redistribution and use in source and binary forms, with or without
|
5
|
+
# modification, are permitted provided that the following conditions are met:
|
6
|
+
#
|
7
|
+
# (1) Redistributions of source code must retain the above copyright notice,
|
8
|
+
# this list of conditions and the following disclaimer.
|
9
|
+
#
|
10
|
+
# (2) Redistributions in binary form must reproduce the above copyright notice,
|
11
|
+
# this list of conditions and the following disclaimer in the documentation
|
12
|
+
# and/or other materials provided with the distribution.
|
13
|
+
#
|
14
|
+
# (3) Neither the name of the copyright holder nor the names of any contributors
|
15
|
+
# may be used to endorse or promote products derived from this software without
|
16
|
+
# specific prior written permission from the respective party.
|
17
|
+
#
|
18
|
+
# (4) Other than as required in clauses (1) and (2), distributions in any form
|
19
|
+
# of modifications or other derivative works may not use the "OpenStudio"
|
20
|
+
# trademark, "OS", "os", or any other confusingly similar designation without
|
21
|
+
# specific prior written permission from Alliance for Sustainable Energy, LLC.
|
22
|
+
#
|
23
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS
|
24
|
+
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
25
|
+
# THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
26
|
+
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE
|
27
|
+
# UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF
|
28
|
+
# THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
29
|
+
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
|
30
|
+
# OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
31
|
+
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
32
|
+
# STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
33
|
+
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
34
|
+
# *******************************************************************************
|
35
|
+
|
36
|
+
require 'erb'
|
37
|
+
# require 'pry'
|
38
|
+
|
39
|
+
# start the measure
|
40
|
+
class ZoneReport < OpenStudio::Measure::ReportingMeasure
|
41
|
+
# define the name that a user will see, this method may be deprecated as
|
42
|
+
# the display name in PAT comes from the name field in measure.xml
|
43
|
+
def name
|
44
|
+
'Zone Report'
|
45
|
+
end
|
46
|
+
|
47
|
+
# define the arguments that the user will input
|
48
|
+
def arguments
|
49
|
+
args = OpenStudio::Measure::OSArgumentVector.new
|
50
|
+
args
|
51
|
+
end # end the arguments method
|
52
|
+
|
53
|
+
# return a vector of IdfObject's to request EnergyPlus objects needed by the run method
|
54
|
+
def energyPlusOutputRequests(runner, user_arguments)
|
55
|
+
super(runner, user_arguments)
|
56
|
+
|
57
|
+
result = OpenStudio::IdfObjectVector.new
|
58
|
+
|
59
|
+
# use the built-in error checking
|
60
|
+
return result unless runner.validateUserArguments(arguments, user_arguments)
|
61
|
+
|
62
|
+
request = OpenStudio::IdfObject.load('Output:Table:SummaryReports,AllSummaryAndSizingPeriod;').get
|
63
|
+
result << request
|
64
|
+
|
65
|
+
result
|
66
|
+
end
|
67
|
+
|
68
|
+
# Method to make class names nicer (remove OS and _)
|
69
|
+
def niceClassName(class_name)
|
70
|
+
class_name = class_name.gsub('OS_', '')
|
71
|
+
class_name = class_name.tr('_', ' ')
|
72
|
+
class_name
|
73
|
+
end
|
74
|
+
|
75
|
+
# Convert a value from the view TabularDataWithStrings from the unit represented there to the given unit.
|
76
|
+
# We make a bunch of special cases for unit aliases used by EnergyPlus which are not in the default
|
77
|
+
# list of OpenStudio conversions
|
78
|
+
# We also special case a few conversions for unitless 'units' so that our final display makes more sense to the user.
|
79
|
+
# When the property is not present (for instance, cooled beams don't have properties in the sql file under 1.5)
|
80
|
+
# this method will return "-" as a placeholder.
|
81
|
+
def eplus_to_openstudio(unitstr)
|
82
|
+
unitstr.gsub('m3', 'm^3').gsub('pa', 'Pa').gsub('m2', 'm^2').gsub(' per ', '/')
|
83
|
+
end
|
84
|
+
|
85
|
+
def convert_prop(property, final_units)
|
86
|
+
return ['-', '-'] unless property
|
87
|
+
|
88
|
+
return [property[0].to_f.round(2), ''] if final_units.empty?
|
89
|
+
return [(property[0].to_f * 100).round(2), '%'] if final_units == '%'
|
90
|
+
return [property[0].to_f.round(2), 'COP'] if final_units == 'COP'
|
91
|
+
|
92
|
+
initial_units = eplus_to_openstudio(property[1])
|
93
|
+
converted = OpenStudio.convert(property[0].to_f, initial_units, final_units)
|
94
|
+
if converted.empty?
|
95
|
+
"Could not convert from #{initial_units} to #{final_units}"
|
96
|
+
else
|
97
|
+
final_units = final_units.gsub('inH_{2}O', 'in. w.c.')
|
98
|
+
[converted.get.round(2), final_units]
|
99
|
+
end
|
100
|
+
end
|
101
|
+
|
102
|
+
def properties_for_cooling_coil(e, props, parent)
|
103
|
+
autosized = case e.iddObjectType
|
104
|
+
when OpenStudio::Model::CoilCoolingDXSingleSpeed.iddObjectType
|
105
|
+
coil = e.to_CoilCoolingDXSingleSpeed.get
|
106
|
+
coil.isRatedTotalCoolingCapacityAutosized && coil.isRatedSensibleHeatRatioAutosized && coil.isRatedAirFlowRateAutosized ? 'Yes' : 'No'
|
107
|
+
when OpenStudio::Model::CoilCoolingWater.iddObjectType
|
108
|
+
coil = e.to_CoilCoolingWater.get
|
109
|
+
coil.isDesignWaterFlowRateAutosized && coil.isDesignAirFlowRateAutosized && coil.isDesignInletWaterTemperatureAutosized ? 'Yes' : 'No'
|
110
|
+
when OpenStudio::Model::CoilCoolingWaterToAirHeatPumpEquationFit.iddObjectType
|
111
|
+
coil = e.to_CoilCoolingWaterToAirHeatPumpEquationFit.get
|
112
|
+
coil.isRatedTotalCoolingCapacityAutosized && coil.isRatedSensibleHeatRatioAutosized && coil.isRatedAirFlowRateAutosized ? 'Yes' : 'No'
|
113
|
+
else
|
114
|
+
'?'
|
115
|
+
end
|
116
|
+
|
117
|
+
coolingcap = convert_prop(props['Nominal Total Capacity'], 'kBtu/hr')
|
118
|
+
coolingeff = convert_prop(props['Nominal Efficiency'], 'COP')
|
119
|
+
sensibleheatratio = convert_prop(props['Nominal Sensible Heat Ratio'], '')
|
120
|
+
parent_name = parent.name.get
|
121
|
+
name = e.name.get
|
122
|
+
coiltype = niceClassName(e.iddObjectType.valueName)
|
123
|
+
|
124
|
+
{ 'Terminal/Zone Equip Name' => parent_name, 'Coil Type' => coiltype, 'Name' => name, 'Autosized' => autosized, 'Nominal Capacity' => coolingcap, 'Nominal Efficiency' => coolingeff, 'Nominal SHR' => sensibleheatratio }
|
125
|
+
end
|
126
|
+
|
127
|
+
def properties_for_heating_coil(e, props, parent)
|
128
|
+
autosized = '-'
|
129
|
+
eff_units = 'COP'
|
130
|
+
heatingcap = ['-', '-']
|
131
|
+
heatingeff = ['-', '-']
|
132
|
+
case e.iddObjectType
|
133
|
+
when OpenStudio::Model::CoilHeatingElectric.iddObjectType
|
134
|
+
coil = e.to_CoilHeatingElectric.get
|
135
|
+
autosized = coil.isNominalCapacityAutosized ? 'Yes' : 'No'
|
136
|
+
heatingcap = convert_prop(props['Nominal Total Capacity'], 'kW')
|
137
|
+
heatingeff = convert_prop(props['Nominal Efficiency'], '%')
|
138
|
+
when OpenStudio::Model::CoilHeatingGas.iddObjectType
|
139
|
+
coil = e.to_CoilHeatingGas.get
|
140
|
+
autosized = coil.isNominalCapacityAutosized ? 'Yes' : 'No'
|
141
|
+
heatingcap = convert_prop(props['Nominal Total Capacity'], 'kBtu/hr')
|
142
|
+
heatingeff = convert_prop(props['Nominal Efficiency'], 'COP')
|
143
|
+
when OpenStudio::Model::CoilHeatingDXSingleSpeed.iddObjectType
|
144
|
+
coil = e.to_CoilHeatingDXSingleSpeed.get
|
145
|
+
autosized = coil.isRatedTotalHeatingCapacityAutosized && coil.isRatedAirFlowRateAutosized ? 'Yes' : 'No'
|
146
|
+
heatingcap = convert_prop(props['Nominal Total Capacity'], 'kBtu/hr')
|
147
|
+
heatingeff = convert_prop(props['Nominal Efficiency'], 'COP')
|
148
|
+
when OpenStudio::Model::CoilHeatingWater.iddObjectType
|
149
|
+
coil = e.to_CoilHeatingWater.get
|
150
|
+
autosized = coil.isMaximumWaterFlowRateAutosized && coil.isRatedCapacityAutosized && coil.isUFactorTimesAreaValueAutosized ? 'Yes' : 'No'
|
151
|
+
heatingcap = convert_prop(props['Nominal Total Capacity'], 'kBtu/hr')
|
152
|
+
heatingeff = ['-', '-']
|
153
|
+
when OpenStudio::Model::CoilHeatingWaterBaseboard.iddObjectType
|
154
|
+
coil = e.to_CoilHeatingWaterBaseboard.get
|
155
|
+
autosized = coil.isMaximumWaterFlowRateAutosized && coil.isUFactorTimesAreaValueAutosized ? 'Yes' : 'No'
|
156
|
+
heatingcap = convert_prop(props['Nominal Total Capacity'], 'kBtu/hr')
|
157
|
+
heatingeff = ['-', '-']
|
158
|
+
when OpenStudio::Model::CoilHeatingWaterToAirHeatPumpEquationFit.iddObjectType
|
159
|
+
coil = e.to_CoilHeatingWaterToAirHeatPumpEquationFit.get
|
160
|
+
autosized = coil.isRatedHeatingCapacityAutosized && coil.isRatedAirFlowRateAutosized && coil.isRatedWaterFlowRateAutosized ? 'Yes' : 'No'
|
161
|
+
heatingcap = convert_prop(props['Nominal Total Capacity'], 'kBtu/hr')
|
162
|
+
heatingeff = convert_prop(props['Nominal Efficiency'], 'COP')
|
163
|
+
else
|
164
|
+
autosized = '?'
|
165
|
+
end
|
166
|
+
|
167
|
+
parent_name = parent.name.get
|
168
|
+
name = e.name.get
|
169
|
+
coiltype = niceClassName(e.iddObjectType.valueName)
|
170
|
+
|
171
|
+
{ 'Terminal/Zone Equip Name' => parent_name, 'Coil Type' => coiltype, 'Name' => name, 'Autosized' => autosized, 'Nominal Capacity' => heatingcap, 'Nominal Efficiency' => heatingeff }
|
172
|
+
end
|
173
|
+
|
174
|
+
def properties_for_fan(equipment, properties, parent)
|
175
|
+
autosized = case equipment.iddObjectType
|
176
|
+
when OpenStudio::Model::FanConstantVolume.iddObjectType
|
177
|
+
fan = equipment.to_FanConstantVolume.get
|
178
|
+
fan.isMaximumFlowRateAutosized ? 'Yes' : 'No'
|
179
|
+
when OpenStudio::Model::FanVariableVolume.iddObjectType
|
180
|
+
fan = equipment.to_FanVariableVolume.get
|
181
|
+
fan.isMaximumFlowRateAutosized ? 'Yes' : 'No'
|
182
|
+
when OpenStudio::Model::FanOnOff.iddObjectType
|
183
|
+
fan = equipment.to_FanOnOff.get
|
184
|
+
fan.isMaximumFlowRateAutosized ? 'Yes' : 'No'
|
185
|
+
when OpenStudio::Model::FanZoneExhaust.iddObjectType
|
186
|
+
'N/A'
|
187
|
+
else
|
188
|
+
'?'
|
189
|
+
end
|
190
|
+
|
191
|
+
name = equipment.name.get
|
192
|
+
parent_name = parent.name.get
|
193
|
+
maxflowrate = convert_prop(properties['Max Air Flow Rate'], 'cfm')
|
194
|
+
fanpressure = convert_prop(properties['Delta Pressure'], 'inH_{2}O')
|
195
|
+
totalefficiency = convert_prop(properties['Total Efficiency'], '%')
|
196
|
+
ratedfanpower = convert_prop(properties['Rated Electric Power'], 'W')
|
197
|
+
{ 'Terminal/Zone Equip Name' => parent_name, 'Fan Type' => niceClassName(equipment.iddObjectType.valueName), 'Name' => name, 'Autosized' => autosized, 'Max Flow' => maxflowrate, 'Efficiency' => totalefficiency, 'Pressure' => fanpressure, 'Power' => ratedfanpower }
|
198
|
+
end
|
199
|
+
|
200
|
+
# Each equipment is grouped into Heating, Cooling or Fans and has properties extracted appropriate for that group.
|
201
|
+
# This method returns a tuple [equipment group, properties] where properties is a hash of name, value or name, [value, units] pairs
|
202
|
+
def properties_for_zone_equipment(equipment, equipment_properties, parent)
|
203
|
+
# Make properties an empty hash if it was nil
|
204
|
+
equipment_properties ||= {}
|
205
|
+
if /^OS_Coil_Cooling/ =~ equipment.iddObjectType.valueName
|
206
|
+
return 'Cooling', properties_for_cooling_coil(equipment, equipment_properties, parent)
|
207
|
+
elsif /^OS_Coil_Heating/ =~ equipment.iddObjectType.valueName
|
208
|
+
return 'Heating', properties_for_heating_coil(equipment, equipment_properties, parent)
|
209
|
+
elsif /^OS_Fan/ =~ equipment.iddObjectType.valueName
|
210
|
+
return 'Fans', properties_for_fan(equipment, equipment_properties, parent)
|
211
|
+
elsif OpenStudio::Model::ZoneHVACBaseboardConvectiveElectric.iddObjectType == equipment.iddObjectType
|
212
|
+
baseboard = equipment.to_ZoneHVACBaseboardConvectiveElectric.get
|
213
|
+
autosized = baseboard.isNominalCapacityAutosized ? 'Yes' : 'No'
|
214
|
+
heatingcap = convert_prop(equipment_properties['Design Size Nominal Capacity'], 'kW')
|
215
|
+
name = baseboard.name.get
|
216
|
+
coiltype = niceClassName(baseboard.iddObjectType.valueName)
|
217
|
+
return 'Heating', { 'Terminal/Zone Equip Name' => name, 'Coil Type' => coiltype, 'Name' => name, 'Autosized' => autosized, 'Nominal Capacity' => heatingcap, 'Nominal Efficiency' => [100, '%'] }
|
218
|
+
else
|
219
|
+
return nil, nil
|
220
|
+
end
|
221
|
+
end
|
222
|
+
|
223
|
+
# define what happens when the measure is run
|
224
|
+
def run(runner, user_arguments)
|
225
|
+
super(runner, user_arguments)
|
226
|
+
|
227
|
+
# use the built-in error checking
|
228
|
+
return false unless runner.validateUserArguments(arguments, user_arguments)
|
229
|
+
|
230
|
+
@runner = runner
|
231
|
+
|
232
|
+
# get the last model and sql file
|
233
|
+
|
234
|
+
model = runner.lastOpenStudioModel
|
235
|
+
if model.empty?
|
236
|
+
runner.registerError('Cannot find last model.')
|
237
|
+
return false
|
238
|
+
end
|
239
|
+
model = model.get
|
240
|
+
|
241
|
+
@sqlFile = runner.lastEnergyPlusSqlFile
|
242
|
+
if @sqlFile.empty?
|
243
|
+
runner.registerError('Cannot find last sql file.')
|
244
|
+
return false
|
245
|
+
end
|
246
|
+
@sqlFile = @sqlFile.get
|
247
|
+
model.setSqlFile(@sqlFile)
|
248
|
+
|
249
|
+
@graph_data = []
|
250
|
+
|
251
|
+
# Collect and collate information about the hvac equipment
|
252
|
+
# When we are done we will have a hash of equipment keyed by equipment name
|
253
|
+
# who's values are hashes keyed by the property name. The values of the
|
254
|
+
# property hashes are [value, unit] pairs
|
255
|
+
# Some types of equipment (for instance CoilCoolingCooledBeams) are not present
|
256
|
+
# in the database as of OS 1.5
|
257
|
+
|
258
|
+
# find table names, etc. in the SupportZoneHVACEquipFields
|
259
|
+
tableNames = getDataByColumn('TableName')
|
260
|
+
columnNames = getDataByColumn('ColumnName')
|
261
|
+
rowNames = getDataByColumn('RowName')
|
262
|
+
units = getDataByColumn('Units')
|
263
|
+
values = getDataByColumn('Value')
|
264
|
+
equipment_rows = tableNames.zip(columnNames, rowNames, units, values)
|
265
|
+
equipment = {}
|
266
|
+
equipment_rows.each do |r|
|
267
|
+
_, field, name, units, value = r
|
268
|
+
next unless name != 'None'
|
269
|
+
eh = equipment[name] || {}
|
270
|
+
eh[field] = [value, units]
|
271
|
+
equipment[name] = eh
|
272
|
+
end
|
273
|
+
|
274
|
+
@zone_collection = []
|
275
|
+
|
276
|
+
@testData = {}
|
277
|
+
|
278
|
+
# Go through each zone in the model and collect all the zone equipment
|
279
|
+
model.getThermalZones.sort.each do |thermalZone|
|
280
|
+
# Skip unconditioned zones
|
281
|
+
if thermalZone.thermostatSetpointDualSetpoint.empty?
|
282
|
+
@runner.registerInfo("Skipping #{thermalZone.name} because it is unconditioned.")
|
283
|
+
next
|
284
|
+
end
|
285
|
+
|
286
|
+
# Skip plenums
|
287
|
+
if thermalZone.isPlenum == true
|
288
|
+
@runner.registerInfo("Skipping #{thermalZone.name} because it is a plenum.")
|
289
|
+
next
|
290
|
+
end
|
291
|
+
|
292
|
+
zone_equipment = {}
|
293
|
+
|
294
|
+
thermalZone.equipment.each do |e|
|
295
|
+
childquipment = e.to_ParentObject.get.children
|
296
|
+
# Look for data on the top level equipment and each of that equipment's children
|
297
|
+
found_primary_heat = false
|
298
|
+
([e] + childquipment).each do |ce|
|
299
|
+
ename = ce.name.get.upcase
|
300
|
+
reporting_type, equipment_properties = properties_for_zone_equipment(ce, equipment[ename], e)
|
301
|
+
next unless equipment_properties
|
302
|
+
# The first heating coil in a zone equipment chain will be primary heating; all
|
303
|
+
# following heating coils will be marked as Backup Heating.
|
304
|
+
if reporting_type == 'Heating'
|
305
|
+
reporting_type = 'Backup Heating' if found_primary_heat
|
306
|
+
found_primary_heat = true
|
307
|
+
end
|
308
|
+
# Find the priority for heating or cooling equipment - the priority is based on the 'parent' zone equipment priority
|
309
|
+
if reporting_type == 'Heating'
|
310
|
+
equipment_properties['Priority'] = (thermalZone.equipmentInHeatingOrder.index { |ze| ze.name.get == e.name.get && ze.iddObjectType == e.iddObjectType }) + 1
|
311
|
+
end
|
312
|
+
if reporting_type == 'Cooling'
|
313
|
+
equipment_properties['Priority'] = (thermalZone.equipmentInCoolingOrder.index { |ze| ze.name.get == e.name.get && ze.iddObjectType == e.iddObjectType }) + 1
|
314
|
+
end
|
315
|
+
# If we don't yet have equipment of this type, make a new array. Then add
|
316
|
+
# our equipment information to the array and update our zone equipment hash.
|
317
|
+
equip_of_type = zone_equipment[reporting_type] || []
|
318
|
+
equip_of_type << equipment_properties
|
319
|
+
zone_equipment[reporting_type] = equip_of_type
|
320
|
+
end
|
321
|
+
end
|
322
|
+
|
323
|
+
zoneMetrics = {}
|
324
|
+
zoneMetrics[:name] = !thermalZone.name.empty? ? thermalZone.name.get : ''
|
325
|
+
zoneMetrics[:equipment] = zone_equipment
|
326
|
+
zoneMetrics[:area] = OpenStudio.convert(thermalZone.floorArea, 'm^2', 'ft^2').get.round(2)
|
327
|
+
|
328
|
+
@currentZoneName = zoneMetrics[:name]
|
329
|
+
|
330
|
+
vals = {}
|
331
|
+
|
332
|
+
vals[:va] = getDetailsData('LightingSummary', 'Entire Facility', 'Interior Lighting', "#{zoneMetrics[:name]}%", 'Lighting Power Density', 'W/m2', 'W/ft^2').round(2)
|
333
|
+
vals[:vb] = getDetailsData('LightingSummary', 'Entire Facility', 'Interior Lighting', "#{zoneMetrics[:name]}%", 'Full Load Hours/Week', 'hr', 'hr').round(2)
|
334
|
+
|
335
|
+
vals[:vc] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Electricity', "InteriorLights:Electricity:Zone:#{zoneMetrics[:name]}", 'Electricity Annual Value', 'GJ', 'kWh').round(2)
|
336
|
+
|
337
|
+
vals[:vd] = getDetailsData('InputVerificationandResultsSummary', 'Entire Facility', 'Zone Summary', zoneMetrics[:name], 'Plug and Process', 'W/m2', 'W/ft^2').round(2)
|
338
|
+
|
339
|
+
vals[:ve] = getDetailsData('InputVerificationandResultsSummary', 'Entire Facility', 'Zone Summary', zoneMetrics[:name], 'Area', 'm2', 'ft^2').round(2)
|
340
|
+
|
341
|
+
vals[:vf] = getDetailsData('InputVerificationandResultsSummary', 'Entire Facility', 'Zone Summary', zoneMetrics[:name], 'Conditioned (Y/N)', '', 's')
|
342
|
+
vals[:vf] = 'No' if vals[:vf] == ''
|
343
|
+
|
344
|
+
vals[:vg] = getDetailsData('HVACSizingSummary', 'Entire Facility', 'Zone Heating', zoneMetrics[:name], 'User Design Load', 'W', 'kBtu/hr').round(2)
|
345
|
+
vals[:vh] = getDetailsData('HVACSizingSummary', 'Entire Facility', 'Zone Heating', zoneMetrics[:name], 'User Design Air Flow', 'm3/s', 'ft^3/s').round(2)
|
346
|
+
vals[:vi] = getDetailsData('HVACSizingSummary', 'Entire Facility', 'Zone Cooling', zoneMetrics[:name], 'User Design Load', 'W', 'kBtu/hr').round(2)
|
347
|
+
vals[:vj] = getDetailsData('HVACSizingSummary', 'Entire Facility', 'Zone Cooling', zoneMetrics[:name], 'User Design Air Flow', 'm3/s', 'ft^3/s').round(2)
|
348
|
+
|
349
|
+
vals[:vk] = getDetailsData('OutdoorAirSummary', 'Entire Facility', 'Average Outdoor Air During Occupied Hours', zoneMetrics[:name], 'Mechanical Ventilation', 'ach', 'ach').round(2)
|
350
|
+
vals[:vl] = getDetailsData('OutdoorAirSummary', 'Entire Facility', 'Average Outdoor Air During Occupied Hours', zoneMetrics[:name], 'Infiltration', 'ach', 'ach').round(2)
|
351
|
+
|
352
|
+
vals[:vm] = getDetailsData('InputVerificationandResultsSummary', 'Entire Facility', 'Zone Summary', zoneMetrics[:name], 'People', 'm2 per person', 'ft^2/person').round(2)
|
353
|
+
|
354
|
+
vals[:vn] = getDetailsData('HVACSizingSummary', 'Entire Facility', 'Zone Cooling', zoneMetrics[:name], 'Date/Time Of Peak', '', 's')
|
355
|
+
vals[:vo] = getDetailsData('HVACSizingSummary', 'Entire Facility', 'Zone Heating', zoneMetrics[:name], 'Date/Time Of Peak', '', 's')
|
356
|
+
|
357
|
+
vals[:vp] = getDetailsData('SystemSummary', 'Entire Facility', 'Time Setpoint Not Met', zoneMetrics[:name], 'During Heating', 'hr', 'hr')
|
358
|
+
vals[:vq] = getDetailsData('SystemSummary', 'Entire Facility', 'Time Setpoint Not Met', zoneMetrics[:name], 'During Cooling', 'hr', 'hr')
|
359
|
+
|
360
|
+
vals[:vr] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Gas', "InteriorEquipment:Gas:Zone:#{zoneMetrics[:name]}", 'Electricity Annual Value', 'GJ', 'Therm').round(2)
|
361
|
+
vals[:vs] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Electricity', "InteriorEquipment:Electricity:Zone:#{zoneMetrics[:name]}", 'Electricity Annual Value', 'GJ', 'kWh').round(2)
|
362
|
+
|
363
|
+
vals[:vt] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Gas', "InteriorEquipment:Gas:Zone:#{zoneMetrics[:name]}", 'Gas Maximum Value', 'W', 'kBtu/hr').round(2)
|
364
|
+
vals[:vu] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Gas', "InteriorEquipment:Gas:Zone:#{zoneMetrics[:name]}", 'Timestamp of Maximum', '', 's')
|
365
|
+
|
366
|
+
vals[:vv] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Electricity', "InteriorLights:Electricity:Zone:#{zoneMetrics[:name]}", 'Electricity Maximum Value', 'W', 'kW').round(2)
|
367
|
+
vals[:vw] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Electricity', "InteriorLights:Electricity:Zone:#{zoneMetrics[:name]}", 'Timestamp of Maximum', '', 's')
|
368
|
+
|
369
|
+
# X unused
|
370
|
+
|
371
|
+
vals[:vy] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Electricity', "InteriorEquipment:Electricity:Zone:#{zoneMetrics[:name]}", 'Electricity Maximum Value', 'W', 'kW').round(2)
|
372
|
+
vals[:vz] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Electricity', "InteriorEquipment:Electricity:Zone:#{zoneMetrics[:name]}", 'Timestamp of Maximum', '', 's')
|
373
|
+
|
374
|
+
vals[:vaa] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Other', "InteriorEquipment:DistrictHeating:Zone:#{zoneMetrics[:name]}", 'Annual Value', 'GJ', 'kBtu').round(2)
|
375
|
+
vals[:vab] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Other', "InteriorEquipment:DistrictHeating:Zone:#{zoneMetrics[:name]}", 'Maximum Value', 'W', 'kBtu/hr').round(2)
|
376
|
+
vals[:vac] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Other', "InteriorEquipment:DistrictHeating:Zone:#{zoneMetrics[:name]}", 'Timestamp of Maximum', '', 's')
|
377
|
+
|
378
|
+
vals[:vad] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Other', "Heating:EnergyTransfer:Zone:#{zoneMetrics[:name]}", 'Annual Value', 'GJ', 'kBtu').round(2)
|
379
|
+
vals[:vae] = (getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Other', "Heating:EnergyTransfer:Zone:#{zoneMetrics[:name]}", 'Maximum Value', 'W', 'kBtu/hr') / zoneMetrics[:area]).round(2)
|
380
|
+
vals[:vaf] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Other', "Heating:EnergyTransfer:Zone:#{zoneMetrics[:name]}", 'Timestamp of Maximum', '', 's')
|
381
|
+
|
382
|
+
vals[:vag] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Other', "Cooling:EnergyTransfer:Zone:#{zoneMetrics[:name]}", 'Annual Value', 'GJ', 'kBtu').round(2)
|
383
|
+
vals[:vah] = (getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Other', "Cooling:EnergyTransfer:Zone:#{zoneMetrics[:name]}", 'Maximum Value', 'W', 'kBtu/hr') / zoneMetrics[:area]).round(2)
|
384
|
+
vals[:vai] = getDetailsData('EnergyMeters', 'Entire Facility', 'Annual and Peak Values - Other', "Cooling:EnergyTransfer:Zone:#{zoneMetrics[:name]}", 'Timestamp of Maximum', '', 's')
|
385
|
+
|
386
|
+
vals[:vaj] = zoneHeatComponentCalc('People', zoneMetrics)
|
387
|
+
vals[:vak] = zoneHeatComponentCalc('Lights', zoneMetrics)
|
388
|
+
vals[:val] = zoneHeatComponentCalc('Equipment', zoneMetrics)
|
389
|
+
vals[:vam] = zoneHeatComponentCalc('Refrigeration', zoneMetrics)
|
390
|
+
vals[:van] = zoneHeatComponentCalc('Water Use Equipment', zoneMetrics)
|
391
|
+
vals[:vao] = zoneHeatComponentCalc('HVAC Equipment Losses', zoneMetrics)
|
392
|
+
vals[:vap] = zoneHeatComponentCalc('Power Generation Equipment', zoneMetrics)
|
393
|
+
vals[:vaq] = zoneHeatComponentCalc('Infiltration', zoneMetrics)
|
394
|
+
vals[:var] = zoneHeatComponentCalc('Zone Ventilation', zoneMetrics)
|
395
|
+
vals[:vas] = zoneHeatComponentCalc('Interzone Mixing', zoneMetrics)
|
396
|
+
vals[:vat] = zoneHeatComponentCalc('Exterior Wall', zoneMetrics)
|
397
|
+
vals[:vau] = zoneHeatComponentCalc('Interzone Wall', zoneMetrics)
|
398
|
+
vals[:vav] = zoneHeatComponentCalc('Ground Contact Wall', zoneMetrics)
|
399
|
+
vals[:vaw] = zoneHeatComponentCalc('Other Wall', zoneMetrics)
|
400
|
+
vals[:vax] = zoneHeatComponentCalc('Opaque Door', zoneMetrics)
|
401
|
+
vals[:vay] = zoneHeatComponentCalc('Roof', zoneMetrics)
|
402
|
+
vals[:vaz] = zoneHeatComponentCalc('Interzone Ceiling', zoneMetrics)
|
403
|
+
vals[:vba] = zoneHeatComponentCalc('Other Roof', zoneMetrics)
|
404
|
+
vals[:vbb] = zoneHeatComponentCalc('Exterior Floor', zoneMetrics)
|
405
|
+
vals[:vbc] = zoneHeatComponentCalc('Interzone Floor', zoneMetrics)
|
406
|
+
vals[:vbd] = zoneHeatComponentCalc('Ground Contact Floor', zoneMetrics)
|
407
|
+
vals[:vbe] = zoneHeatComponentCalc('Other Floor', zoneMetrics)
|
408
|
+
vals[:vbf] = zoneHeatComponentCalc('Fenestration Conduction', zoneMetrics)
|
409
|
+
vals[:vbg] = zoneHeatComponentCalc('Fenestration Solar', zoneMetrics)
|
410
|
+
|
411
|
+
vals[:vbh] = getDetailsData('ZoneComponentLoadSummary', (zoneMetrics[:name]).to_s, 'Heating Peak Conditions', 'Time of Peak Load', 'Value', '', 's')
|
412
|
+
|
413
|
+
vals[:vbi] = zoneCoolComponentCalc('People', zoneMetrics)
|
414
|
+
vals[:vbj] = zoneCoolComponentCalc('Lights', zoneMetrics)
|
415
|
+
vals[:vbk] = zoneCoolComponentCalc('Equipment', zoneMetrics)
|
416
|
+
vals[:vbl] = zoneCoolComponentCalc('Refrigeration', zoneMetrics)
|
417
|
+
vals[:vbm] = zoneCoolComponentCalc('Water Use Equipment', zoneMetrics)
|
418
|
+
vals[:vbn] = zoneCoolComponentCalc('HVAC Equipment Losses', zoneMetrics)
|
419
|
+
vals[:vbo] = zoneCoolComponentCalc('Power Generation Equipment', zoneMetrics)
|
420
|
+
vals[:vbp] = zoneCoolComponentCalc('Infiltration', zoneMetrics)
|
421
|
+
vals[:vbq] = zoneCoolComponentCalc('Zone Ventilation', zoneMetrics)
|
422
|
+
vals[:vbr] = zoneCoolComponentCalc('Interzone Mixing', zoneMetrics)
|
423
|
+
vals[:vbs] = zoneCoolComponentCalc('Exterior Wall', zoneMetrics)
|
424
|
+
vals[:vbt] = zoneCoolComponentCalc('Interzone Wall', zoneMetrics)
|
425
|
+
vals[:vbu] = zoneCoolComponentCalc('Ground Contact Wall', zoneMetrics)
|
426
|
+
vals[:vbv] = zoneCoolComponentCalc('Other Wall', zoneMetrics)
|
427
|
+
vals[:vbw] = zoneCoolComponentCalc('Opaque Door', zoneMetrics)
|
428
|
+
vals[:vbx] = zoneCoolComponentCalc('Roof', zoneMetrics)
|
429
|
+
vals[:vby] = zoneCoolComponentCalc('Interzone Ceiling', zoneMetrics)
|
430
|
+
vals[:vbz] = zoneCoolComponentCalc('Other Roof', zoneMetrics)
|
431
|
+
vals[:vca] = zoneCoolComponentCalc('Exterior Floor', zoneMetrics)
|
432
|
+
vals[:vcb] = zoneCoolComponentCalc('Interzone Floor', zoneMetrics)
|
433
|
+
vals[:vcc] = zoneCoolComponentCalc('Ground Contact Floor', zoneMetrics)
|
434
|
+
vals[:vcd] = zoneCoolComponentCalc('Other Floor', zoneMetrics)
|
435
|
+
vals[:vce] = zoneCoolComponentCalc('Fenestration Conduction', zoneMetrics)
|
436
|
+
vals[:vcf] = zoneCoolComponentCalc('Fenestration Solar', zoneMetrics)
|
437
|
+
|
438
|
+
vals[:vcg] = getDetailsData('ZoneComponentLoadSummary', (zoneMetrics[:name]).to_s, 'Cooling Peak Conditions', 'Time of Peak Load', 'Value', '', 's')
|
439
|
+
|
440
|
+
# vals = loadTestVals( vals )
|
441
|
+
|
442
|
+
vals[:sumBasicHeating] = (vals[:vaj] + vals[:vak] + vals[:val] + vals[:vam] + vals[:vaq] + vals[:var] + vals[:vas]).round(2)
|
443
|
+
vals[:sumBasicCooling] = (vals[:vbi] + vals[:vbj] + vals[:vbk] + vals[:vbl] + vals[:vbp] + vals[:vbq] + vals[:vbr]).round(2)
|
444
|
+
|
445
|
+
vals[:sumOtherHeating] = (vals[:van] + vals[:vao] + vals[:vap]).round(2)
|
446
|
+
vals[:sumOtherCooling] = (vals[:vbm] + vals[:vbn] + vals[:vbo]).round(2)
|
447
|
+
|
448
|
+
vals[:sumWallDoorHeating] = (vals[:vat] + vals[:vau] + vals[:vav] + vals[:vaw] + vals[:vax]).round(2)
|
449
|
+
vals[:sumWallDoorCooling] = (vals[:vbs] + vals[:vbt] + vals[:vbu] + vals[:vbv] + vals[:vbw]).round(2)
|
450
|
+
|
451
|
+
vals[:sumRoofCeilingHeating] = (vals[:vay] + vals[:vaz] + vals[:vba]).round(2)
|
452
|
+
vals[:sumRoofCeilingCooling] = (vals[:vbx] + vals[:vby] + vals[:vbz]).round(2)
|
453
|
+
|
454
|
+
vals[:sumFloorHeating] = (vals[:vbb] + vals[:vbc] + vals[:vbd] + vals[:vbe]).round(2)
|
455
|
+
vals[:sumFloorCooling] = (vals[:vca] + vals[:vcb] + vals[:vcc] + vals[:vcd]).round(2)
|
456
|
+
|
457
|
+
vals[:sumWindowsHeating] = (vals[:vbf] + vals[:vbg]).round(2)
|
458
|
+
vals[:sumWindowsCooling] = (vals[:vce] + vals[:vcf]).round(2)
|
459
|
+
|
460
|
+
vals[:sumHeatingTotal] = (vals[:sumBasicHeating] + vals[:sumOtherHeating] + vals[:sumWallDoorHeating] + vals[:sumRoofCeilingHeating] + vals[:sumFloorHeating] + vals[:sumWindowsHeating]).round(2)
|
461
|
+
vals[:sumCoolingTotal] = (vals[:sumBasicCooling] + vals[:sumOtherCooling] + vals[:sumWallDoorCooling] + vals[:sumRoofCeilingCooling] + vals[:sumFloorCooling] + vals[:sumWindowsCooling]).round(2)
|
462
|
+
|
463
|
+
zoneMetrics[:vals] = vals
|
464
|
+
|
465
|
+
@zone_collection.push(zoneMetrics)
|
466
|
+
|
467
|
+
stacked_bars(zoneMetrics)
|
468
|
+
end
|
469
|
+
|
470
|
+
@zone_collection = @zone_collection.sort_by { |z| z[:name] }
|
471
|
+
|
472
|
+
equipment_lengths = @zone_collection.map { |z| z[:equipment].values.map(&:length) }
|
473
|
+
max_zone_equipments = equipment_lengths.flatten.max
|
474
|
+
|
475
|
+
# Convert the graph data to JSON
|
476
|
+
# This measure requires ruby 2.0.0 to create the JSON for the report graph
|
477
|
+
if RUBY_VERSION >= '2.0.0'
|
478
|
+
require 'json'
|
479
|
+
@graph_data = @graph_data.to_json
|
480
|
+
else
|
481
|
+
runner.registerInfo("This Measure needs Ruby 2.0.0 to generate timeseries graphs on the report. You have Ruby #{RUBY_VERSION}. OpenStudio 1.4.2 and higher user Ruby 2.0.0.")
|
482
|
+
end
|
483
|
+
|
484
|
+
web_asset_path = OpenStudio.getSharedResourcesPath / OpenStudio::Path.new('web_assets')
|
485
|
+
|
486
|
+
html_in = getResourceFileData('report.html.in')
|
487
|
+
|
488
|
+
# configure template with variable values
|
489
|
+
renderer = ERB.new(html_in)
|
490
|
+
html_out = renderer.result(binding)
|
491
|
+
|
492
|
+
writeResourceFileData('report.html', html_out)
|
493
|
+
# copyResourceFile( "graph_resource.js" )
|
494
|
+
# copyResourceFile( "style_resource.css" )
|
495
|
+
|
496
|
+
# closing the sql file
|
497
|
+
@sqlFile.close
|
498
|
+
|
499
|
+
# reporting final condition
|
500
|
+
runner.registerFinalCondition("Successfully finished writing 'Zone Report'.")
|
501
|
+
|
502
|
+
true
|
503
|
+
end # end the run method
|
504
|
+
|
505
|
+
def zoneHeatComponentCalc(component, zoneMetrics)
|
506
|
+
(getDetailsData('ZoneComponentLoadSummary', (zoneMetrics[:name]).to_s, 'Estimated Heating Peak Load Components', component, 'Total', 'W', 'Btu/hr') / zoneMetrics[:area]).round(2)
|
507
|
+
end
|
508
|
+
|
509
|
+
def zoneCoolComponentCalc(component, zoneMetrics)
|
510
|
+
(getDetailsData('ZoneComponentLoadSummary', (zoneMetrics[:name]).to_s, 'Estimated Cooling Peak Load Components', component, 'Total', 'W', 'Btu/hr') / zoneMetrics[:area]).round(2)
|
511
|
+
end
|
512
|
+
|
513
|
+
def stacked_bars(zoneMetrics)
|
514
|
+
# people, lights, equipment, refrigeration, other, infiltration, zone ventilation,
|
515
|
+
# interzone mixing, walls/doors, roof/ceiling, floor, windows
|
516
|
+
|
517
|
+
vals = zoneMetrics[:vals]
|
518
|
+
|
519
|
+
heatingVals = [vals[:vaj],
|
520
|
+
vals[:vak],
|
521
|
+
vals[:val],
|
522
|
+
vals[:vam],
|
523
|
+
vals[:sumOtherHeating],
|
524
|
+
vals[:vaq],
|
525
|
+
vals[:var],
|
526
|
+
vals[:vas],
|
527
|
+
vals[:sumWallDoorHeating],
|
528
|
+
vals[:sumRoofCeilingHeating],
|
529
|
+
vals[:sumFloorHeating],
|
530
|
+
vals[:sumWindowsHeating]]
|
531
|
+
|
532
|
+
coolingVals = [vals[:vbi],
|
533
|
+
vals[:vbj],
|
534
|
+
vals[:vbk],
|
535
|
+
vals[:vbl],
|
536
|
+
vals[:sumOtherCooling],
|
537
|
+
vals[:vbp],
|
538
|
+
vals[:vbq],
|
539
|
+
vals[:vbr],
|
540
|
+
vals[:sumWallDoorCooling],
|
541
|
+
vals[:sumRoofCeilingCooling],
|
542
|
+
vals[:sumFloorCooling],
|
543
|
+
vals[:sumWindowsCooling]]
|
544
|
+
|
545
|
+
positiveHeating = heatingVals.select { |v| v >= 0 }.inject { |sum, x| sum + x } || 0
|
546
|
+
negativeHeating = heatingVals.select { |v| v < 0 }.inject { |sum, x| sum + x } || 0
|
547
|
+
|
548
|
+
positiveCooling = coolingVals.select { |v| v >= 0 }.inject { |sum, x| sum + x } || 0
|
549
|
+
negativeCooling = coolingVals.select { |v| v < 0 }.inject { |sum, x| sum + x } || 0
|
550
|
+
|
551
|
+
maxPositive = positiveHeating > positiveCooling ? positiveHeating : positiveCooling
|
552
|
+
maxNegative = negativeHeating < negativeCooling ? negativeHeating : negativeCooling
|
553
|
+
|
554
|
+
maxPositive = maxPositive == 0 ? maxNegative * -0.20 : maxPositive
|
555
|
+
maxNegative = maxNegative == 0 ? maxPositive * -0.10 : maxNegative
|
556
|
+
|
557
|
+
maxPositive += (maxPositive * 0.20).round(2)
|
558
|
+
maxNegative += (maxNegative * 0.10).round(2)
|
559
|
+
|
560
|
+
stacked_vals = [[0, maxPositive, maxNegative, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
|
561
|
+
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, vals[:sumHeatingTotal], 0],
|
562
|
+
[2,
|
563
|
+
vals[:vaj],
|
564
|
+
vals[:vak],
|
565
|
+
vals[:val],
|
566
|
+
vals[:vam],
|
567
|
+
vals[:sumOtherHeating],
|
568
|
+
vals[:vaq],
|
569
|
+
vals[:var],
|
570
|
+
vals[:vas],
|
571
|
+
vals[:sumWallDoorHeating],
|
572
|
+
vals[:sumRoofCeilingHeating],
|
573
|
+
vals[:sumFloorHeating],
|
574
|
+
vals[:sumWindowsHeating],
|
575
|
+
0,
|
576
|
+
0],
|
577
|
+
[3,
|
578
|
+
vals[:vbi],
|
579
|
+
vals[:vbj],
|
580
|
+
vals[:vbk],
|
581
|
+
vals[:vbl],
|
582
|
+
vals[:sumOtherCooling],
|
583
|
+
vals[:vbp],
|
584
|
+
vals[:vbq],
|
585
|
+
vals[:vbr],
|
586
|
+
vals[:sumWallDoorCooling],
|
587
|
+
vals[:sumRoofCeilingCooling],
|
588
|
+
vals[:sumFloorCooling],
|
589
|
+
vals[:sumWindowsCooling],
|
590
|
+
0,
|
591
|
+
0],
|
592
|
+
[4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, vals[:sumCoolingTotal]],
|
593
|
+
[5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]]
|
594
|
+
|
595
|
+
# Add the hourly load data to JSON for the report.html
|
596
|
+
graph = {}
|
597
|
+
graph['title'] = ''
|
598
|
+
graph['xaxislabel'] = ''
|
599
|
+
graph['yaxislabel'] = 'Contribution Btu/hr/ft2'
|
600
|
+
graph['labels'] = %w[index people lights equipment refrigeration other infiltration zone_ventilation interzone_mixing walls/doors roof/ceiling floor windows net/heating net/cooling]
|
601
|
+
graph['colors'] = ['#888855', '#AAAA55', '#3333AA', '#8888FF', '#888888', '#9999FF', '#AAAAFF', '#AA6666', '#777733', '#888833', '#999933', '#9999FF', '#FF9999', '#9999FF']
|
602
|
+
graph['data'] = stacked_vals
|
603
|
+
|
604
|
+
@graph_data << graph
|
605
|
+
end
|
606
|
+
|
607
|
+
def getPctLoad(val, total)
|
608
|
+
if val != '' && total != 0
|
609
|
+
(val * 100 / total).round(2)
|
610
|
+
else
|
611
|
+
0
|
612
|
+
end
|
613
|
+
end
|
614
|
+
|
615
|
+
def getResourceFileData(fileName)
|
616
|
+
data_in_path = "#{File.dirname(__FILE__)}/resources/#{fileName}"
|
617
|
+
unless File.exist?(data_in_path)
|
618
|
+
data_in_path = "#{File.dirname(__FILE__)}/#{fileName}"
|
619
|
+
end
|
620
|
+
|
621
|
+
html_in = ''
|
622
|
+
File.open(data_in_path, 'r') do |file|
|
623
|
+
html_in = file.read
|
624
|
+
end
|
625
|
+
|
626
|
+
html_in
|
627
|
+
end
|
628
|
+
|
629
|
+
def writeResourceFileData(fileName, data)
|
630
|
+
File.open("./#{fileName}", 'w') do |file|
|
631
|
+
file << data
|
632
|
+
# make sure data is written to the disk one way or the other
|
633
|
+
begin
|
634
|
+
file.fsync
|
635
|
+
rescue StandardError
|
636
|
+
file.flush
|
637
|
+
end
|
638
|
+
end
|
639
|
+
end
|
640
|
+
|
641
|
+
# Fetch a value from the tabulardatawithstrings database view
|
642
|
+
# If final_units is "s" the value is returned unchanged as a string
|
643
|
+
# Otherwise the value is converted from units to final_units - units is specified in energy plus style (m2, m3, etc)
|
644
|
+
# and final_units should be open studio style (m^2, m^3, ...)
|
645
|
+
# If the data is not found or cannot be converted a warning is registered and "" or 0.0 is returned.
|
646
|
+
def getDetailsData(report, forstring, table, row, column, units, final_units)
|
647
|
+
forstring.upcase! if report == 'ZoneComponentLoadSummary'
|
648
|
+
|
649
|
+
str_HVACEquipment_query = 'SELECT Value FROM tabulardatawithstrings WHERE '
|
650
|
+
str_HVACEquipment_query << "ReportName='#{report}' AND "
|
651
|
+
str_HVACEquipment_query << "ReportForString='#{forstring}' AND "
|
652
|
+
str_HVACEquipment_query << "TableName='#{table}' AND "
|
653
|
+
str_HVACEquipment_query << "RowName LIKE '#{row}' AND "
|
654
|
+
str_HVACEquipment_query << "ColumnName='#{column}' AND "
|
655
|
+
str_HVACEquipment_query << "Units='#{units}'"
|
656
|
+
|
657
|
+
query_results = @sqlFile.execAndReturnFirstString(str_HVACEquipment_query)
|
658
|
+
|
659
|
+
if query_results.empty?
|
660
|
+
|
661
|
+
@runner.registerWarning("Could not get data for #{report} #{forstring} #{table} #{row} #{column}.")
|
662
|
+
return final_units == 's' ? '' : 0.0
|
663
|
+
|
664
|
+
else
|
665
|
+
r = query_results.get
|
666
|
+
if report == 'ZoneComponentLoadSummary'
|
667
|
+
@testData["#{@currentZoneName}_#{table}_#{row}"] = r
|
668
|
+
end
|
669
|
+
|
670
|
+
if final_units == 's'
|
671
|
+
return r
|
672
|
+
else
|
673
|
+
converted = OpenStudio.convert(r.to_f, eplus_to_openstudio(units), final_units)
|
674
|
+
if converted.empty?
|
675
|
+
@runner.registerError("Could not convert #{r} from #{units} to #{final_units}")
|
676
|
+
return 0.0
|
677
|
+
else
|
678
|
+
return converted.get.round(2)
|
679
|
+
end
|
680
|
+
end
|
681
|
+
|
682
|
+
end
|
683
|
+
end
|
684
|
+
|
685
|
+
def getDataByColumn(colName)
|
686
|
+
strvec_HVACEquipment_query = "SELECT #{colName} FROM tabulardatawithstrings WHERE "
|
687
|
+
strvec_HVACEquipment_query << "ReportName='EquipmentSummary' and "
|
688
|
+
strvec_HVACEquipment_query << "ReportForString='Entire Facility'"
|
689
|
+
strvec_HVACEquipment_query << 'ORDER BY TableName, ColumnName, RowName, Units, Value'
|
690
|
+
|
691
|
+
query_results = @sqlFile.execAndReturnVectorOfString(strvec_HVACEquipment_query).get
|
692
|
+
|
693
|
+
if query_results.empty?
|
694
|
+
@runner.registerError("Could not get data for requested Column #{colName}.")
|
695
|
+
return []
|
696
|
+
else
|
697
|
+
return query_results
|
698
|
+
end
|
699
|
+
end
|
700
|
+
|
701
|
+
# Accessor to support unit tests
|
702
|
+
attr_reader :zone_collection
|
703
|
+
end # end the measure
|
704
|
+
|
705
|
+
# this allows the measure to be use by the application
|
706
|
+
ZoneReport.new.registerWithApplication
|