/* Copyright (C) 2017-2020 Fredrik Öhrström 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 . */ #ifndef METER_H_ #define METER_H_ #include"util.h" #include"units.h" #include"wmbus.h" #include #include #include #define LIST_OF_METERS \ X(amiplus, T1_bit, ElectricityMeter, AMIPLUS, Amiplus) \ X(apator08, T1_bit, WaterMeter, APATOR08, Apator08) \ X(apator162, C1_bit|T1_bit, WaterMeter, APATOR162, Apator162) \ X(cma12w, C1_bit|T1_bit, TempHygroMeter, CMA12W, CMa12w) \ X(compact5, T1_bit, HeatMeter, COMPACT5, Compact5) \ X(ebzwmbe, T1_bit, ElectricityMeter, EBZWMBE, EBZWMBE) \ X(eurisii, T1_bit, HeatCostAllocationMeter, EURISII, EurisII) \ X(ehzp, T1_bit, ElectricityMeter, EHZP, EHZP) \ X(esyswm, T1_bit, ElectricityMeter, ESYSWM, ESYSWM) \ X(flowiq2200, C1_bit, WaterMeter, FLOWIQ2200, FlowIQ2200) \ X(flowiq3100, C1_bit, WaterMeter, FLOWIQ3100, FlowIQ3100) \ X(em24, C1_bit, ElectricityMeter, EM24, EM24) \ X(emerlin868, T1_bit, WaterMeter, EMERLIN868, EMerlin868) \ X(ev200, T1_bit, WaterMeter, EV200, EV200) \ X(evo868, T1_bit, WaterMeter, EVO868, EVO868) \ X(fhkvdataiii,T1_bit, HeatCostAllocationMeter, FHKVDATAIII, FHKVDataIII) \ X(fhkvdataiv, T1_bit, HeatCostAllocationMeter, FHKVDATAIV, FHKVDataIV) \ X(hydrus, T1_bit, WaterMeter, HYDRUS, Hydrus) \ X(hydrocalm3, T1_bit, HeatMeter, HYDROCALM3, HydrocalM3) \ X(hydrodigit, T1_bit, WaterMeter, HYDRODIGIT, Hydrodigit) \ X(iperl, T1_bit, WaterMeter, IPERL, Iperl) \ X(izar, T1_bit, WaterMeter, IZAR, Izar) \ X(izar3, T1_bit, WaterMeter, IZAR3, Izar3) \ X(lansensm, T1_bit, SmokeDetector, LANSENSM, LansenSM) \ X(lansenth, T1_bit, TempHygroMeter, LANSENTH, LansenTH) \ X(lansendw, T1_bit, DoorWindowDetector, LANSENDW, LansenDW) \ X(lansenpu, T1_bit, PulseCounter, LANSENPU, LansenPU) \ X(mkradio3, T1_bit, WaterMeter, MKRADIO3, MKRadio3) \ X(mkradio4, T1_bit, WaterMeter, MKRADIO4, MKRadio4) \ X(multical21, C1_bit|T1_bit, WaterMeter, MULTICAL21, Multical21) \ X(multical302,C1_bit|T1_bit, HeatMeter, MULTICAL302, Multical302) \ X(multical403,C1_bit, HeatMeter, MULTICAL403, Multical403) \ X(multical603,C1_bit, HeatMeter, MULTICAL603, Multical603) \ X(multical803,C1_bit, HeatMeter, MULTICAL803, Multical803) \ X(omnipower, C1_bit, ElectricityMeter, OMNIPOWER, Omnipower) \ X(rfmamb, T1_bit, TempHygroMeter, RFMAMB, RfmAmb) \ X(rfmtx1, T1_bit, WaterMeter, RFMTX1, RfmTX1) \ X(tsd2, T1_bit, SmokeDetector, TSD2, TSD2) \ X(q400, T1_bit, WaterMeter, Q400, Q400) \ X(qcaloric, C1_bit, HeatCostAllocationMeter, QCALORIC, QCaloric) \ X(sontex868, T1_bit, HeatCostAllocationMeter, SONTEX868, Sontex868) \ X(supercom587,T1_bit, WaterMeter, SUPERCOM587, Supercom587) \ X(ultrimis, T1_bit, WaterMeter, ULTRIMIS, Ultrimis) \ X(vario451, T1_bit, HeatMeter, VARIO451, Vario451) \ X(waterstarm, C1_bit|T1_bit, WaterMeter,WATERSTARM, WaterstarM) \ X(whe46x, S1_bit, HeatCostAllocationMeter, WHE46X, Whe46x) \ X(whe5x, S1_bit, HeatCostAllocationMeter, WHE5X, Whe5x) \ X(topaseskr, T1_bit, WaterMeter, TOPASESKR, TopasEsKr) \ X(sensostar, C1_bit|T1_bit, HeatMeter,SENSOSTAR, Sensostar) \ X(gransystems,T1_bit, ElectricityMeter, CCx01, CCx01) // List of numbers that can be used to detect the meter driver from a telegram. // // meter driver, manufacturer, media, version // #define METER_DETECTION \ X(AMIPLUS, MANUFACTURER_APA, 0x02, 0x02) \ X(AMIPLUS, MANUFACTURER_DEV, 0x37, 0x02) \ X(AMIPLUS, MANUFACTURER_DEV, 0x02, 0x00) \ X(APATOR08, 0x8614/*APT?*/, 0x03, 0x03) \ X(APATOR162, MANUFACTURER_APA, 0x06, 0x05) \ X(APATOR162, MANUFACTURER_APA, 0x07, 0x05) \ X(CMA12W, MANUFACTURER_ELV, 0x1b, 0x20) \ X(COMPACT5, MANUFACTURER_TCH, 0x04, 0x45) \ X(COMPACT5, MANUFACTURER_TCH, 0xc3, 0x45) \ X(COMPACT5, MANUFACTURER_TCH, 0x43, 0x22) \ X(COMPACT5, MANUFACTURER_TCH, 0x43, 0x45) \ X(EBZWMBE, MANUFACTURER_EBZ, 0x37, 0x02) \ X(EURISII, MANUFACTURER_INE, 0x08, 0x55) \ X(EHZP, MANUFACTURER_EMH, 0x02, 0x02) \ X(ESYSWM, MANUFACTURER_ESY, 0x37, 0x30) \ X(FLOWIQ2200,MANUFACTURER_KAW, 0x16, 0x3a) \ X(FLOWIQ3100,MANUFACTURER_KAM, 0x16, 0x1d) \ X(EM24, MANUFACTURER_KAM, 0x02, 0x33) \ X(EMERLIN868,MANUFACTURER_ELR, 0x37, 0x11) \ X(EV200, MANUFACTURER_ELR, 0x07, 0x0d) \ X(EVO868, MANUFACTURER_MAD, 0x07, 0x50) \ X(FHKVDATAIII,MANUFACTURER_TCH, 0x80, 0x69) \ X(FHKVDATAIII,MANUFACTURER_TCH, 0x80, 0x94) \ X(FHKVDATAIV,MANUFACTURER_TCH, 0x08, 0x69) \ X(FHKVDATAIV,MANUFACTURER_TCH, 0x08, 0x94) \ X(HYDRUS, MANUFACTURER_DME, 0x07, 0x70) \ X(HYDRUS, MANUFACTURER_HYD, 0x07, 0x24) \ X(HYDRUS, MANUFACTURER_DME, 0x06, 0x70) \ X(HYDRUS, MANUFACTURER_DME, 0x16, 0x70) \ X(HYDROCALM3,MANUFACTURER_BMT, 0x0d, 0x0b) \ X(HYDRODIGIT,MANUFACTURER_BMT, 0x07, 0x13) \ X(IPERL, MANUFACTURER_SEN, 0x06, 0x68) \ X(IPERL, MANUFACTURER_SEN, 0x07, 0x68) \ X(IPERL, MANUFACTURER_SEN, 0x07, 0x7c) \ X(IZAR, MANUFACTURER_SAP, 0x15, -1) \ X(IZAR, MANUFACTURER_SAP, 0x04, -1) \ X(IZAR, MANUFACTURER_SAP, 0x07, 0x00) \ X(IZAR, MANUFACTURER_DME, 0x07, 0x78) \ X(IZAR3, MANUFACTURER_SAP, 0x00, 0x88) \ X(LANSENSM, MANUFACTURER_LAS, 0x1a, 0x03) \ X(LANSENTH, MANUFACTURER_LAS, 0x1b, 0x07) \ X(LANSENDW, MANUFACTURER_LAS, 0x1d, 0x07) \ X(LANSENPU, MANUFACTURER_LAS, 0x00, 0x14) \ X(LANSENPU, MANUFACTURER_LAS, 0x00, 0x0b) \ X(MKRADIO3, MANUFACTURER_TCH, 0x62, 0x74) \ X(MKRADIO3, MANUFACTURER_TCH, 0x72, 0x74) \ X(MKRADIO4, MANUFACTURER_TCH, 0x62, 0x95) \ X(MKRADIO4, MANUFACTURER_TCH, 0x62, 0x70) \ X(MKRADIO4, MANUFACTURER_TCH, 0x72, 0x95) \ X(MKRADIO4, MANUFACTURER_TCH, 0x72, 0x70) \ X(MULTICAL21, MANUFACTURER_KAM, 0x06, 0x1b) \ X(MULTICAL21, MANUFACTURER_KAM, 0x16, 0x1b) \ X(MULTICAL302,MANUFACTURER_KAM, 0x04, 0x30) \ X(MULTICAL302,MANUFACTURER_KAM, 0x0d, 0x30) \ X(MULTICAL403,MANUFACTURER_KAM, 0x0a, 0x34) \ X(MULTICAL403,MANUFACTURER_KAM, 0x0b, 0x34) \ X(MULTICAL403,MANUFACTURER_KAM, 0x0c, 0x34) \ X(MULTICAL403,MANUFACTURER_KAM, 0x0d, 0x34) \ X(MULTICAL603,MANUFACTURER_KAM, 0x04, 0x35) \ X(MULTICAL803,MANUFACTURER_KAM, 0x04, 0x39) \ X(OMNIPOWER, MANUFACTURER_KAM, 0x02, 0x30) \ X(RFMAMB, MANUFACTURER_BMT, 0x1b, 0x10) \ X(RFMTX1, MANUFACTURER_BMT, 0x07, 0x05) \ X(TSD2, MANUFACTURER_TCH, 0xf0, 0x76) \ X(Q400, MANUFACTURER_AXI, 0x07, 0x10) \ X(QCALORIC, MANUFACTURER_QDS, 0x08, 0x35) \ X(SUPERCOM587,MANUFACTURER_SON, 0x06, 0x3c) \ X(SUPERCOM587,MANUFACTURER_SON, 0x07, 0x3c) \ X(SONTEX868, MANUFACTURER_SON, 0x08, 0x16) \ X(TOPASESKR, MANUFACTURER_AMT, 0x06, 0xf1) \ X(TOPASESKR, MANUFACTURER_AMT, 0x07, 0xf1) \ X(ULTRIMIS, MANUFACTURER_APA, 0x16, 0x01) \ X(VARIO451, MANUFACTURER_TCH, 0x04, 0x27) \ X(VARIO451, MANUFACTURER_TCH, 0xc3, 0x27) \ X(WATERSTARM, MANUFACTURER_DWZ, 0x06, 0x02) \ X(WATERSTARM, MANUFACTURER_DWZ, 0x07, 0x02) \ X(WATERSTARM, MANUFACTURER_EFE, 0x07, 0x03) \ X(WHE46X, MANUFACTURER_LSE, 0x08, 0x18) \ X(WHE5X, MANUFACTURER_LSE, 0x08, 0x34) \ X(SENSOSTAR, MANUFACTURER_EFE, 0x04, 0x00) \ X(CCx01, MANUFACTURER_GSS, 0x02, 0x01) enum class MeterType { #define X(mname,linkmode,info,type,cname) type, LIST_OF_METERS #undef X UNKNOWN }; struct MeterMatch { MeterType driver; int manufacturer; int media; int version; }; // Return a list of matching drivers, like: multical21 void detectMeterDriver(int manufacturer, int media, int version, std::vector *drivers); // When entering the driver, check that the telegram is indeed known to be // compatible with the driver(type), if not then print a warning. bool isMeterDriverValid(MeterType type, int manufacturer, int media, int version); using namespace std; typedef unsigned char uchar; struct MeterInfo { string name; string type; string id; string key; LinkModeSet link_modes; vector shells; vector jsons; // Additional static jsons that are added to each message. MeterInfo() { } MeterInfo(string n, string t, string i, string k, LinkModeSet lms, vector &s, vector &j) { name = n; type = t; id = i; key = k; shells = s; jsons = j; link_modes = lms; } }; struct Print { string vname; // Value name, like: total current previous target Quantity quantity; // Quantity: Energy, Volume Unit default_unit; // Default unit for above quantity: KWH, M3 function getValueDouble; // Callback to fetch the value from the meter. function getValueString; // Callback to fetch the value from the meter. string help; // Helpful information on this meters use of this value. bool field; // If true, print in hr/fields output. bool json; // If true, print in json and shell env variables. string field_name; // Field name for default unit. }; struct Meter { // This meter listens to these ids. virtual vector ids() = 0; // This meter can report these fields, like total_m3, temp_c. virtual vector fields() = 0; virtual vector prints() = 0; virtual string meterName() = 0; virtual string name() = 0; virtual MeterType type() = 0; virtual string datetimeOfUpdateHumanReadable() = 0; virtual string datetimeOfUpdateRobot() = 0; virtual void onUpdate(std::function cb) = 0; virtual int numUpdates() = 0; virtual void printMeter(Telegram *t, string *human_readable, string *fields, char separator, string *json, vector *envs, vector *more_json, vector *selected_fields) = 0; // The handleTelegram expects an input_frame where the DLL crcs have been removed. // Returns true of this meter handled this telegram! virtual bool handleTelegram(AboutTelegram &about, vector input_frame, bool simulated, string *id) = 0; virtual bool isTelegramForMe(Telegram *t) = 0; virtual MeterKeys *meterKeys() = 0; // Dynamically access all data received for the meter. virtual std::vector getRecords() = 0; virtual double getRecordAsDouble(std::string record) = 0; virtual uint16_t getRecordAsUInt16(std::string record) = 0; virtual void addConversions(std::vector cs) = 0; virtual void addShell(std::string cmdline) = 0; virtual vector &shellCmdlines() = 0; virtual ~Meter() = default; }; struct MeterManager { virtual void addMeter(shared_ptr meter) = 0; virtual Meter*lastAddedMeter() = 0; virtual void removeAllMeters() = 0; virtual void forEachMeter(std::function cb) = 0; virtual bool handleTelegram(AboutTelegram &about, vector data, bool simulated) = 0; virtual bool hasAllMetersReceivedATelegram() = 0; virtual bool hasMeters() = 0; virtual void onTelegram(function)> cb) = 0; virtual ~MeterManager() = default; }; shared_ptr createMeterManager(); struct WaterMeter : public virtual Meter { virtual double totalWaterConsumption(Unit u); // m3 virtual bool hasTotalWaterConsumption(); virtual double targetWaterConsumption(Unit u); // m3 virtual bool hasTargetWaterConsumption(); virtual double maxFlow(Unit u); // m3/s virtual bool hasMaxFlow(); virtual double flowTemperature(Unit u); // °C virtual bool hasFlowTemperature(); virtual double externalTemperature(Unit u); // °C virtual bool hasExternalTemperature(); virtual string statusHumanReadable(); virtual string status(); virtual string timeDry(); virtual string timeReversed(); virtual string timeLeaking(); virtual string timeBursting(); }; struct HeatMeter : public virtual Meter { virtual double totalEnergyConsumption(Unit u); // kwh virtual double currentPeriodEnergyConsumption(Unit u); // kwh virtual double previousPeriodEnergyConsumption(Unit u); // kwh virtual double currentPowerConsumption(Unit u); // kw virtual double totalVolume(Unit u); // m3 }; struct ElectricityMeter : public virtual Meter { virtual double totalEnergyConsumption(Unit u); // kwh virtual double totalEnergyProduction(Unit u); // kwh virtual double totalReactiveEnergyConsumption(Unit u); // kvarh virtual double totalReactiveEnergyProduction(Unit u); // kvarh virtual double totalApparentEnergyConsumption(Unit u); // kvah virtual double totalApparentEnergyProduction(Unit u); // kvah virtual double currentPowerConsumption(Unit u); // kw virtual double currentPowerProduction(Unit u); // kw }; struct HeatCostAllocationMeter : public virtual Meter { virtual double currentConsumption(Unit u); virtual string setDate(); virtual double consumptionAtSetDate(Unit u); }; struct TempMeter : public virtual Meter { virtual double currentTemperature(Unit u) = 0; // °C virtual ~TempMeter() = default; }; struct HygroMeter : public virtual Meter { virtual double currentRelativeHumidity() = 0; // RH virtual ~HygroMeter() = default; }; struct TempHygroMeter : public virtual TempMeter, public virtual HygroMeter { virtual ~TempHygroMeter() = default; }; struct SmokeDetector : public virtual Meter { virtual bool smokeDetected() = 0; virtual ~SmokeDetector() = default; }; struct DoorWindowDetector : public virtual Meter { virtual bool open() = 0; virtual ~DoorWindowDetector() = default; }; struct PulseCounter : public virtual Meter { virtual double counterA() = 0; virtual double counterB() = 0; virtual ~PulseCounter() = default; }; struct Generic : public virtual Meter { }; string toMeterName(MeterType mt); MeterType toMeterType(string& type); LinkModeSet toMeterLinkModeSet(string& type); #define X(mname,linkmode,info,type,cname) shared_ptr create##cname(MeterInfo &m); LIST_OF_METERS #undef X Generic *createGeneric(WMBus *bus, MeterInfo &m); #endif