/* Copyright (C) 2022 Fredrik Öhrström (gpl-3.0-or-later) This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #include"meters_common_implementation.h" namespace { struct Driver : public virtual MeterCommonImplementation { Driver(MeterInfo &mi, DriverInfo &di); }; static bool ok = registerDriver([](DriverInfo&di) { di.setName("enercal"); di.setDefaultFields("name,id,status,total_kwh,target_kwh,total_m3,target_m3,timestamp"); di.setMeterType(MeterType::HeatMeter); di.addLinkMode(LinkMode::MBUS); di.addDetection(MANUFACTURER_GWF, 0x04, 0x08); di.setConstructor([](MeterInfo& mi, DriverInfo& di){ return shared_ptr(new Driver(mi, di)); }); }); Driver::Driver(MeterInfo &mi, DriverInfo &di) : MeterCommonImplementation(mi, di) { addStringField( "status", "Meter status.", PrintProperty::STATUS | PrintProperty::INCLUDE_TPL_STATUS); addNumericFieldWithExtractor( "total", "The total energy consumption recorded by this meter.", DEFAULT_PRINT_PROPERTIES, Quantity::Energy, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::AnyEnergyVIF) ); addNumericFieldWithExtractor( "target", "The energy consumption recorded by this meter at the set date.", DEFAULT_PRINT_PROPERTIES, Quantity::Energy, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::AnyEnergyVIF) .set(StorageNr(1)) ); addNumericFieldWithExtractor( "power", "The active power consumption.", DEFAULT_PRINT_PROPERTIES, Quantity::Power, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::AnyPowerVIF) ); addNumericFieldWithExtractor( "flow", "The flow of water.", DEFAULT_PRINT_PROPERTIES, Quantity::Flow, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::VolumeFlow) ); addNumericFieldWithExtractor( "flow_max", "The maximum forward flow of water since the last set date?", DEFAULT_PRINT_PROPERTIES, Quantity::Flow, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Maximum) .set(VIFRange::VolumeFlow) ); addNumericFieldWithExtractor( "forward", "The forward temperature of the water.", DEFAULT_PRINT_PROPERTIES, Quantity::Temperature, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::FlowTemperature) ); addNumericFieldWithExtractor( "return", "The return temperature of the water.", DEFAULT_PRINT_PROPERTIES, Quantity::Temperature, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::ReturnTemperature) ); addNumericFieldWithExtractor( "difference", "The temperature difference forward-return for the water.", DEFAULT_PRINT_PROPERTIES, Quantity::Temperature, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::TemperatureDifference) ); addNumericFieldWithExtractor( "total", "The total amount of water that has passed through this meter.", DEFAULT_PRINT_PROPERTIES, Quantity::Volume, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::Volume) ); addNumericFieldWithExtractor( "target", "The amount of water that had passed through this meter at the set date.", DEFAULT_PRINT_PROPERTIES, Quantity::Volume, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::Volume) .set(StorageNr(1)) ); addNumericFieldWithExtractor( "subunit1", "The amount of water that has passed through subunit 1.", DEFAULT_PRINT_PROPERTIES, Quantity::Volume, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::Volume) .set(SubUnitNr(1)) ); addNumericFieldWithExtractor( "subunit1_target", "The amount of water that had passed through the subunit 1 at the set date.", DEFAULT_PRINT_PROPERTIES, Quantity::Volume, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::Volume) .set(StorageNr(1)) .set(SubUnitNr(1)) ); addNumericFieldWithExtractor( "subunit1", "The current heat cost allocation for subunit 1.", DEFAULT_PRINT_PROPERTIES, Quantity::HCA, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::HeatCostAllocation) .set(SubUnitNr(1)) ); addNumericFieldWithExtractor( "subunit1_target", "The heat cost allocation for subunit 1 at the target date.", DEFAULT_PRINT_PROPERTIES, Quantity::HCA, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::HeatCostAllocation) .set(SubUnitNr(1)) .set(StorageNr(1)) ); addNumericFieldWithExtractor( "subunit2", "The current heat cost allocation for subunit 2.", DEFAULT_PRINT_PROPERTIES, Quantity::HCA, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::HeatCostAllocation) .set(SubUnitNr(2)) ); addNumericFieldWithExtractor( "subunit2_target", "The heat cost allocation for subunit 2 at the target date.", DEFAULT_PRINT_PROPERTIES, Quantity::HCA, VifScaling::Auto, DifSignedness::Signed, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::HeatCostAllocation) .set(SubUnitNr(2)) .set(StorageNr(1)) ); addStringFieldWithExtractor( "target_date", "The most recent billing period date.", DEFAULT_PRINT_PROPERTIES, FieldMatcher::build() .set(MeasurementType::Instantaneous) .set(VIFRange::Date) .set(StorageNr(1)) ); addOptionalLibraryFields("operating_time_h,on_time_h,meter_datetime"); } } // Test: Heat enercal 00309924 NOKEY // Comment: // telegram=|688e8e6808017224993000e61e080406000000040681460600041488f2350084401426e02600025B2600025f220002622a00042247880200042647880200043B00000000042c00000000046d2d0ede2784406e000000008480406e00000000c4800006B1450600c48000143Be43500c4c000142fd22600c4c0006e00000000c480406e00000000c280006cc1271f00000000f416| // {"media":"heat","meter":"enercal","name":"Heat","id":"00309924","status":"OK","total_kwh":411265,"target_kwh":411057,"power_kw":0,"flow_m3h":0,"forward_c":38,"return_c":34,"difference_c":4.2,"total_m3":35354.96,"target_m3":35318.35,"subunit1_m3":25477.5,"subunit1_target_m3":25441.75,"subunit1_hca":0,"subunit1_target_hca":0,"subunit2_hca":0,"subunit2_target_hca":0,"target_date":"2022-07-01","operating_time_h":165959,"on_time_h":165959,"meter_datetime":"2022-07-30 14:45","timestamp":"1111-11-11T11:11:11Z"} // |Heat;00309924;OK;411265;411057;35354.96;35318.35;1111-11-11 11:11.11